1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: wx/msw/ole/oleutils.h
3 // Purpose: OLE helper routines, OLE debugging support &c
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
12 #ifndef _WX_OLEUTILS_H
13 #define _WX_OLEUTILS_H
19 // ole2.h includes windows.h, so include wrapwin.h first
20 #include "wx/msw/wrapwin.h"
21 // get IUnknown, REFIID &c
26 // ============================================================================
27 // General purpose functions and macros
28 // ============================================================================
30 // ----------------------------------------------------------------------------
31 // initialize/cleanup OLE
32 // ----------------------------------------------------------------------------
34 // call OleInitialize() or CoInitialize[Ex]() depending on the platform
36 // return true if ok, false otherwise
37 inline bool wxOleInitialize()
41 hr
= ::CoInitializeEx(NULL
, COINIT_MULTITHREADED
);
43 hr
= ::OleInitialize(NULL
);
46 // RPC_E_CHANGED_MODE indicates that OLE had been already initialized
47 // before, albeit with different mode. Don't consider it to be an error as
48 // we don't actually care ourselves about the mode used so this allows the
49 // main application to call OleInitialize() on its own before we do if it
50 // needs non-default mode.
51 if ( hr
!= RPC_E_CHANGED_MODE
&& FAILED(hr
) )
53 wxLogError(_("Cannot initialize OLE"));
61 inline void wxOleUninitialize()
70 // ----------------------------------------------------------------------------
71 // misc helper functions/macros
72 // ----------------------------------------------------------------------------
74 // release the interface pointer (if !NULL)
75 inline void ReleaseInterface(IUnknown
*pIUnk
)
81 // release the interface pointer (if !NULL) and make it NULL
82 #define RELEASE_AND_NULL(p) if ( (p) != NULL ) { p->Release(); p = NULL; };
84 // return true if the iid is in the array
85 extern WXDLLIMPEXP_CORE
bool IsIidFromList(REFIID riid
, const IID
*aIids
[], size_t nCount
);
87 // ============================================================================
88 // IUnknown implementation helpers
89 // ============================================================================
92 The most dumb implementation of IUnknown methods. We don't support
93 aggregation nor containment, but for 99% of cases this simple
94 implementation is quite enough.
96 Usage is trivial: here is all you should have
97 1) DECLARE_IUNKNOWN_METHODS in your (IUnknown derived!) class declaration
98 2) BEGIN/END_IID_TABLE with ADD_IID in between for all interfaces you
99 support (at least all for which you intent to return 'this' from QI,
100 i.e. you should derive from IFoo if you have ADD_IID(Foo)) somewhere else
101 3) IMPLEMENT_IUNKNOWN_METHODS somewhere also
103 These macros are quite simple: AddRef and Release are trivial and QI does
104 lookup in a static member array of IIDs and returns 'this' if it founds
105 the requested interface in it or E_NOINTERFACE if not.
109 wxAutoULong: this class is used for automatically initalising m_cRef to 0
114 wxAutoULong(ULONG value
= 0) : m_Value(value
) { }
116 operator ULONG
&() { return m_Value
; }
117 ULONG
& operator=(ULONG value
) { m_Value
= value
; return m_Value
; }
119 wxAutoULong
& operator++() { ++m_Value
; return *this; }
120 const wxAutoULong
operator++( int ) { wxAutoULong temp
= *this; ++m_Value
; return temp
; }
122 wxAutoULong
& operator--() { --m_Value
; return *this; }
123 const wxAutoULong
operator--( int ) { wxAutoULong temp
= *this; --m_Value
; return temp
; }
129 // declare the methods and the member variable containing reference count
130 // you must also define the ms_aIids array somewhere with BEGIN_IID_TABLE
131 // and friends (see below)
133 #define DECLARE_IUNKNOWN_METHODS \
135 STDMETHODIMP QueryInterface(REFIID, void **); \
136 STDMETHODIMP_(ULONG) AddRef(); \
137 STDMETHODIMP_(ULONG) Release(); \
139 static const IID *ms_aIids[]; \
142 // macros for declaring supported interfaces
143 // NB: ADD_IID prepends IID_I whereas ADD_RAW_IID does not
144 #define BEGIN_IID_TABLE(cname) const IID *cname::ms_aIids[] = {
145 #define ADD_IID(iid) &IID_I##iid,
146 #define ADD_RAW_IID(iid) &iid,
147 #define END_IID_TABLE }
149 // implementation is as straightforward as possible
150 // Parameter: classname - the name of the class
151 #define IMPLEMENT_IUNKNOWN_METHODS(classname) \
152 STDMETHODIMP classname::QueryInterface(REFIID riid, void **ppv) \
154 wxLogQueryInterface(wxT(#classname), riid); \
156 if ( IsIidFromList(riid, ms_aIids, WXSIZEOF(ms_aIids)) ) { \
165 return (HRESULT) E_NOINTERFACE; \
169 STDMETHODIMP_(ULONG) classname::AddRef() \
171 wxLogAddRef(wxT(#classname), m_cRef); \
176 STDMETHODIMP_(ULONG) classname::Release() \
178 wxLogRelease(wxT(#classname), m_cRef); \
180 if ( --m_cRef == wxAutoULong(0) ) { \
188 // ============================================================================
190 // ============================================================================
192 // VZ: I don't know it's not done for compilers other than VC++ but I leave it
193 // as is. Please note, though, that tracing OLE interface calls may be
194 // incredibly useful when debugging OLE programs.
195 #if defined(__WXDEBUG__) && (( defined(__VISUALC__) && (__VISUALC__ >= 1000) ))
196 // ----------------------------------------------------------------------------
197 // All OLE specific log functions have DebugTrace level (as LogTrace)
198 // ----------------------------------------------------------------------------
200 // tries to translate riid into a symbolic name, if possible
201 WXDLLIMPEXP_CORE
void wxLogQueryInterface(const wxChar
*szInterface
, REFIID riid
);
203 // these functions print out the new value of reference counter
204 WXDLLIMPEXP_CORE
void wxLogAddRef (const wxChar
*szInterface
, ULONG cRef
);
205 WXDLLIMPEXP_CORE
void wxLogRelease(const wxChar
*szInterface
, ULONG cRef
);
208 #define wxLogQueryInterface(szInterface, riid)
209 #define wxLogAddRef(szInterface, cRef)
210 #define wxLogRelease(szInterface, cRef)
213 // wrapper around BSTR type (by Vadim Zeitlin)
215 class WXDLLIMPEXP_CORE wxBasicString
219 wxBasicString(const wxString
& str
);
220 wxBasicString(const wxBasicString
& bstr
);
223 wxBasicString
& operator=(const wxBasicString
& bstr
);
226 // just get the string
227 operator BSTR() const { return m_bstrBuf
; }
228 // retrieve a copy of our string - caller must SysFreeString() it later!
229 BSTR
Get() const { return SysAllocString(m_bstrBuf
); }
238 class WXDLLIMPEXP_FWD_BASE wxVariant
;
240 // wrapper for CURRENCY type used in VARIANT (VARIANT.vt == VT_CY)
241 class WXDLLIMPEXP_CORE wxVariantDataCurrency
: public wxVariantData
244 wxVariantDataCurrency() { VarCyFromR8(0.0, &m_value
); }
245 wxVariantDataCurrency(CURRENCY value
) { m_value
= value
; }
247 CURRENCY
GetValue() const { return m_value
; }
248 void SetValue(CURRENCY value
) { m_value
= value
; }
250 virtual bool Eq(wxVariantData
& data
) const;
252 #if wxUSE_STD_IOSTREAM
253 virtual bool Write(wxSTD ostream
& str
) const;
255 virtual bool Write(wxString
& str
) const;
257 wxVariantData
* Clone() const { return new wxVariantDataCurrency(m_value
); }
258 virtual wxString
GetType() const { return wxS("currency"); }
260 DECLARE_WXANY_CONVERSION()
267 // wrapper for SCODE type used in VARIANT (VARIANT.vt == VT_ERROR)
268 class WXDLLIMPEXP_CORE wxVariantDataErrorCode
: public wxVariantData
271 wxVariantDataErrorCode(SCODE value
= S_OK
) { m_value
= value
; }
273 SCODE
GetValue() const { return m_value
; }
274 void SetValue(SCODE value
) { m_value
= value
; }
276 virtual bool Eq(wxVariantData
& data
) const;
278 #if wxUSE_STD_IOSTREAM
279 virtual bool Write(wxSTD ostream
& str
) const;
281 virtual bool Write(wxString
& str
) const;
283 wxVariantData
* Clone() const { return new wxVariantDataErrorCode(m_value
); }
284 virtual wxString
GetType() const { return wxS("errorcode"); }
286 DECLARE_WXANY_CONVERSION()
292 // wrapper for SAFEARRAY, used for passing multidimensional arrays in wxVariant
293 class WXDLLIMPEXP_CORE wxVariantDataSafeArray
: public wxVariantData
296 wxEXPLICIT
wxVariantDataSafeArray(SAFEARRAY
* value
= NULL
)
301 SAFEARRAY
* GetValue() const { return m_value
; }
302 void SetValue(SAFEARRAY
* value
) { m_value
= value
; }
304 virtual bool Eq(wxVariantData
& data
) const;
306 #if wxUSE_STD_IOSTREAM
307 virtual bool Write(wxSTD ostream
& str
) const;
309 virtual bool Write(wxString
& str
) const;
311 wxVariantData
* Clone() const { return new wxVariantDataSafeArray(m_value
); }
312 virtual wxString
GetType() const { return wxS("safearray"); }
314 DECLARE_WXANY_CONVERSION()
321 WXDLLIMPEXP_CORE
bool wxConvertVariantToOle(const wxVariant
& variant
, VARIANTARG
& oleVariant
);
322 WXDLLIMPEXP_CORE
bool wxConvertOleToVariant(const VARIANTARG
& oleVariant
, wxVariant
& variant
);
323 #endif // wxUSE_VARIANT
325 // Convert string to Unicode
326 WXDLLIMPEXP_CORE BSTR
wxConvertStringToOle(const wxString
& str
);
328 // Convert string from BSTR to wxString
329 WXDLLIMPEXP_CORE wxString
wxConvertStringFromOle(BSTR bStr
);
333 // ----------------------------------------------------------------------------
334 // stub functions to avoid #if wxUSE_OLE in the main code
335 // ----------------------------------------------------------------------------
337 inline bool wxOleInitialize() { return false; }
338 inline void wxOleUninitialize() { }
340 #endif // wxUSE_OLE/!wxUSE_OLE
342 #endif //_WX_OLEUTILS_H