1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/utilsgui.cpp 
   3 // Purpose:     Various utility functions only available in GUI 
   4 // Author:      Vadim Zeitlin 
   6 // Created:     21.06.2003 (extracted from msw/utils.cpp) 
   8 // Copyright:   (c) Julian Smart 
   9 // License:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // for compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  28     #include "wx/cursor.h" 
  29     #include "wx/window.h" 
  33 #include "wx/dynlib.h" 
  35 #include "wx/msw/private.h"     // includes <windows.h> 
  36 #include "wx/msw/registry.h" 
  37 #include <shellapi.h> // needed for SHELLEXECUTEINFO 
  40 // ============================================================================ 
  42 // ============================================================================ 
  44 // --------------------------------------------------------------------------- 
  45 // helper functions for showing a "busy" cursor 
  46 // --------------------------------------------------------------------------- 
  48 static HCURSOR gs_wxBusyCursor 
= 0;     // new, busy cursor 
  49 static HCURSOR gs_wxBusyCursorOld 
= 0;  // old cursor 
  50 static int gs_wxBusyCursorCount 
= 0; 
  52 extern HCURSOR 
wxGetCurrentBusyCursor() 
  54     return gs_wxBusyCursor
; 
  57 // Set the cursor to the busy cursor for all windows 
  58 void wxBeginBusyCursor(const wxCursor 
*cursor
) 
  60     if ( gs_wxBusyCursorCount
++ == 0 ) 
  62         gs_wxBusyCursor 
= (HCURSOR
)cursor
->GetHCURSOR(); 
  63 #ifndef __WXMICROWIN__ 
  64         gs_wxBusyCursorOld 
= ::SetCursor(gs_wxBusyCursor
); 
  67     //else: nothing to do, already set 
  70 // Restore cursor to normal 
  71 void wxEndBusyCursor() 
  73     wxCHECK_RET( gs_wxBusyCursorCount 
> 0, 
  74                  wxT("no matching wxBeginBusyCursor() for wxEndBusyCursor()") ); 
  76     if ( --gs_wxBusyCursorCount 
== 0 ) 
  78 #ifndef __WXMICROWIN__ 
  79         ::SetCursor(gs_wxBusyCursorOld
); 
  81         gs_wxBusyCursorOld 
= 0; 
  85 // true if we're between the above two calls 
  88   return gs_wxBusyCursorCount 
> 0; 
  91 // Check whether this window wants to process messages, e.g. Stop button 
  92 // in long calculations. 
  93 bool wxCheckForInterrupt(wxWindow 
*wnd
) 
  95     wxCHECK( wnd
, false ); 
  98     while ( ::PeekMessage(&msg
, GetHwndOf(wnd
), 0, 0, PM_REMOVE
) ) 
 100         ::TranslateMessage(&msg
); 
 101         ::DispatchMessage(&msg
); 
 107 // MSW only: get user-defined resource from the .res file. 
 108 // Returns NULL or newly-allocated memory, so use delete[] to clean up. 
 110 #ifndef __WXMICROWIN__ 
 111 char *wxLoadUserResource(const wxString
& resourceName
, const wxString
& resourceType
, int* pLen 
/* = NULL */) 
 113     HRSRC hResource 
= ::FindResource(wxGetInstance(), 
 114                                      resourceName
.wx_str(), 
 115                                      resourceType
.wx_str()); 
 116     if ( hResource 
== 0 ) 
 119     HGLOBAL hData 
= ::LoadResource(wxGetInstance(), hResource
); 
 123     void *theText 
= ::LockResource(hData
); 
 127     // Not all compilers put a zero at the end of the resource (e.g. BC++ doesn't). 
 128     // so we need to find the length of the resource. 
 129     int len 
= ::SizeofResource(wxGetInstance(), hResource
); 
 130     char *s 
= new char[len 
+ 1]; 
 131     memcpy(s
, theText
, len
); 
 132     s
[len
] = '\0'; // NUL-terminate in case the resource itself wasn't 
 136     UnlockResource(hData
); 
 140     //  GlobalFree(hData); 
 147 #endif // __WXMICROWIN__ 
 149 // ---------------------------------------------------------------------------- 
 151 // ---------------------------------------------------------------------------- 
 153 // See also the wxGetMousePosition in window.cpp 
 154 // Deprecated: use wxPoint wxGetMousePosition() instead 
 155 void wxGetMousePosition( int* x
, int* y 
) 
 158     GetCursorPos( & pt 
); 
 163 // Return true if we have a colour display 
 164 bool wxColourDisplay() 
 166 #ifdef __WXMICROWIN__ 
 170     // this function is called from wxDC ctor so it is called a *lot* of times 
 171     // hence we optimize it a bit but doing the check only once 
 173     // this should be MT safe as only the GUI thread (holding the GUI mutex) 
 175     static int s_isColour 
= -1; 
 177     if ( s_isColour 
== -1 ) 
 180         int noCols 
= ::GetDeviceCaps(dc
, NUMCOLORS
); 
 182         s_isColour 
= (noCols 
== -1) || (noCols 
> 2); 
 185     return s_isColour 
