]> git.saurik.com Git - wxWidgets.git/blame - src/msw/ole/automtn.cpp
change not only the focus but also last remembered TLW focus in SetValue() to avoid...
[wxWidgets.git] / src / msw / ole / automtn.cpp
CommitLineData
d980b3e1 1/////////////////////////////////////////////////////////////////////////////
11f104e5 2// Name: src/msw/ole/automtn.cpp
d980b3e1
JS
3// Purpose: OLE automation utilities
4// Author: Julian Smart
5// Modified by:
6// Created: 11/6/98
7// RCS-ID: $Id$
8// Copyright: (c) 1998, Julian Smart
65571936 9// Licence: wxWindows licence
d980b3e1
JS
10/////////////////////////////////////////////////////////////////////////////
11
d980b3e1
JS
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#if defined(__BORLANDC__)
7520f3da 16 #pragma hdrstop
d980b3e1
JS
17#endif
18
f6bcfd97 19// With Borland C++, all samples crash if this is compiled in.
ca5c6ac3
VZ
20#if (defined(__BORLANDC__) && (__BORLANDC__ < 0x520)) || defined(__CYGWIN10__)
21 #undef wxUSE_OLE_AUTOMATION
22 #define wxUSE_OLE_AUTOMATION 0
23#endif
24
25#if wxUSE_OLE_AUTOMATION
457e6c54 26
e4db172a
WS
27#ifndef WX_PRECOMP
28 #include "wx/log.h"
18680f86 29 #include "wx/math.h"
e4db172a
WS
30#endif
31
5283098e 32#define _FORCENAMELESSUNION
4676948b 33#include "wx/msw/private.h"
ed5317e5 34#include "wx/msw/ole/oleutils.h"
42e69d6b 35#include "wx/msw/ole/automtn.h"
4676948b
JS
36
37#ifdef __WXWINCE__
38#include "wx/msw/wince/time.h"
39#else
f6bcfd97 40#include <time.h>
4676948b 41#endif
f6bcfd97 42
7dee726c
RS
43#include <wtypes.h>
44#include <unknwn.h>
4676948b 45
7dee726c
RS
46#include <ole2.h>
47#define _huge
4676948b
JS
48
49#ifndef __WXWINCE__
42e69d6b 50#include <ole2ver.h>
4676948b
JS
51#endif
52
42e69d6b 53#include <oleauto.h>
17b74d79 54
61bfe4b0
JS
55#if wxUSE_DATETIME
56#include "wx/datetime.h"
b0f76951 57#endif // wxUSE_DATETIME
d980b3e1
JS
58
59static void ClearVariant(VARIANTARG *pvarg) ;
60static void ReleaseVariant(VARIANTARG *pvarg) ;
61// static void ShowException(LPOLESTR szMember, HRESULT hr, EXCEPINFO *pexcep, unsigned int uiArgErr);
62
63/*
64 * wxAutomationObject
65 */
66
67wxAutomationObject::wxAutomationObject(WXIDISPATCH* dispatchPtr)
68{
0a0e6a5b 69 m_dispatchPtr = dispatchPtr;
d980b3e1
JS
70}
71
72wxAutomationObject::~wxAutomationObject()
73{
0a0e6a5b
WS
74 if (m_dispatchPtr)
75 {
76 ((IDispatch*)m_dispatchPtr)->Release();
77 m_dispatchPtr = NULL;
78 }
d980b3e1
JS
79}
80
81#define INVOKEARG(i) (args ? args[i] : *(ptrArgs[i]))
82
83// For Put/Get, no named arguments are allowed.
84bool wxAutomationObject::Invoke(const wxString& member, int action,
85 wxVariant& retValue, int noArgs, wxVariant args[], const wxVariant* ptrArgs[]) const
86{
0a0e6a5b
WS
87 if (!m_dispatchPtr)
88 return false;
89
90 // nonConstMember is necessary because the wxString class doesn't have enough consts...
91 wxString nonConstMember(member);
92
93 int ch = nonConstMember.Find('.');
94 if (ch != -1)
95 {
96 // Use dot notation to get the next object
97 wxString member2(nonConstMember.Left((size_t) ch));
e4db172a 98 wxString rest(nonConstMember.Right(nonConstMember.length() - ch - 1));
0a0e6a5b
WS
99 wxAutomationObject obj;
100 if (!GetObject(obj, member2))
101 return false;
102 return obj.Invoke(rest, action, retValue, noArgs, args, ptrArgs);
103 }
104
105 VARIANTARG vReturn;
106 ClearVariant(& vReturn);
107
108 VARIANTARG* vReturnPtr = & vReturn;
109
110 // Find number of names args
111 int namedArgCount = 0;
112 int i;
113 for (i = 0; i < noArgs; i++)
114 if (!INVOKEARG(i).GetName().IsNull())
12335fa6 115 {
0a0e6a5b
WS
116 namedArgCount ++;
117 }
118
119 int namedArgStringCount = namedArgCount + 1;
120 BSTR* argNames = new BSTR[namedArgStringCount];
121 argNames[0] = wxConvertStringToOle(member);
122
123 // Note that arguments are specified in reverse order
124 // (all totally logical; hey, we're dealing with OLE here.)
125
126 int j = 0;
127 for (i = 0; i < namedArgCount; i++)
128 {
129 if (!INVOKEARG(i).GetName().IsNull())
130 {
131 argNames[(namedArgCount-j)] = wxConvertStringToOle(INVOKEARG(i).GetName());
132 j ++;
133 }
134 }
135
136 // + 1 for the member name, + 1 again in case we're a 'put'
137 DISPID* dispIds = new DISPID[namedArgCount + 2];
138
139 HRESULT hr;
140 DISPPARAMS dispparams;
141 unsigned int uiArgErr;
142 EXCEPINFO excep;
143
144 // Get the IDs for the member and its arguments. GetIDsOfNames expects the
145 // member name as the first name, followed by argument names (if any).
146 hr = ((IDispatch*)m_dispatchPtr)->GetIDsOfNames(IID_NULL, argNames,
147 1 + namedArgCount, LOCALE_SYSTEM_DEFAULT, dispIds);
148 if (FAILED(hr))
149 {
150// ShowException(szMember, hr, NULL, 0);
151 delete[] argNames;
152 delete[] dispIds;
153 return false;
154 }
155
156 // if doing a property put(ref), we need to adjust the first argument to have a
157 // named arg of DISPID_PROPERTYPUT.
158 if (action & (DISPATCH_PROPERTYPUT | DISPATCH_PROPERTYPUTREF))
159 {
160 namedArgCount = 1;
161 dispIds[1] = DISPID_PROPERTYPUT;
162 vReturnPtr = (VARIANTARG*) NULL;
163 }
164
165 // Convert the wxVariants to VARIANTARGs
166 VARIANTARG* oleArgs = new VARIANTARG[noArgs];
167 for (i = 0; i < noArgs; i++)
168 {
169 // Again, reverse args
170 if (!wxConvertVariantToOle(INVOKEARG((noArgs-1) - i), oleArgs[i]))
171 {
172 delete[] argNames;
173 delete[] dispIds;
12335fa6 174 delete[] oleArgs;
0a0e6a5b
WS
175 return false;
176 }
177 }
178
179 dispparams.rgdispidNamedArgs = dispIds + 1;
180 dispparams.rgvarg = oleArgs;
181 dispparams.cArgs = noArgs;
182 dispparams.cNamedArgs = namedArgCount;
183
184 excep.pfnDeferredFillIn = NULL;
185
186 hr = ((IDispatch*)m_dispatchPtr)->Invoke(dispIds[0], IID_NULL, LOCALE_SYSTEM_DEFAULT,
5c519b6c 187 (WORD)action, &dispparams, vReturnPtr, &excep, &uiArgErr);
0a0e6a5b
WS
188
189 for (i = 0; i < namedArgStringCount; i++)
190 {
191 SysFreeString(argNames[i]);
192 }
193 delete[] argNames;
194 delete[] dispIds;
195
196 for (i = 0; i < noArgs; i++)
197 ReleaseVariant(& oleArgs[i]) ;
198 delete[] oleArgs;
199
200 if (FAILED(hr))
201 {
202 // display the exception information if appropriate:
203// ShowException((const char*) member, hr, &excep, uiArgErr);
204
205 // free exception structure information
206 SysFreeString(excep.bstrSource);
207 SysFreeString(excep.bstrDescription);
208 SysFreeString(excep.bstrHelpFile);
209
210 if (vReturnPtr)
211 ReleaseVariant(vReturnPtr);
212 return false;
213 }
214 else
215 {
216 if (vReturnPtr)
217 {
218 // Convert result to wxVariant form
219 wxConvertOleToVariant(vReturn, retValue);
220 // Mustn't release the dispatch pointer
221 if (vReturn.vt == VT_DISPATCH)
222 {
223 vReturn.pdispVal = (IDispatch*) NULL;
224 }
225 ReleaseVariant(& vReturn);
12335fa6 226 }
0a0e6a5b
WS
227 }
228 return true;
d980b3e1
JS
229}
230
231// Invoke a member function
232wxVariant wxAutomationObject::CallMethod(const wxString& member, int noArgs, wxVariant args[])
233{
0a0e6a5b
WS
234 wxVariant retVariant;
235 if (!Invoke(member, DISPATCH_METHOD, retVariant, noArgs, args))
236 {
237 retVariant.MakeNull();
238 }
239 return retVariant;
d980b3e1
JS
240}
241
24f4ad95
JS
242wxVariant wxAutomationObject::CallMethodArray(const wxString& member, int noArgs, const wxVariant **args)
243{
0a0e6a5b
WS
244 wxVariant retVariant;
245 if (!Invoke(member, DISPATCH_METHOD, retVariant, noArgs, NULL, args))
246 {
247 retVariant.MakeNull();
248 }
249 return retVariant;
24f4ad95
JS
250}
251
d980b3e1 252wxVariant wxAutomationObject::CallMethod(const wxString& member,
0a0e6a5b
WS
253 const wxVariant& arg1, const wxVariant& arg2,
254 const wxVariant& arg3, const wxVariant& arg4,
255 const wxVariant& arg5, const wxVariant& arg6)
d980b3e1 256{
0a0e6a5b
WS
257 const wxVariant** args = new const wxVariant*[6];
258 int i = 0;
259 if (!arg1.IsNull())
260 {
261 args[i] = & arg1;
262 i ++;
263 }
264 if (!arg2.IsNull())
265 {
266 args[i] = & arg2;
267 i ++;
268 }
269 if (!arg3.IsNull())
270 {
271 args[i] = & arg3;
272 i ++;
273 }
274 if (!arg4.IsNull())
275 {
276 args[i] = & arg4;
277 i ++;
278 }
279 if (!arg5.IsNull())
280 {
281 args[i] = & arg5;
282 i ++;
283 }
284 if (!arg6.IsNull())
285 {
286 args[i] = & arg6;
287 i ++;
288 }
289 wxVariant retVariant;
290 if (!Invoke(member, DISPATCH_METHOD, retVariant, i, NULL, args))
291 {
292 retVariant.MakeNull();
293 }
294 delete[] args;
295 return retVariant;
d980b3e1
JS
296}
297
298// Get/Set property
24f4ad95
JS
299wxVariant wxAutomationObject::GetPropertyArray(const wxString& property, int noArgs, const wxVariant **args) const
300{
0a0e6a5b
WS
301 wxVariant retVariant;
302 if (!Invoke(property, DISPATCH_PROPERTYGET, retVariant, noArgs, NULL, args))
303 {
304 retVariant.MakeNull();
305 }
306 return retVariant;
24f4ad95 307}
d980b3e1
JS
308wxVariant wxAutomationObject::GetProperty(const wxString& property, int noArgs, wxVariant args[]) const
309{
0a0e6a5b
WS
310 wxVariant retVariant;
311 if (!Invoke(property, DISPATCH_PROPERTYGET, retVariant, noArgs, args))
312 {
313 retVariant.MakeNull();
314 }
315 return retVariant;
d980b3e1
JS
316}
317
318wxVariant wxAutomationObject::GetProperty(const wxString& property,
0a0e6a5b
WS
319 const wxVariant& arg1, const wxVariant& arg2,
320 const wxVariant& arg3, const wxVariant& arg4,
321 const wxVariant& arg5, const wxVariant& arg6)
d980b3e1 322{
0a0e6a5b
WS
323 const wxVariant** args = new const wxVariant*[6];
324 int i = 0;
325 if (!arg1.IsNull())
326 {
327 args[i] = & arg1;
328 i ++;
329 }
330 if (!arg2.IsNull())
331 {
332 args[i] = & arg2;
333 i ++;
334 }
335 if (!arg3.IsNull())
336 {
337 args[i] = & arg3;
338 i ++;
339 }
340 if (!arg4.IsNull())
341 {
342 args[i] = & arg4;
343 i ++;
344 }
345 if (!arg5.IsNull())
346 {
347 args[i] = & arg5;
348 i ++;
349 }
350 if (!arg6.IsNull())
351 {
352 args[i] = & arg6;
353 i ++;
354 }
355 wxVariant retVariant;
356 if (!Invoke(property, DISPATCH_PROPERTYGET, retVariant, i, NULL, args))
357 {
358 retVariant.MakeNull();
359 }
360 delete[] args;
361 return retVariant;
d980b3e1
JS
362}
363
364bool wxAutomationObject::PutProperty(const wxString& property, int noArgs, wxVariant args[])
365{
0a0e6a5b
WS
366 wxVariant retVariant;
367 if (!Invoke(property, DISPATCH_PROPERTYPUT, retVariant, noArgs, args))
368 {
369 return false;
370 }
371 return true;
d980b3e1
JS
372}
373
24f4ad95
JS
374bool wxAutomationObject::PutPropertyArray(const wxString& property, int noArgs, const wxVariant **args)
375{
0a0e6a5b
WS
376 wxVariant retVariant;
377 if (!Invoke(property, DISPATCH_PROPERTYPUT, retVariant, noArgs, NULL, args))
378 {
379 return false;
380 }
381 return true;
24f4ad95
JS
382}
383
d980b3e1 384bool wxAutomationObject::PutProperty(const wxString& property,
0a0e6a5b
WS
385 const wxVariant& arg1, const wxVariant& arg2,
386 const wxVariant& arg3, const wxVariant& arg4,
387 const wxVariant& arg5, const wxVariant& arg6)
d980b3e1 388{
0a0e6a5b
WS
389 const wxVariant** args = new const wxVariant*[6];
390 int i = 0;
391 if (!arg1.IsNull())
392 {
393 args[i] = & arg1;
394 i ++;
395 }
396 if (!arg2.IsNull())
397 {
398 args[i] = & arg2;
399 i ++;
400 }
401 if (!arg3.IsNull())
402 {
403 args[i] = & arg3;
404 i ++;
405 }
406 if (!arg4.IsNull())
407 {
408 args[i] = & arg4;
409 i ++;
410 }
411 if (!arg5.IsNull())
412 {
413 args[i] = & arg5;
414 i ++;
415 }
416 if (!arg6.IsNull())
417 {
418 args[i] = & arg6;
419 i ++;
420 }
421 wxVariant retVariant;
422 bool ret = Invoke(property, DISPATCH_PROPERTYPUT, retVariant, i, NULL, args);
423 delete[] args;
424 return ret;
d980b3e1
JS
425}
426
427
428// Uses DISPATCH_PROPERTYGET
429// and returns a dispatch pointer. The calling code should call Release
430// on the pointer, though this could be implicit by constructing an wxAutomationObject
431// with it and letting the destructor call Release.
432WXIDISPATCH* wxAutomationObject::GetDispatchProperty(const wxString& property, int noArgs, wxVariant args[]) const
433{
0a0e6a5b
WS
434 wxVariant retVariant;
435 if (Invoke(property, DISPATCH_PROPERTYGET, retVariant, noArgs, args))
436 {
437 if (retVariant.GetType() == wxT("void*"))
438 {
439 return (WXIDISPATCH*) retVariant.GetVoidPtr();
440 }
441 }
442
443 return (WXIDISPATCH*) NULL;
d980b3e1
JS
444}
445
24f4ad95
JS
446// Uses DISPATCH_PROPERTYGET
447// and returns a dispatch pointer. The calling code should call Release
448// on the pointer, though this could be implicit by constructing an wxAutomationObject
449// with it and letting the destructor call Release.
450WXIDISPATCH* wxAutomationObject::GetDispatchProperty(const wxString& property, int noArgs, const wxVariant **args) const
451{
0a0e6a5b
WS
452 wxVariant retVariant;
453 if (Invoke(property, DISPATCH_PROPERTYGET, retVariant, noArgs, NULL, args))
454 {
455 if (retVariant.GetType() == wxT("void*"))
456 {
457 return (WXIDISPATCH*) retVariant.GetVoidPtr();
458 }
459 }
460
461 return (WXIDISPATCH*) NULL;
24f4ad95
JS
462}
463
464
d980b3e1
JS
465// A way of initialising another wxAutomationObject with a dispatch object
466bool wxAutomationObject::GetObject(wxAutomationObject& obj, const wxString& property, int noArgs, wxVariant args[]) const
467{
0a0e6a5b
WS
468 WXIDISPATCH* dispatch = GetDispatchProperty(property, noArgs, args);
469 if (dispatch)
470 {
471 obj.SetDispatchPtr(dispatch);
472 return true;
473 }
474 else
475 return false;
d980b3e1
JS
476}
477
24f4ad95
JS
478// A way of initialising another wxAutomationObject with a dispatch object
479bool wxAutomationObject::GetObject(wxAutomationObject& obj, const wxString& property, int noArgs, const wxVariant **args) const
480{
0a0e6a5b
WS
481 WXIDISPATCH* dispatch = GetDispatchProperty(property, noArgs, args);
482 if (dispatch)
483 {
484 obj.SetDispatchPtr(dispatch);
485 return true;
486 }
487 else
488 return false;
24f4ad95
JS
489}
490
d980b3e1
JS
491// Get a dispatch pointer from the current object associated
492// with a class id
493bool wxAutomationObject::GetInstance(const wxString& classId) const
494{
0a0e6a5b
WS
495 if (m_dispatchPtr)
496 return false;
497
498 CLSID clsId;
499 IUnknown * pUnk = NULL;
500
5149f75b 501 wxBasicString unicodeName(classId);
0a0e6a5b
WS
502
503 if (FAILED(CLSIDFromProgID((BSTR) unicodeName, &clsId)))
504 {
505 wxLogWarning(wxT("Cannot obtain CLSID from ProgID"));
506 return false;
507 }
508
509 if (FAILED(GetActiveObject(clsId, NULL, &pUnk)))
510 {
511 wxLogWarning(wxT("Cannot find an active object"));
512 return false;
513 }
514
515 if (pUnk->QueryInterface(IID_IDispatch, (LPVOID*) &m_dispatchPtr) != S_OK)
516 {
517 wxLogWarning(wxT("Cannot find IDispatch interface"));
518 return false;
519 }
520
521 return true;
d980b3e1
JS
522}
523
524// Get a dispatch pointer from a new object associated
525// with the given class id
526bool wxAutomationObject::CreateInstance(const wxString& classId) const
527{
0a0e6a5b
WS
528 if (m_dispatchPtr)
529 return false;
530
531 CLSID clsId;
532
5149f75b 533 wxBasicString unicodeName(classId);
0a0e6a5b
WS
534
535 if (FAILED(CLSIDFromProgID((BSTR) unicodeName, &clsId)))
536 {
537 wxLogWarning(wxT("Cannot obtain CLSID from ProgID"));
538 return false;
539 }
540
541 // start a new copy of Excel, grab the IDispatch interface
542 if (FAILED(CoCreateInstance(clsId, NULL, CLSCTX_LOCAL_SERVER, IID_IDispatch, (void**)&m_dispatchPtr)))
543 {
544 wxLogWarning(wxT("Cannot start an instance of this class."));
545 return false;
546 }
547
548 return true;
d980b3e1
JS
549}
550
551
e2898549 552WXDLLEXPORT bool wxConvertVariantToOle(const wxVariant& variant, VARIANTARG& oleVariant)
d980b3e1 553{
0a0e6a5b
WS
554 ClearVariant(&oleVariant);
555 if (variant.IsNull())
556 {
557 oleVariant.vt = VT_NULL;
558 return true;
559 }
d980b3e1
JS
560
561 wxString type(variant.GetType());
562
6b978929
JS
563
564 if (type == wxT("long"))
d980b3e1
JS
565 {
566 oleVariant.vt = VT_I4;
567 oleVariant.lVal = variant.GetLong() ;
568 }
6b978929
JS
569 // cVal not always present
570#ifndef __GNUWIN32__
571 else if (type == wxT("char"))
572 {
0a0e6a5b 573 oleVariant.vt=VT_I1; // Signed Char
6b978929
JS
574 oleVariant.cVal=variant.GetChar();
575 }
576#endif
223d09f6 577 else if (type == wxT("double"))
d980b3e1
JS
578 {
579 oleVariant.vt = VT_R8;
580 oleVariant.dblVal = variant.GetDouble();
581 }
223d09f6 582 else if (type == wxT("bool"))
d980b3e1
JS
583 {
584 oleVariant.vt = VT_BOOL;
7c5dc04f 585 // 'bool' required for VC++ 4 apparently
6b12494d 586#if (defined(__VISUALC__) && (__VISUALC__ <= 1000))
7be1f0d9
JS
587 oleVariant.bool = variant.GetBool();
588#else
d980b3e1 589 oleVariant.boolVal = variant.GetBool();
7be1f0d9 590#endif
d980b3e1 591 }
223d09f6 592 else if (type == wxT("string"))
d980b3e1
JS
593 {
594 wxString str( variant.GetString() );
595 oleVariant.vt = VT_BSTR;
ed5317e5 596 oleVariant.bstrVal = wxConvertStringToOle(str);
d980b3e1 597 }
61bfe4b0
JS
598#if wxUSE_DATETIME
599 else if (type == wxT("datetime"))
600 {
601 wxDateTime date( variant.GetDateTime() );
602 oleVariant.vt = VT_DATE;
7520f3da 603
94113cc5 604 long dosDateTime = date.GetAsDOS();
11f104e5
WS
605 short dosDate = short((dosDateTime & 0xFFFF0000) >> 16);
606 short dosTime = short(dosDateTime & 0xFFFF);
7520f3da 607
94113cc5 608 DosDateTimeToVariantTime(dosDate, dosTime, & oleVariant.date);
61bfe4b0 609 }
457e6c54 610#endif
223d09f6 611 else if (type == wxT("void*"))
d980b3e1
JS
612 {
613 oleVariant.vt = VT_DISPATCH;
614 oleVariant.pdispVal = (IDispatch*) variant.GetVoidPtr();
615 }
223d09f6 616 else if (type == wxT("list") || type == wxT("stringlist"))
d980b3e1
JS
617 {
618 oleVariant.vt = VT_VARIANT | VT_ARRAY;
619
0a0e6a5b
WS
620 SAFEARRAY *psa;
621 SAFEARRAYBOUND saBound;
622 VARIANTARG *pvargBase;
623 VARIANTARG *pvarg;
624 int i, j;
d980b3e1
JS
625
626 int iCount = variant.GetCount();
0a0e6a5b
WS
627
628 saBound.lLbound = 0;
629 saBound.cElements = iCount;
630
631 psa = SafeArrayCreate(VT_VARIANT, 1, &saBound);
632 if (psa == NULL)
633 return false;
634
635 SafeArrayAccessData(psa, (void**)&pvargBase);
636
637 pvarg = pvargBase;
638 for (i = 0; i < iCount; i++)
639 {
640 // copy each string in the list of strings
d980b3e1 641 wxVariant eachVariant(variant[i]);
ed5317e5 642 if (!wxConvertVariantToOle(eachVariant, * pvarg))
d980b3e1 643 {
0a0e6a5b
WS
644 // memory failure: back out and free strings alloc'ed up to
645 // now, and then the array itself.
646 pvarg = pvargBase;
647 for (j = 0; j < i; j++)
648 {
649 SysFreeString(pvarg->bstrVal);
650 pvarg++;
651 }
652 SafeArrayDestroy(psa);
653 return false;
654 }
655 pvarg++;
656 }
657
658 SafeArrayUnaccessData(psa);
d980b3e1
JS
659
660 oleVariant.parray = psa;
661 }
662 else
663 {
664 oleVariant.vt = VT_NULL;
0a0e6a5b 665 return false;
d980b3e1 666 }
0a0e6a5b 667 return true;
d980b3e1
JS
668}
669
670#ifndef VT_TYPEMASK
671#define VT_TYPEMASK 0xfff
672#endif
673
242019ee
VZ
674WXDLLEXPORT bool
675wxConvertOleToVariant(const VARIANTARG& oleVariant, wxVariant& variant)
d980b3e1 676{
e54266ad 677 bool ok = true;
242019ee 678 if ( oleVariant.vt & VT_ARRAY )
0a0e6a5b 679 {
242019ee
VZ
680 variant.ClearList();
681
682 // Compute the total number of elements in all array dimensions
683 int cElements = 1;
684 for ( int cDims = 0; cDims < oleVariant.parray->cDims; cDims++ )
685 cElements *= oleVariant.parray->rgsabound[cDims].cElements;
686
687 // Get a pointer to the data
688 VARIANTARG* pvdata;
689 HRESULT hr = SafeArrayAccessData(oleVariant.parray, (void **)&pvdata);
690 if ( FAILED(hr) )
691 return false;
692
242019ee 693 for ( int i = 0; i < cElements; i++ )
0a0e6a5b 694 {
242019ee
VZ
695 VARIANTARG& oleElement = pvdata[i];
696 wxVariant vElement;
697 if ( !wxConvertOleToVariant(oleElement, vElement) )
698 {
699 ok = false;
700 variant.ClearList();
701 break;
702 }
703
704 variant.Append(vElement);
0a0e6a5b 705 }
242019ee
VZ
706
707 SafeArrayUnaccessData(oleVariant.parray);
708 }
709 else if ( oleVariant.vt & VT_BYREF )
710 {
711 switch ( oleVariant.vt & VT_TYPEMASK )
0a0e6a5b 712 {
242019ee
VZ
713 case VT_VARIANT:
714 {
715 VARIANTARG& oleReference = *((LPVARIANT)oleVariant.byref);
716 if (!wxConvertOleToVariant(oleReference,variant))
717 return false;
718 break;
719 }
720
721 default:
722 wxLogError(wxT("wxAutomationObject::ConvertOleToVariant: [as yet] unhandled reference %X"),oleVariant.vt);
723 return false;
0a0e6a5b 724 }
242019ee
VZ
725 }
726 else // simply type (not array or reference)
727 {
728 switch (oleVariant.vt & VT_TYPEMASK)
0a0e6a5b 729 {
242019ee
VZ
730 case VT_BSTR:
731 {
732 wxString str(wxConvertStringFromOle(oleVariant.bstrVal));
733 variant = str;
734 break;
735 }
736 case VT_DATE:
737 {
738#if wxUSE_DATETIME
739 unsigned short dosDate = 0;
740 unsigned short dosTime = 0;
741 VariantTimeToDosDateTime(oleVariant.date, & dosDate, & dosTime);
742
743 long dosDateTime = (dosDate << 16) | dosTime;
744 wxDateTime date;
745 date.SetFromDOS(dosDateTime);
746 variant = date;
747#endif
748 break;
749 }
750 case VT_I4:
751 {
752 variant = (long) oleVariant.lVal;
753 break;
754 }
755 case VT_I2:
756 {
757 variant = (long) oleVariant.iVal;
758 break;
759 }
0a0e6a5b 760
242019ee
VZ
761 case VT_BOOL:
762 {
6b12494d 763#if (defined(_MSC_VER) && (_MSC_VER <= 1000) && !defined(__MWERKS__) ) //GC
25889d3c 764#ifndef HAVE_BOOL // Can't use bool operator if no native bool type
242019ee 765 variant = (long) (oleVariant.bool != 0);
25889d3c 766#else
242019ee 767 variant = (bool) (oleVariant.bool != 0);
25889d3c 768#endif
f6bcfd97
BP
769#else
770#ifndef HAVE_BOOL // Can't use bool operator if no native bool type
242019ee 771 variant = (long) (oleVariant.boolVal != 0);
7be1f0d9 772#else
242019ee 773 variant = (bool) (oleVariant.boolVal != 0);
f6bcfd97 774#endif
7be1f0d9 775#endif
242019ee
VZ
776 break;
777 }
778 case VT_R8:
779 {
780 variant = oleVariant.dblVal;
781 break;
782 }
783 case VT_DISPATCH:
784 {
785 variant = (void*) oleVariant.pdispVal;
786 break;
787 }
788 case VT_NULL:
789 {
790 variant.MakeNull();
791 break;
792 }
793 case VT_EMPTY:
794 {
795 break; // Ignore Empty Variant, used only during destruction of objects
796 }
797 default:
798 {
799 wxLogError(wxT("wxAutomationObject::ConvertOleToVariant: Unknown variant value type %X -> %X"),oleVariant.vt,oleVariant.vt&VT_TYPEMASK);
0a0e6a5b 800 return false;
242019ee 801 }
0a0e6a5b
WS
802 }
803 }
242019ee 804
e54266ad 805 return ok;
d980b3e1
JS
806}
807
d980b3e1
JS
808/*
809 * ClearVariant
810 *
811 * Zeros a variant structure without regard to current contents
812 */
813static void ClearVariant(VARIANTARG *pvarg)
814{
0a0e6a5b
WS
815 pvarg->vt = VT_EMPTY;
816 pvarg->wReserved1 = 0;
817 pvarg->wReserved2 = 0;
818 pvarg->wReserved3 = 0;
819 pvarg->lVal = 0;
d980b3e1
JS
820}
821
822/*
823 * ReleaseVariant
824 *
825 * Clears a particular variant structure and releases any external objects
826 * or memory contained in the variant. Supports the data types listed above.
827 */
828static void ReleaseVariant(VARIANTARG *pvarg)
829{
0a0e6a5b
WS
830 VARTYPE vt;
831 VARIANTARG _huge *pvargArray;
56fb1bca 832 LONG lLBound, lUBound, l;
0a0e6a5b 833
5c519b6c 834 vt = (VARTYPE)(pvarg->vt & 0xfff); // mask off flags
0a0e6a5b
WS
835
836 // check if an array. If so, free its contents, then the array itself.
837 if (V_ISARRAY(pvarg))
838 {
839 // variant arrays are all this routine currently knows about. Since a
840 // variant can contain anything (even other arrays), call ourselves
841 // recursively.
842 if (vt == VT_VARIANT)
843 {
844 SafeArrayGetLBound(pvarg->parray, 1, &lLBound);
845 SafeArrayGetUBound(pvarg->parray, 1, &lUBound);
846
847 if (lUBound > lLBound)
848 {
849 lUBound -= lLBound;
850
851 SafeArrayAccessData(pvarg->parray, (void**)&pvargArray);
852
853 for (l = 0; l < lUBound; l++)
854 {
855 ReleaseVariant(pvargArray);
856 pvargArray++;
857 }
858
859 SafeArrayUnaccessData(pvarg->parray);
860 }
861 }
862 else
863 {
864 wxLogWarning(wxT("ReleaseVariant: Array contains non-variant type"));
865 }
866
867 // Free the array itself.
868 SafeArrayDestroy(pvarg->parray);
869 }
870 else
871 {
872 switch (vt)
873 {
874 case VT_DISPATCH:
875 if (pvarg->pdispVal)
876 pvarg->pdispVal->Release();
877 break;
878
879 case VT_BSTR:
880 SysFreeString(pvarg->bstrVal);
881 break;
882
883 case VT_I2:
0a0e6932 884 case VT_I4:
0a0e6a5b
WS
885 case VT_BOOL:
886 case VT_R8:
887 case VT_ERROR: // to avoid erroring on an error return from Excel
0a0e6932 888 case VT_EMPTY:
b0fa01cc 889 case VT_DATE:
0a0e6a5b
WS
890 // no work for these types
891 break;
892
893 default:
894 wxLogWarning(wxT("ReleaseVariant: Unknown type"));
895 break;
896 }
897 }
898
899 ClearVariant(pvarg);
d980b3e1
JS
900}
901
902#if 0
903
904void ShowException(LPOLESTR szMember, HRESULT hr, EXCEPINFO *pexcep, unsigned int uiArgErr)
905{
0a0e6a5b
WS
906 TCHAR szBuf[512];
907
908 switch (GetScode(hr))
909 {
910 case DISP_E_UNKNOWNNAME:
911 wsprintf(szBuf, L"%s: Unknown name or named argument.", szMember);
912 break;
913
914 case DISP_E_BADPARAMCOUNT:
915 wsprintf(szBuf, L"%s: Incorrect number of arguments.", szMember);
916 break;
917
918 case DISP_E_EXCEPTION:
919 wsprintf(szBuf, L"%s: Error %d: ", szMember, pexcep->wCode);
920 if (pexcep->bstrDescription != NULL)
921 lstrcat(szBuf, pexcep->bstrDescription);
922 else
923 lstrcat(szBuf, L"<<No Description>>");
924 break;
925
926 case DISP_E_MEMBERNOTFOUND:
927 wsprintf(szBuf, L"%s: method or property not found.", szMember);
928 break;
929
930 case DISP_E_OVERFLOW:
931 wsprintf(szBuf, L"%s: Overflow while coercing argument values.", szMember);
932 break;
933
934 case DISP_E_NONAMEDARGS:
935 wsprintf(szBuf, L"%s: Object implementation does not support named arguments.",
936 szMember);
937 break;
938
939 case DISP_E_UNKNOWNLCID:
940 wsprintf(szBuf, L"%s: The locale ID is unknown.", szMember);
941 break;
942
943 case DISP_E_PARAMNOTOPTIONAL:
944 wsprintf(szBuf, L"%s: Missing a required parameter.", szMember);
945 break;
946
947 case DISP_E_PARAMNOTFOUND:
948 wsprintf(szBuf, L"%s: Argument not found, argument %d.", szMember, uiArgErr);
949 break;
950
951 case DISP_E_TYPEMISMATCH:
952 wsprintf(szBuf, L"%s: Type mismatch, argument %d.", szMember, uiArgErr);
953 break;
954
955 default:
3103e8a9 956 wsprintf(szBuf, L"%s: Unknown error occurred.", szMember);
0a0e6a5b
WS
957 break;
958 }
959
960 wxLogWarning(szBuf);
d980b3e1
JS
961}
962
963#endif
964
ca5c6ac3 965#endif // wxUSE_OLE_AUTOMATION