]> git.saurik.com Git - wxWidgets.git/blame - src/os2/utils.cpp
wxFontEnumerator mostly works for wxMSW
[wxWidgets.git] / src / os2 / utils.cpp
CommitLineData
0e320a79
DW
1/////////////////////////////////////////////////////////////////////////////
2// Name: utils.cpp
3// Purpose: Various utilities
45fcbf3b 4// Author: David Webster
0e320a79 5// Modified by:
45fcbf3b 6// Created: 09/17/99
0e320a79 7// RCS-ID: $Id$
45fcbf3b
DW
8// Copyright: (c) David Webster
9// Licence: wxWindows license
0e320a79
DW
10/////////////////////////////////////////////////////////////////////////////
11
45fcbf3b
DW
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#ifndef WX_PRECOMP
16 #include "wx/setup.h"
17 #include "wx/utils.h"
18 #include "wx/app.h"
19 #include "wx/cursor.h"
20#endif //WX_PRECOMP
21
22#include "wx/os2/private.h"
23#include "wx/timer.h"
24#include "wx/intl.h"
25
0e320a79 26#include <ctype.h>
45fcbf3b
DW
27#include <direct.h>
28
29#include "wx/log.h"
30
31#include <io.h>
0e320a79
DW
32
33#include <stdio.h>
34#include <stdlib.h>
35#include <string.h>
45fcbf3b 36#include <errno.h>
0e320a79
DW
37#include <stdarg.h>
38
d90895ac 39#define INCL_DOS
86de7616 40#define INCL_PM
d90895ac 41#define INCL_GPI
86de7616 42#include <os2.h>
45fcbf3b 43#include<netdb.h>
d90895ac 44#define PURE_32
7cdc2f1e
DW
45#include <upm.h>
46#include <netcons.h>
47#include <netbios.h>
45fcbf3b 48
86de7616
DW
49static const wxChar WX_SECTION[] = _T("wxWindows");
50static const wxChar eHOSTNAME[] = _T("HostName");
51static const wxChar eUSERID[] = _T("UserId");
52static const wxChar eUSERNAME[] = _T("UserName");
45fcbf3b
DW
53
54// For the following functions we SHOULD fill in support
55// for Windows-NT (which I don't know) as I assume it begin
56// a POSIX Unix (so claims MS) that it has some special
57// functions beyond those provided by WinSock
58
0e320a79 59// Get full hostname (eg. DoDo.BSn-Germany.crg.de)
10e5b930
DW
60bool wxGetHostName(
61 wxChar* zBuf
62, int nMaxSize
63)
0e320a79 64{
d90895ac 65#if wxUSE_NET_API
10e5b930
DW
66 char zServer[256];
67 char zComputer[256];
004fd0c8 68 unsigned short nLevel = 0;
10e5b930 69 unsigned char* zBuffer;
004fd0c8
DW
70 unsigned short nBuffer;
71 unsigned short* pnTotalAvail;
10e5b930 72
dde11e60
DW
73 NetBios32GetInfo( (const unsigned char*)zServer
74 ,(const unsigned char*)zComputer
75 ,nLevel
76 ,zBuffer
77 ,nBuffer
78 ,pnTotalAvail
79 );
10e5b930 80 strcpy(zBuf, zServer);
45fcbf3b 81#else
10e5b930
DW
82 wxChar* zSysname;
83 const wxChar* zDefaultHost = _T("noname");
45fcbf3b 84
10e5b930
DW
85 if ((zSysname = wxGetenv(_T("SYSTEM_NAME"))) == NULL)
86 {
87 ULONG n = ::PrfQueryProfileString( HINI_PROFILE
88 ,(PSZ)WX_SECTION
89 ,(PSZ)eHOSTNAME
90 ,(PSZ)zDefaultHost
91 ,(void*)zBuf
92 ,(ULONG)nMaxSize - 1
93 );
94 }
95 else
96 wxStrncpy(zBuf, zSysname, nMaxSize - 1);
97 zBuf[nMaxSize] = _T('\0');
45fcbf3b 98#endif
10e5b930 99 return *zBuf ? TRUE : FALSE;
0e320a79
DW
100}
101
102// Get user ID e.g. jacs
10e5b930
DW
103bool wxGetUserId(
104 wxChar* zBuf
2bd5bbc9 105, int nType
10e5b930 106)
0e320a79 107{
2bd5bbc9
DW
108 long lrc;
109 // UPM procs return 0 on success
110 lrc = U32ELOCU((unsigned char*)zBuf, (unsigned long *)&nType);
111 if (lrc == 0) return TRUE;
112 return FALSE;
0e320a79
DW
113}
114
10e5b930
DW
115bool wxGetUserName(
116 wxChar* zBuf
117, int nMaxSize
118)
0e320a79 119{
45fcbf3b 120#ifdef USE_NET_API
10e5b930
DW
121 wxGetUserId( zBuf
122 ,nMaxSize
123 );
45fcbf3b 124#else
10e5b930 125 wxStrncpy(zBuf, _T("Unknown User"), nMaxSize);
45fcbf3b 126#endif
10e5b930 127 return TRUE;
0e320a79
DW
128}
129
10e5b930
DW
130int wxKill(
131 long lPid
132, int nSig
133)
0e320a79 134{
10e5b930 135 return((int)::DosKillProcess(0, (PID)lPid));
0e320a79
DW
136}
137
138//
139// Execute a program in an Interactive Shell
140//
10e5b930
DW
141bool wxShell(
142 const wxString& rCommand
143)
0e320a79 144{
10e5b930 145 wxChar* zShell;
45fcbf3b 146
10e5b930
DW
147 if ((zShell = wxGetenv(_T("COMSPEC"))) == NULL)
148 zShell = _T("\\CMD.EXE");
45fcbf3b 149
10e5b930
DW
150 wxChar zTmp[255];
151
152 if (rCommand != "")
153 wxSprintf( zTmp
154 ,"%s /c %s"
155 ,zShell
156 ,WXSTRINGCAST rCommand
157 );
158 else
159 wxStrcpy(zTmp, zShell);
160
161 return (wxExecute((wxChar*)zTmp, FALSE) != 0);
0e320a79
DW
162}
163
164// Get free memory in bytes, or -1 if cannot determine amount (e.g. on UNIX)
10e5b930
DW
165long wxGetFreeMemory(
166 void* pMemptr
167)
0e320a79 168{
78d50441
DW
169 ULONG lSize;
170 ULONG lMemFlags;
171 APIRET rc;
172
173 lMemFlags = PAG_FREE;
10e5b930 174 rc = ::DosQueryMem(pMemptr, &lSize, &lMemFlags);
78d50441
DW
175 if (rc != 0)
176 return -1L;
177 return (long)lSize;
45fcbf3b
DW
178}
179
180// Sleep for nSecs seconds. Attempt a Windows implementation using timers.
181static bool inTimer = FALSE;
182
183class wxSleepTimer: public wxTimer
184{
10e5b930
DW
185public:
186 inline void Notify()
187 {
188 inTimer = FALSE;
189 Stop();
190 }
45fcbf3b
DW
191};
192
10e5b930 193static wxTimer* wxTheSleepTimer = NULL;
45fcbf3b 194
10e5b930
DW
195void wxUsleep(
196 unsigned long ulMilliseconds
197)
45fcbf3b 198{
10e5b930 199 ::DosSleep(ulMilliseconds);
0e320a79
DW
200}
201
10e5b930
DW
202void wxSleep(
203 int nSecs
204)
0e320a79 205{
10e5b930 206 ::DosSleep(1000 * nSecs);
0e320a79
DW
207}
208
209// Consume all events until no more left
210void wxFlushEvents()
211{
45fcbf3b 212// wxYield();
0e320a79
DW
213}
214
45fcbf3b 215// Output a debug mess., in a system dependent fashion.
10e5b930
DW
216void wxDebugMsg(
217 const wxChar* zFmt ...
218)
219{
220 va_list vAp;
221 static wxChar zBuffer[512];
222
223 if (!wxTheApp->GetWantDebugOutput())
224 return ;
225 va_start(vAp, zFmt);
226 sprintf(zBuffer, zFmt, vAp) ;
227 va_end(vAp);
0e320a79
DW
228}
229
230// Non-fatal error: pop up message box and (possibly) continue
10e5b930
DW
231void wxError(
232 const wxString& rMsg
233, const wxString& rTitle
234)
235{
236 wxSprintf(wxBuffer, "%s\nContinue?", WXSTRINGCAST rMsg);
237 if (::WinMessageBox( HWND_DESKTOP
238 ,NULL
239 ,(PSZ)wxBuffer
240 ,(PSZ)WXSTRINGCAST rTitle
241 ,0
242 ,MB_ICONEXCLAMATION | MB_YESNO
243 ) == MBID_YES)
0e320a79
DW
244 wxExit();
245}
246
247// Fatal error: pop up message box and abort
10e5b930
DW
248void wxFatalError(
249 const wxString& rMsg
250, const wxString& rTitle
251)
252{
253 unsigned long ulRc;
254
255 ulRc = ::WinMessageBox( HWND_DESKTOP
256 ,NULL
257 ,WXSTRINGCAST rMsg
258 ,WXSTRINGCAST rTitle
259 ,0
260 ,MB_NOICON | MB_OK
261 );
262 DosExit(EXIT_PROCESS, ulRc);
0e320a79
DW
263}
264
265// Emit a beeeeeep
266void wxBell()
267{
45fcbf3b 268 DosBeep(1000,1000); // 1kHz during 1 sec.
0e320a79
DW
269}
270
45fcbf3b
DW
271// Chris Breeze 27/5/98: revised WIN32 code to
272// detect WindowsNT correctly
10e5b930
DW
273int wxGetOsVersion(
274 int* pMajorVsn
275, int* pMinorVsn
276)
277{
278 ULONG ulSysInfo[QSV_MAX] = {0};
279
280 if (::DosQuerySysInfo( 1L
281 ,QSV_MAX
282 ,(PVOID)ulSysInfo
283 ,sizeof(ULONG) * QSV_MAX
284 ))
285 {
286 *pMajorVsn = ulSysInfo[QSV_VERSION_MAJOR];
287 *pMinorVsn = ulSysInfo[QSV_VERSION_MINOR];
288 return wxWINDOWS_OS2;
289 }
290 return wxWINDOWS; // error if we get here, return generic value
0e320a79
DW
291}
292
293// Reading and writing resources (eg WIN.INI, .Xdefaults)
294#if wxUSE_RESOURCES
10e5b930
DW
295bool wxWriteResource(
296 const wxString& rSection
297, const wxString& rEntry
298, const wxString& rValue
299, const wxString& rFile
300)
0e320a79 301{
78d50441
DW
302 HAB hab;
303 HINI hIni;
304
10e5b930 305 if (rFile != "")
78d50441 306 {
10e5b930 307 hIni = ::PrfOpenProfile(hab, (PSZ)WXSTRINGCAST rFile);
78d50441
DW
308 if (hIni != 0L)
309 {
310 return (::PrfWriteProfileString( hIni
10e5b930
DW
311 ,(PSZ)WXSTRINGCAST rSection
312 ,(PSZ)WXSTRINGCAST rEntry
313 ,(PSZ)WXSTRINGCAST rValue
78d50441
DW
314 ));
315 }
316 }
317 else
318 return (::PrfWriteProfileString( HINI_PROFILE
10e5b930
DW
319 ,(PSZ)WXSTRINGCAST rSection
320 ,(PSZ)WXSTRINGCAST rEntry
321 ,(PSZ)WXSTRINGCAST rValue
78d50441 322 ));
d90895ac 323 return FALSE;
0e320a79
DW
324}
325
10e5b930
DW
326bool wxWriteResource(
327 const wxString& rSection
328, const wxString& rEntry
329, float fValue
330, const wxString& rFile
331)
332{
333 wxChar zBuf[50];
334
335 wxSprintf(zBuf, "%.4f", fValue);
336 return wxWriteResource( rSection
337 ,rEntry
338 ,zBuf
339 ,rFile
340 );
0e320a79
DW
341}
342
1be7f92a
DW
343bool wxWriteResource(
344 const wxString& rSection
345, const wxString& rEntry
346, long lValue
347, const wxString& rFile
348)
0e320a79 349{
1be7f92a
DW
350 wxChar zBuf[50];
351
352 wxSprintf(zBuf, "%ld", lValue);
353 return wxWriteResource( rSection
354 ,rEntry
355 ,zBuf
356 ,rFile
357 );
0e320a79
DW
358}
359
1be7f92a
DW
360bool wxWriteResource(
361 const wxString& rSection
362, const wxString& rEntry
363, int lValue
364, const wxString& rFile
365)
0e320a79 366{
1be7f92a
DW
367 wxChar zBuf[50];
368
369 wxSprintf(zBuf, "%d", lValue);
370 return wxWriteResource( rSection
371 ,rEntry
372 ,zBuf
373 ,rFile
374 );
0e320a79
DW
375}
376
1be7f92a
DW
377bool wxGetResource(
378 const wxString& rSection
379, const wxString& rEntry
380, wxChar** ppValue
381, const wxString& rFile
382)
0e320a79 383{
78d50441
DW
384 HAB hab;
385 HINI hIni;
1be7f92a 386 static const wxChar zDefunkt[] = _T("$$default");
78d50441 387
1be7f92a 388 if (rFile != "")
78d50441 389 {
1be7f92a 390 hIni = ::PrfOpenProfile(hab, (PSZ)WXSTRINGCAST rFile);
78d50441
DW
391 if (hIni != 0L)
392 {
393 ULONG n = ::PrfQueryProfileString( hIni
1be7f92a
DW
394 ,(PSZ)WXSTRINGCAST rSection
395 ,(PSZ)WXSTRINGCAST rEntry
396 ,(PSZ)zDefunkt
78d50441
DW
397 ,(void*)wxBuffer
398 ,1000
399 );
1be7f92a 400 if (n == 0L || wxStrcmp(wxBuffer, zDefunkt) == 0)
78d50441
DW
401 return FALSE;
402 }
403 else
404 return FALSE;
405 }
406 else
407 {
408 ULONG n = ::PrfQueryProfileString( HINI_PROFILE
1be7f92a
DW
409 ,(PSZ)WXSTRINGCAST rSection
410 ,(PSZ)WXSTRINGCAST rEntry
411 ,(PSZ)zDefunkt
78d50441
DW
412 ,(void*)wxBuffer
413 ,1000
414 );
1be7f92a 415 if (n == 0L || wxStrcmp(wxBuffer, zDefunkt) == 0)
78d50441
DW
416 return FALSE;
417 }
1be7f92a
DW
418 if (*ppValue)
419 delete[] (*ppValue);
420 *ppValue = copystring(wxBuffer);
78d50441 421 return TRUE;
d90895ac 422}
0e320a79 423
1be7f92a
DW
424bool wxGetResource(
425 const wxString& rSection
426, const wxString& rEntry
427, float* pValue
428, const wxString& rFile
429)
0e320a79 430{
1be7f92a
DW
431 wxChar* zStr = NULL;
432 bool bSucc = wxGetResource( rSection
433 ,rEntry
434 ,(wxChar **)&zStr
435 ,rFile
436 );
437
438 if (bSucc)
439 {
440 *pValue = (float)wxStrtod(zStr, NULL);
441 delete[] zStr;
442 return TRUE;
443 }
444 else return FALSE;
0e320a79
DW
445}
446
1be7f92a
DW
447bool wxGetResource(
448 const wxString& rSection
449, const wxString& rEntry
450, long* pValue
451, const wxString& rFile
452)
0e320a79 453{
1be7f92a
DW
454 wxChar* zStr = NULL;
455 bool bSucc = wxGetResource( rSection
456 ,rEntry
457 ,(wxChar **)&zStr
458 ,rFile
459 );
460
461 if (bSucc)
462 {
463 *pValue = wxStrtol(zStr, NULL, 10);
464 delete[] zStr;
465 return TRUE;
466 }
467 else return FALSE;
0e320a79
DW
468}
469
1be7f92a
DW
470bool wxGetResource(
471 const wxString& rSection
472, const wxString& rEntry
473, int* pValue
474, const wxString& rFile
475)
0e320a79 476{
1be7f92a
DW
477 wxChar* zStr = NULL;
478 bool bSucc = wxGetResource( rSection
479 ,rEntry
480 ,(wxChar **)&zStr
481 ,rFile
482 );
483
484 if (bSucc)
485 {
486 *pValue = (int)wxStrtol(zStr, NULL, 10);
487 delete[] zStr;
488 return TRUE;
489 }
490 else return FALSE;
0e320a79
DW
491}
492#endif // wxUSE_RESOURCES
493
45fcbf3b
DW
494// ---------------------------------------------------------------------------
495// helper functions for showing a "busy" cursor
496// ---------------------------------------------------------------------------
497
498HCURSOR gs_wxBusyCursor = 0; // new, busy cursor
499HCURSOR gs_wxBusyCursorOld = 0; // old cursor
500static int gs_wxBusyCursorCount = 0;
0e320a79
DW
501
502// Set the cursor to the busy cursor for all windows
1be7f92a
DW
503void wxBeginBusyCursor(
504 wxCursor* pCursor
505)
0e320a79 506{
45fcbf3b
DW
507 if ( gs_wxBusyCursorCount++ == 0 )
508 {
1be7f92a 509 gs_wxBusyCursor = (HCURSOR)pCursor->GetHCURSOR();
45fcbf3b
DW
510 ::WinSetPointer(HWND_DESKTOP, (HPOINTER)gs_wxBusyCursor);
511 }
512 //else: nothing to do, already set
0e320a79
DW
513}
514
515// Restore cursor to normal
516void wxEndBusyCursor()
517{
1be7f92a
DW
518 wxCHECK_RET( gs_wxBusyCursorCount > 0
519 ,_T("no matching wxBeginBusyCursor() for wxEndBusyCursor()")
520 );
45fcbf3b 521
1be7f92a 522 if (--gs_wxBusyCursorCount == 0)
45fcbf3b
DW
523 {
524 ::WinSetPointer(HWND_DESKTOP, (HPOINTER)gs_wxBusyCursorOld);
525 gs_wxBusyCursorOld = 0;
526 }
0e320a79
DW
527}
528
529// TRUE if we're between the above two calls
530bool wxIsBusy()
531{
1be7f92a 532 return (gs_wxBusyCursorCount > 0);
45fcbf3b
DW
533}
534
535// ---------------------------------------------------------------------------
1be7f92a
DW
536const wxChar* wxGetHomeDir(
537 wxString* pStr
538)
45fcbf3b 539{
1be7f92a 540 wxString& rStrDir = *pStr;
0e320a79 541
1be7f92a
DW
542 // OS/2 has no idea about home,
543 // so use the working directory instead?
45fcbf3b 544
1be7f92a 545 // 256 was taken from os2def.h
45fcbf3b
DW
546#ifndef MAX_PATH
547# define MAX_PATH 256
548#endif
549
1be7f92a
DW
550 char zDirName[256];
551 ULONG ulDirLen;
45fcbf3b 552
1be7f92a
DW
553 ::DosQueryCurrentDir(0, zDirName, &ulDirLen);
554 rStrDir = zDirName;
555 return rStrDir.c_str();
45fcbf3b
DW
556}
557
10e5b930 558// Hack for OS/2
1be7f92a
DW
559wxChar* wxGetUserHome (
560 const wxString& rUser
561)
562{
563 wxChar* zHome;
564 wxString sUser1(rUser);
565
566 if (sUser1 != _T(""))
45fcbf3b 567 {
1be7f92a
DW
568 wxChar zTmp[64];
569
570 if (wxGetUserId( zTmp
571 ,sizeof(zTmp)/sizeof(char)
572 ))
573 {
574 // Guests belong in the temp dir
575 if (wxStricmp(zTmp, _T("annonymous")) == 0)
576 {
577 if ((zHome = wxGetenv(_T("TMP"))) != NULL ||
578 (zHome = wxGetenv(_T("TMPDIR"))) != NULL ||
579 (zHome = wxGetenv(_T("TEMP"))) != NULL)
580 return *zHome ? zHome : (wxChar*)_T("\\");
581 }
582 if (wxStricmp(zTmp, WXSTRINGCAST sUser1) == 0)
583 sUser1 = _T("");
584 }
45fcbf3b 585 }
1be7f92a
DW
586 if (sUser1 == _T(""))
587 if ((zHome = wxGetenv(_T("HOME"))) != NULL)
588 {
589 wxStrcpy(wxBuffer, zHome);
590 Unix2DosFilename(wxBuffer);
591 return wxBuffer;
592 }
593 return NULL; // No home known!
0e320a79
DW
594}
595
596// Check whether this window wants to process messages, e.g. Stop button
597// in long calculations.
1be7f92a
DW
598bool wxCheckForInterrupt(
599 wxWindow* pWnd
600)
0e320a79 601{
1be7f92a
DW
602 if(pWnd)
603 {
604 QMSG vMsg;
605 HAB hab;
606 HWND hwndFilter;
607 HWND hwndWin= (HWND) pWnd->GetHWND();
45fcbf3b 608
1be7f92a
DW
609 while(::WinPeekMsg(hab, &vMsg, hwndFilter, 0, 0, PM_REMOVE))
610 {
611 ::WinDispatchMsg(hab, &vMsg);
612 }
613 return TRUE;//*** temporary?
614 }
615 else
45fcbf3b 616 {
1be7f92a
DW
617 wxFAIL_MSG(_T("pWnd==NULL !!!"));
618 return FALSE;//*** temporary?
45fcbf3b 619 }
45fcbf3b
DW
620}
621
1be7f92a
DW
622void wxGetMousePosition(
623 int* pX
624, int* pY
625)
0e320a79 626{
1be7f92a
DW
627 POINTL vPt;
628
629 ::WinQueryPointerPos(HWND_DESKTOP, &vPt);
630 *pX = vPt.x;
631 *pY = vPt.y;
0e320a79
DW
632};
633
634// Return TRUE if we have a colour display
635bool wxColourDisplay()
636{
1be7f92a
DW
637 HPS hpsScreen;
638 HDC hdcScreen;
639 LONG lColors;
640
641 hpsScreen = ::WinGetScreenPS(HWND_DESKTOP);
642 hdcScreen = ::GpiQueryDevice(hpsScreen);
643 ::DevQueryCaps(hdcScreen, CAPS_COLORS, 1L, &lColors);
644 return(lColors > 1L);
0e320a79
DW
645}
646
647// Returns depth of screen
648int wxDisplayDepth()
649{
1be7f92a
DW
650 HPS hpsScreen;
651 HDC hdcScreen;
652 LONG lPlanes;
653 LONG lBitsPerPixel;
654 LONG nDepth;
655
656 hpsScreen = ::WinGetScreenPS(HWND_DESKTOP);
657 hdcScreen = ::GpiQueryDevice(hpsScreen);
658 ::DevQueryCaps(hdcScreen, CAPS_COLOR_PLANES, 1L, &lPlanes);
659 ::DevQueryCaps(hdcScreen, CAPS_COLOR_BITCOUNT, 1L, &lBitsPerPixel);
660
661 nDepth = (int)(lPlanes * lBitsPerPixel);
662 DevCloseDC(hdcScreen);
d90895ac 663 return (nDepth);
0e320a79
DW
664}
665
666// Get size of display
1be7f92a
DW
667void wxDisplaySize(
668 int* pWidth
669, int* pHeight
670)
0e320a79 671{
1be7f92a
DW
672 HPS hpsScreen;
673 HDC hdcScreen;
45fcbf3b 674
1be7f92a
DW
675 hpsScreen = ::WinGetScreenPS(HWND_DESKTOP);
676 hdcScreen = ::GpiQueryDevice(hpsScreen);
677 ::DevQueryCaps(hdcScreen, CAPS_WIDTH, 1L, (PLONG)pWidth);
678 ::DevQueryCaps(hdcScreen, CAPS_HEIGHT, 1L, (PLONG)pHeight);
679 DevCloseDC(hdcScreen);
45fcbf3b
DW
680}
681
1be7f92a
DW
682bool wxDirExists(
683 const wxString& rDir
684)
45fcbf3b 685{
1be7f92a 686 return (::DosSetCurrentDir(WXSTRINGCAST rDir));
45fcbf3b
DW
687}
688
689// ---------------------------------------------------------------------------
690// window information functions
691// ---------------------------------------------------------------------------
692
1be7f92a
DW
693wxString WXDLLEXPORT wxGetWindowText(
694 WXHWND hWnd
695)
45fcbf3b 696{
1be7f92a
DW
697 wxString vStr;
698 long lLen = ::WinQueryWindowTextLength((HWND)hWnd) + 1;
699
700 ::WinQueryWindowText((HWND)hWnd, lLen, vStr.GetWriteBuf((int)lLen));
701 vStr.UngetWriteBuf();
45fcbf3b 702
1be7f92a 703 return vStr;
45fcbf3b
DW
704}
705
1be7f92a
DW
706wxString WXDLLEXPORT wxGetWindowClass(
707 WXHWND hWnd
708)
45fcbf3b 709{
1be7f92a
DW
710 wxString vStr;
711 int nLen = 256; // some starting value
45fcbf3b
DW
712
713 for ( ;; )
714 {
1be7f92a 715 int nCount = ::WinQueryClassName((HWND)hWnd, nLen, vStr.GetWriteBuf(nLen));
45fcbf3b 716
1be7f92a
DW
717 vStr.UngetWriteBuf();
718 if (nCount == nLen )
45fcbf3b
DW
719 {
720 // the class name might have been truncated, retry with larger
721 // buffer
1be7f92a 722 nLen *= 2;
45fcbf3b
DW
723 }
724 else
725 {
726 break;
727 }
728 }
1be7f92a 729 return vStr;
45fcbf3b
DW
730}
731
1be7f92a
DW
732WXWORD WXDLLEXPORT wxGetWindowId(
733 WXHWND hWnd
734)
45fcbf3b
DW
735{
736 return ::WinQueryWindowUShort((HWND)hWnd, QWS_ID);
0e320a79
DW
737}
738