!= 0; 
 189 // Returns depth of screen 
 193     return GetDeviceCaps(dc
, PLANES
) * GetDeviceCaps(dc
, BITSPIXEL
); 
 196 // Get size of display 
 197 void wxDisplaySize(int *width
, int *height
) 
 199 #ifdef __WXMICROWIN__ 
 201     HWND hWnd 
= GetDesktopWindow(); 
 202     ::GetWindowRect(hWnd
, & rect
); 
 205         *width 
= rect
.right 
- rect
.left
; 
 207         *height 
= rect
.bottom 
- rect
.top
; 
 208 #else // !__WXMICROWIN__ 
 212         *width 
= ::GetDeviceCaps(dc
, HORZRES
); 
 214         *height 
= ::GetDeviceCaps(dc
, VERTRES
); 
 215 #endif // __WXMICROWIN__/!__WXMICROWIN__ 
 218 void wxDisplaySizeMM(int *width
, int *height
) 
 220 #ifdef __WXMICROWIN__ 
 230         *width 
= ::GetDeviceCaps(dc
, HORZSIZE
); 
 232         *height 
= ::GetDeviceCaps(dc
, VERTSIZE
); 
 236 void wxClientDisplayRect(int *x
, int *y
, int *width
, int *height
) 
 238 #if defined(__WXMICROWIN__) 
 240     wxDisplaySize(width
, height
); 
 242     // Determine the desktop dimensions minus the taskbar and any other 
 243     // special decorations... 
 246     SystemParametersInfo(SPI_GETWORKAREA
, 0, &r
, 0); 
 249     if (width
)  *width 
= r
.right 
- r
.left
; 
 250     if (height
) *height 
= r
.bottom 
- r
.top
; 
 254 // --------------------------------------------------------------------------- 
 255 // window information functions 
 256 // --------------------------------------------------------------------------- 
 258 wxString WXDLLEXPORT 
wxGetWindowText(WXHWND hWnd
) 
 264         int len 
= GetWindowTextLength((HWND
)hWnd
) + 1; 
 265         ::GetWindowText((HWND
)hWnd
, wxStringBuffer(str
, len
), len
); 
 271 wxString WXDLLEXPORT 
wxGetWindowClass(WXHWND hWnd
) 
 276 #ifndef __WXMICROWIN__ 
 279         int len 
= 256; // some starting value 
 283             int count 
= ::GetClassName((HWND
)hWnd
, wxStringBuffer(str
, len
), len
); 
 287                 // the class name might have been truncated, retry with larger 
 297 #endif // !__WXMICROWIN__ 
 302 int WXDLLEXPORT 
wxGetWindowId(WXHWND hWnd
) 
 304     return ::GetWindowLong((HWND
)hWnd
, GWL_ID
); 
 307 // ---------------------------------------------------------------------------- 
 309 // ---------------------------------------------------------------------------- 
 311 void PixelToHIMETRIC(LONG 
*x
, LONG 
*y
, HDC hdcRef
) 
 313     int iWidthMM 
= GetDeviceCaps(hdcRef
, HORZSIZE
), 
 314         iHeightMM 
= GetDeviceCaps(hdcRef
, VERTSIZE
), 
 315         iWidthPels 
= GetDeviceCaps(hdcRef
, HORZRES
), 
 316         iHeightPels 
= GetDeviceCaps(hdcRef
, VERTRES
); 
 318     *x 
*= (iWidthMM 
* 100); 
 320     *y 
*= (iHeightMM 
* 100); 
 324 void HIMETRICToPixel(LONG 
*x
, LONG 
*y
, HDC hdcRef
) 
 326     int iWidthMM 
= GetDeviceCaps(hdcRef
, HORZSIZE
), 
 327         iHeightMM 
= GetDeviceCaps(hdcRef
, VERTSIZE
), 
 328         iWidthPels 
= GetDeviceCaps(hdcRef
, HORZRES
), 
 329         iHeightPels 
= GetDeviceCaps(hdcRef
, VERTRES
); 
 332     *x 
/= (iWidthMM 
* 100); 
 334     *y 
/= (iHeightMM 
* 100); 
 337 void HIMETRICToPixel(LONG 
*x
, LONG 
*y
) 
 339     HIMETRICToPixel(x
, y
, ScreenHDC()); 
 342 void PixelToHIMETRIC(LONG 
*x
, LONG 
*y
) 
 344     PixelToHIMETRIC(x
, y
, ScreenHDC()); 
 347 void wxDrawLine(HDC hdc
, int x1
, int y1
, int x2
, int y2
) 
 355     Polyline(hdc
, points
, 2); 
 357     MoveToEx(hdc
, x1
, y1
, NULL
); LineTo((HDC
) hdc
, x2
, y2
); 
 362 // ---------------------------------------------------------------------------- 
 363 // Shell API wrappers 
 364 // ---------------------------------------------------------------------------- 
 366 extern bool wxEnableFileNameAutoComplete(HWND hwnd
) 
 368 #if wxUSE_DYNLIB_CLASS 
 369     typedef HRESULT (WINAPI 
*SHAutoComplete_t
)(HWND
, DWORD
); 
 371     static SHAutoComplete_t s_pfnSHAutoComplete 
