]> git.saurik.com Git - wxWidgets.git/blame - src/msw/utils.cpp
controls can now be put in the toolbars (MSW only so far, preliminary version)
[wxWidgets.git] / src / msw / utils.cpp
CommitLineData
2bda0e17
KB
1/////////////////////////////////////////////////////////////////////////////
2// Name: utils.cpp
3// Purpose: Various utilities
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart and Markus Holzem
743e0a66 9// Licence: wxWindows license
2bda0e17
KB
10/////////////////////////////////////////////////////////////////////////////
11
12#ifdef __GNUG__
a3b46648 13// #pragma implementation "utils.h" // Note: this is done in utilscmn.cpp now.
2bda0e17
KB
14#endif
15
16// For compilers that support precompilation, includes "wx.h".
17#include "wx/wxprec.h"
18
19#ifdef __BORLANDC__
a4f96412 20 #pragma hdrstop
2bda0e17
KB
21#endif
22
23#ifndef WX_PRECOMP
a4f96412
VZ
24 #include "wx/setup.h"
25 #include "wx/utils.h"
26 #include "wx/app.h"
27 #include "wx/cursor.h"
743e0a66 28#endif //WX_PRECOMP
2bda0e17
KB
29
30#include "wx/msw/private.h"
31#include "wx/timer.h"
57c208c5 32#include "wx/intl.h"
2bda0e17 33
c030b70f
JS
34#include <windows.h>
35
2bda0e17
KB
36#include <ctype.h>
37
5ea105e0 38#if !defined(__GNUWIN32__) && !defined(__WXWINE__) && !defined(__SALFORDC__)
a4f96412 39 #include <direct.h>
ce3ed50d 40
a4f96412
VZ
41 #ifndef __MWERKS__
42 #include <dos.h>
43 #endif
743e0a66 44#endif //GNUWIN32
2bda0e17 45
57c208c5 46#if defined(__GNUWIN32__) && !defined(__TWIN32__)
a4f96412
VZ
47 #include <sys/unistd.h>
48 #include <sys/stat.h>
743e0a66
VZ
49#endif //GNUWIN32
50
51#include "wx/log.h"
2bda0e17
KB
52
53#ifdef __BORLANDC__ // Please someone tell me which version of Borland needs
54 // this (3.1 I believe) and how to test for it.
55 // If this works for Borland 4.0 as well, then no worries.
a4f96412 56 #include <dir.h>
2bda0e17
KB
57#endif
58
a4f96412
VZ
59// VZ: there is some code using NetXXX() functions to get the full user name:
60// I don't think it's a good idea because they don't work under Win95 and
61// seem to return the same as wxGetUserId() under NT. If you really want
62// to use them, just #define USE_NET_API
63#undef USE_NET_API
64
65#ifdef USE_NET_API
66 #include <lm.h>
67#endif // USE_NET_API
68
5ea105e0 69#if defined(__WIN32__) && !defined(__WXWINE__)
a4f96412 70 #include <io.h>
2bda0e17 71
a4f96412
VZ
72 #ifndef __GNUWIN32__
73 #include <shellapi.h>
74 #endif
2bda0e17
KB
75#endif
76
77#include <stdio.h>
78#include <stdlib.h>
79#include <string.h>
80#ifndef __WATCOMC__
3f4a0c5b
VZ
81 #if !(defined(_MSC_VER) && (_MSC_VER > 800))
82 #include <errno.h>
83 #endif
2bda0e17
KB
84#endif
85#include <stdarg.h>
86
c030b70f 87//// BEGIN for console support: VC++ only
631f1bfe 88#ifdef __VISUALC__
c030b70f 89
3f4a0c5b 90#include "wx/msw/msvcrt.h"
c030b70f 91
3f4a0c5b 92#include <fcntl.h>
c030b70f 93
3f4a0c5b 94#include "wx/ioswrap.h"
c030b70f
JS
95
96#if wxUSE_IOSTREAMH
97// N.B. BC++ doesn't have istream.h, ostream.h
c030b70f
JS
98# include <io.h>
99# include <fstream.h>
c030b70f 100#else
c030b70f 101# include <fstream>
c030b70f
JS
102#endif
103
104/* Need to undef new if including crtdbg.h */
105# ifdef new
106# undef new
107# endif
108
1e6d9499 109#ifndef __WIN16__
c030b70f 110# include <crtdbg.h>
1e6d9499 111#endif
c030b70f
JS
112
113# if defined(__WXDEBUG__) && wxUSE_GLOBAL_MEMORY_OPERATORS && wxUSE_DEBUG_NEW_ALWAYS
114# define new new(__FILE__,__LINE__)
115# endif
116
631f1bfe
JS
117#endif
118 // __VISUALC__
c030b70f
JS
119/// END for console support
120
2bda0e17 121// In the WIN.INI file
223d09f6
KB
122static const wxChar WX_SECTION[] = wxT("wxWindows");
123static const wxChar eHOSTNAME[] = wxT("HostName");
124static const wxChar eUSERID[] = wxT("UserId");
125static const wxChar eUSERNAME[] = wxT("UserName");
2bda0e17 126
2bda0e17
KB
127// For the following functions we SHOULD fill in support
128// for Windows-NT (which I don't know) as I assume it begin
129// a POSIX Unix (so claims MS) that it has some special
130// functions beyond those provided by WinSock
131
132// Get full hostname (eg. DoDo.BSn-Germany.crg.de)
837e5743 133bool wxGetHostName(wxChar *buf, int maxSize)
2bda0e17 134{
57c208c5 135#if defined(__WIN32__) && !defined(__TWIN32__)
2bda0e17
KB
136 DWORD nSize = maxSize;
137 return (::GetComputerName(buf, &nSize) != 0);
138#else
837e5743 139 wxChar *sysname;
223d09f6 140 const wxChar *default_host = wxT("noname");
2bda0e17 141
223d09f6 142 if ((sysname = wxGetenv(wxT("SYSTEM_NAME"))) == NULL) {
2bda0e17
KB
143 GetProfileString(WX_SECTION, eHOSTNAME, default_host, buf, maxSize - 1);
144 } else
837e5743 145 wxStrncpy(buf, sysname, maxSize - 1);
223d09f6 146 buf[maxSize] = wxT('\0');
2bda0e17
KB
147 return *buf ? TRUE : FALSE;
148#endif
149}
150
151// Get user ID e.g. jacs
837e5743 152bool wxGetUserId(wxChar *buf, int maxSize)
2bda0e17 153{
57c208c5 154#if defined(__WIN32__) && !defined(__win32s__) && !defined(__TWIN32__)
0a144271
VZ
155 DWORD nSize = maxSize;
156 if ( ::GetUserName(buf, &nSize) == 0 )
157 {
a4f96412 158 // actually, it does happen on Win9x if the user didn't log on
223d09f6 159 DWORD res = ::GetEnvironmentVariable(wxT("username"), buf, maxSize);
a4f96412
VZ
160 if ( res == 0 )
161 {
162 // not found
163 return FALSE;
164 }
0a144271
VZ
165 }
166
167 return TRUE;
0a144271 168#else // Win16 or Win32s
837e5743 169 wxChar *user;
223d09f6 170 const wxChar *default_id = wxT("anonymous");
2bda0e17
KB
171
172 // Can't assume we have NIS (PC-NFS) or some other ID daemon
173 // So we ...
223d09f6
KB
174 if ( (user = wxGetenv(wxT("USER"))) == NULL &&
175 (user = wxGetenv(wxT("LOGNAME"))) == NULL )
a4f96412 176 {
2bda0e17
KB
177 // Use wxWindows configuration data (comming soon)
178 GetProfileString(WX_SECTION, eUSERID, default_id, buf, maxSize - 1);
a4f96412
VZ
179 }
180 else
181 {
837e5743 182 wxStrncpy(buf, user, maxSize - 1);
a4f96412
VZ
183 }
184
2bda0e17
KB
185 return *buf ? TRUE : FALSE;
186#endif
187}
188
189// Get user name e.g. Julian Smart
837e5743 190bool wxGetUserName(wxChar *buf, int maxSize)
2bda0e17 191{
0a144271 192#if wxUSE_PENWINDOWS && !defined(__WATCOMC__) && !defined(__GNUWIN32__)
81d66cf3
JS
193 extern HANDLE g_hPenWin; // PenWindows Running?
194 if (g_hPenWin)
2bda0e17
KB
195 {
196 // PenWindows Does have a user concept!
197 // Get the current owner of the recognizer
198 GetPrivateProfileString("Current", "User", default_name, wxBuffer, maxSize - 1, "PENWIN.INI");
199 strncpy(buf, wxBuffer, maxSize - 1);
200 }
201 else
202#endif
203 {
a4f96412
VZ
204#ifdef USE_NET_API
205 CHAR szUserName[256];
206 if ( !wxGetUserId(szUserName, WXSIZEOF(szUserName)) )
207 return FALSE;
208
209 // TODO how to get the domain name?
210 CHAR *szDomain = "";
211
212 // the code is based on the MSDN example (also see KB article Q119670)
213 WCHAR wszUserName[256]; // Unicode user name
214 WCHAR wszDomain[256];
215 LPBYTE ComputerName;
216
217 USER_INFO_2 *ui2; // User structure
218
219 // Convert ANSI user name and domain to Unicode
220 MultiByteToWideChar( CP_ACP, 0, szUserName, strlen(szUserName)+1,
221 wszUserName, WXSIZEOF(wszUserName) );
222 MultiByteToWideChar( CP_ACP, 0, szDomain, strlen(szDomain)+1,
223 wszDomain, WXSIZEOF(wszDomain) );
224
225 // Get the computer name of a DC for the domain.
226 if ( NetGetDCName( NULL, wszDomain, &ComputerName ) != NERR_Success )
227 {
223d09f6 228 wxLogError(wxT("Can not find domain controller"));
a4f96412
VZ
229
230 goto error;
231 }
232
233 // Look up the user on the DC
234 NET_API_STATUS status = NetUserGetInfo( (LPWSTR)ComputerName,
235 (LPWSTR)&wszUserName,
236 2, // level - we want USER_INFO_2
237 (LPBYTE *) &ui2 );
238 switch ( status )
239 {
240 case NERR_Success:
241 // ok
242 break;
243
244 case NERR_InvalidComputer:
223d09f6 245 wxLogError(wxT("Invalid domain controller name."));
a4f96412
VZ
246
247 goto error;
248
249 case NERR_UserNotFound:
223d09f6 250 wxLogError(wxT("Invalid user name '%s'."), szUserName);
a4f96412
VZ
251
252 goto error;
253
254 default:
223d09f6 255 wxLogSysError(wxT("Can't get information about user"));
a4f96412
VZ
256
257 goto error;
258 }
259
260 // Convert the Unicode full name to ANSI
261 WideCharToMultiByte( CP_ACP, 0, ui2->usri2_full_name, -1,
262 buf, maxSize, NULL, NULL );
263
264 return TRUE;
265
266error:
223d09f6 267 wxLogError(wxT("Couldn't look up full user name."));
a4f96412
VZ
268
269 return FALSE;
270#else // !USE_NET_API
2bda0e17 271 // Could use NIS, MS-Mail or other site specific programs
b07135ba 272 // Use wxWindows configuration data
223d09f6 273 bool ok = GetProfileString(WX_SECTION, eUSERNAME, wxT(""), buf, maxSize - 1) != 0;
0a144271
VZ
274 if ( !ok )
275 {
276 ok = wxGetUserId(buf, maxSize);
277 }
278
279 if ( !ok )
280 {
223d09f6 281 wxStrncpy(buf, wxT("Unknown User"), maxSize);
0a144271 282 }
a4f96412 283#endif // Win32/16
2bda0e17 284 }
0a144271
VZ
285
286 return TRUE;
2bda0e17
KB
287}
288
2bda0e17
KB
289int wxKill(long pid, int sig)
290{
291 return 0;
292}
293
294//
295// Execute a program in an Interactive Shell
296//
297bool
298wxShell(const wxString& command)
299{
837e5743 300 wxChar *shell;
223d09f6
KB
301 if ((shell = wxGetenv(wxT("COMSPEC"))) == NULL)
302 shell = wxT("\\COMMAND.COM");
2bda0e17 303
837e5743 304 wxChar tmp[255];
223d09f6
KB
305 if (command != wxT(""))
306 wxSprintf(tmp, wxT("%s /c %s"), shell, WXSTRINGCAST command);
2bda0e17 307 else
837e5743 308 wxStrcpy(tmp, shell);
2bda0e17 309
837e5743 310 return (wxExecute((wxChar *)tmp, FALSE) != 0);
2bda0e17
KB
311}
312
313// Get free memory in bytes, or -1 if cannot determine amount (e.g. on UNIX)
634903fd 314long wxGetFreeMemory()
2bda0e17 315{
57c208c5 316#if defined(__WIN32__) && !defined(__BORLANDC__) && !defined(__TWIN32__)
2bda0e17
KB
317 MEMORYSTATUS memStatus;
318 memStatus.dwLength = sizeof(MEMORYSTATUS);
319 GlobalMemoryStatus(&memStatus);
320 return memStatus.dwAvailPhys;
321#else
322 return (long)GetFreeSpace(0);
323#endif
324}
325
326// Sleep for nSecs seconds. Attempt a Windows implementation using timers.
327static bool inTimer = FALSE;
328class wxSleepTimer: public wxTimer
329{
330 public:
634903fd 331 inline void Notify()
2bda0e17
KB
332 {
333 inTimer = FALSE;
334 Stop();
335 }
336};
337
338static wxTimer *wxTheSleepTimer = NULL;
339
afb74891
VZ
340void wxUsleep(unsigned long milliseconds)
341{
58a33cb4 342#ifdef __WIN32__
afb74891 343 ::Sleep(milliseconds);
58a33cb4
JS
344#else
345 if (inTimer)
346 return;
347
348 wxTheSleepTimer = new wxSleepTimer;
349 inTimer = TRUE;
350 wxTheSleepTimer->Start(milliseconds);
351 while (inTimer)
352 {
353 if (wxTheApp->Pending())
354 wxTheApp->Dispatch();
355 }
356 delete wxTheSleepTimer;
357 wxTheSleepTimer = NULL;
358#endif
afb74891
VZ
359}
360
2bda0e17
KB
361void wxSleep(int nSecs)
362{
363#if 0 // WIN32 hangs app
364 Sleep( 1000*nSecs );
365#else
366 if (inTimer)
367 return;
368
369 wxTheSleepTimer = new wxSleepTimer;
370 inTimer = TRUE;
371 wxTheSleepTimer->Start(nSecs*1000);
372 while (inTimer)
373 {
374 if (wxTheApp->Pending())
375 wxTheApp->Dispatch();
376 }
377 delete wxTheSleepTimer;
378 wxTheSleepTimer = NULL;
379#endif
380}
381
382// Consume all events until no more left
634903fd 383void wxFlushEvents()
2bda0e17
KB
384{
385// wxYield();
386}
387
388// Output a debug mess., in a system dependent fashion.
837e5743 389void wxDebugMsg(const wxChar *fmt ...)
2bda0e17
KB
390{
391 va_list ap;
837e5743 392 static wxChar buffer[512];
2bda0e17
KB
393
394 if (!wxTheApp->GetWantDebugOutput())
395 return ;
396
397 va_start(ap, fmt);
398
399 wvsprintf(buffer,fmt,ap) ;
837e5743 400 OutputDebugString((LPCTSTR)buffer) ;
2bda0e17
KB
401
402 va_end(ap);
403}
404
405// Non-fatal error: pop up message box and (possibly) continue
406void wxError(const wxString& msg, const wxString& title)
407{
223d09f6 408 wxSprintf(wxBuffer, wxT("%s\nContinue?"), WXSTRINGCAST msg);
837e5743 409 if (MessageBox(NULL, (LPCTSTR)wxBuffer, (LPCTSTR)WXSTRINGCAST title,
2bda0e17
KB
410 MB_ICONSTOP | MB_YESNO) == IDNO)
411 wxExit();
412}
413
414// Fatal error: pop up message box and abort
415void wxFatalError(const wxString& msg, const wxString& title)
416{
223d09f6 417 wxSprintf(wxBuffer, wxT("%s: %s"), WXSTRINGCAST title, WXSTRINGCAST msg);
837e5743 418 FatalAppExit(0, (LPCTSTR)wxBuffer);
2bda0e17
KB
419}
420
421// Emit a beeeeeep
634903fd 422void wxBell()
2bda0e17 423{
b07135ba
RD
424 // Removed by RD because IHMO syncronous sound is a Bad Thing. MessageBeep
425 // will do a similar thing anyway if there is no sound card...
426//#ifdef __WIN32__
427// Beep(1000,1000) ; // 1kHz during 1 sec.
428//#else
ce200637 429 MessageBeep((UINT)-1) ;
b07135ba 430//#endif
2bda0e17
KB
431}
432
6f65e337
JS
433// Chris Breeze 27/5/98: revised WIN32 code to
434// detect WindowsNT correctly
2bda0e17
KB
435int wxGetOsVersion(int *majorVsn, int *minorVsn)
436{
6f65e337
JS
437 if (majorVsn) *majorVsn = 0;
438 if (minorVsn) *minorVsn = 0;
439
2432b92d 440#if defined(__WIN32__) && !defined(__SC__)
6f65e337
JS
441 OSVERSIONINFO info;
442 memset(&info, 0, sizeof(OSVERSIONINFO));
443 info.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
444 if (GetVersionEx(&info))
445 {
446 if (majorVsn) *majorVsn = info.dwMajorVersion;
447 if (minorVsn) *minorVsn = info.dwMinorVersion;
448 switch (info.dwPlatformId)
743e0a66
VZ
449 {
450 case VER_PLATFORM_WIN32s:
451 return wxWIN32S;
452 break;
453 case VER_PLATFORM_WIN32_WINDOWS:
454 return wxWIN95;
455 break;
456 case VER_PLATFORM_WIN32_NT:
457 return wxWINDOWS_NT;
458 break;
6f65e337 459 }
743e0a66
VZ
460 }
461 return wxWINDOWS; // error if we get here, return generic value
6f65e337
JS
462#else
463 // Win16 code...
2432b92d 464 int retValue = 0;
6f65e337 465# ifdef __WINDOWS_386__
2bda0e17 466 retValue = wxWIN386;
6f65e337 467# else
47d67540 468# if !defined(__WATCOMC__) && !defined(GNUWIN32) && wxUSE_PENWINDOWS
81d66cf3
JS
469 extern HANDLE g_hPenWin;
470 retValue = g_hPenWin ? wxPENWINDOWS : wxWINDOWS ;
6f65e337
JS
471# endif
472# endif
2bda0e17
KB
473 // @@@@ To be completed. I don't have the manual here...
474 if (majorVsn) *majorVsn = 3 ;
475 if (minorVsn) *minorVsn = 1 ;
476 return retValue ;
6f65e337 477#endif
2bda0e17
KB
478}
479
480// Reading and writing resources (eg WIN.INI, .Xdefaults)
47d67540 481#if wxUSE_RESOURCES
2bda0e17
KB
482bool wxWriteResource(const wxString& section, const wxString& entry, const wxString& value, const wxString& file)
483{
223d09f6 484 if (file != wxT(""))
8fb3a512
JS
485// Don't know what the correct cast should be, but it doesn't
486// compile in BC++/16-bit without this cast.
487#if !defined(__WIN32__)
488 return (WritePrivateProfileString((const char*) section, (const char*) entry, (const char*) value, (const char*) file) != 0);
489#else
837e5743 490 return (WritePrivateProfileString((LPCTSTR)WXSTRINGCAST section, (LPCTSTR)WXSTRINGCAST entry, (LPCTSTR)value, (LPCTSTR)WXSTRINGCAST file) != 0);
8fb3a512 491#endif
2bda0e17 492 else
837e5743 493 return (WriteProfileString((LPCTSTR)WXSTRINGCAST section, (LPCTSTR)WXSTRINGCAST entry, (LPCTSTR)WXSTRINGCAST value) != 0);
2bda0e17
KB
494}
495
496bool wxWriteResource(const wxString& section, const wxString& entry, float value, const wxString& file)
497{
837e5743 498 wxChar buf[50];
223d09f6 499 wxSprintf(buf, wxT("%.4f"), value);
2bda0e17
KB
500 return wxWriteResource(section, entry, buf, file);
501}
502
503bool wxWriteResource(const wxString& section, const wxString& entry, long value, const wxString& file)
504{
837e5743 505 wxChar buf[50];
223d09f6 506 wxSprintf(buf, wxT("%ld"), value);
2bda0e17
KB
507 return wxWriteResource(section, entry, buf, file);
508}
509
510bool wxWriteResource(const wxString& section, const wxString& entry, int value, const wxString& file)
511{
837e5743 512 wxChar buf[50];
223d09f6 513 wxSprintf(buf, wxT("%d"), value);
2bda0e17
KB
514 return wxWriteResource(section, entry, buf, file);
515}
516
837e5743 517bool wxGetResource(const wxString& section, const wxString& entry, wxChar **value, const wxString& file)
2bda0e17 518{
223d09f6
KB
519 static const wxChar defunkt[] = wxT("$$default");
520 if (file != wxT(""))
2bda0e17 521 {
837e5743
OK
522 int n = GetPrivateProfileString((LPCTSTR)WXSTRINGCAST section, (LPCTSTR)WXSTRINGCAST entry, (LPCTSTR)defunkt,
523 (LPTSTR)wxBuffer, 1000, (LPCTSTR)WXSTRINGCAST file);
524 if (n == 0 || wxStrcmp(wxBuffer, defunkt) == 0)
2bda0e17
KB
525 return FALSE;
526 }
527 else
528 {
837e5743
OK
529 int n = GetProfileString((LPCTSTR)WXSTRINGCAST section, (LPCTSTR)WXSTRINGCAST entry, (LPCTSTR)defunkt,
530 (LPTSTR)wxBuffer, 1000);
531 if (n == 0 || wxStrcmp(wxBuffer, defunkt) == 0)
2bda0e17
KB
532 return FALSE;
533 }
534 if (*value) delete[] (*value);
535 *value = copystring(wxBuffer);
536 return TRUE;
537 }
538
539bool wxGetResource(const wxString& section, const wxString& entry, float *value, const wxString& file)
540{
837e5743
OK
541 wxChar *s = NULL;
542 bool succ = wxGetResource(section, entry, (wxChar **)&s, file);
2bda0e17
KB
543 if (succ)
544 {
837e5743 545 *value = (float)wxStrtod(s, NULL);
2bda0e17
KB
546 delete[] s;
547 return TRUE;
548 }
549 else return FALSE;
550}
551
552bool wxGetResource(const wxString& section, const wxString& entry, long *value, const wxString& file)
553{
837e5743
OK
554 wxChar *s = NULL;
555 bool succ = wxGetResource(section, entry, (wxChar **)&s, file);
2bda0e17
KB
556 if (succ)
557 {
837e5743 558 *value = wxStrtol(s, NULL, 10);
2bda0e17
KB
559 delete[] s;
560 return TRUE;
561 }
562 else return FALSE;
563}
564
565bool wxGetResource(const wxString& section, const wxString& entry, int *value, const wxString& file)
566{
837e5743
OK
567 wxChar *s = NULL;
568 bool succ = wxGetResource(section, entry, (wxChar **)&s, file);
2bda0e17
KB
569 if (succ)
570 {
837e5743 571 *value = (int)wxStrtol(s, NULL, 10);
b07135ba 572 delete[] s;
2bda0e17
KB
573 return TRUE;
574 }
575 else return FALSE;
576}
47d67540 577#endif // wxUSE_RESOURCES
2bda0e17 578
634903fd 579// ---------------------------------------------------------------------------
25889d3c 580// helper functions for showing a "busy" cursor
634903fd
VZ
581// ---------------------------------------------------------------------------
582
25889d3c
JS
583HCURSOR gs_wxBusyCursor = 0; // new, busy cursor
584HCURSOR gs_wxBusyCursorOld = 0; // old cursor
634903fd 585static int gs_wxBusyCursorCount = 0;
2bda0e17
KB
586
587// Set the cursor to the busy cursor for all windows
588void wxBeginBusyCursor(wxCursor *cursor)
589{
634903fd
VZ
590 if ( gs_wxBusyCursorCount++ == 0 )
591 {
592 gs_wxBusyCursor = (HCURSOR)cursor->GetHCURSOR();
593 gs_wxBusyCursorOld = ::SetCursor(gs_wxBusyCursor);
594 }
595 //else: nothing to do, already set
2bda0e17
KB
596}
597
598// Restore cursor to normal
634903fd 599void wxEndBusyCursor()
2bda0e17 600{
634903fd 601 wxCHECK_RET( gs_wxBusyCursorCount > 0,
223d09f6 602 wxT("no matching wxBeginBusyCursor() for wxEndBusyCursor()") );
b07135ba 603
634903fd
VZ
604 if ( --gs_wxBusyCursorCount == 0 )
605 {
606 ::SetCursor(gs_wxBusyCursorOld);
607
608 gs_wxBusyCursorOld = 0;
609 }
2bda0e17
KB
610}
611
612// TRUE if we're between the above two calls
634903fd 613bool wxIsBusy()
2bda0e17 614{
634903fd 615 return (gs_wxBusyCursorCount > 0);
b07135ba 616}
2bda0e17 617
634903fd 618// ---------------------------------------------------------------------------
837e5743 619const wxChar* wxGetHomeDir(wxString *pstr)
743e0a66
VZ
620{
621 wxString& strDir = *pstr;
622
57c208c5 623 #if defined(__UNIX__) && !defined(__TWIN32__)
837e5743 624 const wxChar *szHome = wxGetenv("HOME");
743e0a66
VZ
625 if ( szHome == NULL ) {
626 // we're homeless...
627 wxLogWarning(_("can't find user's HOME, using current directory."));
223d09f6 628 strDir = wxT(".");
743e0a66
VZ
629 }
630 else
631 strDir = szHome;
632
633 // add a trailing slash if needed
223d09f6
KB
634 if ( strDir.Last() != wxT('/') )
635 strDir << wxT('/');
743e0a66
VZ
636 #else // Windows
637 #ifdef __WIN32__
223d09f6 638 const wxChar *szHome = wxGetenv(wxT("HOMEDRIVE"));
743e0a66
VZ
639 if ( szHome != NULL )
640 strDir << szHome;
223d09f6 641 szHome = wxGetenv(wxT("HOMEPATH"));
743e0a66
VZ
642 if ( szHome != NULL ) {
643 strDir << szHome;
644
645 // the idea is that under NT these variables have default values
646 // of "%systemdrive%:" and "\\". As we don't want to create our
647 // config files in the root directory of the system drive, we will
648 // create it in our program's dir. However, if the user took care
649 // to set HOMEPATH to something other than "\\", we suppose that he
650 // knows what he is doing and use the supplied value.
223d09f6 651 if ( wxStrcmp(szHome, wxT("\\")) != 0 )
743e0a66
VZ
652 return strDir.c_str();
653 }
654
655 #else // Win16
656 // Win16 has no idea about home, so use the working directory instead
657 #endif // WIN16/32
658
659 // 260 was taken from windef.h
660 #ifndef MAX_PATH
661 #define MAX_PATH 260
662 #endif
663
664 wxString strPath;
665 ::GetModuleFileName(::GetModuleHandle(NULL),
666 strPath.GetWriteBuf(MAX_PATH), MAX_PATH);
667 strPath.UngetWriteBuf();
668
669 // extract the dir name
670 wxSplitPath(strPath, &strDir, NULL, NULL);
671
672 #endif // UNIX/Win
673
674 return strDir.c_str();
675}
676
2bda0e17 677// Hack for MS-DOS
837e5743 678wxChar *wxGetUserHome (const wxString& user)
2bda0e17 679{
837e5743 680 wxChar *home;
2bda0e17
KB
681 wxString user1(user);
682
223d09f6 683 if (user1 != wxT("")) {
837e5743 684 wxChar tmp[64];
2bda0e17
KB
685 if (wxGetUserId(tmp, sizeof(tmp)/sizeof(char))) {
686 // Guests belong in the temp dir
223d09f6
KB
687 if (wxStricmp(tmp, wxT("annonymous")) == 0) {
688 if ((home = wxGetenv(wxT("TMP"))) != NULL ||
689 (home = wxGetenv(wxT("TMPDIR"))) != NULL ||
690 (home = wxGetenv(wxT("TEMP"))) != NULL)
691 return *home ? home : (wxChar*)wxT("\\");
2bda0e17 692 }
837e5743 693 if (wxStricmp(tmp, WXSTRINGCAST user1) == 0)
223d09f6 694 user1 = wxT("");
2bda0e17
KB
695 }
696 }
223d09f6
KB
697 if (user1 == wxT(""))
698 if ((home = wxGetenv(wxT("HOME"))) != NULL)
2bda0e17 699 {
837e5743 700 wxStrcpy(wxBuffer, home);
2bda0e17
KB
701 Unix2DosFilename(wxBuffer);
702 return wxBuffer;
703 }
704 return NULL; // No home known!
705}
706
707// Check whether this window wants to process messages, e.g. Stop button
708// in long calculations.
709bool wxCheckForInterrupt(wxWindow *wnd)
710{
743e0a66
VZ
711 if(wnd){
712 MSG msg;
713 HWND win= (HWND) wnd->GetHWND();
714 while(PeekMessage(&msg,win,0,0,PM_REMOVE)){
715 TranslateMessage(&msg);
716 DispatchMessage(&msg);
717 }
718 return TRUE;//*** temporary?
719 }
720 else{
223d09f6 721 wxFAIL_MSG(wxT("wnd==NULL !!!"));
634903fd 722
743e0a66
VZ
723 return FALSE;//*** temporary?
724 }
2bda0e17
KB
725}
726
727// MSW only: get user-defined resource from the .res file.
728// Returns NULL or newly-allocated memory, so use delete[] to clean up.
729
2049ba38 730#ifdef __WXMSW__
837e5743 731wxChar *wxLoadUserResource(const wxString& resourceName, const wxString& resourceType)
2bda0e17 732{
837e5743 733 wxChar *s = NULL;
57c208c5 734#if !defined(__WIN32__) || defined(__TWIN32__)
2bda0e17
KB
735 HRSRC hResource = ::FindResource(wxGetInstance(), WXSTRINGCAST resourceName, WXSTRINGCAST resourceType);
736#else
737#ifdef UNICODE
738 HRSRC hResource = ::FindResourceW(wxGetInstance(), WXSTRINGCAST resourceName, WXSTRINGCAST resourceType);
739#else
740 HRSRC hResource = ::FindResourceA(wxGetInstance(), WXSTRINGCAST resourceName, WXSTRINGCAST resourceType);
741#endif
742#endif
743
744 if (hResource == 0)
745 return NULL;
746 HGLOBAL hData = ::LoadResource(wxGetInstance(), hResource);
747 if (hData == 0)
748 return NULL;
837e5743 749 wxChar *theText = (wxChar *)LockResource(hData);
2bda0e17
KB
750 if (!theText)
751 return NULL;
b07135ba 752
2bda0e17
KB
753 s = copystring(theText);
754
755 // Obsolete in WIN32
756#ifndef __WIN32__
757 UnlockResource(hData);
758#endif
759
760 // No need??
761// GlobalFree(hData);
762
763 return s;
764}
765#endif
766
767void wxGetMousePosition( int* x, int* y )
768{
769 POINT pt;
770 GetCursorPos( & pt );
771 *x = pt.x;
772 *y = pt.y;
773};
774
775// Return TRUE if we have a colour display
634903fd 776bool wxColourDisplay()
2bda0e17 777{
57c208c5 778 HDC dc = ::GetDC((HWND) NULL);
2bda0e17
KB
779 bool flag;
780 int noCols = GetDeviceCaps(dc, NUMCOLORS);
781 if ((noCols == -1) || (noCols > 2))
782 flag = TRUE;
783 else
784 flag = FALSE;
57c208c5 785 ReleaseDC((HWND) NULL, dc);
2bda0e17
KB
786 return flag;
787}
788
789// Returns depth of screen
634903fd 790int wxDisplayDepth()
2bda0e17 791{
57c208c5 792 HDC dc = ::GetDC((HWND) NULL);
2bda0e17
KB
793 int planes = GetDeviceCaps(dc, PLANES);
794 int bitsPerPixel = GetDeviceCaps(dc, BITSPIXEL);
795 int depth = planes*bitsPerPixel;
57c208c5 796 ReleaseDC((HWND) NULL, dc);
2bda0e17
KB
797 return depth;
798}
799
800// Get size of display
801void wxDisplaySize(int *width, int *height)
802{
57c208c5 803 HDC dc = ::GetDC((HWND) NULL);
2bda0e17 804 *width = GetDeviceCaps(dc, HORZRES); *height = GetDeviceCaps(dc, VERTRES);
57c208c5 805 ReleaseDC((HWND) NULL, dc);
2bda0e17
KB
806}
807
7f555861
JS
808bool wxDirExists(const wxString& dir)
809{
810 /* MATTHEW: [6] Always use same code for Win32, call FindClose */
811#if defined(__WIN32__)
812 WIN32_FIND_DATA fileInfo;
813#else
814#ifdef __BORLANDC__
815 struct ffblk fileInfo;
816#else
817 struct find_t fileInfo;
818#endif
819#endif
820
821#if defined(__WIN32__)
3f4a0c5b
VZ
822 HANDLE h = FindFirstFile((LPTSTR) WXSTRINGCAST dir,(LPWIN32_FIND_DATA)&fileInfo);
823
824 if (h==INVALID_HANDLE_VALUE)
825 return FALSE;
826 else {
827 FindClose(h);
828 return ((fileInfo.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == FILE_ATTRIBUTE_DIRECTORY);
829 }
7f555861
JS
830#else
831 // In Borland findfirst has a different argument
832 // ordering from _dos_findfirst. But _dos_findfirst
833 // _should_ be ok in both MS and Borland... why not?
834#ifdef __BORLANDC__
835 return ((findfirst(WXSTRINGCAST dir, &fileInfo, _A_SUBDIR) == 0 && (fileInfo.ff_attrib & _A_SUBDIR) != 0));
836#else
837 return (((_dos_findfirst(WXSTRINGCAST dir, _A_SUBDIR, &fileInfo) == 0) && (fileInfo.attrib & _A_SUBDIR)) != 0);
838#endif
839#endif
840}
841
cc2b7472
VZ
842// ---------------------------------------------------------------------------
843// window information functions
844// ---------------------------------------------------------------------------
845
1c4a764c
VZ
846wxString WXDLLEXPORT wxGetWindowText(WXHWND hWnd)
847{
848 wxString str;
849 int len = GetWindowTextLength((HWND)hWnd) + 1;
850 GetWindowText((HWND)hWnd, str.GetWriteBuf(len), len);
851 str.UngetWriteBuf();
852
853 return str;
854}
855
cc2b7472
VZ
856wxString WXDLLEXPORT wxGetWindowClass(WXHWND hWnd)
857{
858 wxString str;
859
860 int len = 256; // some starting value
861
862 for ( ;; )
863 {
864 // as we've #undefined GetClassName we must now manually choose the
865 // right function to call
866 int count =
867
868 #ifndef __WIN32__
869 GetClassName
870 #else // Win32
871 #ifdef UNICODE
872 GetClassNameW
873 #else // !Unicode
874 #ifdef __TWIN32__
875 GetClassName
876 #else // !Twin32
877 GetClassNameA
878 #endif // Twin32/!Twin32
879 #endif // Unicode/ANSI
880 #endif // Win16/32
881 ((HWND)hWnd, str.GetWriteBuf(len), len);
882
883 str.UngetWriteBuf();
884 if ( count == len )
885 {
886 // the class name might have been truncated, retry with larger
887 // buffer
888 len *= 2;
889 }
890 else
891 {
892 break;
893 }
894 }
895
896 return str;
897}
898
42e69d6b 899WXWORD WXDLLEXPORT wxGetWindowId(WXHWND hWnd)
cc2b7472
VZ
900{
901#ifndef __WIN32__
42e69d6b 902 return GetWindowWord((HWND)hWnd, GWW_ID);
cc2b7472 903#else // Win32
42e69d6b 904 return GetWindowLong((HWND)hWnd, GWL_ID);
cc2b7472
VZ
905#endif // Win16/32
906}
907
7f555861
JS
908#if 0
909//------------------------------------------------------------------------
910// wild character routines
911//------------------------------------------------------------------------
912
913bool wxIsWild( const wxString& pattern )
914{
915 wxString tmp = pattern;
916 char *pat = WXSTRINGCAST(tmp);
917 while (*pat) {
3f4a0c5b
VZ
918 switch (*pat++) {
919 case '?': case '*': case '[': case '{':
920 return TRUE;
921 case '\\':
922 if (!*pat++)
923 return FALSE;
924 }
7f555861
JS
925 }
926 return FALSE;
927};
928
929
930bool wxMatchWild( const wxString& pat, const wxString& text, bool dot_special )
931{
932 wxString tmp1 = pat;
933 char *pattern = WXSTRINGCAST(tmp1);
934 wxString tmp2 = text;
935 char *str = WXSTRINGCAST(tmp2);
936 char c;
937 char *cp;
938 bool done = FALSE, ret_code, ok;
939 // Below is for vi fans
940 const char OB = '{', CB = '}';
941
942 // dot_special means '.' only matches '.'
943 if (dot_special && *str == '.' && *pattern != *str)
3f4a0c5b 944 return FALSE;
7f555861
JS
945
946 while ((*pattern != '\0') && (!done)
947 && (((*str=='\0')&&((*pattern==OB)||(*pattern=='*')))||(*str!='\0'))) {
3f4a0c5b
VZ
948 switch (*pattern) {
949 case '\\':
950 pattern++;
951 if (*pattern != '\0')
952 pattern++;
953 break;
954 case '*':
955 pattern++;
956 ret_code = FALSE;
957 while ((*str!='\0')
958 && (!(ret_code=wxMatchWild(pattern, str++, FALSE))))
959 /*loop*/;
960 if (ret_code) {
961 while (*str != '\0')
962 str++;
963 while (*pattern != '\0')
964 pattern++;
965 }
966 break;
967 case '[':
968 pattern++;
969 repeat:
970 if ((*pattern == '\0') || (*pattern == ']')) {
971 done = TRUE;
972 break;
973 }
974 if (*pattern == '\\') {
975 pattern++;
976 if (*pattern == '\0') {
977 done = TRUE;
978 break;
979 }
980 }
981 if (*(pattern + 1) == '-') {
982 c = *pattern;
983 pattern += 2;
984 if (*pattern == ']') {
985 done = TRUE;
986 break;
987 }
988 if (*pattern == '\\') {
989 pattern++;
990 if (*pattern == '\0') {
991 done = TRUE;
992 break;
993 }
994 }
995 if ((*str < c) || (*str > *pattern)) {
996 pattern++;
997 goto repeat;
998 }
999 } else if (*pattern != *str) {
1000 pattern++;
1001 goto repeat;
1002 }
1003 pattern++;
1004 while ((*pattern != ']') && (*pattern != '\0')) {
1005 if ((*pattern == '\\') && (*(pattern + 1) != '\0'))
1006 pattern++;
1007 pattern++;
1008 }
1009 if (*pattern != '\0') {
1010 pattern++, str++;
1011 }
1012 break;
1013 case '?':
1014 pattern++;
1015 str++;
1016 break;
1017 case OB:
1018 pattern++;
1019 while ((*pattern != CB) && (*pattern != '\0')) {
1020 cp = str;
1021 ok = TRUE;
1022 while (ok && (*cp != '\0') && (*pattern != '\0')
1023 && (*pattern != ',') && (*pattern != CB)) {
1024 if (*pattern == '\\')
1025 pattern++;
1026 ok = (*pattern++ == *cp++);
1027 }
1028 if (*pattern == '\0') {
1029 ok = FALSE;
1030 done = TRUE;
1031 break;
1032 } else if (ok) {
1033 str = cp;
1034 while ((*pattern != CB) && (*pattern != '\0')) {
1035 if (*++pattern == '\\') {
1036 if (*++pattern == CB)
1037 pattern++;
1038 }
1039 }
1040 } else {
1041 while (*pattern!=CB && *pattern!=',' && *pattern!='\0') {
1042 if (*++pattern == '\\') {
7f555861 1043 if (*++pattern == CB || *pattern == ',')
3f4a0c5b
VZ
1044 pattern++;
1045 }
1046 }
1047 }
1048 if (*pattern != '\0')
1049 pattern++;
1050 }
1051 break;
1052 default:
1053 if (*str == *pattern) {
1054 str++, pattern++;
1055 } else {
1056 done = TRUE;
1057 }
1058 }
7f555861
JS
1059 }
1060 while (*pattern == '*')
3f4a0c5b 1061 pattern++;
7f555861
JS
1062 return ((*str == '\0') && (*pattern == '\0'));
1063};
1064
1065#endif
1066
a0a302dc
JS
1067#if defined(__WIN95__) && defined(__WXDEBUG__) && wxUSE_DBWIN32
1068
1069/*
1070When I started programming with Visual C++ v4.0, I missed one of my favorite
1071tools -- DBWIN. Finding the code for a simple debug trace utility, DBMON,
1072on MSDN was a step in the right direction, but it is a console application
1073and thus has limited features and extensibility. DBWIN32 is my creation
1074to solve this problem.
1075
b07135ba 1076The code is essentially a merging of a stripped down version of the DBWIN code
a0a302dc
JS
1077from VC 1.5 and DBMON.C with a few 32 bit changes.
1078
b07135ba 1079As of version 1.2B, DBWIN32 supports both Win95 and NT. The NT support is
a0a302dc
JS
1080built into the operating system and works just by running DBWIN32. The Win95
1081team decided not to support this hook, so I have provided code that will do
1082this for you. See the file WIN95.TXT for instructions on installing this.
1083
1084If you have questions, problems or suggestions about DBWIN32, I welcome your
1085feedback and plan to actively maintain the code.
1086
1087Andrew Tucker
1088ast@halcyon.com
1089
1090To download dbwin32, see e.g.:
1091
1092http://ftp.digital.com/pub/micro/NT/WinSite/programr/dbwin32.zip
1093*/
1094
57c208c5 1095#if !defined(__MWERKS__) && !defined(__SALFORDC__) && !defined(__TWIN32__)
a0a302dc 1096#include <process.h>
17dff81c 1097#endif
a0a302dc 1098
837e5743 1099void OutputDebugStringW95(const wxChar* lpOutputString, ...)
a0a302dc
JS
1100{
1101 HANDLE heventDBWIN; /* DBWIN32 synchronization object */
1102 HANDLE heventData; /* data passing synch object */
1103 HANDLE hSharedFile; /* memory mapped file shared data */
1104 LPSTR lpszSharedMem;
837e5743 1105 wxChar achBuffer[500];
a0a302dc
JS
1106
1107 /* create the output buffer */
1108 va_list args;
1109 va_start(args, lpOutputString);
837e5743 1110 wxVsprintf(achBuffer, lpOutputString, args);
a0a302dc
JS
1111 va_end(args);
1112
b07135ba
RD
1113 /*
1114 Do a regular OutputDebugString so that the output is
a0a302dc
JS
1115 still seen in the debugger window if it exists.
1116
b07135ba 1117 This ifdef is necessary to avoid infinite recursion
a0a302dc
JS
1118 from the inclusion of W95TRACE.H
1119 */
1120#ifdef _UNICODE
1121 ::OutputDebugStringW(achBuffer);
57c208c5
JS
1122#else
1123#ifdef __TWIN32__
1124 ::OutputDebugString(achBuffer);
a0a302dc
JS
1125#else
1126 ::OutputDebugStringA(achBuffer);
57c208c5 1127#endif
a0a302dc
JS
1128#endif
1129
1130 /* bail if it's not Win95 */
1131 {
1132 OSVERSIONINFO VerInfo;
1133 VerInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
1134 GetVersionEx(&VerInfo);
1135 if ( VerInfo.dwPlatformId != VER_PLATFORM_WIN32_WINDOWS )
1136 return;
1137 }
1138
1139 /* make sure DBWIN is open and waiting */
223d09f6 1140 heventDBWIN = OpenEvent(EVENT_MODIFY_STATE, FALSE, wxT("DBWIN_BUFFER_READY"));
a0a302dc
JS
1141 if ( !heventDBWIN )
1142 {
223d09f6 1143 //MessageBox(NULL, wxT("DBWIN_BUFFER_READY nonexistent"), NULL, MB_OK);
b07135ba 1144 return;
a0a302dc
JS
1145 }
1146
1147 /* get a handle to the data synch object */
223d09f6 1148 heventData = OpenEvent(EVENT_MODIFY_STATE, FALSE, wxT("DBWIN_DATA_READY"));
a0a302dc
JS
1149 if ( !heventData )
1150 {
223d09f6 1151 // MessageBox(NULL, wxT("DBWIN_DATA_READY nonexistent"), NULL, MB_OK);
a0a302dc 1152 CloseHandle(heventDBWIN);
b07135ba 1153 return;
a0a302dc 1154 }
b07135ba 1155
223d09f6 1156 hSharedFile = CreateFileMapping((HANDLE)-1, NULL, PAGE_READWRITE, 0, 4096, wxT("DBWIN_BUFFER"));
b07135ba 1157 if (!hSharedFile)
a0a302dc 1158 {
223d09f6 1159 //MessageBox(NULL, wxT("DebugTrace: Unable to create file mapping object DBWIN_BUFFER"), wxT("Error"), MB_OK);
a0a302dc
JS
1160 CloseHandle(heventDBWIN);
1161 CloseHandle(heventData);
1162 return;
1163 }
1164
1165 lpszSharedMem = (LPSTR)MapViewOfFile(hSharedFile, FILE_MAP_WRITE, 0, 0, 512);
b07135ba 1166 if (!lpszSharedMem)
a0a302dc 1167 {
223d09f6 1168 //MessageBox(NULL, wxT("DebugTrace: Unable to map shared memory"), wxT("Error"), MB_OK);
a0a302dc
JS
1169 CloseHandle(heventDBWIN);
1170 CloseHandle(heventData);
1171 return;
1172 }
1173
1174 /* wait for buffer event */
1175 WaitForSingleObject(heventDBWIN, INFINITE);
1176
1177 /* write it to the shared memory */
ce3ed50d 1178#if defined( __BORLANDC__ ) || defined( __MWERKS__ ) || defined(__SALFORDC__)
a0a302dc
JS
1179 *((LPDWORD)lpszSharedMem) = getpid();
1180#else
1181 *((LPDWORD)lpszSharedMem) = _getpid();
1182#endif
1183
223d09f6 1184 wsprintf((LPTSTR)(lpszSharedMem + sizeof(DWORD)), wxT("%s"), achBuffer);
a0a302dc
JS
1185
1186 /* signal data ready event */
1187 SetEvent(heventData);
1188
1189 /* clean up handles */
1190 CloseHandle(hSharedFile);
1191 CloseHandle(heventData);
1192 CloseHandle(heventDBWIN);
1193
1194 return;
1195}
1196
1197
1198#endif
1199
c030b70f 1200
cf45d1f4 1201#if 0
c030b70f
JS
1202
1203// maximum mumber of lines the output console should have
1204static const WORD MAX_CONSOLE_LINES = 500;
1205
1206BOOL WINAPI MyConsoleHandler( DWORD dwCtrlType ) { // control signal type
3f4a0c5b
VZ
1207 FreeConsole();
1208 return TRUE;
c030b70f
JS
1209}
1210
1211void wxRedirectIOToConsole()
1212{
1213 int hConHandle;
1214 long lStdHandle;
1215 CONSOLE_SCREEN_BUFFER_INFO coninfo;
1216 FILE *fp;
1217
1218 // allocate a console for this app
1219 AllocConsole();
1220
1221 // set the screen buffer to be big enough to let us scroll text
3f4a0c5b 1222 GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE),
c030b70f
JS
1223 &coninfo);
1224 coninfo.dwSize.Y = MAX_CONSOLE_LINES;
3f4a0c5b 1225 SetConsoleScreenBufferSize(GetStdHandle(STD_OUTPUT_HANDLE),
c030b70f
JS
1226 coninfo.dwSize);
1227
1228 // redirect unbuffered STDOUT to the console
1229 lStdHandle = (long)GetStdHandle(STD_OUTPUT_HANDLE);
1230 hConHandle = _open_osfhandle(lStdHandle, _O_TEXT);
1231 if(hConHandle <= 0) return;
1232 fp = _fdopen( hConHandle, "w" );
1233 *stdout = *fp;
1234 setvbuf( stdout, NULL, _IONBF, 0 );
1235
1236 // redirect unbuffered STDIN to the console
1237 lStdHandle = (long)GetStdHandle(STD_INPUT_HANDLE);
1238 hConHandle = _open_osfhandle(lStdHandle, _O_TEXT);
1239 if(hConHandle <= 0) return;
1240 fp = _fdopen( hConHandle, "r" );
1241 *stdin = *fp;
1242 setvbuf( stdin, NULL, _IONBF, 0 );
1243
1244 // redirect unbuffered STDERR to the console
1245 lStdHandle = (long)GetStdHandle(STD_ERROR_HANDLE);
1246 hConHandle = _open_osfhandle(lStdHandle, _O_TEXT);
1247 if(hConHandle <= 0) return;
1248 fp = _fdopen( hConHandle, "w" );
1249 *stderr = *fp;
1250 setvbuf( stderr, NULL, _IONBF, 0 );
3f4a0c5b
VZ
1251
1252 // make cout, wcout, cin, wcin, wcerr, cerr, wclog and clog
c030b70f
JS
1253 // point to console as well
1254 ios::sync_with_stdio();
1255
3f4a0c5b 1256 SetConsoleCtrlHandler(MyConsoleHandler, TRUE);
c030b70f
JS
1257}
1258#else
1259// Not supported
1260void wxRedirectIOToConsole()
1261{
1262}
1263#endif
1264
1265