2 * Copyright (C) 2007 Apple Inc. All rights reserved.
\r
4 * Redistribution and use in source and binary forms, with or without
\r
5 * modification, are permitted provided that the following conditions
\r
8 * 1. Redistributions of source code must retain the above copyright
\r
9 * notice, this list of conditions and the following disclaimer.
\r
10 * 2. Redistributions in binary form must reproduce the above copyright
\r
11 * notice, this list of conditions and the following disclaimer in the
\r
12 * documentation and/or other materials provided with the distribution.
\r
13 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
\r
14 * its contributors may be used to endorse or promote products derived
\r
15 * from this software without specific prior written permission.
\r
17 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
\r
18 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
\r
19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
\r
20 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
\r
21 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
\r
22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
\r
23 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
\r
24 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
\r
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
\r
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
\r
29 #ifndef COMEnumVariant_h
\r
30 #define COMEnumVariant_h
\r
35 #include <WebCore/BString.h>
\r
36 #include <wtf/Assertions.h>
\r
37 #include <wtf/Noncopyable.h>
\r
43 template<typename T> struct COMVariantSetter {};
\r
45 template<> struct COMVariantSetter<WebCore::String>
\r
47 static void setVariant(VARIANT* variant, const WebCore::String& value)
\r
49 ASSERT(V_VT(variant) == VT_EMPTY);
\r
51 V_VT(variant) = VT_BSTR;
\r
52 V_BSTR(variant) = WebCore::BString(value).release();
\r
56 template<> struct COMVariantSetter<unsigned long long>
\r
58 static void setVariant(VARIANT* variant, unsigned long long value)
\r
60 ASSERT(V_VT(variant) == VT_EMPTY);
\r
62 V_VT(variant) = VT_UI8;
\r
63 V_UI8(variant) = value;
\r
67 template<typename COMType, typename UnderlyingType>
\r
68 struct COMIUnknownVariantSetter
\r
70 static void setVariant(VARIANT* variant, const UnderlyingType& value)
\r
72 ASSERT(V_VT(variant) == VT_EMPTY);
\r
74 V_VT(variant) = VT_UNKNOWN;
\r
75 V_UNKNOWN(variant) = COMType::createInstance(value);
\r
79 template<typename ContainerType>
\r
80 class COMEnumVariant : public IEnumVARIANT, Noncopyable {
\r
82 static COMEnumVariant* adopt(ContainerType&);
\r
83 static COMEnumVariant* createInstance(const ContainerType&);
\r
86 virtual HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, void** ppvObject);
\r
87 virtual ULONG STDMETHODCALLTYPE AddRef();
\r
88 virtual ULONG STDMETHODCALLTYPE Release();
\r
91 virtual HRESULT STDMETHODCALLTYPE Next(ULONG celt, VARIANT* rgVar, ULONG* pCeltFetched);
\r
92 virtual HRESULT STDMETHODCALLTYPE Skip(ULONG celt);
\r
93 virtual HRESULT STDMETHODCALLTYPE Reset();
\r
94 virtual HRESULT STDMETHODCALLTYPE Clone(IEnumVARIANT** ppEnum);
\r
102 COMEnumVariant(const ContainerType& container)
\r
104 , m_container(container)
\r
105 , m_currentPos(m_container.begin())
\r
109 ~COMEnumVariant() {}
\r
113 ContainerType m_container;
\r
114 typename ContainerType::const_iterator m_currentPos;
\r
117 // COMEnumVariant ------------------------------------------------------------------
\r
118 template<typename ContainerType>
\r
119 COMEnumVariant<typename ContainerType>* COMEnumVariant<ContainerType>::adopt(ContainerType& container)
\r
121 COMEnumVariant* instance = new COMEnumVariant;
\r
122 instance->m_container.swap(container);
\r
123 instance->m_currentPos = instance->m_container.begin();
\r
124 instance->AddRef();
\r
128 template<typename ContainerType>
\r
129 COMEnumVariant<typename ContainerType>* COMEnumVariant<ContainerType>::createInstance(const ContainerType& container)
\r
131 COMEnumVariant* instance = new COMEnumVariant(container);
\r
132 instance->AddRef();
\r
136 // IUnknown ------------------------------------------------------------------------
\r
137 template<typename ContainerType>
\r
138 HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::QueryInterface(REFIID riid, void** ppvObject)
\r
141 if (IsEqualGUID(riid, IID_IUnknown))
\r
142 *ppvObject = static_cast<COMEnumVariant*>(this);
\r
143 else if (IsEqualGUID(riid, IID_IEnumVARIANT))
\r
144 *ppvObject = static_cast<COMEnumVariant*>(this);
\r
146 return E_NOINTERFACE;
\r
152 template<typename ContainerType>
\r
153 ULONG STDMETHODCALLTYPE COMEnumVariant<ContainerType>::AddRef()
\r
155 return ++m_refCount;
\r
158 template<typename ContainerType>
\r
159 ULONG STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Release()
\r
161 ULONG newRef = --m_refCount;
\r
168 // IEnumVARIANT --------------------------------------------------------------------
\r
169 template<typename ContainerType>
\r
170 HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Next(ULONG celt, VARIANT* rgVar, ULONG* pCeltFetched)
\r
176 for (unsigned i = 0 ; i < celt; i++)
\r
177 VariantInit(&rgVar[i]);
\r
179 for (unsigned i = 0; i < celt; i++) {
\r
180 if (m_currentPos == m_container.end())
\r
183 COMVariantSetter<ContainerType::ValueType>::setVariant(&rgVar[i], *m_currentPos);
\r
192 template<typename ContainerType>
\r
193 HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Skip(ULONG celt)
\r
195 for (unsigned i = 0; i < celt; i++) {
\r
196 if (m_currentPos == m_container.end())
\r
204 template<typename ContainerType>
\r
205 HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Reset()
\r
207 m_currentPos = m_container.begin();
\r
211 template<typename ContainerType>
\r
212 HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Clone(IEnumVARIANT** ppEnum)
\r