summaryrefslogtreecommitdiffstats
path: root/WebKit/win/COMEnumVariant.h
diff options
context:
space:
mode:
Diffstat (limited to 'WebKit/win/COMEnumVariant.h')
-rw-r--r--WebKit/win/COMEnumVariant.h221
1 files changed, 221 insertions, 0 deletions
diff --git a/WebKit/win/COMEnumVariant.h b/WebKit/win/COMEnumVariant.h
new file mode 100644
index 0000000..be0ac8c
--- /dev/null
+++ b/WebKit/win/COMEnumVariant.h
@@ -0,0 +1,221 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifndef COMEnumVariant_h
+#define COMEnumVariant_h
+
+#define NOMINMAX
+#include <unknwn.h>
+
+#include <WebCore/BString.h>
+#include <wtf/Assertions.h>
+#include <wtf/Noncopyable.h>
+
+namespace WebCore {
+ class String;
+}
+
+template<typename T> struct COMVariantSetter {};
+
+template<> struct COMVariantSetter<WebCore::String>
+{
+ static void setVariant(VARIANT* variant, const WebCore::String& value)
+ {
+ ASSERT(V_VT(variant) == VT_EMPTY);
+
+ V_VT(variant) = VT_BSTR;
+ V_BSTR(variant) = WebCore::BString(value).release();
+ }
+};
+
+template<> struct COMVariantSetter<unsigned long long>
+{
+ static void setVariant(VARIANT* variant, unsigned long long value)
+ {
+ ASSERT(V_VT(variant) == VT_EMPTY);
+
+ V_VT(variant) = VT_UI8;
+ V_UI8(variant) = value;
+ }
+};
+
+template<typename COMType, typename UnderlyingType>
+struct COMIUnknownVariantSetter
+{
+ static void setVariant(VARIANT* variant, const UnderlyingType& value)
+ {
+ ASSERT(V_VT(variant) == VT_EMPTY);
+
+ V_VT(variant) = VT_UNKNOWN;
+ V_UNKNOWN(variant) = COMType::createInstance(value);
+ }
+};
+
+template<typename ContainerType>
+class COMEnumVariant : public IEnumVARIANT, Noncopyable {
+public:
+ static COMEnumVariant* adopt(ContainerType&);
+ static COMEnumVariant* createInstance(const ContainerType&);
+
+ // IUnknown
+ virtual HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, void** ppvObject);
+ virtual ULONG STDMETHODCALLTYPE AddRef();
+ virtual ULONG STDMETHODCALLTYPE Release();
+
+ // IEnumVARIANT
+ virtual HRESULT STDMETHODCALLTYPE Next(ULONG celt, VARIANT* rgVar, ULONG* pCeltFetched);
+ virtual HRESULT STDMETHODCALLTYPE Skip(ULONG celt);
+ virtual HRESULT STDMETHODCALLTYPE Reset();
+ virtual HRESULT STDMETHODCALLTYPE Clone(IEnumVARIANT** ppEnum);
+
+private:
+ COMEnumVariant()
+ : m_refCount(0)
+ {
+ }
+
+ COMEnumVariant(const ContainerType& container)
+ : m_refCount(0)
+ , m_container(container)
+ , m_currentPos(m_container.begin())
+ {
+ }
+
+ ~COMEnumVariant() {}
+
+ ULONG m_refCount;
+
+ ContainerType m_container;
+ typename ContainerType::const_iterator m_currentPos;
+};
+
+// COMEnumVariant ------------------------------------------------------------------
+template<typename ContainerType>
+COMEnumVariant<typename ContainerType>* COMEnumVariant<ContainerType>::adopt(ContainerType& container)
+{
+ COMEnumVariant* instance = new COMEnumVariant;
+ instance->m_container.swap(container);
+ instance->m_currentPos = instance->m_container.begin();
+ instance->AddRef();
+ return instance;
+}
+
+template<typename ContainerType>
+COMEnumVariant<typename ContainerType>* COMEnumVariant<ContainerType>::createInstance(const ContainerType& container)
+{
+ COMEnumVariant* instance = new COMEnumVariant(container);
+ instance->AddRef();
+ return instance;
+}
+
+// IUnknown ------------------------------------------------------------------------
+template<typename ContainerType>
+HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::QueryInterface(REFIID riid, void** ppvObject)
+{
+ *ppvObject = 0;
+ if (IsEqualGUID(riid, IID_IUnknown))
+ *ppvObject = static_cast<COMEnumVariant*>(this);
+ else if (IsEqualGUID(riid, IID_IEnumVARIANT))
+ *ppvObject = static_cast<COMEnumVariant*>(this);
+ else
+ return E_NOINTERFACE;
+
+ AddRef();
+ return S_OK;
+}
+
+template<typename ContainerType>
+ULONG STDMETHODCALLTYPE COMEnumVariant<ContainerType>::AddRef()
+{
+ return ++m_refCount;
+}
+
+template<typename ContainerType>
+ULONG STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Release()
+{
+ ULONG newRef = --m_refCount;
+ if (!newRef)
+ delete this;
+
+ return newRef;
+}
+
+// IEnumVARIANT --------------------------------------------------------------------
+template<typename ContainerType>
+HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Next(ULONG celt, VARIANT* rgVar, ULONG* pCeltFetched)
+{
+ if (pCeltFetched)
+ *pCeltFetched = 0;
+ if (!rgVar)
+ return E_POINTER;
+ for (unsigned i = 0 ; i < celt; i++)
+ VariantInit(&rgVar[i]);
+
+ for (unsigned i = 0; i < celt; i++) {
+ if (m_currentPos == m_container.end())
+ return S_FALSE;
+
+ COMVariantSetter<ContainerType::ValueType>::setVariant(&rgVar[i], *m_currentPos);
+ ++m_currentPos;
+ if (pCeltFetched)
+ (*pCeltFetched)++;
+ }
+
+ return S_OK;
+}
+
+template<typename ContainerType>
+HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Skip(ULONG celt)
+{
+ for (unsigned i = 0; i < celt; i++) {
+ if (m_currentPos == m_container.end())
+ return S_FALSE;
+
+ ++m_currentPos;
+ }
+ return S_OK;
+}
+
+template<typename ContainerType>
+HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Reset()
+{
+ m_currentPos = m_container.begin();
+ return S_OK;
+}
+
+template<typename ContainerType>
+HRESULT STDMETHODCALLTYPE COMEnumVariant<ContainerType>::Clone(IEnumVARIANT** ppEnum)
+{
+ if (!ppEnum)
+ return E_POINTER;
+
+ *ppEnum = 0;
+ return E_NOTIMPL;
+}
+
+#endif