= NULL
; 
 372     static bool s_initialized 
= false; 
 374     if ( !s_initialized 
) 
 376         s_initialized 
= true; 
 379         wxDynamicLibrary 
dll(wxT("shlwapi.dll")); 
 380         if ( dll
.IsLoaded() ) 
 382             s_pfnSHAutoComplete 
= 
 383                 (SHAutoComplete_t
)dll
.GetSymbol(wxT("SHAutoComplete")); 
 384             if ( s_pfnSHAutoComplete 
) 
 386                 // won't be unloaded until the process termination, no big deal 
 392     if ( !s_pfnSHAutoComplete 
) 
 395     HRESULT hr 
= s_pfnSHAutoComplete(hwnd
, 0x10 /* SHACF_FILESYS_ONLY */); 
 398         wxLogApiError(wxT("SHAutoComplete"), hr
); 
 406 #endif // wxUSE_DYNLIB_CLASS/!wxUSE_DYNLIB_CLASS 
 409 // ---------------------------------------------------------------------------- 
 410 // Launch document with default app 
 411 // ---------------------------------------------------------------------------- 
 413 bool wxLaunchDefaultApplication(const wxString
& document
, int flags
) 
 417     WinStruct
<SHELLEXECUTEINFO
> sei
; 
 418     sei
.lpFile 
= document
.wx_str(); 
 419     sei
.lpVerb 
= wxT("open"); 
 421     sei
.nShow 
= SW_SHOWNORMAL
; // SW_SHOWDEFAULT not defined under CE (#10216) 
 423     sei
.nShow 
= SW_SHOWDEFAULT
; 
 426     // avoid Windows message box in case of error for consistency with 
 427     // wxLaunchDefaultBrowser() even if don't show the error ourselves in this 
 429     sei
.fMask 
= SEE_MASK_FLAG_NO_UI
; 
 431     if ( ::ShellExecuteEx(&sei
) ) 
 437 // ---------------------------------------------------------------------------- 
 438 // Launch default browser 
 439 // ---------------------------------------------------------------------------- 
 441 bool wxDoLaunchDefaultBrowser(const wxString
& url
, const wxString
& scheme
, int flags
) 
 446     if ( flags 
& wxBROWSER_NEW_WINDOW 
) 
 448         // ShellExecuteEx() opens the URL in an existing window by default so 
 449         // we can't use it if we need a new window 
 450         wxRegKey 
key(wxRegKey::HKCR
, scheme 
+ wxT("\\shell\\open")); 
 453             // try the default browser, it must be registered at least for http URLs 
 454             key
.SetName(wxRegKey::HKCR
, wxT("http\\shell\\open")); 
 459             wxRegKey 
keyDDE(key
, wxT("DDEExec")); 
 460             if ( keyDDE
.Exists() ) 
 462                 // we only know the syntax of WWW_OpenURL DDE request for IE, 
 463                 // optimistically assume that all other browsers are compatible 
 465                 static const wxChar 
*TOPIC_OPEN_URL 
= wxT("WWW_OpenURL"); 
 467                 wxRegKey 
keyTopic(keyDDE
, wxT("topic")); 
 468                 bool ok 
= keyTopic
.Exists() && 
 469                             keyTopic
.QueryDefaultValue() == TOPIC_OPEN_URL
; 
 472                     ddeCmd 
= keyDDE
.QueryDefaultValue(); 
 473                     ok 
= !ddeCmd
.empty(); 
 478                     // for WWW_OpenURL, the index of the window to open the URL 
 479                     // in is -1 (meaning "current") by default, replace it with 
 480                     // 0 which means "new" (see KB article 160957) 
 481                     ok 
= ddeCmd
.Replace(wxT("-1"), wxT("0"), 
 482                                         false /* only first occurrence */) == 1; 
 487                     // and also replace the parameters: the topic should 
 488                     // contain a placeholder for the URL 
 489                     ok 
= ddeCmd
.Replace(wxT("%1"), url
, false) == 1; 
 494                     // try to send it the DDE request now but ignore the errors 
 497                     const wxString ddeServer 
= wxRegKey(keyDDE
, wxT("application")); 
 498                     if ( wxExecuteDDE(ddeServer
, TOPIC_OPEN_URL
, ddeCmd
) ) 
 501                     // this is not necessarily an error: maybe browser is 
 502                     // simply not running, but no matter, in any case we're 
 503                     // going to launch it using ShellExecuteEx() below now and 
 504                     // we shouldn't try to open a new window if we open a new 
 512     WinStruct
<SHELLEXECUTEINFO
> sei
; 
 513     sei
.lpFile 
= url
.c_str(); 
 514     sei
.lpVerb 
= wxT("open"); 
 515     sei
.nShow 
= SW_SHOWNORMAL
; 
 516     sei
.fMask 
= SEE_MASK_FLAG_NO_UI
; // we give error message ourselves 
 518     if ( ::ShellExecuteEx(&sei
) )