]> git.saurik.com Git - wxWidgets.git/blame - src/os2/window.cpp
GetMacXXXX to MacGetXXXX
[wxWidgets.git] / src / os2 / window.cpp
CommitLineData
0e320a79
DW
1/////////////////////////////////////////////////////////////////////////////
2// Name: windows.cpp
3// Purpose: wxWindow
cdf1e714 4// Author: David Webster
0e320a79 5// Modified by:
cdf1e714 6// Created: 10/12/99
0e320a79 7// RCS-ID: $Id$
cdf1e714
DW
8// Copyright: (c) David Webster
9// Licence: wxWindows licence
0e320a79
DW
10/////////////////////////////////////////////////////////////////////////////
11
a885d89a 12//
cdf1e714 13// For compilers that support precompilation, includes "wx.h".
a885d89a 14//
cdf1e714 15#include "wx/wxprec.h"
0e320a79 16
849949b1
DW
17#ifndef WX_PRECOMP
18 #define INCL_DOS
19 #define INCL_PM
20 #include <os2.h>
21 #include "wx/window.h"
22 #include "wx/accel.h"
23 #include "wx/setup.h"
24 #include "wx/menu.h"
25 #include "wx/dc.h"
26 #include "wx/dcclient.h"
27 #include "wx/utils.h"
28 #include "wx/app.h"
29 #include "wx/panel.h"
30 #include "wx/layout.h"
31 #include "wx/dialog.h"
32 #include "wx/frame.h"
33 #include "wx/listbox.h"
34 #include "wx/button.h"
35 #include "wx/msgdlg.h"
19193a2c 36 #include "wx/scrolwin.h"
849949b1
DW
37
38 #include <stdio.h>
39#endif
40
41#if wxUSE_OWNER_DRAWN
42 #include "wx/ownerdrw.h"
43#endif
44
45#if wxUSE_DRAG_AND_DROP
46 #include "wx/dnd.h"
47#endif
0e320a79
DW
48
49#include "wx/menuitem.h"
50#include "wx/log.h"
51
cdf1e714
DW
52#include "wx/os2/private.h"
53
849949b1
DW
54#if wxUSE_TOOLTIPS
55 #include "wx/tooltip.h"
0e320a79
DW
56#endif
57
849949b1
DW
58#if wxUSE_CARET
59 #include "wx/caret.h"
60#endif // wxUSE_CARET
61
62#include "wx/intl.h"
63#include "wx/log.h"
64
65
66#include "wx/textctrl.h"
67
0e320a79
DW
68#include <string.h>
69
a885d89a
DW
70//
71// Place compiler, OS specific includes here
72//
849949b1 73
a885d89a
DW
74//
75// Standard macros -- these are for OS/2 PM, but most GUI's have something similar
76//
849949b1 77#ifndef GET_X_LPARAM
a885d89a 78//
849949b1 79// SHORT1FROMMP -- LOWORD
a885d89a 80//
849949b1 81 #define GET_X_LPARAM(mp) ((unsigned short)(unsigned long)(mp))
a885d89a 82//
849949b1 83// SHORT2FROMMP -- HIWORD
a885d89a 84//
849949b1
DW
85 #define GET_Y_LPARAM(mp) ((unsigned short)(unsigned long)(mp >> 16))
86#endif // GET_X_LPARAM
87
29a99be3
DW
88#ifndef CW_USEDEFAULT
89# define CW_USEDEFAULT ((int)0x80000000)
90#endif
91
849949b1
DW
92// ---------------------------------------------------------------------------
93// global variables
94// ---------------------------------------------------------------------------
95
a885d89a 96//
19193a2c 97// The last PM message we got (MT-UNSAFE)
a885d89a 98//
61243a51 99QMSG s_currentMsg;
849949b1 100
19193a2c 101#if wxUSE_MENUS_NATIVE
a885d89a 102wxMenu* wxCurrentPopupMenu = NULL;
19193a2c
KB
103#endif // wxUSE_MENUS_NATIVE
104
8138a3b0
SN
105#if !defined(__VISAGECPP__) || (__IBMCPP__ < 400)
106extern wxChar wxCanvasClassName[];
61243a51 107#endif
a885d89a 108wxList* wxWinHandleList = NULL;
849949b1
DW
109
110// ---------------------------------------------------------------------------
111// private functions
112// ---------------------------------------------------------------------------
a885d89a
DW
113
114//
849949b1 115// the window proc for all our windows; most gui's have something similar
a885d89a 116//
f23208ca
DW
117MRESULT EXPENTRY wxWndProc( HWND hWnd
118 ,ULONG message
119 ,MPARAM mp1
120 ,MPARAM mp2
121 );
11e59d47
DW
122
123#ifdef __WXDEBUG__
124 const char *wxGetMessageName(int message);
125#endif //__WXDEBUG__
126
19193a2c
KB
127void wxRemoveHandleAssociation(wxWindowOS2* pWin);
128void wxAssociateWinWithHandle( HWND hWnd
129 ,wxWindowOS2* pWin
a885d89a
DW
130 );
131wxWindow* wxFindWinFromHandle(WXHWND hWnd);
849949b1 132
a885d89a
DW
133//
134// This magical function is used to translate VK_APPS key presses to right
135// mouse clicks
136//
19193a2c
KB
137// Unused?
138#if 0
a885d89a
DW
139static void TranslateKbdEventToMouse( wxWindow* pWin
140 ,int* pX
141 ,int* pY
142 ,MPARAM* pFlags
143 );
19193a2c 144#endif
a885d89a
DW
145//
146// get the current state of SHIFT/CTRL keys
147//
148static inline bool IsShiftDown() { return (::WinGetKeyState(HWND_DESKTOP, VK_SHIFT) & 0x8000) != 0; }
149static inline bool IsCtrlDown() { return (::WinGetKeyState(HWND_DESKTOP, VK_CTRL) & 0x8000) != 0; }
849949b1
DW
150// ---------------------------------------------------------------------------
151// event tables
152// ---------------------------------------------------------------------------
0e320a79 153
19193a2c
KB
154// in wxUniv-OS/2 this class is abstract because it doesn't have DoPopupMenu()
155// method
156#ifdef __WXUNIVERSAL__
157 IMPLEMENT_ABSTRACT_CLASS(wxWindowOS2, wxWindowBase)
158#else // __WXPM__
849949b1 159 IMPLEMENT_DYNAMIC_CLASS(wxWindow, wxWindowBase)
19193a2c 160#endif // __WXUNIVERSAL__/__WXPM__
0e320a79 161
19193a2c 162BEGIN_EVENT_TABLE(wxWindowOS2, wxWindowBase)
0367c1c0
DW
163 EVT_ERASE_BACKGROUND(wxWindowOS2::OnEraseBackground)
164 EVT_SYS_COLOUR_CHANGED(wxWindowOS2::OnSysColourChanged)
165 EVT_INIT_DIALOG(wxWindowOS2::OnInitDialog)
166 EVT_IDLE(wxWindowOS2::OnIdle)
167 EVT_SET_FOCUS(wxWindowOS2::OnSetFocus)
849949b1 168END_EVENT_TABLE()
0e320a79 169
849949b1
DW
170// ===========================================================================
171// implementation
172// ===========================================================================
0e320a79 173
19193a2c
KB
174// ---------------------------------------------------------------------------
175// wxWindow utility functions
176// ---------------------------------------------------------------------------
177
a885d89a 178//
776d87d5 179// Find an item given the PM Window id
a885d89a 180//
0367c1c0 181wxWindow* wxWindowOS2::FindItem(
a885d89a 182 long lId
776d87d5 183) const
cdf1e714 184{
19193a2c
KB
185#if wxUSE_CONTROLS
186 wxControl* pItem = wxDynamicCast(this, wxControl);
a885d89a
DW
187
188 if (pItem)
189 {
190 //
191 // I it we or one of our "internal" children?
192 //
19193a2c
KB
193 if (pItem->GetId() == lId
194#ifndef __WXUNIVERSAL__
195 || (pItem->GetSubcontrols().Index(lId) != wxNOT_FOUND)
196#endif
197 )
a885d89a
DW
198 {
199 return pItem;
200 }
201 }
19193a2c 202#endif // wxUSE_CONTROLS
a885d89a 203
776d87d5
DW
204 wxWindowList::Node* pCurrent = GetChildren().GetFirst();
205
206 while (pCurrent)
cdf1e714 207 {
776d87d5 208 wxWindow* pChildWin = pCurrent->GetData();
a885d89a 209 wxWindow* pWnd = pChildWin->FindItem(lId);
cdf1e714 210
776d87d5
DW
211 if (pWnd)
212 return pWnd;
cdf1e714 213
776d87d5 214 pCurrent = pCurrent->GetNext();
cdf1e714 215 }
776d87d5 216 return(NULL);
0367c1c0 217} // end of wxWindowOS2::FindItem
cdf1e714 218
a885d89a 219//
776d87d5 220// Find an item given the PM Window handle
a885d89a 221//
0367c1c0 222wxWindow* wxWindowOS2::FindItemByHWND(
776d87d5
DW
223 WXHWND hWnd
224, bool bControlOnly
225) const
cdf1e714 226{
776d87d5
DW
227 wxWindowList::Node* pCurrent = GetChildren().GetFirst();
228
229 while (pCurrent)
cdf1e714 230 {
776d87d5 231 wxWindow* pParent = pCurrent->GetData();
cdf1e714 232
a885d89a 233 //
cdf1e714 234 // Do a recursive search.
a885d89a 235 //
776d87d5
DW
236 wxWindow* pWnd = pParent->FindItemByHWND(hWnd);
237
238 if (pWnd)
239 return(pWnd);
cdf1e714 240
19193a2c
KB
241 if (!bControlOnly
242#if wxUSE_CONTROLS
243 || pParent->IsKindOf(CLASSINFO(wxControl))
244#endif // wxUSE_CONTROLS
245 )
cdf1e714 246 {
776d87d5
DW
247 wxWindow* pItem = pCurrent->GetData();
248
249 if (pItem->GetHWND() == hWnd)
250 return(pItem);
cdf1e714
DW
251 else
252 {
776d87d5
DW
253 if (pItem->ContainsHWND(hWnd))
254 return(pItem);
cdf1e714
DW
255 }
256 }
776d87d5 257 pCurrent = pCurrent->GetNext();
cdf1e714 258 }
776d87d5 259 return(NULL);
0367c1c0 260} // end of wxWindowOS2::FindItemByHWND
cdf1e714 261
a885d89a 262//
cdf1e714 263// Default command handler
a885d89a 264//
0367c1c0 265bool wxWindowOS2::OS2Command(
776d87d5
DW
266 WXUINT WXUNUSED(uParam)
267, WXWORD WXUNUSED(uId)
268)
cdf1e714 269{
776d87d5 270 return(FALSE);
cdf1e714
DW
271}
272
273// ----------------------------------------------------------------------------
274// constructors and such
275// ----------------------------------------------------------------------------
276
0367c1c0 277void wxWindowOS2::Init()
0e320a79 278{
a885d89a
DW
279 //
280 // Generic
281 //
849949b1 282 InitBase();
0e320a79 283
a885d89a 284 //
849949b1 285 // PM specific
a885d89a 286 //
776d87d5
DW
287 m_bDoubleClickAllowed = 0;
288 m_bWinCaptured = FALSE;
cdf1e714 289
849949b1 290 m_isBeingDeleted = FALSE;
776d87d5
DW
291 m_fnOldWndProc = 0;
292 m_bUseCtl3D = FALSE;
293 m_bMouseInWindow = FALSE;
0e320a79 294
a885d89a 295 //
849949b1 296 // wxWnd
a885d89a 297 //
c8b5f745
DW
298 m_hMenu = 0L;
299 m_hWnd = 0L;
300 m_hWndScrollBarHorz = 0L;
301 m_hWndScrollBarVert = 0L;
0e320a79 302
a885d89a
DW
303 //
304 // Pass WM_GETDLGCODE to DefWindowProc()
849949b1 305 m_lDlgCode = 0;
0e320a79 306
776d87d5
DW
307 m_nXThumbSize = 0;
308 m_nYThumbSize = 0;
309 m_bBackgroundTransparent = FALSE;
0e320a79 310
a885d89a
DW
311 //
312 // As all windows are created with WS_VISIBLE style...
313 //
849949b1 314 m_isShown = TRUE;
0e320a79 315
cdf1e714 316#if wxUSE_MOUSEEVENT_HACK
776d87d5
DW
317 m_lLastMouseX =
318 m_lLastMouseY = -1;
319 m_nLastMouseEvent = -1;
cdf1e714 320#endif // wxUSE_MOUSEEVENT_HACK
0367c1c0 321} // wxWindowOS2::Init
0e320a79 322
a885d89a 323//
849949b1 324// Destructor
a885d89a 325//
0367c1c0 326wxWindowOS2::~wxWindowOS2()
0e320a79 327{
849949b1 328 m_isBeingDeleted = TRUE;
0e320a79 329
cdf1e714 330 OS2DetachWindowMenu();
1ec46a5b
DW
331 for (wxWindow* pWin = GetParent(); pWin; pWin = pWin->GetParent())
332 {
333 wxFrame* pFrame = wxDynamicCast(pWin, wxFrame);
334
335 if (pFrame)
336 {
337 if (pFrame->GetLastFocus() == this)
338 pFrame->SetLastFocus((wxWindow*)NULL);
1ec46a5b
DW
339 }
340 }
19193a2c
KB
341
342 DestroyChildren();
343
849949b1
DW
344 if (m_parent)
345 m_parent->RemoveChild(this);
e604d44b 346
849949b1
DW
347 if (m_hWnd)
348 {
a885d89a 349 if(!::WinDestroyWindow(GetHWND()))
223d09f6 350 wxLogLastError(wxT("DestroyWindow"));
a885d89a 351 //
849949b1 352 // remove hWnd <-> wxWindow association
a885d89a 353 //
849949b1
DW
354 wxRemoveHandleAssociation(this);
355 }
0367c1c0 356} // end of wxWindowOS2::~wxWindowOS2
0e320a79 357
19193a2c 358// real construction (Init() must have been called before!)
0367c1c0 359bool wxWindowOS2::Create(
776d87d5
DW
360 wxWindow* pParent
361, wxWindowID vId
362, const wxPoint& rPos
363, const wxSize& rSize
364, long lStyle
365, const wxString& rName
366)
367{
f23208ca 368 HWND hParent = NULLHANDLE;
45bfc84b 369 wxPoint vPos = rPos; // The OS/2 position
19193a2c
KB
370 ULONG ulCreateFlags = 0;
371 WXDWORD dwExStyle = 0;
f23208ca 372
776d87d5
DW
373 wxCHECK_MSG(pParent, FALSE, wxT("can't create wxWindow without parent"));
374
375 if ( !CreateBase( pParent
376 ,vId
377 ,rPos
378 ,rSize
379 ,lStyle
380 ,wxDefaultValidator
381 ,rName
382 ))
383 return(FALSE);
384
f23208ca
DW
385 if (pParent)
386 {
b7d8f285
DW
387 int nTempy;
388
f23208ca
DW
389 pParent->AddChild(this);
390 hParent = GetWinHwnd(pParent);
cb71578c 391
cb71578c
DW
392 if ( pParent->IsKindOf(CLASSINFO(wxGenericScrolledWindow)) ||
393 pParent->IsKindOf(CLASSINFO(wxScrolledWindow))
394 )
395 ulCreateFlags |= WS_CLIPSIBLINGS;
f23208ca 396 }
776d87d5 397
a885d89a 398 //
f23208ca
DW
399 // Most wxSTYLES are really PM Class specific styles and will be
400 // set in those class create procs. PM's basic windows styles are
401 // very limited.
a885d89a 402 //
51c1d535
DW
403 ulCreateFlags |= WS_VISIBLE;
404
405
19193a2c
KB
406#ifdef __WXUNIVERSAL__
407 // no 3d effects, we draw them ourselves
408 WXDWORD exStyle = 0;
409#else // !wxUniversal
cb71578c 410 if (lStyle & wxCLIP_SIBLINGS)
51c1d535
DW
411 ulCreateFlags |= WS_CLIPSIBLINGS;
412
a885d89a 413 if (lStyle & wxCLIP_CHILDREN )
f23208ca 414 ulCreateFlags |= WS_CLIPCHILDREN;
a885d89a 415
f23208ca 416 //
008089f6 417 //
f23208ca 418 //
776d87d5 419 bool bWant3D;
19193a2c
KB
420 dwExStyle = Determine3DEffects(WS_EX_CLIENTEDGE, &bWant3D);
421
422#endif
008089f6
DW
423
424 //
425 // Add the simple border style as we'll use this to draw borders
426 //
427 if (lStyle & wxSIMPLE_BORDER)
428 dwExStyle |= wxSIMPLE_BORDER;
776d87d5 429
19193a2c
KB
430 if (lStyle & wxPOPUP_WINDOW)
431 {
432 // a popup window floats on top of everything
433//TODO: fix this...
434// exStyle |= WS_EX_TOPMOST | WS_EX_TOOLWINDOW;
435
436 // it is also created hidden as other top level windows
437 ulCreateFlags &= ~WS_VISIBLE;
438 m_isShown = FALSE;
439 }
440
a885d89a 441 //
f23208ca
DW
442 // Generic OS/2 Windows are created with no owner, no Z Order, no Control data,
443 // and no presentation parameters
a885d89a 444 //
f23208ca
DW
445 OS2Create( hParent
446 ,(PSZ)wxCanvasClassName
447 ,rName.c_str()
448 ,ulCreateFlags
45bfc84b
DW
449 ,vPos.x
450 ,vPos.y
776d87d5
DW
451 ,WidthDefault(rSize.x)
452 ,HeightDefault(rSize.y)
f23208ca
DW
453 ,NULLHANDLE
454 ,NULLHANDLE
455 ,m_windowId
008089f6
DW
456 ,NULL
457 ,NULL
458 ,dwExStyle
776d87d5 459 );
e604d44b 460
776d87d5 461 return(TRUE);
0367c1c0 462} // end of wxWindowOS2::Create
0e320a79 463
cdf1e714
DW
464// ---------------------------------------------------------------------------
465// basic operations
466// ---------------------------------------------------------------------------
0e320a79 467
0367c1c0 468void wxWindowOS2::SetFocus()
0e320a79 469{
bbdc9a87 470 HWND hWnd = GetHwnd();
19193a2c 471 wxCHECK_RET( hWnd, _T("can't set focus to invalid window") );
bbdc9a87
DW
472
473 if (hWnd)
474 ::WinSetFocus(HWND_DESKTOP, hWnd);
0367c1c0 475} // end of wxWindowOS2::SetFocus
0e320a79 476
11e59d47 477wxWindow* wxWindowBase::FindFocus()
0e320a79 478{
bbdc9a87
DW
479 HWND hWnd = ::WinQueryFocus(HWND_DESKTOP);
480
481 if (hWnd)
482 {
483 return wxFindWinFromHandle((WXHWND)hWnd);
484 }
485 return NULL;
a885d89a 486} // wxWindowBase::FindFocus
0e320a79 487
0367c1c0 488bool wxWindowOS2::Enable(
bbdc9a87
DW
489 bool bEnable
490)
0e320a79 491{
bbdc9a87
DW
492 if (!wxWindowBase::Enable(bEnable))
493 return(FALSE);
494
495 HWND hWnd = GetHwnd();
496
497 if ( hWnd )
498 ::WinEnableWindow(hWnd, (BOOL)bEnable);
499
500 wxWindowList::Node* pNode = GetChildren().GetFirst();
501
502 while (pNode)
503 {
504 wxWindow* pChild = pNode->GetData();
505
506 pChild->Enable(bEnable);
507 pNode = pNode->GetNext();
508 }
19193a2c 509 return TRUE;
0367c1c0 510} // end of wxWindowOS2::Enable
0e320a79 511
0367c1c0 512bool wxWindowOS2::Show(
bbdc9a87
DW
513 bool bShow
514)
0e320a79 515{
bbdc9a87
DW
516 if (!wxWindowBase::Show(bShow))
517 return(FALSE);
518
519 HWND hWnd = GetHwnd();
520
521 ::WinShowWindow(hWnd, bShow);
522
523 if (bShow)
524 {
a885d89a 525 ::WinSetWindowPos(hWnd, HWND_TOP, 0, 0, 0, 0, SWP_ACTIVATE | SWP_ZORDER);
bbdc9a87 526 }
19193a2c 527 return TRUE;
0367c1c0 528} // end of wxWindowOS2::Show
0e320a79 529
0367c1c0 530void wxWindowOS2::Raise()
0e320a79 531{
a885d89a 532 ::WinSetWindowPos(GetHwnd(), HWND_TOP, 0, 0, 0, 0, SWP_ZORDER | SWP_ACTIVATE);
0367c1c0 533} // end of wxWindowOS2::Raise
0e320a79 534
0367c1c0 535void wxWindowOS2::Lower()
0e320a79 536{
a885d89a 537 ::WinSetWindowPos(GetHwnd(), HWND_BOTTOM, 0, 0, 0, 0, SWP_ZORDER | SWP_DEACTIVATE);
0367c1c0 538} // end of wxWindowOS2::Lower
0e320a79 539
0367c1c0 540void wxWindowOS2::SetTitle(
bbdc9a87
DW
541 const wxString& rTitle
542)
0e320a79 543{
f38374d0 544 ::WinSetWindowText(GetHwnd(), rTitle.c_str());
0367c1c0 545} // end of wxWindowOS2::SetTitle
0e320a79 546
0367c1c0 547wxString wxWindowOS2::GetTitle() const
0e320a79 548{
cdf1e714 549 return wxGetWindowText(GetHWND());
0367c1c0 550} // end of wxWindowOS2::GetTitle
0e320a79 551
0367c1c0 552void wxWindowOS2::CaptureMouse()
0e320a79 553{
f38374d0
DW
554 HWND hWnd = GetHwnd();
555
556 if (hWnd && !m_bWinCaptured)
557 {
558 ::WinSetCapture(HWND_DESKTOP, hWnd);
559 m_bWinCaptured = TRUE;
560 }
0367c1c0 561} // end of wxWindowOS2::GetTitle
0e320a79 562
0367c1c0 563void wxWindowOS2::ReleaseMouse()
0e320a79 564{
a885d89a 565 if (m_bWinCaptured)
f38374d0
DW
566 {
567 ::WinSetCapture(HWND_DESKTOP, NULLHANDLE);
568 m_bWinCaptured = FALSE;
569 }
0367c1c0 570} // end of wxWindowOS2::ReleaseMouse
0e320a79 571
0367c1c0
DW
572/* static */ wxWindow* wxWindowBase::GetCapture()
573{
574 HWND hwnd = ::WinQueryCapture(HWND_DESKTOP);
575 return hwnd ? wxFindWinFromHandle((WXHWND)hwnd) : (wxWindow *)NULL;
576} // end of wxWindowBase::GetCapture
577
578bool wxWindowOS2::SetFont(
f38374d0
DW
579 const wxFont& rFont
580)
0e320a79 581{
f38374d0
DW
582 if (!wxWindowBase::SetFont(rFont))
583 {
584 // nothing to do
585 return(FALSE);
586 }
587
588 HWND hWnd = GetHwnd();
589
590 if (hWnd != 0)
591 {
987da0d4
DW
592 char zFont[128];
593 char zFacename[30];
594 char zWeight[30];
595 char zStyle[30];
596
597 //
598 // The fonts available for Presentation Params are just three
599 // outline fonts, the rest are available to the GPI, so we must
600 // map the families to one of these three
601 //
602 switch(rFont.GetFamily())
603 {
604 case wxSCRIPT:
605 case wxDECORATIVE:
606 case wxROMAN:
607 strcpy(zFacename,"Times New Roman");
608 break;
f38374d0 609
987da0d4
DW
610 case wxTELETYPE:
611 case wxMODERN:
612 strcpy(zFacename, "Courier");
613 break;
614
615 case wxSWISS:
616 case wxDEFAULT:
617 default:
618 strcpy(zFacename, "Helvetica");
619 break;
620 }
621
622 switch(rFont.GetWeight())
623 {
624 default:
625 case wxNORMAL:
626 case wxLIGHT:
627 zWeight[0] = '\0';
628 break;
629
630 case wxBOLD:
631 case wxFONTWEIGHT_MAX:
632 strcpy(zWeight, "Bold");
633 break;
634 }
635 switch(rFont.GetStyle())
636 {
637 case wxITALIC:
638 case wxSLANT:
639 strcpy(zStyle, "Italic");
640 break;
641
642 default:
643 zStyle[0] = '\0';
644 break;
645 }
646 sprintf(zFont, "%d.%s", rFont.GetPointSize(), zFacename);
647 if (zWeight[0] != '\0')
648 {
649 strcat(zFont, " ");
650 strcat(zFont, zWeight);
651 }
652 if (zStyle[0] != '\0')
653 {
654 strcat(zFont, " ");
655 strcat(zFont, zStyle);
656 }
657 ::WinSetPresParam(hWnd, PP_FONTNAMESIZE, strlen(zFont) + 1, (PVOID)zFont);
f38374d0 658 }
cdf1e714 659 return(TRUE);
0e320a79
DW
660}
661
0367c1c0 662bool wxWindowOS2::SetCursor(
f38374d0
DW
663 const wxCursor& rCursor
664) // check if base implementation is OK
0e320a79 665{
f38374d0
DW
666 if ( !wxWindowBase::SetCursor(rCursor))
667 {
668 // no change
669 return FALSE;
670 }
671
19193a2c
KB
672 if ( m_cursor.Ok() ) {
673 HWND hWnd = GetHwnd();
674 POINTL vPoint;
675 RECTL vRect;
f38374d0 676
19193a2c
KB
677 ::WinQueryPointerPos(HWND_DESKTOP, &vPoint);
678 ::WinQueryWindowRect(hWnd, &vRect);
f38374d0 679
19193a2c
KB
680 if (::WinPtInRect(vHabmain, &vRect, &vPoint) && !wxIsBusy())
681 {
682 ::WinSetPointer(HWND_DESKTOP, (HPOINTER)m_cursor.GetHCURSOR());
683 }
f38374d0
DW
684 }
685 return TRUE;
0367c1c0 686} // end of wxWindowOS2::SetCursor
0e320a79 687
0367c1c0 688void wxWindowOS2::WarpPointer(
a885d89a
DW
689 int nXPos
690, int nYPos
691)
0e320a79 692{
a885d89a
DW
693 int nX = nXPos;
694 int nY = nYPos;
695 RECTL vRect;
696
697 ::WinQueryWindowRect(GetHwnd(), &vRect);
698 nX += vRect.xLeft;
699 nY += vRect.yBottom;
700
701 ::WinSetPointerPos(HWND_DESKTOP, (LONG)nX, (LONG)(nY));
0367c1c0 702} // end of wxWindowOS2::WarpPointer
0e320a79 703
cdf1e714 704#if WXWIN_COMPATIBILITY
0367c1c0 705void wxWindowOS2::OS2DeviceToLogical (float *x, float *y) const
0e320a79 706{
0e320a79 707}
cdf1e714 708#endif // WXWIN_COMPATIBILITY
0e320a79 709
cdf1e714
DW
710// ---------------------------------------------------------------------------
711// scrolling stuff
712// ---------------------------------------------------------------------------
0e320a79 713
cdf1e714 714#if WXWIN_COMPATIBILITY
0367c1c0 715void wxWindowOS2::SetScrollRange(
a885d89a
DW
716 int nOrient
717, int nRange
718, bool bRefresh
719)
0e320a79 720{
b7d8f285
DW
721 int nRange1 = nRange;
722 int nPageSize = GetScrollPage(nOrient);
723
724 if (nPpageSize > 1 && nRange > 0)
725 {
726 nRange1 += (nPageSize - 1);
727 }
728
729 if (nOrient == wxHORIZONTAL)
730 {
731 ::WinSendMsg(m_hWndScrollBarHorz, SBM_SETSCROLLBAR, (MPARAM)0, MPFROM2SHORT(0, (SHORT)nRange1));
732 ::WinSendMsg(m_hWndScrollBarHorz, SBM_SETTHUMBSIZE, MPFROM2SHORT((SHORT)nThumbVisible, (SHORT)nRange1), (MPARAM)0);
733 }
734 else
735 {
736 ::WinSendMsg(m_hWndScrollBarVert, SBM_SETSCROLLBAR, (MPARAM)0, MPFROM2SHORT(0, (SHORT)nRange1));
737 ::WinSendMsg(m_hWndScrollBarVert, SBM_SETTHUMBSIZE, MPFROM2SHORT((SHORT)nThumbVisible, (SHORT)nRange1), (MPARAM)0);
738 }
0367c1c0 739} // end of wxWindowOS2::SetScrollRange
0e320a79 740
0367c1c0 741void wxWindowOS2::SetScrollPage(
a885d89a
DW
742 int nOrient
743, int nPage
744, bool bRefresh
745)
0e320a79 746{
b7d8f285
DW
747 if (nOrient == wxHORIZONTAL )
748 m_nXThumbSize = nPage;
a885d89a 749 else
b7d8f285 750 m_nYThumbSize = nPage;
0367c1c0 751} // end of wxWindowOS2::SetScrollPage
0e320a79 752
0367c1c0 753int wxWindowOS2::OldGetScrollRange(
a885d89a
DW
754 int nOrient
755) const
0e320a79 756{
a885d89a
DW
757 MRESULT mRc;
758 HWND hWnd = GetHwnd();
0e320a79 759
a885d89a
DW
760 if (hWnd)
761 {
762 mRc = WinSendMsg(hWnd, SBM_QUERYRANGE, (MPARAM)0L, (MPARAM)0L);
763 return(SHORT2FROMMR(mRc));
764 }
765 return 0;
0367c1c0 766} // end of wxWindowOS2::OldGetScrollRange
a885d89a 767
0367c1c0 768int wxWindowOS2::GetScrollPage(
a885d89a
DW
769 int nOrient
770) const
0e320a79 771{
a885d89a 772 if (nOrient == wxHORIZONTAL)
05fc6c0a 773 return m_nXThumbSize;
a885d89a 774 else
05fc6c0a 775 return m_nYThumbSize;
0367c1c0 776} // end of wxWindowOS2::GetScrollPage
11e59d47 777#endif // WXWIN_COMPATIBILITY
0e320a79 778
0367c1c0 779int wxWindowOS2::GetScrollPos(
05fc6c0a
DW
780 int nOrient
781) const
0e320a79 782{
b7d8f285
DW
783 if (nOrient == wxHORIZONTAL)
784 return((int)::WinSendMsg(m_hWndScrollBarHorz, SBM_QUERYPOS, (MPARAM)NULL, (MPARAM)NULL));
785 else
786 return((int)::WinSendMsg(m_hWndScrollBarVert, SBM_QUERYPOS, (MPARAM)NULL, (MPARAM)NULL));
0367c1c0 787} // end of wxWindowOS2::GetScrollPos
0e320a79 788
0367c1c0 789int wxWindowOS2::GetScrollRange(
05fc6c0a
DW
790 int nOrient
791) const
0e320a79 792{
05fc6c0a 793 MRESULT mr;
0e320a79 794
b7d8f285
DW
795 if (nOrient == wxHORIZONTAL)
796 mr = ::WinSendMsg(m_hWndScrollBarHorz, SBM_QUERYRANGE, (MPARAM)NULL, (MPARAM)NULL);
797 else
798 mr = ::WinSendMsg(m_hWndScrollBarVert, SBM_QUERYRANGE, (MPARAM)NULL, (MPARAM)NULL);
05fc6c0a 799 return((int)SHORT2FROMMR(mr));
0367c1c0 800} // end of wxWindowOS2::GetScrollRange
05fc6c0a 801
0367c1c0 802int wxWindowOS2::GetScrollThumb(
05fc6c0a
DW
803 int nOrient
804) const
0e320a79 805{
b7d8f285
DW
806 if (nOrient == wxHORIZONTAL )
807 return m_nXThumbSize;
808 else
809 return m_nYThumbSize;
0367c1c0 810} // end of wxWindowOS2::GetScrollThumb
0e320a79 811
0367c1c0 812void wxWindowOS2::SetScrollPos(
05fc6c0a
DW
813 int nOrient
814, int nPos
19193a2c 815, bool WXUNUSED(bRefresh)
05fc6c0a 816)
0e320a79 817{
b7d8f285
DW
818 if (nOrient == wxHORIZONTAL )
819 ::WinSendMsg(m_hWndScrollBarHorz, SBM_SETPOS, (MPARAM)nPos, (MPARAM)NULL);
820 else
821 ::WinSendMsg(m_hWndScrollBarVert, SBM_SETPOS, (MPARAM)nPos, (MPARAM)NULL);
19193a2c 822} // end of wxWindowOS2::SetScrollPos
0e320a79 823
0367c1c0 824void wxWindowOS2::SetScrollbar(
05fc6c0a
DW
825 int nOrient
826, int nPos
827, int nThumbVisible
828, int nRange
19193a2c 829, bool WXUNUSED(bRefresh)
05fc6c0a 830)
0e320a79 831{
1c84ee88
DW
832 int nOldRange = nRange - nThumbVisible;
833 int nRange1 = nOldRange;
834 int nPageSize = nThumbVisible;
835 SBCDATA vInfo;
836 HWND hWnd = GetHwnd();
e661fcd7 837 ULONG ulStyle = WS_VISIBLE | WS_SYNCPAINT;
1c84ee88
DW
838 RECTL vRect;
839
840 ::WinQueryWindowRect(hWnd, &vRect);
841 if (nPageSize > 1 && nRange > 0)
05fc6c0a 842 {
1c84ee88
DW
843 nRange1 += (nPageSize - 1);
844 }
845
846 vInfo.cb = sizeof(SBCDATA);
847 vInfo.posFirst = 0;
848 vInfo.posLast = (SHORT)nRange1;
849 vInfo.posThumb = nPos;
850
851 if (nOrient == wxHORIZONTAL )
852 {
853 ulStyle |= SBS_HORZ;
c8b5f745 854 if (m_hWndScrollBarHorz == 0L)
1c84ee88 855 {
b7d8f285
DW
856 //
857 // We create the scrollbars with the desktop so that they are not
858 // registered as child windows of the window in order that child
859 // windows may be scrolled without scrolling the scrollbars themselves!
860 //
e661fcd7 861 m_hWndScrollBarHorz = ::WinCreateWindow( hWnd
c8b5f745
DW
862 ,WC_SCROLLBAR
863 ,(PSZ)NULL
864 ,ulStyle
865 ,vRect.xLeft
866 ,vRect.yBottom
867 ,vRect.xRight - vRect.xLeft
868 ,20
869 ,hWnd
870 ,HWND_TOP
b7d8f285 871 ,FID_HORZSCROLL
c8b5f745
DW
872 ,&vInfo
873 ,NULL
874 );
1c84ee88
DW
875 }
876 else
877 {
c8b5f745
DW
878 RECTL vRect2;
879
880 //
881 // Only want to resize the scrollbar if it changes, otherwise
882 // we'd probably end up in a recursive loop until we crash the call stack
883 // because this method is called in a ScrolledWindow OnSize event and SWP_MOVE | SWP_SIZE
884 // generates those events.
885 //
886 ::WinQueryWindowRect(m_hWndScrollBarHorz, &vRect2);
887 if (!(vRect2.xLeft == vRect.xLeft &&
888 vRect2.xRight == vRect.xRight &&
889 vRect2.yBottom == vRect.yBottom &&
890 vRect2.yTop == vRect.yTop
891 ) )
892 {
893 ::WinSetWindowPos( m_hWndScrollBarHorz
894 ,HWND_TOP
895 ,vRect.xLeft
896 ,vRect.yBottom
897 ,vRect.xRight - vRect.xLeft
898 ,20
899 ,SWP_ACTIVATE | SWP_MOVE | SWP_SIZE | SWP_SHOW
900 );
901 }
902 ::WinSendMsg(m_hWndScrollBarHorz, SBM_SETSCROLLBAR, (MPARAM)nPos, MPFROM2SHORT(0, (SHORT)nRange1));
2be39ac9 903 ::WinSendMsg(m_hWndScrollBarHorz, SBM_SETTHUMBSIZE, MPFROM2SHORT((SHORT)nThumbVisible, (SHORT)nRange1), (MPARAM)0);
1c84ee88 904 }
05fc6c0a
DW
905 }
906 else
907 {
1c84ee88 908 ulStyle |= SBS_VERT;
c8b5f745 909 if (m_hWndScrollBarVert == 0L)
1c84ee88 910 {
e661fcd7 911 m_hWndScrollBarVert = ::WinCreateWindow( hWnd
c8b5f745
DW
912 ,WC_SCROLLBAR
913 ,(PSZ)NULL
914 ,ulStyle
915 ,vRect.xRight - 20
833615dd 916 ,vRect.yBottom + 20
c8b5f745 917 ,20
833615dd 918 ,vRect.yTop - (vRect.yBottom + 20)
c8b5f745
DW
919 ,hWnd
920 ,HWND_TOP
b7d8f285 921 ,FID_VERTSCROLL
c8b5f745
DW
922 ,&vInfo
923 ,NULL
924 );
1c84ee88
DW
925 }
926 else
927 {
c8b5f745
DW
928 RECTL vRect2;
929
930 //
931 // Only want to resize the scrollbar if it changes, otherwise
932 // we'd probably end up in a recursive loop until we crash the call stack
933 // because this method is called in a ScrolledWindow OnSize event and SWP_MOVE | SWP_SIZE
934 // generates those events.
935 //
936 ::WinQueryWindowRect(m_hWndScrollBarVert, &vRect2);
937 if (!(vRect2.xLeft == vRect.xLeft &&
938 vRect2.xRight == vRect.xRight &&
939 vRect2.yBottom == vRect.yBottom &&
940 vRect2.yTop == vRect.yTop
941 ) )
942 {
943 ::WinSetWindowPos( m_hWndScrollBarVert
944 ,HWND_TOP
945 ,vRect.xRight - 20
833615dd 946 ,vRect.yBottom + 20
c8b5f745 947 ,20
833615dd 948 ,vRect.yTop - (vRect.yBottom + 20)
c8b5f745
DW
949 ,SWP_ACTIVATE | SWP_MOVE | SWP_SIZE | SWP_SHOW
950 );
951 }
952 ::WinSendMsg(m_hWndScrollBarVert, SBM_SETSCROLLBAR, (MPARAM)nPos, MPFROM2SHORT(0, (SHORT)nRange1));
2be39ac9 953 ::WinSendMsg(m_hWndScrollBarVert, SBM_SETTHUMBSIZE, MPFROM2SHORT((SHORT)nThumbVisible, (SHORT)nRange1), (MPARAM)0);
1c84ee88 954 }
05fc6c0a
DW
955 m_nYThumbSize = nThumbVisible;
956 }
0367c1c0 957} // end of wxWindowOS2::SetScrollbar
0e320a79 958
0367c1c0 959void wxWindowOS2::ScrollWindow(
05fc6c0a
DW
960 int nDx
961, int nDy
962, const wxRect* pRect
963)
0e320a79 964{
b7d8f285 965 RECTL vRect;
05fc6c0a
DW
966 RECTL vRect2;
967
b7d8f285 968 nDy *= -1; // flip the sign of Dy as OS/2 is opposite wxWin.
05fc6c0a
DW
969 if (pRect)
970 {
971 vRect2.xLeft = pRect->x;
b7d8f285 972 vRect2.yTop = pRect->y + pRect->height;
05fc6c0a 973 vRect2.xRight = pRect->x + pRect->width;
b7d8f285 974 vRect2.yBottom = pRect->y;
05fc6c0a 975 }
e661fcd7
DW
976 else
977 {
978 ::WinQueryWindowRect(GetHwnd(), &vRect2);
979 ::WinQueryWindowRect(m_hWndScrollBarHorz, &vRect);
980 vRect2.yBottom += vRect.yTop - vRect.yBottom;
981 ::WinQueryWindowRect(m_hWndScrollBarVert, &vRect);
982 vRect2.xRight -= vRect.xRight - vRect.xLeft;
05fc6c0a 983
e661fcd7 984 }
05fc6c0a 985 if (pRect)
b7d8f285
DW
986 ::WinScrollWindow( GetHwnd()
987 ,(LONG)nDx
988 ,(LONG)nDy
989 ,&vRect2
990 ,NULL
991 ,NULLHANDLE
992 ,NULL
e661fcd7 993 ,SW_INVALIDATERGN
b7d8f285 994 );
05fc6c0a 995 else
b7d8f285
DW
996 ::WinScrollWindow( GetHwnd()
997 ,nDx
998 ,nDy
999 ,NULL
1000 ,NULL
1001 ,NULLHANDLE
1002 ,NULL
e661fcd7
DW
1003 ,SW_INVALIDATERGN
1004 );
1005
1006 //
1007 // Move the children
1008 wxWindowList::Node* pCurrent = GetChildren().GetFirst();
1009 SWP vSwp;
1010
1011 while (pCurrent)
1012 {
1013 wxWindow* pChildWin = pCurrent->GetData();
1014
cb71578c
DW
1015 if (pChildWin->GetHWND() != NULLHANDLE)
1016 {
1017 ::WinQueryWindowPos(pChildWin->GetHWND(), &vSwp);
1018 ::WinQueryWindowRect(pChildWin->GetHWND(), &vRect);
1019 if (pChildWin->GetHWND() == m_hWndScrollBarVert ||
1020 pChildWin->GetHWND() == m_hWndScrollBarHorz)
1021 {
1022 ::WinSetWindowPos( pChildWin->GetHWND()
1023 ,HWND_TOP
1024 ,vSwp.x + nDx
1025 ,vSwp.y + nDy
1026 ,0
1027 ,0
1028 ,SWP_MOVE | SWP_SHOW | SWP_ZORDER
1029 );
1030 }
1031 else
1032 {
1033 ::WinSetWindowPos( pChildWin->GetHWND()
1034 ,HWND_BOTTOM
1035 ,vSwp.x + nDx
1036 ,vSwp.y + nDy
1037 ,0
1038 ,0
1039 ,SWP_MOVE | SWP_ZORDER
1040 );
1041 ::WinInvalidateRect(pChildWin->GetHWND(), &vRect, FALSE);
1042 }
1043 }
e661fcd7
DW
1044 pCurrent = pCurrent->GetNext();
1045 }
0367c1c0 1046} // end of wxWindowOS2::ScrollWindow
0e320a79 1047
cdf1e714
DW
1048// ---------------------------------------------------------------------------
1049// subclassing
1050// ---------------------------------------------------------------------------
0e320a79 1051
0367c1c0 1052void wxWindowOS2::SubclassWin(
c86c44a0
DW
1053 WXHWND hWnd
1054)
0e320a79 1055{
05fc6c0a
DW
1056 HWND hwnd = (HWND)hWnd;
1057
776d87d5 1058 wxASSERT_MSG( !m_fnOldWndProc, wxT("subclassing window twice?") );
61243a51 1059 wxCHECK_RET(::WinIsWindow(vHabmain, hwnd), wxT("invalid HWND in SubclassWin") );
542875a8 1060 m_fnOldWndProc = (WXFARPROC) ::WinSubclassWindow(hwnd, (PFNWP)wxWndProc);
0367c1c0 1061} // end of wxWindowOS2::SubclassWin
0e320a79 1062
0367c1c0 1063void wxWindowOS2::UnsubclassWin()
0e320a79 1064{
c86c44a0 1065 //
cdf1e714 1066 // Restore old Window proc
c86c44a0 1067 //
e604d44b 1068 HWND hwnd = GetHWND();
c86c44a0 1069
e604d44b 1070 if (m_hWnd)
cdf1e714 1071 {
61243a51 1072 wxCHECK_RET( ::WinIsWindow(vHabmain, hwnd), wxT("invalid HWND in UnsubclassWin") );
0e320a79 1073
e604d44b 1074 PFNWP fnProc = (PFNWP)::WinQueryWindowPtr(hwnd, QWP_PFNWP);
51c1d535 1075
c86c44a0 1076 if ( (m_fnOldWndProc != 0) && (fnProc != (PFNWP) m_fnOldWndProc))
cdf1e714 1077 {
c86c44a0
DW
1078 WinSubclassWindow(hwnd, (PFNWP)m_fnOldWndProc);
1079 m_fnOldWndProc = 0;
cdf1e714
DW
1080 }
1081 }
0367c1c0 1082} // end of wxWindowOS2::UnsubclassWin
0e320a79 1083
05fc6c0a 1084//
cdf1e714 1085// Make a Windows extended style from the given wxWindows window style
05fc6c0a 1086//
0367c1c0 1087WXDWORD wxWindowOS2::MakeExtendedStyle(
05fc6c0a
DW
1088 long lStyle
1089, bool bEliminateBorders
1090)
0e320a79 1091{
05fc6c0a 1092 //
008089f6
DW
1093 // Simply fill out with wxWindow extended styles. We'll conjure
1094 // something up in OS2Create and all window redrawing pieces later
05fc6c0a 1095 //
008089f6
DW
1096 WXDWORD dwStyle = 0;
1097
1098 if (lStyle & wxTRANSPARENT_WINDOW )
1099 dwStyle |= wxTRANSPARENT_WINDOW;
1100
1101 if (!bEliminateBorders)
1102 {
1103 if (lStyle & wxSUNKEN_BORDER)
1104 dwStyle |= wxSUNKEN_BORDER;
1105 if (lStyle & wxDOUBLE_BORDER)
1106 dwStyle |= wxDOUBLE_BORDER;
1107 if (lStyle & wxRAISED_BORDER )
1108 dwStyle |= wxRAISED_BORDER;
1109 if (lStyle & wxSTATIC_BORDER)
1110 dwStyle |= wxSTATIC_BORDER;
1111 }
1112 return dwStyle;
0367c1c0 1113} // end of wxWindowOS2::MakeExtendedStyle
cdf1e714 1114
c86c44a0 1115//
008089f6 1116// Determines whether simulated 3D effects or CTL3D should be used,
cdf1e714 1117// applying a default border style if required, and returning an extended
008089f6 1118// style to pass to OS2Create.
c86c44a0 1119//
0367c1c0 1120WXDWORD wxWindowOS2::Determine3DEffects(
c86c44a0 1121 WXDWORD dwDefaultBorderStyle
542875a8 1122, bool* pbWant3D
c86c44a0
DW
1123) const
1124{
542875a8
DW
1125 WXDWORD dwStyle = 0L;
1126
c86c44a0 1127 //
008089f6
DW
1128 // Native PM does not have any specialize 3D effects like WIN32 does,
1129 // so we have to try and invent them.
1130 //
1131
1132 //
1133 // If matches certain criteria, then assume no 3D effects
1134 // unless specifically requested (dealt with in MakeExtendedStyle)
1135 //
1136 if (!GetParent() ||
1137 !IsKindOf(CLASSINFO(wxControl)) ||
1138 (m_windowStyle & wxNO_BORDER)
1139 )
1140 {
1141 *pbWant3D = FALSE;
1142 return MakeExtendedStyle(m_windowStyle, FALSE);
1143 }
1144
1145 //
1146 // 1) App can specify global 3D effects
c86c44a0 1147 //
008089f6
DW
1148 *pbWant3D = wxTheApp->GetAuto3D();
1149
1150 //
1151 // 2) If the parent is being drawn with user colours, or simple border
1152 // specified, switch effects off.
1153 //
1154 if (GetParent() &&
1155 (GetParent()->GetWindowStyleFlag() & wxUSER_COLOURS) ||
1156 (m_windowStyle & wxSIMPLE_BORDER)
1157 )
1158 *pbWant3D = FALSE;
1159
1160 //
1161 // 3) Control can override this global setting by defining
1162 // a border style, e.g. wxSUNKEN_BORDER
1163 //
1164 if ((m_windowStyle & wxDOUBLE_BORDER) ||
1165 (m_windowStyle & wxRAISED_BORDER) ||
1166 (m_windowStyle & wxSTATIC_BORDER) ||
1167 (m_windowStyle & wxSUNKEN_BORDER)
1168 )
1169 *pbWant3D = TRUE;
1170
1171 dwStyle = MakeExtendedStyle( m_windowStyle
1172 ,FALSE
1173 );
1174
1175 //
1176 // If we want 3D, but haven't specified a border here,
1177 // apply the default border style specified.
1178 //
1179 if (dwDefaultBorderStyle && (*pbWant3D) &&
1180 !((m_windowStyle & wxDOUBLE_BORDER) ||
1181 (m_windowStyle & wxRAISED_BORDER) ||
1182 (m_windowStyle & wxSTATIC_BORDER) ||
1183 (m_windowStyle & wxSIMPLE_BORDER)
1184 )
1185 )
1186 dwStyle |= dwDefaultBorderStyle;
c86c44a0 1187 return dwStyle;
0367c1c0 1188} // end of wxWindowOS2::Determine3DEffects
cdf1e714
DW
1189
1190#if WXWIN_COMPATIBILITY
0367c1c0 1191void wxWindowOS2::OnCommand(
c86c44a0
DW
1192 wxWindow& rWin
1193, wxCommandEvent& rEvent
1194)
cdf1e714 1195{
c86c44a0
DW
1196 if (GetEventHandler()->ProcessEvent(rEvent))
1197 return;
1198 if (m_parent)
1199 m_parent->GetEventHandler()->OnCommand( rWin
1200 ,rEvent
1201 );
0367c1c0 1202} // end of wxWindowOS2::OnCommand
cdf1e714 1203
0367c1c0 1204wxObject* wxWindowOS2::GetChild(
c86c44a0
DW
1205 int nNumber
1206) const
cdf1e714 1207{
c86c44a0
DW
1208 //
1209 // Return a pointer to the Nth object in the Panel
1210 //
1211 wxNode* pNode = GetChildren().First();
1212 int n = nNumber;
cdf1e714 1213
c86c44a0
DW
1214 while (pNode && n--)
1215 pNode = pNode->Next();
1216 if (pNode)
1217 {
1218 wxObject* pObj = (wxObject*)pNode->Data();
1219 return(pObj);
1220 }
1221 else
1222 return NULL;
0367c1c0 1223} // end of wxWindowOS2::GetChild
cdf1e714
DW
1224
1225#endif // WXWIN_COMPATIBILITY
1226
c86c44a0 1227//
cdf1e714 1228// Setup background and foreground colours correctly
c86c44a0 1229//
0367c1c0 1230void wxWindowOS2::SetupColours()
cdf1e714
DW
1231{
1232 if ( GetParent() )
1233 SetBackgroundColour(GetParent()->GetBackgroundColour());
0367c1c0 1234} // end of wxWindowOS2::SetupColours
cdf1e714 1235
0367c1c0 1236void wxWindowOS2::OnIdle(
19193a2c 1237 wxIdleEvent& WXUNUSED(rEvent)
c86c44a0 1238)
cdf1e714 1239{
c86c44a0
DW
1240 //
1241 // Check if we need to send a LEAVE event
1242 //
1243 if (m_bMouseInWindow)
1244 {
1245 POINTL vPoint;
cdf1e714 1246
c86c44a0
DW
1247 ::WinQueryPointerPos(HWND_DESKTOP, &vPoint);
1248 if (::WinWindowFromPoint(HWND_DESKTOP, &vPoint, FALSE) != (HWND)GetHwnd())
1249 {
1250 //
1251 // Generate a LEAVE event
1252 //
1253 m_bMouseInWindow = FALSE;
1254
1255 //
1256 // Unfortunately the mouse button and keyboard state may have changed
1257 // by the time the OnIdle function is called, so 'state' may be
1258 // meaningless.
1259 //
1260 int nState = 0;
1261
1262 if (::WinGetKeyState(HWND_DESKTOP, VK_SHIFT) != 0)
1263 nState |= VK_SHIFT;
542875a8 1264 if (::WinGetKeyState(HWND_DESKTOP, VK_CTRL) != 0);
c86c44a0
DW
1265 nState |= VK_CTRL;
1266
1267 wxMouseEvent rEvent(wxEVT_LEAVE_WINDOW);
1268
1269 InitMouseEvent( rEvent
1270 ,vPoint.x
1271 ,vPoint.y
1272 ,nState
1273 );
542875a8 1274 (void)GetEventHandler()->ProcessEvent(rEvent);
c86c44a0
DW
1275 }
1276 }
1277 UpdateWindowUI();
0367c1c0 1278} // end of wxWindowOS2::OnIdle
c86c44a0
DW
1279
1280//
cdf1e714 1281// Set this window to be the child of 'parent'.
c86c44a0 1282//
0367c1c0 1283bool wxWindowOS2::Reparent(
c86c44a0
DW
1284 wxWindow* pParent
1285)
cdf1e714 1286{
c86c44a0 1287 if (!wxWindowBase::Reparent(pParent))
cdf1e714 1288 return FALSE;
c86c44a0
DW
1289
1290 HWND hWndChild = GetHwnd();
1291 HWND hWndParent = GetParent() ? GetWinHwnd(GetParent()) : (HWND)0;
1292
1293 ::WinSetParent(hWndChild, hWndParent, TRUE);
1294 return TRUE;
0367c1c0 1295} // end of wxWindowOS2::Reparent
cdf1e714 1296
0367c1c0 1297void wxWindowOS2::Clear()
cdf1e714 1298{
19193a2c 1299 wxClientDC vDc((wxWindow*)this);
c86c44a0
DW
1300 wxBrush vBrush( GetBackgroundColour()
1301 ,wxSOLID
1302 );
1303
1304 vDc.SetBackground(vBrush);
1305 vDc.Clear();
0367c1c0 1306} // end of wxWindowOS2::Clear
cdf1e714 1307
0367c1c0 1308void wxWindowOS2::Refresh(
c86c44a0
DW
1309 bool bEraseBack
1310, const wxRect* pRect
1311)
cdf1e714 1312{
c86c44a0
DW
1313 HWND hWnd = GetHwnd();
1314
1315 if (hWnd)
1316 {
1317 if (pRect)
1318 {
1319 RECTL vOs2Rect;
1320
1321 vOs2Rect.xLeft = pRect->x;
542875a8 1322 vOs2Rect.yTop = pRect->y;
c86c44a0
DW
1323 vOs2Rect.xRight = pRect->x + pRect->width;
1324 vOs2Rect.yBottom = pRect->y + pRect->height;
1325
1326 ::WinInvalidateRect(hWnd, &vOs2Rect, bEraseBack);
1327 }
1328 else
1329 ::WinInvalidateRect(hWnd, NULL, bEraseBack);
1330 }
0367c1c0 1331} // end of wxWindowOS2::Refresh
cdf1e714
DW
1332
1333// ---------------------------------------------------------------------------
1334// drag and drop
1335// ---------------------------------------------------------------------------
1336
1337#if wxUSE_DRAG_AND_DROP
0367c1c0 1338void wxWindowOS2::SetDropTarget(
c86c44a0
DW
1339 wxDropTarget* pDropTarget
1340)
cdf1e714 1341{
542875a8 1342 if (m_dropTarget != 0)
c86c44a0
DW
1343 {
1344 m_dropTarget->Revoke(m_hWnd);
1345 delete m_dropTarget;
1346 }
1347 m_dropTarget = pDropTarget;
1348 if (m_dropTarget != 0)
1349 m_dropTarget->Register(m_hWnd);
0367c1c0 1350} // end of wxWindowOS2::SetDropTarget
cdf1e714
DW
1351#endif
1352
c86c44a0 1353//
cdf1e714
DW
1354// old style file-manager drag&drop support: we retain the old-style
1355// DragAcceptFiles in parallel with SetDropTarget.
c86c44a0 1356//
0367c1c0 1357void wxWindowOS2::DragAcceptFiles(
c86c44a0
DW
1358 bool bAccept
1359)
cdf1e714 1360{
c86c44a0
DW
1361 HWND hWnd = GetHwnd();
1362
1363 if (hWnd && bAccept)
542875a8 1364 ::DrgAcceptDroppedFiles(hWnd, NULL, NULL, DO_COPY, 0L);
0367c1c0 1365} // end of wxWindowOS2::DragAcceptFiles
cdf1e714
DW
1366
1367// ----------------------------------------------------------------------------
1368// tooltips
1369// ----------------------------------------------------------------------------
1370
1371#if wxUSE_TOOLTIPS
1372
0367c1c0 1373void wxWindowOS2::DoSetToolTip(
c86c44a0
DW
1374 wxToolTip* pTooltip
1375)
cdf1e714 1376{
c86c44a0 1377 wxWindowBase::DoSetToolTip(pTooltip);
cdf1e714 1378
542875a8 1379 if (m_tooltip)
cdf1e714 1380 m_tooltip->SetWindow(this);
0367c1c0 1381} // end of wxWindowOS2::DoSetToolTip
cdf1e714
DW
1382
1383#endif // wxUSE_TOOLTIPS
1384
1385// ---------------------------------------------------------------------------
1386// moving and resizing
1387// ---------------------------------------------------------------------------
1388
1389// Get total size
0367c1c0 1390void wxWindowOS2::DoGetSize(
a7ef993c
DW
1391 int* pWidth
1392, int* pHeight
1393) const
cdf1e714 1394{
a7ef993c
DW
1395 HWND hWnd = GetHwnd();
1396 RECTL vRect;
c86c44a0 1397
a7ef993c
DW
1398 ::WinQueryWindowRect(hWnd, &vRect);
1399
1400 if (pWidth)
1401 *pWidth = vRect.xRight - vRect.xLeft;
1402 if (pHeight )
1403 // OS/2 PM is backwards from windows
1404 *pHeight = vRect.yTop - vRect.yBottom;
0367c1c0 1405} // end of wxWindowOS2::DoGetSize
cdf1e714 1406
0367c1c0 1407void wxWindowOS2::DoGetPosition(
a7ef993c
DW
1408 int* pX
1409, int* pY
1410) const
cdf1e714 1411{
a7ef993c 1412 HWND hWnd = GetHwnd();
8330166c 1413 SWP vSwp;
a7ef993c 1414 POINTL vPoint;
8330166c 1415 wxWindow* pParent = GetParent();
c86c44a0 1416
8330166c
DW
1417 //
1418 // It would seem that WinQueryWindowRect would be the correlary to
1419 // the WIN32 WinGetRect, but unlike WinGetRect which returns the window
1420 // origin position in screen coordinates, WinQueryWindowRect returns it
1421 // relative to itself, i.e. (0,0). To get the same under PM we must
1422 // us WinQueryWindowPos. This call, unlike the WIN32 call, however,
1423 // returns a position relative to it's parent, so no parent adujstments
1424 // are needed under OS/2. Also, windows should be created using
1425 // wxWindow coordinates, i.e 0,0 is the TOP left so vSwp will already
1426 // reflect that.
1427 //
1428 ::WinQueryWindowPos(hWnd, &vSwp);
c86c44a0 1429
8330166c
DW
1430 vPoint.x = vSwp.x;
1431 vPoint.y = vSwp.y;
c86c44a0 1432
a7ef993c 1433 //
8330166c
DW
1434 // We may be faking the client origin. So a window that's really at (0,
1435 // 30) may appear (to wxWin apps) to be at (0, 0).
a7ef993c 1436 //
8330166c 1437 if (pParent)
c86c44a0 1438 {
8330166c 1439 wxPoint vPt(pParent->GetClientAreaOrigin());
a7ef993c 1440
8330166c
DW
1441 vPoint.x -= vPt.x;
1442 vPoint.y -= vPt.y;
c86c44a0
DW
1443 }
1444
a7ef993c
DW
1445 if (pX)
1446 *pX = vPoint.x;
542875a8 1447 if (pY)
a7ef993c 1448 *pY = vPoint.y;
0367c1c0 1449} // end of wxWindowOS2::DoGetPosition
cdf1e714 1450
0367c1c0 1451void wxWindowOS2::DoScreenToClient(
a7ef993c
DW
1452 int* pX
1453, int* pY
1454) const
cdf1e714 1455{
a7ef993c 1456 HWND hWnd = GetHwnd();
8330166c 1457 SWP vSwp;
542875a8 1458
8330166c 1459 ::WinQueryWindowPos(hWnd, &vSwp);
cdf1e714 1460
a7ef993c 1461 if (pX)
8330166c 1462 *pX += vSwp.x;
a7ef993c 1463 if (pY)
8330166c 1464 *pY += vSwp.y;
0367c1c0 1465} // end of wxWindowOS2::DoScreenToClient
a7ef993c 1466
0367c1c0 1467void wxWindowOS2::DoClientToScreen(
a7ef993c
DW
1468 int* pX
1469, int* pY
1470) const
cdf1e714 1471{
a7ef993c 1472 HWND hWnd = GetHwnd();
8330166c 1473 SWP vSwp;
19193a2c 1474
8330166c 1475 ::WinQueryWindowPos(hWnd, &vSwp);
c86c44a0 1476
a7ef993c 1477 if (pX)
8330166c 1478 *pX += vSwp.x;
a7ef993c 1479 if (pY)
8330166c 1480 *pY += vSwp.y;
0367c1c0 1481} // end of wxWindowOS2::DoClientToScreen
cdf1e714 1482
a7ef993c 1483//
cdf1e714 1484// Get size *available for subwindows* i.e. excluding menu bar etc.
a7ef993c
DW
1485// Must be a frame type window
1486//
0367c1c0 1487void wxWindowOS2::DoGetClientSize(
a7ef993c
DW
1488 int* pWidth
1489, int* pHeight
1490) const
cdf1e714 1491{
a7ef993c
DW
1492 HWND hWnd = GetHwnd();
1493 HWND hWndClient;
1494 RECTL vRect;
1495
833615dd
DW
1496 if (IsKindOf(CLASSINFO(wxFrame)))
1497 hWndClient = ::WinWindowFromID(GetHwnd(), FID_CLIENT);
1498 else
1499 hWndClient = NULLHANDLE;
51c1d535
DW
1500 if( hWndClient == NULLHANDLE)
1501 ::WinQueryWindowRect(GetHwnd(), &vRect);
1502 else
1503 ::WinQueryWindowRect(hWndClient, &vRect);
a7ef993c 1504
542875a8 1505 if (pWidth)
a7ef993c 1506 *pWidth = vRect.xRight;
542875a8 1507 if (pHeight)
a7ef993c 1508 *pHeight = vRect.yTop;
0367c1c0 1509} // end of wxWindowOS2::DoGetClientSize
cdf1e714 1510
0367c1c0 1511void wxWindowOS2::DoMoveWindow(
a7ef993c
DW
1512 int nX
1513, int nY
1514, int nWidth
1515, int nHeight
1516)
cdf1e714 1517{
39c26ef2
DW
1518 RECTL vRect;
1519 HWND hParent;
1520 wxWindow* pParent = GetParent();
1521
1522 if (pParent)
987da0d4 1523 {
39c26ef2 1524 hParent = GetWinHwnd(pParent);
987da0d4
DW
1525 if (pParent->IsKindOf(CLASSINFO(wxFrame)))
1526 {
1527 if (IsKindOf(CLASSINFO(wxStatusBar)) ||
1528 IsKindOf(CLASSINFO(wxMenuBar)) ||
1529 IsKindOf(CLASSINFO(wxToolBar))
1530 )
1531 nY = pParent->GetSize().y - (nY + nHeight);
1532 else
1533 nY = pParent->GetClientSize().y - (nY + nHeight);
1534 }
1535 else
1536 nY = pParent->GetSize().y - (nY + nHeight);
1537 }
39c26ef2 1538 else
c86c44a0 1539 {
987da0d4
DW
1540 RECTL vRect;
1541
1542 ::WinQueryWindowRect(HWND_DESKTOP, &vRect);
1543 nY = vRect.yTop - (nY + nHeight);
1544 }
1545 ::WinSetWindowPos( GetHwnd()
1546 ,HWND_TOP
1547 ,(LONG)nX
1548 ,(LONG)nY
1549 ,(LONG)nWidth
1550 ,(LONG)nHeight
1551 ,SWP_ZORDER | SWP_SIZE | SWP_MOVE | SWP_SHOW
1552 );
0367c1c0 1553} // end of wxWindowOS2::DoMoveWindow
cdf1e714 1554
c86c44a0
DW
1555//
1556// Set the size of the window: if the dimensions are positive, just use them,
cdf1e714
DW
1557// but if any of them is equal to -1, it means that we must find the value for
1558// it ourselves (unless sizeFlags contains wxSIZE_ALLOW_MINUS_ONE flag, in
1559// which case -1 is a valid value for x and y)
1560//
1561// If sizeFlags contains wxSIZE_AUTO_WIDTH/HEIGHT flags (default), we calculate
1562// the width/height to best suit our contents, otherwise we reuse the current
1563// width/height
c86c44a0 1564//
0367c1c0 1565void wxWindowOS2::DoSetSize(
a7ef993c
DW
1566 int nX
1567, int nY
1568, int nWidth
1569, int nHeight
1570, int nSizeFlags
1571)
cdf1e714 1572{
a7ef993c
DW
1573 //
1574 // Get the current size and position...
1575 //
1576 int nCurrentX;
1577 int nCurrentY;
1578 int nCurrentWidth;
1579 int nCurrentHeight;
542875a8 1580 wxSize vSize(-1, -1);
a7ef993c 1581
19193a2c
KB
1582 GetPosition(&nCurrentX, &nCurrentY);
1583 GetSize(&nCurrentWidth, &nCurrentHeight);
c86c44a0
DW
1584
1585 // ... and don't do anything (avoiding flicker) if it's already ok
19193a2c
KB
1586 if (nX == nCurrentX && nY == nCurrentY &&
1587 nWidth == nCurrentWidth && nHeight == nCurrentHeight)
c86c44a0
DW
1588 {
1589 return;
1590 }
1591
a7ef993c
DW
1592 if (nX == -1 && !(nSizeFlags & wxSIZE_ALLOW_MINUS_ONE))
1593 nX = nCurrentX;
542875a8 1594 if (nY == -1 && !(nSizeFlags & wxSIZE_ALLOW_MINUS_ONE))
a7ef993c 1595 nY = nCurrentY;
c86c44a0 1596
19193a2c 1597 AdjustForParentClientOrigin(nX, nY, nSizeFlags);
c86c44a0 1598
a7ef993c 1599 if (nWidth == -1)
c86c44a0 1600 {
a7ef993c 1601 if (nSizeFlags & wxSIZE_AUTO_WIDTH)
c86c44a0 1602 {
a7ef993c
DW
1603 vSize = DoGetBestSize();
1604 nWidth = vSize.x;
c86c44a0
DW
1605 }
1606 else
1607 {
a7ef993c
DW
1608 //
1609 // Just take the current one
1610 //
1611 nWidth = nCurrentWidth;
c86c44a0
DW
1612 }
1613 }
1614
a7ef993c 1615 if (nHeight == -1)
c86c44a0 1616 {
a7ef993c 1617 if (nSizeFlags & wxSIZE_AUTO_HEIGHT)
c86c44a0 1618 {
a7ef993c 1619 if (vSize.x == -1)
c86c44a0 1620 {
a7ef993c 1621 vSize = DoGetBestSize();
c86c44a0 1622 }
a7ef993c 1623 nHeight = vSize.y;
c86c44a0
DW
1624 }
1625 else
1626 {
1627 // just take the current one
a7ef993c 1628 nHeight = nCurrentHeight;
c86c44a0
DW
1629 }
1630 }
1631
a7ef993c
DW
1632 DoMoveWindow( nX
1633 ,nY
1634 ,nWidth
1635 ,nHeight
1636 );
0367c1c0 1637} // end of wxWindowOS2::DoSetSize
cdf1e714 1638
0367c1c0 1639void wxWindowOS2::DoSetClientSize(
a7ef993c
DW
1640 int nWidth
1641, int nHeight
1642)
cdf1e714 1643{
a7ef993c
DW
1644 wxWindow* pParent = GetParent();
1645 HWND hWnd = GetHwnd();
1646 HWND hParentWnd = (HWND)0;
8330166c 1647 POINTL vPoint;
a7ef993c 1648 RECTL vRect;
8330166c
DW
1649 RECTL vRect2;
1650 RECTL vRect3;
1651 HWND hClientWnd = (HWND)0;
c86c44a0 1652
8330166c 1653 hClientWnd = ::WinWindowFromID(hWnd, FID_CLIENT);
a7ef993c 1654 ::WinQueryWindowRect(hClientWnd, &vRect2);
a7ef993c
DW
1655 ::WinQueryWindowRect(hWnd, &vRect);
1656 ::WinQueryWindowRect(hParentWnd, &vRect3);
8330166c 1657
a7ef993c
DW
1658 int nActualWidth = vRect2.xRight - vRect2.xLeft - vRect.xRight + nWidth;
1659 int nActualHeight = vRect2.yTop - vRect2.yBottom - vRect.yTop + nHeight;
c86c44a0 1660
a7ef993c
DW
1661 vPoint.x = vRect2.xLeft;
1662 vPoint.y = vRect2.yBottom;
1663 if (pParent)
8330166c 1664 {
a7ef993c 1665 vPoint.x -= vRect3.xLeft;
542875a8 1666 vPoint.y -= vRect3.yBottom;
c86c44a0 1667 }
19193a2c
KB
1668
1669 DoMoveWindow(vPoint.x, vPoint.y, nActualWidth, nActualHeight);
1670
1671 wxSizeEvent vEvent(wxSize(nWidth, nHeight), m_windowId);
a7ef993c
DW
1672 vEvent.SetEventObject(this);
1673 GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 1674} // end of wxWindowOS2::DoSetClientSize
cdf1e714 1675
0367c1c0 1676wxPoint wxWindowOS2::GetClientAreaOrigin() const
cdf1e714
DW
1677{
1678 return wxPoint(0, 0);
0367c1c0 1679} // end of wxWindowOS2::GetClientAreaOrigin
cdf1e714 1680
cdf1e714
DW
1681// ---------------------------------------------------------------------------
1682// text metrics
1683// ---------------------------------------------------------------------------
1684
0367c1c0 1685int wxWindowOS2::GetCharHeight() const
cdf1e714 1686{
a7ef993c
DW
1687 HPS hPs;
1688 FONTMETRICS vFontMetrics;
cdf1e714 1689
a7ef993c
DW
1690 hPs = ::WinGetPS(GetHwnd());
1691
542875a8 1692 if(!GpiQueryFontMetrics(hPs, sizeof(FONTMETRICS), &vFontMetrics))
b7084589
DW
1693 {
1694 ::WinReleasePS(hPs);
a7ef993c 1695 return (0);
b7084589 1696 }
a7ef993c 1697 ::WinReleasePS(hPs);
b7084589 1698 return(vFontMetrics.lMaxAscender + vFontMetrics.lMaxDescender);
0367c1c0 1699} // end of wxWindowOS2::GetCharHeight
a7ef993c 1700
0367c1c0 1701int wxWindowOS2::GetCharWidth() const
cdf1e714 1702{
542875a8
DW
1703 HPS hPs;
1704 FONTMETRICS vFontMetrics;
1705
a7ef993c 1706 hPs = ::WinGetPS(GetHwnd());
cdf1e714 1707
542875a8 1708 if(!GpiQueryFontMetrics(hPs, sizeof(FONTMETRICS), &vFontMetrics))
b7084589
DW
1709 {
1710 ::WinReleasePS(hPs);
a7ef993c 1711 return (0);
b7084589 1712 }
a7ef993c 1713 ::WinReleasePS(hPs);
b7084589 1714 return(vFontMetrics.lAveCharWidth);
0367c1c0 1715} // end of wxWindowOS2::GetCharWidth
a7ef993c 1716
0367c1c0 1717void wxWindowOS2::GetTextExtent(
a7ef993c
DW
1718 const wxString& rString
1719, int* pX
1720, int* pY
1721, int* pDescent
1722, int* pExternalLeading
1723, const wxFont* pTheFont
1724) const
cdf1e714 1725{
987da0d4
DW
1726 POINTL avPoint[TXTBOX_COUNT];
1727 POINTL vPtMin;
1728 POINTL vPtMax;
1729 int i;
1730 int l;
1731 FONTMETRICS vFM; // metrics structure
1732 BOOL bRc;
1733 char* pStr;
1734 ERRORID vErrorCode; // last error id code
1735 HPS hPS;
19193a2c 1736
a7ef993c 1737
987da0d4 1738 hPS = ::WinGetPS(GetHwnd());
a7ef993c 1739
987da0d4
DW
1740 l = rString.Length();
1741 if (l > 0L)
1742 {
1743 pStr = (PCH)rString.c_str();
a7ef993c 1744
987da0d4
DW
1745 //
1746 // In world coordinates.
1747 //
1748 bRc = ::GpiQueryTextBox( hPS
1749 ,l
1750 ,pStr
1751 ,TXTBOX_COUNT // return maximum information
1752 ,avPoint // array of coordinates points
1753 );
1754 if (bRc)
1755 {
1756 vPtMin.x = avPoint[0].x;
1757 vPtMax.x = avPoint[0].x;
1758 vPtMin.y = avPoint[0].y;
1759 vPtMax.y = avPoint[0].y;
1760 for (i = 1; i < 4; i++)
1761 {
1762 if(vPtMin.x > avPoint[i].x) vPtMin.x = avPoint[i].x;
1763 if(vPtMin.y > avPoint[i].y) vPtMin.y = avPoint[i].y;
1764 if(vPtMax.x < avPoint[i].x) vPtMax.x = avPoint[i].x;
1765 if(vPtMax.y < avPoint[i].y) vPtMax.y = avPoint[i].y;
1766 }
1767 bRc = ::GpiQueryFontMetrics( hPS
1768 ,sizeof(FONTMETRICS)
1769 ,&vFM
1770 );
1771 if (!bRc)
1772 {
1773 vPtMin.x = 0;
1774 vPtMin.y = 0;
1775 vPtMax.x = 0;
1776 vPtMax.y = 0;
1777 }
1778 }
1779 else
1780 {
1781 vPtMin.x = 0;
1782 vPtMin.y = 0;
1783 vPtMax.x = 0;
1784 vPtMax.y = 0;
1785 }
1786 }
1787 else
a7ef993c 1788 {
987da0d4
DW
1789 vPtMin.x = 0;
1790 vPtMin.y = 0;
1791 vPtMax.x = 0;
1792 vPtMax.y = 0;
a7ef993c 1793 }
987da0d4
DW
1794 if (pX)
1795 *pX = (vPtMax.x - vPtMin.x + 1);
1796 if (pY)
1797 *pY = (vPtMax.y - vPtMin.y + 1);
1798 if (pDescent)
1799 {
1800 if (bRc)
1801 *pDescent = vFM.lMaxDescender;
1802 else
1803 *pDescent = 0;
1804 }
1805 if (pExternalLeading)
1806 {
1807 if (bRc)
1808 *pExternalLeading = vFM.lExternalLeading;
1809 else
1810 *pExternalLeading = 0;
1811 }
1812 ::WinReleasePS(hPS);
1813} // end of wxWindow::GetTextExtent
cdf1e714
DW
1814
1815#if wxUSE_CARET && WXWIN_COMPATIBILITY
1816// ---------------------------------------------------------------------------
1817// Caret manipulation
1818// ---------------------------------------------------------------------------
1819
0367c1c0 1820void wxWindowOS2::CreateCaret(
a7ef993c
DW
1821 int nWidth
1822, int nHeight
1823)
cdf1e714 1824{
a7ef993c
DW
1825 SetCaret(new wxCaret( this
1826 ,nWidth
1827 ,nHeight
1828 ));
0367c1c0 1829} // end of wxWindowOS2::CreateCaret
a7ef993c 1830
0367c1c0 1831void wxWindowOS2::CreateCaret(
a7ef993c
DW
1832 const wxBitmap* pBitmap
1833)
cdf1e714 1834{
a7ef993c 1835 wxFAIL_MSG("not implemented");
0367c1c0 1836} // end of wxWindowOS2::CreateCaret
cdf1e714 1837
0367c1c0 1838void wxWindowOS2::ShowCaret(
a7ef993c
DW
1839 bool bShow
1840)
cdf1e714 1841{
a7ef993c
DW
1842 wxCHECK_RET( m_caret, "no caret to show" );
1843
1844 m_caret->Show(bShow);
0367c1c0 1845} // end of wxWindowOS2::ShowCaret
cdf1e714 1846
0367c1c0 1847void wxWindowOS2::DestroyCaret()
cdf1e714 1848{
a7ef993c 1849 SetCaret(NULL);
0367c1c0 1850} // end of wxWindowOS2::DestroyCaret
cdf1e714 1851
0367c1c0 1852void wxWindowOS2::SetCaretPos(
a7ef993c
DW
1853 int nX
1854, int nY)
cdf1e714 1855{
a7ef993c
DW
1856 wxCHECK_RET( m_caret, "no caret to move" );
1857
1858 m_caret->Move( nX
1859 ,nY
1860 );
0367c1c0 1861} // end of wxWindowOS2::SetCaretPos
cdf1e714 1862
0367c1c0 1863void wxWindowOS2::GetCaretPos(
a7ef993c
DW
1864 int* pX
1865, int* pY
1866) const
cdf1e714 1867{
a7ef993c
DW
1868 wxCHECK_RET( m_caret, "no caret to get position of" );
1869
1870 m_caret->GetPosition( pX
1871 ,pY
1872 );
0367c1c0 1873} // end of wxWindowOS2::GetCaretPos
cdf1e714
DW
1874
1875#endif //wxUSE_CARET
1876
1877// ---------------------------------------------------------------------------
1878// popup menu
1879// ---------------------------------------------------------------------------
19193a2c
KB
1880//
1881#if wxUSE_MENUS_NATIVE
3febf684
DW
1882static void wxYieldForCommandsOnly()
1883{
1884 //
1885 // Peek all WM_COMMANDs (it will always return WM_QUIT too but we don't
1886 // want to process it here)
1887 //
1888 QMSG vMsg;
1889
19193a2c
KB
1890 while (::WinPeekMsg(vHabmain, &vMsg, (HWND)0, WM_COMMAND,
1891 WM_COMMAND,PM_REMOVE) && vMsg.msg != WM_QUIT)
3febf684
DW
1892 {
1893 wxTheApp->DoMessage((WXMSG*)&vMsg);
1894 }
1895}
19193a2c 1896#endif // wxUSE_MENUS_NATIVE
3febf684 1897
19193a2c 1898#if wxUSE_MENUS_NATIVE
0367c1c0 1899bool wxWindowOS2::DoPopupMenu(
61243a51
DW
1900 wxMenu* pMenu
1901, int nX
1902, int nY
1903)
cdf1e714 1904{
61243a51
DW
1905 HWND hWnd = GetHwnd();
1906 HWND hWndParent = GetParent() ? GetWinHwnd(GetParent()) : (HWND)0;
1907 HWND hMenu = GetHmenuOf(pMenu);
1908
1909 pMenu->SetInvokingWindow(this);
1910 pMenu->UpdateUI();
1911
1912 DoClientToScreen( &nX
1913 ,&nY
1914 );
1915 wxCurrentPopupMenu = pMenu;
1916
1917 ::WinPopupMenu( hWndParent
1918 ,hWnd
1919 ,hMenu
1920 ,nX
1921 ,nY
1922 ,0L
1923 ,PU_MOUSEBUTTON2DOWN | PU_MOUSEBUTTON2 | PU_KEYBOARD
1924 );
3febf684 1925 // we need to do it righ now as otherwise the events are never going to be
19193a2c 1926 // sent to wxCurrentPopupMenu from ;()
3febf684
DW
1927 //
1928 // note that even eliminating (ugly) wxCurrentPopupMenu global wouldn't
1929 // help and we'd still need wxYieldForCommandsOnly() as the menu may be
1930 // destroyed as soon as we return (it can be a local variable in the caller
1931 // for example) and so we do need to process the event immediately
1932 wxYieldForCommandsOnly();
61243a51
DW
1933 wxCurrentPopupMenu = NULL;
1934
1935 pMenu->SetInvokingWindow(NULL);
1936 return TRUE;
0367c1c0 1937} // end of wxWindowOS2::DoPopupMenu
19193a2c 1938#endif // wxUSE_MENUS_NATIVE
cdf1e714
DW
1939
1940// ===========================================================================
1941// pre/post message processing
1942// ===========================================================================
1943
0367c1c0 1944MRESULT wxWindowOS2::OS2DefWindowProc(
61243a51
DW
1945 WXUINT uMsg
1946, WXWPARAM wParam
1947, WXLPARAM lParam
1948)
cdf1e714 1949{
61243a51 1950 if (m_fnOldWndProc)
19193a2c 1951 return (MRESULT)m_fnOldWndProc(GetHWND(), uMsg, (MPARAM)wParam, (MPARAM)lParam);
61243a51 1952 else
19193a2c 1953 return ::WinDefWindowProc(GetHWND(), uMsg, (MPARAM)wParam, (MPARAM)lParam);
0367c1c0 1954} // end of wxWindowOS2::OS2DefWindowProc
cdf1e714 1955
0367c1c0 1956bool wxWindowOS2::OS2ProcessMessage(
61243a51
DW
1957 WXMSG* pMsg
1958)
cdf1e714 1959{
19193a2c
KB
1960// wxUniversal implements tab traversal itself
1961#ifndef __WXUNIVERSAL__
61243a51
DW
1962 QMSG* pQMsg = (QMSG*)pMsg;
1963
1964 if (m_hWnd != 0 && (GetWindowStyleFlag() & wxTAB_TRAVERSAL))
1965 {
1966 //
1967 // Intercept dialog navigation keys
1968 //
1969 bool bProcess = TRUE;
1970 USHORT uKeyFlags = SHORT1FROMMP(pQMsg->mp1);
1971
1972 if (uKeyFlags & KC_KEYUP)
1973 bProcess = FALSE;
1974
1975 if (uKeyFlags & KC_ALT)
1976 bProcess = FALSE;
1977
1978 if (!(uKeyFlags & KC_VIRTUALKEY))
1979 bProcess = FALSE;
1980
1981 if (bProcess)
1982 {
1983 bool bCtrlDown = IsCtrlDown();
1984 bool bShiftDown = IsShiftDown();
1985
1986 //
1987 // WM_QUERYDLGCODE: ask the control if it wants the key for itself,
1988 // don't process it if it's the case (except for Ctrl-Tab/Enter
1989 // combinations which are always processed)
1990 //
1991 ULONG ulDlgCode = 0;
1992
1993 if (!bCtrlDown)
1994 {
1995 ulDlgCode = (ULONG)::WinSendMsg(pQMsg->hwnd, WM_QUERYDLGCODE, pQMsg, 0);
1996 }
1997
1998 bool bForward = TRUE;
1999 bool bWindowChange = FALSE;
2000
2001 switch (SHORT2FROMMP(pQMsg->mp2))
2002 {
2003 //
2004 // Going to make certain assumptions about specific types of controls
2005 // here, so we may have to alter some things later if they prove invalid
2006 //
2007 case VK_TAB:
2008 //
2009 // Shift tabl will always be a nav-key but tabs may be wanted
2010 //
2011 if (!bShiftDown)
2012 {
2013 bProcess = FALSE;
2014 }
2015 else
2016 {
2017 //
2018 // Entry Fields want tabs for themselve usually
2019 //
2020 switch (ulDlgCode)
2021 {
2022 case DLGC_ENTRYFIELD:
2023 case DLGC_MLE:
2024 bProcess = TRUE;
2025 break;
2026
2027 default:
2028 bProcess = FALSE;
2029 }
2030
2031 //
2032 // Ctrl-Tab cycles thru notebook pages
2033 //
2034 bWindowChange = bCtrlDown;
2035 bForward = !bShiftDown;
2036 }
2037 break;
2038
2039 case VK_UP:
2040 case VK_LEFT:
2041 if (bCtrlDown)
2042 bProcess = FALSE;
2043 else
2044 bForward = FALSE;
2045 break;
2046
2047 case VK_DOWN:
2048 case VK_RIGHT:
2049 if (bCtrlDown)
2050 bProcess = FALSE;
2051 break;
2052
2053 case VK_ENTER:
2054 {
2055 if (bCtrlDown)
2056 {
2057 //
2058 // ctrl-enter is not processed
2059 //
2060 return FALSE;
2061 }
2062 else if (ulDlgCode & DLGC_BUTTON)
2063 {
2064 //
2065 // buttons want process Enter themselevs
2066 //
2067 bProcess = FALSE;
2068 }
2069 else
2070 {
1bcfc0e1
DW
2071 wxButton* pBtn = wxDynamicCast( GetDefaultItem()
2072 ,wxButton
2073 );
61243a51
DW
2074
2075 if (pBtn && pBtn->IsEnabled())
2076 {
2077 //
2078 // If we do have a default button, do press it
2079 //
2080 pBtn->OS2Command(BN_CLICKED, 0 /* unused */);
2081 return TRUE;
2082 }
2083 // else: but if it does not it makes sense to make
2084 // it work like a TAB - and that's what we do.
2085 // Note that Ctrl-Enter always works this way.
2086 }
2087 }
2088 break;
2089
2090 default:
2091 bProcess = FALSE;
2092 }
2093
2094 if (bProcess)
2095 {
2096 wxNavigationKeyEvent vEvent;
2097
2098 vEvent.SetDirection(bForward);
2099 vEvent.SetWindowChange(bWindowChange);
2100 vEvent.SetEventObject(this);
2101
2102 if (GetEventHandler()->ProcessEvent(vEvent))
2103 {
2104 wxButton* pBtn = wxDynamicCast(FindFocus(), wxButton);
2105
2106 if (pBtn)
2107 {
2108 //
2109 // The button which has focus should be default
2110 //
2111 pBtn->SetDefault();
2112 }
2113 return TRUE;
2114 }
2115 }
2116 }
dae16775
DW
2117 //
2118 // Let Dialogs process
2119 //
2120 if (::WinSendMsg(pQMsg->hwnd, WM_QUERYDLGCODE, pQMsg, 0));
2121 return TRUE;
61243a51 2122 }
19193a2c
KB
2123#else
2124 pMsg = pMsg; // just shut up the compiler
2125#endif // __WXUNIVERSAL__
61243a51
DW
2126
2127#if wxUSE_TOOLTIPS
2128 if ( m_tooltip )
2129 {
2130 // relay mouse move events to the tooltip control
2131 QMSG* pQMsg = (QMSG*)pMsg;
2132
2133 if (pQMsg->msg == WM_MOUSEMOVE )
2134 m_tooltip->RelayEvent(pMsg);
2135 }
2136#endif // wxUSE_TOOLTIPS
2137
cdf1e714 2138 return FALSE;
0367c1c0 2139} // end of wxWindowOS2::OS2ProcessMessage
cdf1e714 2140
0367c1c0 2141bool wxWindowOS2::OS2TranslateMessage(
61243a51
DW
2142 WXMSG* pMsg
2143)
cdf1e714 2144{
19193a2c 2145#if wxUSE_ACCEL && !defined(__WXUNIVERSAL__)
7e99520b
DW
2146 return m_acceleratorTable.Translate(m_hWnd, pMsg);
2147#else
19193a2c 2148 pMsg = pMsg;
7e99520b
DW
2149 return FALSE;
2150#endif //wxUSE_ACCEL
0367c1c0 2151} // end of wxWindowOS2::OS2TranslateMessage
cdf1e714
DW
2152
2153// ---------------------------------------------------------------------------
61243a51 2154// message params unpackers
cdf1e714
DW
2155// ---------------------------------------------------------------------------
2156
0367c1c0 2157void wxWindowOS2::UnpackCommand(
61243a51
DW
2158 WXWPARAM wParam
2159, WXLPARAM lParam
2160, WORD* pId
2161, WXHWND* phWnd
2162, WORD* pCmd
2163)
cdf1e714 2164{
5b3ed311
DW
2165 *pId = LOWORD(wParam);
2166 *phWnd = NULL; // or may be GetHWND() ?
2167 *pCmd = LOWORD(lParam);
0367c1c0 2168} // end of wxWindowOS2::UnpackCommand
cdf1e714 2169
0367c1c0 2170void wxWindowOS2::UnpackActivate(
61243a51
DW
2171 WXWPARAM wParam
2172, WXLPARAM lParam
2173, WXWORD* pState
2174, WXHWND* phWnd
2175)
cdf1e714 2176{
61243a51
DW
2177 *pState = LOWORD(wParam);
2178 *phWnd = (WXHWND)lParam;
0367c1c0 2179} // end of wxWindowOS2::UnpackActivate
cdf1e714 2180
0367c1c0 2181void wxWindowOS2::UnpackScroll(
61243a51
DW
2182 WXWPARAM wParam
2183, WXLPARAM lParam
2184, WXWORD* pCode
2185, WXWORD* pPos
2186, WXHWND* phWnd
2187)
cdf1e714 2188{
b7d8f285
DW
2189 ULONG ulId;
2190 HWND hWnd;
2191
2192 ulId = (ULONG)LONGFROMMP(wParam);
2193 hWnd = ::WinWindowFromID(GetHwnd(), ulId);
2194 if (hWnd == m_hWndScrollBarHorz || hWnd == m_hWndScrollBarVert)
2195 *phWnd = NULLHANDLE;
2196 else
2197 *phWnd = hWnd;
2198
2199 *pPos = SHORT1FROMMP(lParam);
2200 *pCode = SHORT2FROMMP(lParam);
0367c1c0 2201} // end of wxWindowOS2::UnpackScroll
61243a51 2202
0367c1c0 2203void wxWindowOS2::UnpackMenuSelect(
61243a51
DW
2204 WXWPARAM wParam
2205, WXLPARAM lParam
2206, WXWORD* pItem
2207, WXWORD* pFlags
2208, WXHMENU* phMenu
2209)
cdf1e714 2210{
61243a51
DW
2211 *pItem = (WXWORD)LOWORD(wParam);
2212 *pFlags = HIWORD(wParam);
2213 *phMenu = (WXHMENU)lParam;
0367c1c0 2214} // end of wxWindowOS2::UnpackMenuSelect
cdf1e714
DW
2215
2216// ---------------------------------------------------------------------------
2217// Main wxWindows window proc and the window proc for wxWindow
2218// ---------------------------------------------------------------------------
2219
61243a51 2220//
cdf1e714
DW
2221// Hook for new window just as it's being created, when the window isn't yet
2222// associated with the handle
61243a51 2223//
19193a2c 2224wxWindowOS2* wxWndHook = NULL;
cdf1e714 2225
61243a51 2226//
cdf1e714 2227// Main window proc
61243a51 2228//
f23208ca 2229MRESULT EXPENTRY wxWndProc(
61243a51
DW
2230 HWND hWnd
2231, ULONG ulMsg
2232, MPARAM wParam
2233, MPARAM lParam
2234)
cdf1e714 2235{
61243a51
DW
2236 //
2237 // Trace all ulMsgs - useful for the debugging
2238 //
cdf1e714
DW
2239#ifdef __WXDEBUG__
2240 wxLogTrace(wxTraceMessages, wxT("Processing %s(wParam=%8lx, lParam=%8lx)"),
61243a51 2241 wxGetMessageName(ulMsg), wParam, lParam);
cdf1e714
DW
2242#endif // __WXDEBUG__
2243
19193a2c 2244 wxWindowOS2* pWnd = wxFindWinFromHandle((WXHWND)hWnd);
cdf1e714 2245
61243a51
DW
2246 //
2247 // When we get the first message for the HWND we just created, we associate
cdf1e714 2248 // it with wxWindow stored in wxWndHook
61243a51
DW
2249 //
2250 if (!pWnd && wxWndHook)
cdf1e714 2251 {
cdf1e714 2252 wxAssociateWinWithHandle(hWnd, wxWndHook);
61243a51 2253 pWnd = wxWndHook;
cdf1e714 2254 wxWndHook = NULL;
61243a51 2255 pWnd->SetHWND((WXHWND)hWnd);
cdf1e714
DW
2256 }
2257
5b3ed311 2258 MRESULT rc = (MRESULT)0;
e604d44b 2259
cdf1e714 2260
61243a51 2261 //
cdf1e714 2262 // Stop right here if we don't have a valid handle in our wxWindow object.
61243a51
DW
2263 //
2264 if (pWnd && !pWnd->GetHWND())
cdf1e714 2265 {
61243a51
DW
2266 pWnd->SetHWND((WXHWND) hWnd);
2267 rc = pWnd->OS2DefWindowProc(ulMsg, wParam, lParam );
2268 pWnd->SetHWND(0);
cdf1e714
DW
2269 }
2270 else
2271 {
61243a51
DW
2272 if (pWnd)
2273 rc = pWnd->OS2WindowProc(ulMsg, wParam, lParam);
e604d44b 2274 else
61243a51 2275 rc = ::WinDefWindowProc(hWnd, ulMsg, wParam, lParam);
cdf1e714 2276 }
5b3ed311 2277
cdf1e714 2278 return rc;
61243a51 2279} // end of wxWndProc
cdf1e714 2280
61243a51
DW
2281//
2282// We will add (or delete) messages we need to handle at this default
2283// level as we go
2284//
0367c1c0 2285MRESULT wxWindowOS2::OS2WindowProc(
61243a51
DW
2286 WXUINT uMsg
2287, WXWPARAM wParam
2288, WXLPARAM lParam
2289)
cdf1e714 2290{
61243a51
DW
2291 //
2292 // Did we process the uMsg?
2293 //
2294 bool bProcessed = FALSE;
e604d44b 2295 MRESULT mResult;
cdf1e714 2296
61243a51
DW
2297 //
2298 // For most messages we should return 0 when we do process the message
2299 //
e604d44b 2300 mResult = (MRESULT)0;
61243a51
DW
2301
2302 switch (uMsg)
cdf1e714
DW
2303 {
2304 case WM_CREATE:
2305 {
61243a51
DW
2306 bool bMayCreate;
2307
2308 bProcessed = HandleCreate( (WXLPCREATESTRUCT)lParam
2309 ,&bMayCreate
2310 );
2311 if (bProcessed)
cdf1e714 2312 {
61243a51
DW
2313 //
2314 // Return 0 to bAllow window creation
2315 //
e604d44b 2316 mResult = (MRESULT)(bMayCreate ? 0 : -1);
cdf1e714
DW
2317 }
2318 }
2319 break;
2320
2321 case WM_DESTROY:
e604d44b
DW
2322 HandleDestroy();
2323 bProcessed = TRUE;
e604d44b 2324 break;
cdf1e714
DW
2325
2326 case WM_MOVE:
61243a51
DW
2327 bProcessed = HandleMove( LOWORD(lParam)
2328 ,HIWORD(lParam)
2329 );
cdf1e714
DW
2330 break;
2331
2332 case WM_SIZE:
61243a51
DW
2333 bProcessed = HandleSize( LOWORD(lParam)
2334 ,HIWORD(lParam)
2335 ,(WXUINT)wParam
2336 );
cdf1e714
DW
2337 break;
2338
2339 case WM_ACTIVATE:
2340 {
61243a51
DW
2341 WXWORD wState;
2342 WXHWND hWnd;
2343
2344 UnpackActivate( wParam
2345 ,lParam
2346 ,&wState
2347 ,&hWnd
2348 );
2349
2350 bProcessed = HandleActivate( wState
2351 ,(WXHWND)hWnd
2352 );
e604d44b 2353 bProcessed = FALSE;
cdf1e714
DW
2354 }
2355 break;
2356
2357 case WM_SETFOCUS:
61243a51
DW
2358 if (SHORT1FROMMP((MPARAM)lParam) == TRUE)
2359 bProcessed = HandleSetFocus((WXHWND)(HWND)wParam);
2360 else
2361 bProcessed = HandleKillFocus((WXHWND)(HWND)wParam);
cdf1e714
DW
2362 break;
2363
2364 case WM_PAINT:
61243a51 2365 bProcessed = HandlePaint();
cdf1e714
DW
2366 break;
2367
2368 case WM_CLOSE:
61243a51
DW
2369 //
2370 // Don't let the DefWindowProc() destroy our window - we'll do it
cdf1e714 2371 // ourselves in ~wxWindow
61243a51
DW
2372 //
2373 bProcessed = TRUE;
e604d44b 2374 mResult = (MRESULT)TRUE;
cdf1e714
DW
2375 break;
2376
61243a51
DW
2377 case WM_SHOW:
2378 bProcessed = HandleShow(wParam != 0, (int)lParam);
cdf1e714
DW
2379 break;
2380
61243a51
DW
2381 //
2382 // Under OS2 PM Joysticks are treated just like mouse events
2383 // The "Motion" events will be prevelent in joysticks
2384 //
cdf1e714 2385 case WM_MOUSEMOVE:
61243a51
DW
2386 case WM_BUTTON1DOWN:
2387 case WM_BUTTON1UP:
2388 case WM_BUTTON1DBLCLK:
2389 case WM_BUTTON1MOTIONEND:
2390 case WM_BUTTON1MOTIONSTART:
2391 case WM_BUTTON2DOWN:
2392 case WM_BUTTON2UP:
2393 case WM_BUTTON2DBLCLK:
2394 case WM_BUTTON2MOTIONEND:
2395 case WM_BUTTON2MOTIONSTART:
2396 case WM_BUTTON3DOWN:
2397 case WM_BUTTON3UP:
2398 case WM_BUTTON3DBLCLK:
2399 case WM_BUTTON3MOTIONEND:
2400 case WM_BUTTON3MOTIONSTART:
cdf1e714
DW
2401 {
2402 short x = LOWORD(lParam);
2403 short y = HIWORD(lParam);
2404
61243a51 2405 bProcessed = HandleMouseEvent(uMsg, x, y, (WXUINT)wParam);
cdf1e714
DW
2406 }
2407 break;
cdf1e714 2408 case WM_SYSCOMMAND:
61243a51 2409 bProcessed = HandleSysCommand(wParam, lParam);
cdf1e714
DW
2410 break;
2411
2412 case WM_COMMAND:
2413 {
2414 WORD id, cmd;
2415 WXHWND hwnd;
2416 UnpackCommand(wParam, lParam, &id, &hwnd, &cmd);
2417
61243a51 2418 bProcessed = HandleCommand(id, cmd, hwnd);
cdf1e714
DW
2419 }
2420 break;
2421
61243a51
DW
2422 //
2423 // For these messages we must return TRUE if process the message
2424 //
cdf1e714
DW
2425 case WM_DRAWITEM:
2426 case WM_MEASUREITEM:
2427 {
45bedfdd 2428 int nIdCtrl = (UINT)wParam;
45bedfdd 2429
61243a51 2430 if ( uMsg == WM_DRAWITEM )
cdf1e714 2431 {
45bedfdd 2432 bProcessed = OS2OnDrawItem(nIdCtrl,
cdf1e714
DW
2433 (WXDRAWITEMSTRUCT *)lParam);
2434 }
2435 else
2436 {
45bedfdd 2437 bProcessed = OS2OnMeasureItem(nIdCtrl,
cdf1e714
DW
2438 (WXMEASUREITEMSTRUCT *)lParam);
2439 }
2440
61243a51 2441 if ( bProcessed )
e604d44b 2442 mResult = (MRESULT)TRUE;
cdf1e714
DW
2443 }
2444 break;
2445
61243a51 2446 case WM_QUERYDLGCODE:
cdf1e714
DW
2447 if ( m_lDlgCode )
2448 {
e604d44b 2449 mResult = (MRESULT)m_lDlgCode;
61243a51 2450 bProcessed = TRUE;
cdf1e714 2451 }
61243a51 2452 //
cdf1e714 2453 //else: get the dlg code from the DefWindowProc()
61243a51 2454 //
cdf1e714
DW
2455 break;
2456
61243a51
DW
2457 //
2458 // In OS/2 PM all keyboard events are of the WM_CHAR type. Virtual key and key-up
2459 // and key-down events are obtained from the WM_CHAR params.
2460 //
2461 case WM_CHAR:
cdf1e714 2462 {
61243a51 2463 USHORT uKeyFlags = SHORT1FROMMP((MPARAM)wParam);
cdf1e714 2464
61243a51
DW
2465 if (uKeyFlags & KC_KEYUP)
2466 {
19193a2c 2467 //TODO: check if the cast to WXWORD isn't causing trouble
987da0d4 2468 bProcessed = HandleKeyUp((WXDWORD)wParam, lParam);
cdf1e714 2469 break;
61243a51
DW
2470 }
2471 else // keydown event
2472 {
2473 //
2474 // If this has been processed by an event handler,
2475 // return 0 now (we've handled it). DON't RETURN
2476 // we still need to process further
2477 //
987da0d4 2478 HandleKeyDown((WXDWORD)wParam, lParam);
61243a51 2479 if (uKeyFlags & KC_VIRTUALKEY)
cdf1e714 2480 {
61243a51
DW
2481 USHORT uVk = SHORT2FROMMP((MPARAM)lParam);
2482
2483 //
2484 // We consider these message "not interesting" to OnChar
2485 //
2486 if (uVk == VK_SHIFT || uVk == VK_CTRL )
2487 {
2488 bProcessed = TRUE;
2489 break;
2490 }
2491 switch(uVk)
2492 {
2493 //
2494 // Avoid duplicate messages to OnChar for these ASCII keys: they
2495 // will be translated by TranslateMessage() and received in WM_CHAR
2496 case VK_ESC:
2497 case VK_SPACE:
2498 case VK_ENTER:
2499 case VK_BACKSPACE:
2500 case VK_TAB:
2501 // But set processed to FALSE, not TRUE to still pass them to
2502 // the control's default window proc - otherwise built-in
2503 // keyboard handling won't work
2504 bProcessed = FALSE;
2505 break;
2506
2507 case VK_LEFT:
2508 case VK_RIGHT:
2509 case VK_DOWN:
2510 case VK_UP:
2511 default:
987da0d4 2512 bProcessed = HandleChar((WXDWORD)wParam, lParam);
61243a51
DW
2513 }
2514 break;
cdf1e714 2515 }
61243a51
DW
2516 else // WM_CHAR -- Always an ASCII character
2517 {
987da0d4 2518 bProcessed = HandleChar((WXDWORD)wParam, lParam, TRUE);
61243a51
DW
2519 break;
2520 }
2521 }
cdf1e714 2522 }
cdf1e714
DW
2523
2524 case WM_HSCROLL:
2525 case WM_VSCROLL:
2526 {
61243a51
DW
2527 WXWORD wCode;
2528 WXWORD wPos;
2529 WXHWND hWnd;
2530 UnpackScroll( wParam
2531 ,lParam
2532 ,&wCode
2533 ,&wPos
2534 ,&hWnd
2535 );
2536
2537 bProcessed = OS2OnScroll( uMsg == WM_HSCROLL ? wxHORIZONTAL
2538 : wxVERTICAL
2539 ,wCode
2540 ,wPos
2541 ,hWnd
2542 );
cdf1e714
DW
2543 }
2544 break;
2545
3c299c3a
DW
2546 case WM_CONTROL:
2547 switch(SHORT2FROMMP(wParam))
2548 {
2549 case SPBN_UPARROW:
2550 case SPBN_DOWNARROW:
2551 case SPBN_CHANGE:
2552 {
2553 char zVal[10];
2554 long lVal;
2555
2556 ::WinSendMsg( HWNDFROMMP(lParam)
2557 ,SPBM_QUERYVALUE
2558 ,&zVal
2559 ,MPFROM2SHORT( (USHORT)10
2560 ,(USHORT)SPBQ_UPDATEIFVALID
2561 )
2562 );
2563 lVal = atol(zVal);
2564 bProcessed = OS2OnScroll( wxVERTICAL
2565 ,(int)SHORT2FROMMP(wParam)
2566 ,(int)lVal
2567 ,HWNDFROMMP(lParam)
2568 );
2569 }
2570 break;
2571
2572 case SLN_SLIDERTRACK:
2573 {
2574 HWND hWnd = ::WinWindowFromID(GetHWND(), SHORT1FROMMP(wParam));
2575 wxWindowOS2* pChild = wxFindWinFromHandle(hWnd);
2576
2577 if (pChild->IsKindOf(CLASSINFO(wxSlider)))
2578 bProcessed = OS2OnScroll( wxVERTICAL
2579 ,(int)SHORT2FROMMP(wParam)
2580 ,(int)LONGFROMMP(lParam)
2581 ,hWnd
2582 );
2583 }
2584 break;
2585 }
2586 break;
2587
61243a51
DW
2588#if defined(__VISAGECPP__) && (__IBMCPP__ >= 400)
2589 case WM_CTLCOLORCHANGE:
cdf1e714 2590 {
e604d44b 2591 bProcessed = HandleCtlColor(&hBrush);
61243a51
DW
2592 }
2593 break;
2594#endif
8d854fa9
DW
2595 case WM_ERASEBACKGROUND:
2596 //
2597 // Returning TRUE to requestw PM to paint the window background
2598 // in SYSCLR_WINDOW. We don't really want that
2599 //
2600 bProcessed = HandleEraseBkgnd((WXHDC)(HPS)wParam);
2601 mResult = (MRESULT)(FALSE);
2602 break;
2603
cdf1e714
DW
2604 // the return value for this message is ignored
2605 case WM_SYSCOLORCHANGE:
61243a51 2606 bProcessed = HandleSysColorChange();
cdf1e714
DW
2607 break;
2608
61243a51
DW
2609 case WM_REALIZEPALETTE:
2610 bProcessed = HandlePaletteChanged();
cdf1e714
DW
2611 break;
2612
61243a51
DW
2613 // move all drag and drops to wxDrg
2614 case WM_ENDDRAG:
2615 bProcessed = HandleEndDrag(wParam);
cdf1e714
DW
2616 break;
2617
61243a51
DW
2618 case WM_INITDLG:
2619 bProcessed = HandleInitDialog((WXHWND)(HWND)wParam);
cdf1e714 2620
61243a51 2621 if ( bProcessed )
cdf1e714
DW
2622 {
2623 // we never set focus from here
e604d44b 2624 mResult = FALSE;
cdf1e714
DW
2625 }
2626 break;
2627
61243a51
DW
2628 // wxFrame specific message
2629 case WM_MINMAXFRAME:
f6bcfd97 2630 bProcessed = HandleGetMinMaxInfo((PSWP)wParam);
cdf1e714
DW
2631 break;
2632
61243a51
DW
2633 case WM_SYSVALUECHANGED:
2634 // TODO: do something
e604d44b 2635 mResult = (MRESULT)TRUE;
cdf1e714
DW
2636 break;
2637
61243a51
DW
2638 //
2639 // Comparable to WM_SETPOINTER for windows, only for just controls
2640 //
2641 case WM_CONTROLPOINTER:
2642 bProcessed = HandleSetCursor( SHORT1FROMMP(wParam) // Control ID
2643 ,(HWND)lParam // Cursor Handle
2644 );
2645 if (bProcessed )
cdf1e714 2646 {
61243a51
DW
2647 //
2648 // Returning TRUE stops the DefWindowProc() from further
cdf1e714
DW
2649 // processing this message - exactly what we need because we've
2650 // just set the cursor.
61243a51 2651 //
e604d44b 2652 mResult = (MRESULT)TRUE;
cdf1e714
DW
2653 }
2654 break;
2655 }
61243a51 2656 if (!bProcessed)
cdf1e714
DW
2657 {
2658#ifdef __WXDEBUG__
2659 wxLogTrace(wxTraceMessages, wxT("Forwarding %s to DefWindowProc."),
61243a51 2660 wxGetMessageName(uMsg));
cdf1e714 2661#endif // __WXDEBUG__
d08f23a7
DW
2662 if (IsKindOf(CLASSINFO(wxFrame)))
2663 mResult = ::WinDefWindowProc(m_hWnd, uMsg, wParam, lParam);
2664 else
2665 mResult = OS2DefWindowProc(uMsg, wParam, lParam);
cdf1e714 2666 }
e604d44b 2667 return mResult;
0367c1c0 2668} // end of wxWindowOS2::OS2WindowProc
cdf1e714 2669
61243a51 2670//
cdf1e714 2671// Dialog window proc
61243a51
DW
2672//
2673MRESULT wxDlgProc(
19193a2c 2674 HWND WXUNUSED(hWnd)
61243a51 2675, UINT uMsg
19193a2c
KB
2676, MPARAM WXUNUSED(wParam)
2677, MPARAM WXUNUSED(lParam))
cdf1e714 2678{
61243a51 2679 if (uMsg == WM_INITDLG)
cdf1e714 2680 {
61243a51
DW
2681 //
2682 // For this message, returning TRUE tells system to set focus to the
cdf1e714 2683 // first control in the dialog box
61243a51
DW
2684 //
2685 return (MRESULT)TRUE;
cdf1e714
DW
2686 }
2687 else
2688 {
61243a51
DW
2689 //
2690 // For all the other ones, FALSE means that we didn't process the
cdf1e714 2691 // message
61243a51
DW
2692 //
2693 return (MRESULT)0;
cdf1e714 2694 }
61243a51 2695} // end of wxDlgProc
cdf1e714 2696
61243a51
DW
2697wxWindow* wxFindWinFromHandle(
2698 WXHWND hWnd
2699)
cdf1e714 2700{
61243a51
DW
2701 wxNode* pNode = wxWinHandleList->Find((long)hWnd);
2702
2703 if (!pNode)
cdf1e714 2704 return NULL;
61243a51
DW
2705 return (wxWindow *)pNode->Data();
2706} // end of wxFindWinFromHandle
cdf1e714 2707
61243a51
DW
2708void wxAssociateWinWithHandle(
2709 HWND hWnd
19193a2c 2710, wxWindowOS2* pWin
61243a51 2711)
cdf1e714 2712{
61243a51
DW
2713 //
2714 // Adding NULL hWnd is (first) surely a result of an error and
cdf1e714 2715 // (secondly) breaks menu command processing
61243a51 2716 //
cdf1e714
DW
2717 wxCHECK_RET( hWnd != (HWND)NULL,
2718 wxT("attempt to add a NULL hWnd to window list ignored") );
2719
2720
61243a51
DW
2721 wxWindow* pOldWin = wxFindWinFromHandle((WXHWND) hWnd);
2722
2723 if (pOldWin && (pOldWin != pWin))
cdf1e714 2724 {
61243a51
DW
2725 wxString str(pWin->GetClassInfo()->GetClassName());
2726 wxLogError( "Bug! Found existing HWND %X for new window of class %s"
2727 ,(int)hWnd
2728 ,(const char*)str
2729 );
cdf1e714 2730 }
61243a51 2731 else if (!pOldWin)
cdf1e714 2732 {
61243a51
DW
2733 wxWinHandleList->Append( (long)hWnd
2734 ,pWin
2735 );
cdf1e714 2736 }
61243a51 2737} // end of wxAssociateWinWithHandle
cdf1e714 2738
61243a51 2739void wxRemoveHandleAssociation(
19193a2c 2740 wxWindowOS2* pWin
61243a51 2741)
cdf1e714 2742{
61243a51
DW
2743 wxWinHandleList->DeleteObject(pWin);
2744} // end of wxRemoveHandleAssociation
cdf1e714 2745
61243a51 2746//
cdf1e714
DW
2747// Default destroyer - override if you destroy it in some other way
2748// (e.g. with MDI child windows)
61243a51 2749//
0367c1c0 2750void wxWindowOS2::OS2DestroyWindow()
cdf1e714
DW
2751{
2752}
2753
0367c1c0 2754void wxWindowOS2::OS2DetachWindowMenu()
cdf1e714 2755{
19193a2c 2756#ifndef __WXUNIVERSAL__
61243a51 2757 if (m_hMenu)
cdf1e714 2758 {
61243a51 2759 HMENU hMenu = (HMENU)m_hMenu;
cdf1e714 2760
61243a51
DW
2761 int nN = (int)::WinSendMsg(hMenu, MM_QUERYITEMCOUNT, 0, 0);
2762 int i;
2763
2764 for (i = 0; i < nN; i++)
cdf1e714 2765 {
61243a51
DW
2766 wxChar zBuf[100];
2767 int nChars = (int)::WinSendMsg( hMenu
2768 ,MM_QUERYITEMTEXT
2769 ,MPFROM2SHORT(i, nN)
2770 ,zBuf
2771 );
2772 if (!nChars)
cdf1e714
DW
2773 {
2774 wxLogLastError(wxT("GetMenuString"));
cdf1e714
DW
2775 continue;
2776 }
2777
61243a51 2778 if (wxStrcmp(zBuf, wxT("&Window")) == 0)
cdf1e714 2779 {
61243a51 2780 ::WinSendMsg(hMenu, MM_DELETEITEM, MPFROM2SHORT(i, TRUE), 0);
cdf1e714
DW
2781 break;
2782 }
2783 }
2784 }
19193a2c 2785#endif // __WXUNIVERSAL__
0367c1c0 2786} // end of wxWindowOS2::OS2DetachWindowMenu
cdf1e714 2787
0367c1c0 2788bool wxWindowOS2::OS2Create(
f23208ca
DW
2789 WXHWND hParent
2790, PSZ zClass
61243a51 2791, const wxChar* zTitle
61243a51 2792, WXDWORD dwStyle
f23208ca
DW
2793, long lX
2794, long lY
2795, long lWidth
2796, long lHeight
2797, WXHWND hOwner
19193a2c 2798, WXHWND WXUNUSED(hZOrder)
f23208ca
DW
2799, unsigned long ulId
2800, void* pCtlData
2801, void* pPresParams
008089f6 2802, WXDWORD dwExStyle
61243a51
DW
2803)
2804{
914589c2
DW
2805 ERRORID vError;
2806 wxString sError;
5b3ed311 2807 long lX1 = 0L;
f23208ca 2808 long lY1 = 0L;
5b3ed311
DW
2809 long lWidth1 = 20L;
2810 long lHeight1 = 20L;
f23208ca 2811 int nControlId = 0;
51c1d535
DW
2812 int nNeedsubclass = 0;
2813 PCSZ pszClass = zClass;
cdf1e714 2814
61243a51 2815 //
cdf1e714
DW
2816 // Find parent's size, if it exists, to set up a possible default
2817 // panel size the size of the parent window
61243a51 2818 //
cb71578c
DW
2819 lX1 = lX;
2820 lY1 = lY;
f23208ca
DW
2821 if (lWidth > -1L)
2822 lWidth1 = lWidth;
2823 if (lHeight > -1L)
2824 lHeight1 = lHeight;
cdf1e714
DW
2825
2826 wxWndHook = this;
2827
f23208ca
DW
2828 //
2829 // check to see if the new window is a standard control
2830 //
2831 if ((ULONG)zClass == (ULONG)WC_BUTTON ||
2832 (ULONG)zClass == (ULONG)WC_COMBOBOX ||
2833 (ULONG)zClass == (ULONG)WC_CONTAINER ||
2834 (ULONG)zClass == (ULONG)WC_ENTRYFIELD ||
40bd6154 2835 (ULONG)zClass == (ULONG)WC_FRAME ||
f23208ca
DW
2836 (ULONG)zClass == (ULONG)WC_LISTBOX ||
2837 (ULONG)zClass == (ULONG)WC_MENU ||
2838 (ULONG)zClass == (ULONG)WC_NOTEBOOK ||
2839 (ULONG)zClass == (ULONG)WC_SCROLLBAR ||
2840 (ULONG)zClass == (ULONG)WC_SPINBUTTON ||
2841 (ULONG)zClass == (ULONG)WC_STATIC ||
2842 (ULONG)zClass == (ULONG)WC_TITLEBAR ||
2843 (ULONG)zClass == (ULONG)WC_VALUESET
2844 )
cdf1e714 2845 {
f23208ca 2846 nControlId = ulId;
cdf1e714 2847 }
51c1d535
DW
2848 else
2849 {
2850 // no standard controls
2851 if(wxString (wxT("wxFrameClass")) == wxString(zClass) )
2852 {
2853 pszClass = WC_FRAME;
2854 nNeedsubclass = 1;
2855 }
2856 else
2857 {
2858 nControlId = ulId;
2859 if(nControlId < 0)
2860 nControlId = FID_CLIENT;
2861 }
2862 }
cdf1e714 2863
19193a2c
KB
2864 HWND parent;
2865 if ( GetWindowStyleFlag() & wxPOPUP_WINDOW )
2866 {
2867 // popup windows should have desktop as parent because they shouldn't
2868 // be limited to the parents client area as child windows usually are
2869 parent = HWND_DESKTOP;
2870 }
2871 else if ( hParent )
2872 {
2873 parent = hParent;
2874 }
2875 else
2876 {
2877 // top level window
2878 parent = NULL;
2879 }
2880
f23208ca
DW
2881 //
2882 // We will either have a registered class via string name or a standard PM Class via a long
2883 //
19193a2c
KB
2884 m_hWnd = (WXHWND)::WinCreateWindow(parent, zClass,
2885 (PSZ)zTitle ? zTitle : wxT(""),
2886 dwStyle, lX1, lY1, lWidth, lHeight,
2887 hOwner, HWND_TOP, (ULONG)nControlId,
2888 pCtlData, pPresParams);
2889
f23208ca
DW
2890 if (!m_hWnd)
2891 {
2892 vError = ::WinGetLastError(vHabmain);
2893 sError = wxPMErrorToStr(vError);
2894 wxLogError("Can't create window of class %s!. Error: %s\n", zClass, sError);
2895 return FALSE;
cdf1e714 2896 }
008089f6 2897 m_dwExStyle = dwExStyle;
f6bcfd97 2898 ::WinSetWindowULong(m_hWnd, QWL_USER, (ULONG) this);
cdf1e714 2899 wxWndHook = NULL;
61243a51 2900
cdf1e714 2901#ifdef __WXDEBUG__
61243a51
DW
2902 wxNode* pNode = wxWinHandleList->Member(this);
2903
2904 if (pNode)
cdf1e714 2905 {
61243a51
DW
2906 HWND hWnd = (HWND)pNode->GetKeyInteger();
2907
2908 if (hWnd != (HWND)m_hWnd)
2909
cdf1e714
DW
2910 {
2911 wxLogError("A second HWND association is being added for the same window!");
2912 }
2913 }
2914#endif
61243a51
DW
2915 wxAssociateWinWithHandle((HWND)m_hWnd
2916 ,this
2917 );
cd212ee4 2918 //
e604d44b
DW
2919 // Now need to subclass window.
2920 //
51c1d535
DW
2921 if(!nNeedsubclass)
2922 {
2923 wxAssociateWinWithHandle((HWND)m_hWnd,this);
2924 }
2925 else
2926 {
2927 SubclassWin(GetHWND());
2928 }
cdf1e714 2929 return TRUE;
0367c1c0 2930} // end of wxWindowOS2::OS2Create
cdf1e714
DW
2931
2932// ===========================================================================
2933// OS2 PM message handlers
2934// ===========================================================================
2935
2936// ---------------------------------------------------------------------------
61243a51 2937// window creation/destruction
cdf1e714
DW
2938// ---------------------------------------------------------------------------
2939
0367c1c0 2940bool wxWindowOS2::HandleCreate(
19193a2c 2941 WXLPCREATESTRUCT WXUNUSED(vCs)
61243a51
DW
2942, bool* pbMayCreate
2943)
cdf1e714 2944{
19193a2c 2945 wxWindowCreateEvent vEvent((wxWindow*)this);
cdf1e714 2946
61243a51
DW
2947 (void)GetEventHandler()->ProcessEvent(vEvent);
2948 *pbMayCreate = TRUE;
cdf1e714 2949 return TRUE;
0367c1c0 2950} // end of wxWindowOS2::HandleCreate
cdf1e714 2951
0367c1c0 2952bool wxWindowOS2::HandleDestroy()
cdf1e714 2953{
19193a2c 2954 wxWindowDestroyEvent vEvent((wxWindow*)this);
cdf1e714 2955
61243a51
DW
2956 (void)GetEventHandler()->ProcessEvent(vEvent);
2957
2958 //
2959 // Delete our drop target if we've got one
2960 //
cdf1e714 2961#if wxUSE_DRAG_AND_DROP
61243a51 2962 if (m_dropTarget != NULL)
cdf1e714 2963 {
61243a51 2964 m_dropTarget->Revoke(m_hWnd);
cdf1e714
DW
2965 delete m_dropTarget;
2966 m_dropTarget = NULL;
2967 }
2968#endif // wxUSE_DRAG_AND_DROP
2969
61243a51 2970 //
cdf1e714 2971 // WM_DESTROY handled
61243a51 2972 //
cdf1e714 2973 return TRUE;
0367c1c0 2974} // end of wxWindowOS2::HandleDestroy
cdf1e714
DW
2975
2976// ---------------------------------------------------------------------------
2977// activation/focus
2978// ---------------------------------------------------------------------------
0367c1c0 2979void wxWindowOS2::OnSetFocus(
61243a51
DW
2980 wxFocusEvent& rEvent
2981)
cdf1e714 2982{
61243a51 2983 rEvent.Skip();
0367c1c0 2984} // end of wxWindowOS2::OnSetFocus
61243a51 2985
0367c1c0 2986bool wxWindowOS2::HandleActivate(
61243a51
DW
2987 int nState
2988, WXHWND WXUNUSED(hActivate)
2989)
2990{
2991 wxActivateEvent vEvent( wxEVT_ACTIVATE
2992 ,(bool)nState
2993 ,m_windowId
2994 );
2995 vEvent.SetEventObject(this);
2996 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 2997} // end of wxWindowOS2::HandleActivate
61243a51 2998
0367c1c0 2999bool wxWindowOS2::HandleSetFocus(
61243a51
DW
3000 WXHWND WXUNUSED(hWnd)
3001)
cdf1e714
DW
3002{
3003#if wxUSE_CARET
61243a51 3004 //
cdf1e714 3005 // Deal with caret
61243a51
DW
3006 //
3007 if (m_caret)
cdf1e714
DW
3008 {
3009 m_caret->OnSetFocus();
3010 }
3011#endif // wxUSE_CARET
3012
1bcfc0e1
DW
3013#if wxUSE_TEXTCTRL
3014 // If it's a wxTextCtrl don't send the event as it will be done
3015 // after the control gets to process it from EN_FOCUS handler
3016 if ( wxDynamicCastThis(wxTextCtrl) )
cdf1e714 3017 {
1bcfc0e1 3018 return FALSE;
cdf1e714 3019 }
1bcfc0e1 3020#endif // wxUSE_TEXTCTRL
cdf1e714 3021
61243a51 3022 wxFocusEvent vEvent(wxEVT_SET_FOCUS, m_windowId);
cdf1e714 3023
61243a51
DW
3024 vEvent.SetEventObject(this);
3025 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3026} // end of wxWindowOS2::HandleSetFocus
cdf1e714 3027
0367c1c0 3028bool wxWindowOS2::HandleKillFocus(
61243a51
DW
3029 WXHWND WXUNUSED(hWnd)
3030)
cdf1e714
DW
3031{
3032#if wxUSE_CARET
61243a51 3033 //
cdf1e714 3034 // Deal with caret
61243a51
DW
3035 //
3036 if (m_caret)
cdf1e714
DW
3037 {
3038 m_caret->OnKillFocus();
3039 }
3040#endif // wxUSE_CARET
3041
61243a51
DW
3042 wxFocusEvent vEvent( wxEVT_KILL_FOCUS
3043 ,m_windowId
3044 );
cdf1e714 3045
61243a51
DW
3046 vEvent.SetEventObject(this);
3047 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3048} // end of wxWindowOS2::HandleKillFocus
cdf1e714
DW
3049
3050// ---------------------------------------------------------------------------
3051// miscellaneous
3052// ---------------------------------------------------------------------------
3053
0367c1c0 3054bool wxWindowOS2::HandleShow(
61243a51 3055 bool bShow
19193a2c 3056, int WXUNUSED(nStatus)
61243a51 3057)
cdf1e714 3058{
19193a2c 3059 wxShowEvent vEvent(GetId(), bShow);
cdf1e714 3060
61243a51
DW
3061 vEvent.m_eventObject = this;
3062 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3063} // end of wxWindowOS2::HandleShow
cdf1e714 3064
0367c1c0 3065bool wxWindowOS2::HandleInitDialog(
61243a51
DW
3066 WXHWND WXUNUSED(hWndFocus)
3067)
cdf1e714 3068{
61243a51 3069 wxInitDialogEvent vEvent(GetId());
cdf1e714 3070
61243a51
DW
3071 vEvent.m_eventObject = this;
3072 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3073} // end of wxWindowOS2::HandleInitDialog
cdf1e714 3074
19193a2c 3075bool wxWindowOS2::HandleEndDrag(WXWPARAM WXUNUSED(wParam))
cdf1e714 3076{
61243a51 3077 // TODO: We'll handle drag and drop later
cdf1e714
DW
3078 return FALSE;
3079}
3080
0367c1c0 3081bool wxWindowOS2::HandleSetCursor(
19193a2c 3082 USHORT WXUNUSED(vId)
61243a51
DW
3083, WXHWND hPointer
3084)
cdf1e714 3085{
61243a51
DW
3086 //
3087 // Under OS/2 PM this allows the pointer to be changed
3088 // as it passes over a control
3089 //
3090 ::WinSetPointer(HWND_DESKTOP, (HPOINTER)hPointer);
3091 return TRUE;
0367c1c0 3092} // end of wxWindowOS2::HandleSetCursor
cdf1e714
DW
3093
3094// ---------------------------------------------------------------------------
3095// owner drawn stuff
3096// ---------------------------------------------------------------------------
0367c1c0 3097bool wxWindowOS2::OS2OnDrawItem(
61243a51
DW
3098 int vId
3099, WXDRAWITEMSTRUCT* pItemStruct
3100)
cdf1e714 3101{
19193a2c 3102#if wxUSE_OWNER_DRAWN
402e2f7c 3103 wxDC vDc;
61243a51 3104
19193a2c 3105#if wxUSE_MENUS_NATIVE
61243a51 3106 //
402e2f7c 3107 // Is it a menu item?
61243a51 3108 //
402e2f7c
DW
3109 if (vId == 0)
3110 {
5afb9458
DW
3111 ERRORID vError;
3112 wxString sError;
402e2f7c 3113 POWNERITEM pMeasureStruct = (POWNERITEM)pItemStruct;
23122f8c
DW
3114 wxFrame* pFrame = (wxFrame*)this;
3115 wxMenuItem* pMenuItem = pFrame->GetMenuBar()->FindItem(pMeasureStruct->idItem, pMeasureStruct->hItem);
402e2f7c 3116 HDC hDC = ::GpiQueryDevice(pMeasureStruct->hps);
23122f8c
DW
3117 wxRect vRect( pMeasureStruct->rclItem.xLeft
3118 ,pMeasureStruct->rclItem.yBottom
3119 ,pMeasureStruct->rclItem.xRight - pMeasureStruct->rclItem.xLeft
3120 ,pMeasureStruct->rclItem.yTop - pMeasureStruct->rclItem.yBottom
3121 );
402e2f7c
DW
3122 vDc.SetHDC( hDC
3123 ,FALSE
3124 );
3125 vDc.SetHPS(pMeasureStruct->hps);
5afb9458
DW
3126 //
3127 // Load the wxWindows Pallete and set to RGB mode
3128 //
3129 if (!::GpiCreateLogColorTable( pMeasureStruct->hps
3130 ,0L
3131 ,LCOLF_CONSECRGB
3132 ,0L
3133 ,(LONG)wxTheColourDatabase->m_nSize
3134 ,(PLONG)wxTheColourDatabase->m_palTable
3135 ))
3136 {
3137 vError = ::WinGetLastError(vHabmain);
3138 sError = wxPMErrorToStr(vError);
3139 wxLogError("Unable to set current color table. Error: %s\n", sError);
3140 }
3141 //
3142 // Set the color table to RGB mode
3143 //
3144 if (!::GpiCreateLogColorTable( pMeasureStruct->hps
3145 ,0L
3146 ,LCOLF_RGB
3147 ,0L
3148 ,0L
3149 ,NULL
3150 ))
3151 {
3152 vError = ::WinGetLastError(vHabmain);
3153 sError = wxPMErrorToStr(vError);
3154 wxLogError("Unable to set current color table. Error: %s\n", sError);
3155 }
cdf1e714 3156
23122f8c
DW
3157 wxCHECK( pMenuItem->IsKindOf(CLASSINFO(wxMenuItem)), FALSE );
3158
3159
3160 int eAction = 0;
3161 int eStatus = 0;
cdf1e714 3162
402e2f7c 3163 if (pMeasureStruct->fsAttribute == pMeasureStruct->fsAttributeOld)
23122f8c
DW
3164 {
3165 //
3166 // Entire Item needs to be redrawn (either it has reappeared from
3167 // behind another window or is being displayed for the first time
3168 //
402e2f7c 3169 eAction = wxOwnerDrawn::wxODDrawAll;
23122f8c
DW
3170
3171 if (pMeasureStruct->fsAttribute & MIA_HILITED)
3172 {
3173 //
3174 // If it is currently selected we let the system handle it
3175 //
3176 eStatus |= wxOwnerDrawn::wxODSelected;
3177 }
3178 if (pMeasureStruct->fsAttribute & MIA_CHECKED)
3179 {
3180 //
3181 // If it is currently checked we draw our own
3182 //
3183 eStatus |= wxOwnerDrawn::wxODChecked;
3184 pMeasureStruct->fsAttributeOld = pMeasureStruct->fsAttribute &= ~MIA_CHECKED;
3185 }
3186 if (pMeasureStruct->fsAttribute & MIA_DISABLED)
3187 {
3188 //
3189 // If it is currently disabled we let the system handle it
3190 //
3191 eStatus |= wxOwnerDrawn::wxODDisabled;
3192 }
3193 //
3194 // Don't really care about framed (indicationg focus) or NoDismiss
3195 //
3196 }
402e2f7c 3197 else
23122f8c 3198 {
5afb9458
DW
3199 if (pMeasureStruct->fsAttribute & MIA_HILITED)
3200 {
3201 eAction = wxOwnerDrawn::wxODDrawAll;
3202 eStatus |= wxOwnerDrawn::wxODSelected;
3203 //
3204 // Keep the system from trying to highlight with its bogus colors
3205 //
3206 pMeasureStruct->fsAttributeOld = pMeasureStruct->fsAttribute &= ~MIA_HILITED;
3207 }
3208 else if (!(pMeasureStruct->fsAttribute & MIA_HILITED))
3209 {
3210 eAction = wxOwnerDrawn::wxODDrawAll;
3211 eStatus = 0;
3212 //
3213 // Keep the system from trying to highlight with its bogus colors
3214 //
3215 pMeasureStruct->fsAttribute = pMeasureStruct->fsAttributeOld &= ~MIA_HILITED;
3216 }
3217 else
3218 {
3219 //
3220 // For now we don't care about anything else
3221 // just ignore the entire message!
3222 //
3223 return TRUE;
3224 }
23122f8c
DW
3225 }
3226 //
3227 // Now redraw the item
3228 //
45bedfdd
DW
3229 return(pMenuItem->OnDrawItem( vDc
3230 ,vRect
23122f8c
DW
3231 ,(wxOwnerDrawn::wxODAction)eAction
3232 ,(wxOwnerDrawn::wxODStatus)eStatus
45bedfdd 3233 ));
402e2f7c
DW
3234 //
3235 // leave the fsAttribute and fsOldAttribute unchanged. If different,
3236 // the system will do the highlight or fraeming or disabling for us,
3237 // otherwise, we'd have to do it ourselves.
61243a51 3238 //
cdf1e714 3239 }
19193a2c 3240#endif // wxUSE_MENUS_NATIVE
cdf1e714 3241
402e2f7c
DW
3242 wxWindow* pItem = FindItem(vId);
3243
3244 if (pItem && pItem->IsKindOf(CLASSINFO(wxControl)))
3245 {
3246 return ((wxControl *)pItem)->OS2OnDraw(pItemStruct);
3247 }
19193a2c
KB
3248#else
3249 vId = vId;
3250 pItemStruct = pItemStruct;
61243a51 3251#endif
402e2f7c 3252 return FALSE;
0367c1c0 3253} // end of wxWindowOS2::OS2OnDrawItem
cdf1e714 3254
0367c1c0 3255bool wxWindowOS2::OS2OnMeasureItem(
402e2f7c
DW
3256 int lId
3257, WXMEASUREITEMSTRUCT* pItemStruct
3258)
0e320a79 3259{
19193a2c 3260#if wxUSE_OWNER_DRAWN
402e2f7c
DW
3261 //
3262 // Is it a menu item?
3263 //
45bedfdd 3264 if (lId == 65536) // I really don't like this...has to be a better indicator
cdf1e714 3265 {
4049cc1c 3266 if (IsKindOf(CLASSINFO(wxFrame))) // we'll assume if Frame then a menu
45bedfdd 3267 {
4049cc1c
DW
3268 size_t nWidth;
3269 size_t nHeight;
3270 POWNERITEM pMeasureStruct = (POWNERITEM)pItemStruct;
45bedfdd
DW
3271 wxFrame* pFrame = (wxFrame*)this;
3272 wxMenuItem* pMenuItem = pFrame->GetMenuBar()->FindItem(pMeasureStruct->idItem, pMeasureStruct->hItem);
cdf1e714 3273
45bedfdd 3274 wxCHECK( pMenuItem->IsKindOf(CLASSINFO(wxMenuItem)), FALSE );
4049cc1c
DW
3275 nWidth = 0L;
3276 nHeight = 0L;
3277 if (pMenuItem->OnMeasureItem( &nWidth
3278 ,&nHeight
3279 ))
3280 {
3281 pMeasureStruct->rclItem.xRight = nWidth;
3282 pMeasureStruct->rclItem.xLeft = 0L;
3283 pMeasureStruct->rclItem.yTop = nHeight;
3284 pMeasureStruct->rclItem.yBottom = 0L;
3285 return TRUE;
3286 }
3287 return FALSE;
45bedfdd 3288 }
cdf1e714 3289 }
f15b4952 3290 wxWindow* pItem = FindItem(lId);
cdf1e714 3291
402e2f7c 3292 if (pItem && pItem->IsKindOf(CLASSINFO(wxControl)))
cdf1e714 3293 {
402e2f7c 3294 return ((wxControl *)pItem)->OS2OnMeasure(pItemStruct);
cdf1e714 3295 }
19193a2c
KB
3296#else
3297 lId = lId;
3298 pItemStruct = pItemStruct;
3299#endif // wxUSE_OWNER_DRAWN
cdf1e714 3300 return FALSE;
0e320a79
DW
3301}
3302
cdf1e714
DW
3303// ---------------------------------------------------------------------------
3304// colours and palettes
3305// ---------------------------------------------------------------------------
849949b1 3306
0367c1c0 3307bool wxWindowOS2::HandleSysColorChange()
0e320a79 3308{
61243a51 3309 wxSysColourChangedEvent vEvent;
cdf1e714 3310
61243a51
DW
3311 vEvent.SetEventObject(this);
3312 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3313} // end of wxWindowOS2::HandleSysColorChange
0e320a79 3314
0367c1c0 3315bool wxWindowOS2::HandleCtlColor(
19193a2c 3316 WXHBRUSH* WXUNUSED(phBrush)
61243a51 3317)
0e320a79 3318{
61243a51
DW
3319 //
3320 // Not much provided with message. So not sure I can do anything with it
3321 //
3322 return TRUE;
0367c1c0 3323} // end of wxWindowOS2::HandleCtlColor
cdf1e714 3324
0e320a79 3325
cdf1e714 3326// Define for each class of dialog and control
19193a2c
KB
3327WXHBRUSH wxWindowOS2::OnCtlColor(WXHDC WXUNUSED(hDC),
3328 WXHWND WXUNUSED(hWnd),
3329 WXUINT WXUNUSED(nCtlColor),
3330 WXUINT WXUNUSED(message),
3331 WXWPARAM WXUNUSED(wParam),
3332 WXLPARAM WXUNUSED(lParam))
0e320a79 3333{
cdf1e714 3334 return (WXHBRUSH)0;
0e320a79
DW
3335}
3336
0367c1c0 3337bool wxWindowOS2::HandlePaletteChanged()
0e320a79 3338{
61243a51
DW
3339 // need to set this to something first
3340 WXHWND hWndPalChange = NULLHANDLE;
cdf1e714 3341
61243a51
DW
3342 wxPaletteChangedEvent vEvent(GetId());
3343
3344 vEvent.SetEventObject(this);
3345 vEvent.SetChangedWindow(wxFindWinFromHandle(hWndPalChange));
0e320a79 3346
61243a51 3347 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3348} // end of wxWindowOS2::HandlePaletteChanged
61243a51 3349
61243a51 3350//
cdf1e714 3351// Responds to colour changes: passes event on to children.
61243a51 3352//
0367c1c0 3353void wxWindowOS2::OnSysColourChanged(
61243a51
DW
3354 wxSysColourChangedEvent& rEvent
3355)
0e320a79 3356{
61243a51
DW
3357 wxNode* pNode = GetChildren().First();
3358
3359 while (pNode)
cdf1e714 3360 {
61243a51 3361 //
cdf1e714 3362 // Only propagate to non-top-level windows
61243a51
DW
3363 //
3364 wxWindow* pWin = (wxWindow *)pNode->Data();
3365
3366 if (pWin->GetParent())
cdf1e714 3367 {
61243a51 3368 wxSysColourChangedEvent vEvent;
cdf1e714 3369
61243a51
DW
3370 rEvent.m_eventObject = pWin;
3371 pWin->GetEventHandler()->ProcessEvent(vEvent);
3372 }
3373 pNode = pNode->Next();
cdf1e714 3374 }
0367c1c0 3375} // end of wxWindowOS2::OnSysColourChanged
0e320a79 3376
cdf1e714
DW
3377// ---------------------------------------------------------------------------
3378// painting
3379// ---------------------------------------------------------------------------
3380
0367c1c0 3381bool wxWindowOS2::HandlePaint()
0e320a79 3382{
19193a2c
KB
3383 HRGN hRgn;
3384 wxPaintEvent vEvent(m_windowId);
40bd6154
DW
3385 HPS hPS;
3386 RECTL vRect;
19193a2c
KB
3387 bool bProcessed;
3388
3389 // Create empty region
3390 // TODO: get HPS somewhere else if possible
3391 hPS = ::WinGetPS(GetHwnd());
3392 hRgn = ::GpiCreateRegion(hPS, 0, NULL);
61243a51 3393
19193a2c 3394 if (::WinQueryUpdateRegion(GetHwnd(), hRgn) == RGN_ERROR)
61243a51
DW
3395 {
3396 wxLogLastError("CreateRectRgn");
3397 return FALSE;
3398 }
7e99520b 3399
19193a2c
KB
3400 m_updateRegion = wxRegion(hRgn, hPS);
3401
61243a51 3402 vEvent.SetEventObject(this);
19193a2c
KB
3403 bProcessed = GetEventHandler()->ProcessEvent(vEvent);
3404
19193a2c 3405 return GetEventHandler()->ProcessEvent(vEvent); //bProcessed;
0367c1c0 3406} // end of wxWindowOS2::HandlePaint
0e320a79 3407
0367c1c0 3408bool wxWindowOS2::HandleEraseBkgnd(
8d854fa9
DW
3409 WXHDC hDC
3410)
0e320a79 3411{
8d854fa9 3412 SWP vSwp;
19193a2c 3413 bool rc;
8d854fa9
DW
3414
3415 ::WinQueryWindowPos(GetHwnd(), &vSwp);
3416 if (vSwp.fl & SWP_MINIMIZE)
61243a51 3417 return TRUE;
cdf1e714 3418
8d854fa9
DW
3419 wxDC vDC;
3420
3421 vDC.m_hPS = (HPS)hDC; // this is really a PS
19193a2c 3422 vDC.SetWindow((wxWindow*)this);
8d854fa9 3423 vDC.BeginDrawing();
cdf1e714 3424
8d854fa9
DW
3425 wxEraseEvent vEvent(m_windowId, &vDC);
3426
3427 vEvent.SetEventObject(this);
cdf1e714 3428
19193a2c 3429 rc = GetEventHandler()->ProcessEvent(vEvent);
cdf1e714 3430
8d854fa9
DW
3431 vDC.EndDrawing();
3432 vDC.m_hPS = NULLHANDLE;
61243a51 3433 return TRUE;
0367c1c0 3434} // end of wxWindowOS2::HandleEraseBkgnd
cdf1e714 3435
0367c1c0 3436void wxWindowOS2::OnEraseBackground(
61243a51
DW
3437 wxEraseEvent& rEvent
3438)
0e320a79 3439{
8d854fa9
DW
3440 RECTL vRect;
3441 HPS hPS = rEvent.m_dc->m_hPS;
19193a2c
KB
3442 APIRET rc;
3443 LONG lColor = m_backgroundColour.GetPixel();
8d854fa9 3444
19193a2c
KB
3445 rc = ::WinQueryWindowRect(GetHwnd(), &vRect);
3446 rc = ::WinFillRect(hPS, &vRect, lColor);
0367c1c0 3447} // end of wxWindowOS2::OnEraseBackground
0e320a79 3448
cdf1e714
DW
3449// ---------------------------------------------------------------------------
3450// moving and resizing
3451// ---------------------------------------------------------------------------
0e320a79 3452
0367c1c0 3453bool wxWindowOS2::HandleMinimize()
0e320a79 3454{
61243a51 3455 wxIconizeEvent vEvent(m_windowId);
cdf1e714 3456
61243a51
DW
3457 vEvent.SetEventObject(this);
3458 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3459} // end of wxWindowOS2::HandleMinimize
0e320a79 3460
0367c1c0 3461bool wxWindowOS2::HandleMaximize()
0e320a79 3462{
61243a51 3463 wxMaximizeEvent vEvent(m_windowId);
cdf1e714 3464
61243a51
DW
3465 vEvent.SetEventObject(this);
3466 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3467} // end of wxWindowOS2::HandleMaximize
0e320a79 3468
0367c1c0 3469bool wxWindowOS2::HandleMove(
61243a51
DW
3470 int nX
3471, int nY
3472)
0e320a79 3473{
19193a2c 3474 wxMoveEvent vEvent(wxPoint(nX, nY), m_windowId);
cdf1e714 3475
61243a51
DW
3476 vEvent.SetEventObject(this);
3477 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3478} // end of wxWindowOS2::HandleMove
0e320a79 3479
0367c1c0 3480bool wxWindowOS2::HandleSize(
61243a51
DW
3481 int nWidth
3482, int nHeight
3483, WXUINT WXUNUSED(nFlag)
3484)
0e320a79 3485{
19193a2c 3486 wxSizeEvent vEvent(wxSize(nWidth, nHeight), m_windowId);
cdf1e714 3487
61243a51
DW
3488 vEvent.SetEventObject(this);
3489 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3490} // end of wxWindowOS2::HandleSize
0e320a79 3491
0367c1c0 3492bool wxWindowOS2::HandleGetMinMaxInfo(
61243a51
DW
3493 PSWP pSwp
3494)
0e320a79 3495{
61243a51
DW
3496 bool bRc = FALSE;
3497 POINTL vPoint;
cdf1e714 3498
61243a51 3499 switch(pSwp->fl)
cdf1e714 3500 {
61243a51
DW
3501 case SWP_MAXIMIZE:
3502 ::WinGetMaxPosition(GetHwnd(), pSwp);
3503 m_maxWidth = pSwp->cx;
3504 m_maxHeight = pSwp->cy;
3505 break;
cdf1e714 3506
61243a51
DW
3507 case SWP_MINIMIZE:
3508 ::WinGetMinPosition(GetHwnd(), pSwp, &vPoint);
3509 m_minWidth = pSwp->cx;
3510 m_minHeight = pSwp->cy;
3511 break;
cdf1e714 3512
61243a51
DW
3513 default:
3514 return FALSE;
cdf1e714 3515 }
61243a51 3516 return TRUE;
0367c1c0 3517} // end of wxWindowOS2::HandleGetMinMaxInfo
0e320a79 3518
cdf1e714
DW
3519// ---------------------------------------------------------------------------
3520// command messages
3521// ---------------------------------------------------------------------------
0367c1c0 3522bool wxWindowOS2::HandleCommand(
61243a51
DW
3523 WXWORD wId
3524, WXWORD wCmd
3525, WXHWND hControl
3526)
0e320a79 3527{
19193a2c 3528#if wxUSE_MENUS_NATIVE
61243a51 3529 if (wxCurrentPopupMenu)
0e320a79 3530 {
61243a51 3531 wxMenu* pPopupMenu = wxCurrentPopupMenu;
0e320a79 3532
61243a51
DW
3533 wxCurrentPopupMenu = NULL;
3534 return pPopupMenu->OS2Command(wCmd, wId);
cdf1e714 3535 }
19193a2c 3536#endif // wxUSE_MENUS_NATIVE
0e320a79 3537
61243a51
DW
3538 wxWindow* pWin = FindItem(wId);
3539
3540 if (!pWin)
cdf1e714 3541 {
61243a51 3542 pWin = wxFindWinFromHandle(hControl);
0e320a79 3543 }
cdf1e714 3544
61243a51 3545 if (pWin)
19193a2c
KB
3546 return pWin->OS2Command(wCmd, wId);
3547
cdf1e714 3548 return FALSE;
0367c1c0 3549} // end of wxWindowOS2::HandleCommand
de44a9f0 3550
0367c1c0 3551bool wxWindowOS2::HandleSysCommand(
61243a51 3552 WXWPARAM wParam
19193a2c 3553, WXLPARAM WXUNUSED(lParam)
61243a51 3554)
0e320a79 3555{
61243a51
DW
3556 //
3557 // 4 bits are reserved
3558 //
3559 switch (SHORT1FROMMP(wParam))
3560 {
3561 case SC_MAXIMIZE:
3562 return HandleMaximize();
3563
3564 case SC_MINIMIZE:
3565 return HandleMinimize();
3566 }
cdf1e714 3567 return FALSE;
0367c1c0 3568} // end of wxWindowOS2::HandleSysCommand
cdf1e714
DW
3569
3570// ---------------------------------------------------------------------------
3571// mouse events
3572// ---------------------------------------------------------------------------
19193a2c 3573//TODO!!! check against MSW
0367c1c0 3574void wxWindowOS2::InitMouseEvent(
61243a51
DW
3575 wxMouseEvent& rEvent
3576, int nX
3577, int nY
3578, WXUINT uFlags
3579)
cdf1e714 3580{
61243a51
DW
3581 rEvent.m_x = nX;
3582 rEvent.m_y = nY;
3583 rEvent.m_shiftDown = ((uFlags & VK_SHIFT) != 0);
3584 rEvent.m_controlDown = ((uFlags & VK_CTRL) != 0);
3585 rEvent.m_leftDown = ((uFlags & VK_BUTTON1) != 0);
3586 rEvent.m_middleDown = ((uFlags & VK_BUTTON3) != 0);
3587 rEvent.m_rightDown = ((uFlags & VK_BUTTON2) != 0);
3588 rEvent.SetTimestamp(s_currentMsg.time);
3589 rEvent.m_eventObject = this;
cdf1e714
DW
3590
3591#if wxUSE_MOUSEEVENT_HACK
61243a51
DW
3592 m_lastMouseX = nX;
3593 m_lastMouseY = nY;
3594 m_lastMouseEvent = rEvent.GetEventType();
cdf1e714 3595#endif // wxUSE_MOUSEEVENT_HACK
0367c1c0 3596} // end of wxWindowOS2::InitMouseEvent
0e320a79 3597
0367c1c0 3598bool wxWindowOS2::HandleMouseEvent(
61243a51
DW
3599 WXUINT uMsg
3600, int nX
3601, int nY
3602, WXUINT uFlags
3603)
0e320a79 3604{
61243a51
DW
3605 //
3606 // The mouse events take consecutive IDs from WM_MOUSEFIRST to
cdf1e714
DW
3607 // WM_MOUSELAST, so it's enough to substract WM_MOUSEMOVE == WM_MOUSEFIRST
3608 // from the message id and take the value in the table to get wxWin event
3609 // id
61243a51 3610 //
cdf1e714
DW
3611 static const wxEventType eventsMouse[] =
3612 {
3613 wxEVT_MOTION,
3614 wxEVT_LEFT_DOWN,
3615 wxEVT_LEFT_UP,
3616 wxEVT_LEFT_DCLICK,
3617 wxEVT_RIGHT_DOWN,
3618 wxEVT_RIGHT_UP,
3619 wxEVT_RIGHT_DCLICK,
3620 wxEVT_MIDDLE_DOWN,
3621 wxEVT_MIDDLE_UP,
3622 wxEVT_MIDDLE_DCLICK
3623 };
3624
61243a51 3625 wxMouseEvent vEvent(eventsMouse[uMsg - WM_MOUSEMOVE]);
cdf1e714 3626
61243a51
DW
3627 InitMouseEvent( vEvent
3628 ,nX
3629 ,nY
3630 ,uFlags
3631 );
0e320a79 3632
61243a51 3633 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3634} // end of wxWindowOS2::HandleMouseEvent
61243a51 3635
0367c1c0 3636bool wxWindowOS2::HandleMouseMove(
61243a51
DW
3637 int nX
3638, int nY
3639, WXUINT uFlags
3640)
0e320a79 3641{
61243a51 3642 if (!m_bMouseInWindow)
cdf1e714 3643 {
61243a51 3644 //
cdf1e714 3645 // Generate an ENTER event
61243a51 3646 //
776d87d5 3647 m_bMouseInWindow = TRUE;
cdf1e714 3648
61243a51 3649 wxMouseEvent vEvent(wxEVT_ENTER_WINDOW);
cdf1e714 3650
61243a51
DW
3651 InitMouseEvent( vEvent
3652 ,nX
3653 ,nY
3654 ,uFlags
3655 );
cdf1e714 3656
61243a51 3657 (void)GetEventHandler()->ProcessEvent(vEvent);
cdf1e714 3658 }
61243a51
DW
3659 return HandleMouseEvent( WM_MOUSEMOVE
3660 ,nX
3661 ,nY
3662 ,uFlags
3663 );
0367c1c0 3664} // end of wxWindowOS2::HandleMouseMove
0e320a79 3665
cdf1e714
DW
3666// ---------------------------------------------------------------------------
3667// keyboard handling
3668// ---------------------------------------------------------------------------
3669
61243a51
DW
3670//
3671// Create the key event of the given type for the given key - used by
3672// HandleChar and HandleKeyDown/Up
3673//
0367c1c0 3674wxKeyEvent wxWindowOS2::CreateKeyEvent(
61243a51
DW
3675 wxEventType eType
3676, int nId
3677, WXLPARAM lParam
3678) const
3679{
3680 wxKeyEvent vEvent(eType);
3681
3682 vEvent.SetId(GetId());
3683 vEvent.m_shiftDown = IsShiftDown();
3684 vEvent.m_controlDown = IsCtrlDown();
3685 vEvent.m_altDown = (HIWORD(lParam) & KC_ALT) == KC_ALT;
3686
3687 vEvent.m_eventObject = (wxWindow *)this; // const_cast
3688 vEvent.m_keyCode = nId;
3689 vEvent.SetTimestamp(s_currentMsg.time);
3690
3691 //
3692 // Translate the position to client coords
3693 //
3694 POINTL vPoint;
3695 RECTL vRect;
3696
3697 ::WinQueryPointerPos(HWND_DESKTOP, &vPoint);
3698 ::WinQueryWindowRect( GetHwnd()
3699 ,&vRect
3700 );
3701
3702 vPoint.x -= vRect.xLeft;
3703 vPoint.y -= vRect.yBottom;
3704
3705 vEvent.m_x = vPoint.x;
3706 vEvent.m_y = vPoint.y;
3707
3708 return vEvent;
0367c1c0 3709} // end of wxWindowOS2::CreateKeyEvent
61243a51
DW
3710
3711//
cdf1e714
DW
3712// isASCII is TRUE only when we're called from WM_CHAR handler and not from
3713// WM_KEYDOWN one
61243a51 3714//
0367c1c0 3715bool wxWindowOS2::HandleChar(
987da0d4 3716 WXDWORD wParam
61243a51
DW
3717, WXLPARAM lParam
3718, bool isASCII
3719)
86de7616 3720{
61243a51
DW
3721 bool bCtrlDown = FALSE;
3722 int vId;
3723
3724 if (isASCII)
3725 {
3726 //
3727 // If 1 -> 26, translate to CTRL plus a letter.
3728 //
3729 vId = wParam;
3730 if ((vId > 0) && (vId < 27))
3731 {
3732 switch (vId)
3733 {
3734 case 13:
3735 vId = WXK_RETURN;
3736 break;
3737
3738 case 8:
3739 vId = WXK_BACK;
3740 break;
3741
3742 case 9:
3743 vId = WXK_TAB;
3744 break;
3745
3746 default:
3747 bCtrlDown = TRUE;
3748 vId = vId + 96;
3749 }
3750 }
3751 }
3752 else if ( (vId = wxCharCodeOS2ToWX(wParam)) == 0)
3753 {
3754 //
3755 // It's ASCII and will be processed here only when called from
3756 // WM_CHAR (i.e. when isASCII = TRUE), don't process it now
3757 //
3758 vId = -1;
3759 }
3760
3761 if (vId != -1)
3762 {
3763 wxKeyEvent vEvent(CreateKeyEvent( wxEVT_CHAR
3764 ,vId
3765 ,lParam
3766 ));
3767
3768 if (bCtrlDown)
3769 {
3770 vEvent.m_controlDown = TRUE;
3771 }
3772
3773 if (GetEventHandler()->ProcessEvent(vEvent))
3774 return TRUE;
3775 }
3776 return FALSE;
cdf1e714 3777}
86de7616 3778
0367c1c0 3779bool wxWindowOS2::HandleKeyDown(
61243a51
DW
3780 WXWORD wParam
3781, WXLPARAM lParam
3782)
cdf1e714 3783{
61243a51 3784 int nId = wxCharCodeOS2ToWX(wParam);
86de7616 3785
61243a51
DW
3786 if (!nId)
3787 {
3788 //
3789 // Normal ASCII char
3790 //
3791 nId = wParam;
3792 }
3793
3794 if (nId != -1)
3795 {
3796 wxKeyEvent vEvent(CreateKeyEvent( wxEVT_KEY_DOWN
3797 ,nId
3798 ,lParam
3799 ));
3800
3801 if (GetEventHandler()->ProcessEvent(vEvent))
3802 {
3803 return TRUE;
3804 }
3805 }
3806 return FALSE;
0367c1c0 3807} // end of wxWindowOS2::HandleKeyDown
61243a51 3808
0367c1c0 3809bool wxWindowOS2::HandleKeyUp(
987da0d4 3810 WXDWORD wParam
61243a51
DW
3811, WXLPARAM lParam
3812)
86de7616 3813{
61243a51
DW
3814 int nId = wxCharCodeOS2ToWX(wParam);
3815
3816 if (!nId)
3817 {
3818 //
3819 // Normal ASCII char
3820 //
3821 nId = wParam;
3822 }
3823
3824 if (nId != -1)
3825 {
3826 wxKeyEvent vEvent(CreateKeyEvent( wxEVT_KEY_UP
3827 ,nId
3828 ,lParam
3829 ));
3830
3831 if (GetEventHandler()->ProcessEvent(vEvent))
3832 return TRUE;
3833 }
3834 return FALSE;
0367c1c0 3835} // end of wxWindowOS2::HandleKeyUp
86de7616 3836
cdf1e714
DW
3837// ---------------------------------------------------------------------------
3838// joystick
3839// ---------------------------------------------------------------------------
86de7616 3840
cdf1e714
DW
3841// ---------------------------------------------------------------------------
3842// scrolling
3843// ---------------------------------------------------------------------------
3844
0367c1c0 3845bool wxWindowOS2::OS2OnScroll(
61243a51
DW
3846 int nOrientation
3847, WXWORD wParam
3848, WXWORD wPos
3849, WXHWND hControl
3850)
cdf1e714 3851{
61243a51 3852 if (hControl)
86de7616 3853 {
61243a51
DW
3854 wxWindow* pChild = wxFindWinFromHandle(hControl);
3855
3856 if (pChild )
3857 return pChild->OS2OnScroll( nOrientation
3858 ,wParam
3859 ,wPos
3860 ,hControl
3861 );
cdf1e714 3862 }
86de7616 3863
61243a51
DW
3864 wxScrollWinEvent vEvent;
3865
3866 vEvent.SetPosition(wPos);
3867 vEvent.SetOrientation(nOrientation);
3868 vEvent.m_eventObject = this;
3869
3870 switch (wParam)
3871 {
3872 case SB_LINEUP:
3873 vEvent.m_eventType = wxEVT_SCROLLWIN_LINEUP;
3874 break;
3875
3876 case SB_LINEDOWN:
3877 vEvent.m_eventType = wxEVT_SCROLLWIN_LINEDOWN;
3878 break;
3879
3880 case SB_PAGEUP:
3881 vEvent.m_eventType = wxEVT_SCROLLWIN_PAGEUP;
3882 break;
3883
3884 case SB_PAGEDOWN:
3885 vEvent.m_eventType = wxEVT_SCROLLWIN_PAGEDOWN;
3886 break;
3887
3888 case SB_SLIDERPOSITION:
3889 vEvent.m_eventType = wxEVT_SCROLLWIN_THUMBRELEASE;
3890 break;
3891
3892 case SB_SLIDERTRACK:
3893 vEvent.m_eventType = wxEVT_SCROLLWIN_THUMBTRACK;
3894 break;
3895
3896 default:
3897 return FALSE;
3898 }
3899 return GetEventHandler()->ProcessEvent(vEvent);
0367c1c0 3900} // end of wxWindowOS2::OS2OnScroll
86de7616 3901
cdf1e714
DW
3902// ===========================================================================
3903// global functions
3904// ===========================================================================
3905
61243a51 3906void wxGetCharSize(
987da0d4
DW
3907 WXHWND hWnd
3908, int* pX
3909, int* pY
19193a2c 3910,wxFont* WXUNUSED(pTheFont)
61243a51 3911)
cdf1e714 3912{
987da0d4
DW
3913 FONTMETRICS vFM;
3914 HPS hPS;
3915 BOOL rc;
3916
3917 hPS =::WinGetPS(hWnd);
3918
3919 rc = ::GpiQueryFontMetrics(hPS, sizeof(FONTMETRICS), &vFM);
3920 if (rc)
3921 {
3922 if (pX)
3923 *pX = vFM.lAveCharWidth;
3924 if (pY)
3925 *pY = vFM.lEmHeight + vFM.lExternalLeading;
3926 }
3927 else
3928 {
3929 if (pX)
3930 *pX = 10;
3931 if (pY)
3932 *pY = 15;
3933 }
3934 ::WinReleasePS(hPS);
61243a51 3935} // end of wxGetCharSize
cdf1e714 3936
61243a51 3937//
cdf1e714
DW
3938// Returns 0 if was a normal ASCII value, not a special key. This indicates that
3939// the key should be ignored by WM_KEYDOWN and processed by WM_CHAR instead.
61243a51
DW
3940//
3941int wxCharCodeOS2ToWX(
3942 int nKeySym
3943)
cdf1e714 3944{
61243a51
DW
3945 int nId = 0;
3946
3947 switch (nKeySym)
3948 {
3949 case VK_BACKTAB: nId = WXK_BACK; break;
3950 case VK_TAB: nId = WXK_TAB; break;
3951 case VK_CLEAR: nId = WXK_CLEAR; break;
3952 case VK_ENTER: nId = WXK_RETURN; break;
3953 case VK_SHIFT: nId = WXK_SHIFT; break;
3954 case VK_CTRL: nId = WXK_CONTROL; break;
3955 case VK_PAUSE: nId = WXK_PAUSE; break;
3956 case VK_SPACE: nId = WXK_SPACE; break;
3957 case VK_ESC: nId = WXK_ESCAPE; break;
3958 case VK_END: nId = WXK_END; break;
3959 case VK_HOME : nId = WXK_HOME; break;
3960 case VK_LEFT : nId = WXK_LEFT; break;
3961 case VK_UP: nId = WXK_UP; break;
3962 case VK_RIGHT: nId = WXK_RIGHT; break;
3963 case VK_DOWN : nId = WXK_DOWN; break;
3964 case VK_PRINTSCRN: nId = WXK_PRINT; break;
3965 case VK_INSERT: nId = WXK_INSERT; break;
3966 case VK_DELETE: nId = WXK_DELETE; break;
3967 case VK_F1: nId = WXK_F1; break;
3968 case VK_F2: nId = WXK_F2; break;
3969 case VK_F3: nId = WXK_F3; break;
3970 case VK_F4: nId = WXK_F4; break;
3971 case VK_F5: nId = WXK_F5; break;
3972 case VK_F6: nId = WXK_F6; break;
3973 case VK_F7: nId = WXK_F7; break;
3974 case VK_F8: nId = WXK_F8; break;
3975 case VK_F9: nId = WXK_F9; break;
3976 case VK_F10: nId = WXK_F10; break;
3977 case VK_F11: nId = WXK_F11; break;
3978 case VK_F12: nId = WXK_F12; break;
3979 case VK_F13: nId = WXK_F13; break;
3980 case VK_F14: nId = WXK_F14; break;
3981 case VK_F15: nId = WXK_F15; break;
3982 case VK_F16: nId = WXK_F16; break;
3983 case VK_F17: nId = WXK_F17; break;
3984 case VK_F18: nId = WXK_F18; break;
3985 case VK_F19: nId = WXK_F19; break;
3986 case VK_F20: nId = WXK_F20; break;
3987 case VK_F21: nId = WXK_F21; break;
3988 case VK_F22: nId = WXK_F22; break;
3989 case VK_F23: nId = WXK_F23; break;
3990 case VK_F24: nId = WXK_F24; break;
3991 case VK_NUMLOCK: nId = WXK_NUMLOCK; break;
3992 case VK_SCRLLOCK: nId = WXK_SCROLL; break;
3993 default:
86de7616 3994 {
cdf1e714 3995 return 0;
86de7616
DW
3996 }
3997 }
61243a51
DW
3998 return nId;
3999} // end of wxCharCodeOS2ToWX
86de7616 4000
61243a51
DW
4001int wxCharCodeWXToOS2(
4002 int nId
4003, bool* bIsVirtual
4004)
86de7616 4005{
61243a51
DW
4006 int nKeySym = 0;
4007
4008 *bIsVirtual = TRUE;
4009 switch (nId)
4010 {
4011 case WXK_CLEAR: nKeySym = VK_CLEAR; break;
4012 case WXK_SHIFT: nKeySym = VK_SHIFT; break;
4013 case WXK_CONTROL: nKeySym = VK_CTRL; break;
4014 case WXK_PAUSE: nKeySym = VK_PAUSE; break;
4015 case WXK_END: nKeySym = VK_END; break;
4016 case WXK_HOME : nKeySym = VK_HOME; break;
4017 case WXK_LEFT : nKeySym = VK_LEFT; break;
4018 case WXK_UP: nKeySym = VK_UP; break;
4019 case WXK_RIGHT: nKeySym = VK_RIGHT; break;
4020 case WXK_DOWN : nKeySym = VK_DOWN; break;
4021 case WXK_PRINT: nKeySym = VK_PRINTSCRN; break;
4022 case WXK_INSERT: nKeySym = VK_INSERT; break;
4023 case WXK_DELETE: nKeySym = VK_DELETE; break;
4024 case WXK_F1: nKeySym = VK_F1; break;
4025 case WXK_F2: nKeySym = VK_F2; break;
4026 case WXK_F3: nKeySym = VK_F3; break;
4027 case WXK_F4: nKeySym = VK_F4; break;
4028 case WXK_F5: nKeySym = VK_F5; break;
4029 case WXK_F6: nKeySym = VK_F6; break;
4030 case WXK_F7: nKeySym = VK_F7; break;
4031 case WXK_F8: nKeySym = VK_F8; break;
4032 case WXK_F9: nKeySym = VK_F9; break;
4033 case WXK_F10: nKeySym = VK_F10; break;
4034 case WXK_F11: nKeySym = VK_F11; break;
4035 case WXK_F12: nKeySym = VK_F12; break;
4036 case WXK_F13: nKeySym = VK_F13; break;
4037 case WXK_F14: nKeySym = VK_F14; break;
4038 case WXK_F15: nKeySym = VK_F15; break;
4039 case WXK_F16: nKeySym = VK_F16; break;
4040 case WXK_F17: nKeySym = VK_F17; break;
4041 case WXK_F18: nKeySym = VK_F18; break;
4042 case WXK_F19: nKeySym = VK_F19; break;
4043 case WXK_F20: nKeySym = VK_F20; break;
4044 case WXK_F21: nKeySym = VK_F21; break;
4045 case WXK_F22: nKeySym = VK_F22; break;
4046 case WXK_F23: nKeySym = VK_F23; break;
4047 case WXK_F24: nKeySym = VK_F24; break;
4048 case WXK_NUMLOCK: nKeySym = VK_NUMLOCK; break;
4049 case WXK_SCROLL: nKeySym = VK_SCRLLOCK; break;
4050 default:
cdf1e714 4051 {
61243a51
DW
4052 *bIsVirtual = FALSE;
4053 nKeySym = nId;
cdf1e714
DW
4054 break;
4055 }
86de7616 4056 }
61243a51
DW
4057 return nKeySym;
4058} // end of wxCharCodeWXToOS2
86de7616 4059
61243a51 4060wxWindow* wxGetActiveWindow()
cdf1e714 4061{
61243a51 4062 HWND hWnd = ::WinQueryActiveWindow(HWND_DESKTOP);
86de7616 4063
61243a51
DW
4064 if (hWnd != 0)
4065 {
4066 return wxFindWinFromHandle((WXHWND)hWnd);
4067 }
4068 return NULL;
4069} // end of wxGetActiveWindow
cdf1e714
DW
4070
4071#ifdef __WXDEBUG__
61243a51
DW
4072const char* wxGetMessageName(
4073 int nMessage)
cdf1e714 4074{
61243a51 4075 switch (nMessage)
86de7616 4076 {
cdf1e714
DW
4077 case 0x0000: return "WM_NULL";
4078 case 0x0001: return "WM_CREATE";
4079 case 0x0002: return "WM_DESTROY";
61243a51
DW
4080 case 0x0004: return "WM_ENABLE";
4081 case 0x0005: return "WM_SHOW";
4082 case 0x0006: return "WM_MOVE";
4083 case 0x0007: return "WM_SIZE";
4084 case 0x0008: return "WM_ADJUSTWINDOWPOS";
4085 case 0x0009: return "WM_CALCVALIDRECTS";
4086 case 0x000A: return "WM_SETWINDOWPARAMS";
4087 case 0x000B: return "WM_QUERYWINDOWPARAMS";
4088 case 0x000C: return "WM_HITTEST";
4089 case 0x000D: return "WM_ACTIVATE";
4090 case 0x000F: return "WM_SETFOCUS";
4091 case 0x0010: return "WM_SETSELECTION";
4092 case 0x0011: return "WM_PPAINT";
4093 case 0x0012: return "WM_PSETFOCUS";
4094 case 0x0013: return "WM_PSYSCOLORCHANGE";
4095 case 0x0014: return "WM_PSIZE";
4096 case 0x0015: return "WM_PACTIVATE";
4097 case 0x0016: return "WM_PCONTROL";
4098 case 0x0020: return "WM_COMMAND";
4099 case 0x0021: return "WM_SYSCOMMAND";
4100 case 0x0022: return "WM_HELP";
4101 case 0x0023: return "WM_PAINT";
4102 case 0x0024: return "WM_TIMER";
4103 case 0x0025: return "WM_SEM1";
4104 case 0x0026: return "WM_SEM2";
4105 case 0x0027: return "WM_SEM3";
4106 case 0x0028: return "WM_SEM4";
4107 case 0x0029: return "WM_CLOSE";
4108 case 0x002A: return "WM_QUIT";
4109 case 0x002B: return "WM_SYSCOLORCHANGE";
4110 case 0x002D: return "WM_SYSVALUECHANGE";
4111 case 0x002E: return "WM_APPTERMINATENOTIFY";
4112 case 0x002F: return "WM_PRESPARAMCHANGED";
4113 // Control notification messages
4114 case 0x0030: return "WM_CONTROL";
4115 case 0x0031: return "WM_VSCROLL";
4116 case 0x0032: return "WM_HSCROLL";
4117 case 0x0033: return "WM_INITMENU";
4118 case 0x0034: return "WM_MENUSELECT";
4119 case 0x0035: return "WM_MENUSEND";
4120 case 0x0036: return "WM_DRAWITEM";
4121 case 0x0037: return "WM_MEASUREITEM";
4122 case 0x0038: return "WM_CONTROLPOINTER";
4123 case 0x003A: return "WM_QUERYDLGCODE";
4124 case 0x003B: return "WM_INITDLG";
4125 case 0x003C: return "WM_SUBSTITUTESTRING";
4126 case 0x003D: return "WM_MATCHMNEMONIC";
4127 case 0x003E: return "WM_SAVEAPPLICATION";
4128 case 0x0129: return "WM_CTLCOLORCHANGE";
4129 case 0x0130: return "WM_QUERYCTLTYPE";
4130 // Frame messages
4131 case 0x0040: return "WM_FLASHWINDOW";
4132 case 0x0041: return "WM_FORMATFRAME";
4133 case 0x0042: return "WM_UPDATEFRAME";
4134 case 0x0043: return "WM_FOCUSCHANGE";
4135 case 0x0044: return "WM_SETBORDERSIZE";
4136 case 0x0045: return "WM_TRACKFRAME";
4137 case 0x0046: return "WM_MINMAXFRAME";
4138 case 0x0047: return "WM_SETICON";
4139 case 0x0048: return "WM_QUERYICON";
4140 case 0x0049: return "WM_SETACCELTABLE";
4141 case 0x004A: return "WM_QUERYACCELTABLE";
4142 case 0x004B: return "WM_TRANSLATEACCEL";
4143 case 0x004C: return "WM_QUERYTRACKINFO";
4144 case 0x004D: return "WM_QUERYBORDERSIZE";
4145 case 0x004E: return "WM_NEXTMENU";
4146 case 0x004F: return "WM_ERASEBACKGROUND";
4147 case 0x0050: return "WM_QUERYFRAMEINFO";
4148 case 0x0051: return "WM_QUERYFOCUSCHAIN";
4149 case 0x0052: return "WM_OWNERPOSCHANGE";
4150 case 0x0053: return "WM_CACLFRAMERECT";
4151 case 0x0055: return "WM_WINDOWPOSCHANGED";
4152 case 0x0056: return "WM_ADJUSTFRAMEPOS";
4153 case 0x0059: return "WM_QUERYFRAMECTLCOUNT";
4154 case 0x005B: return "WM_QUERYHELPINFO";
4155 case 0x005C: return "WM_SETHELPINFO";
4156 case 0x005D: return "WM_ERROR";
4157 case 0x005E: return "WM_REALIZEPALETTE";
4158 // Clipboard messages
4159 case 0x0060: return "WM_RENDERFMT";
4160 case 0x0061: return "WM_RENDERALLFMTS";
4161 case 0x0062: return "WM_DESTROYCLIPBOARD";
4162 case 0x0063: return "WM_PAINTCLIPBOARD";
4163 case 0x0064: return "WM_SIZECLIPBOARD";
4164 case 0x0065: return "WM_HSCROLLCLIPBOARD";
4165 case 0x0066: return "WM_VSCROLLCLIPBOARD";
4166 case 0x0067: return "WM_DRAWCLIPBOARD";
4167 // mouse messages
4168 case 0x0070: return "WM_MOUSEMOVE";
4169 case 0x0071: return "WM_BUTTON1DOWN";
4170 case 0x0072: return "WM_BUTTON1UP";
4171 case 0x0073: return "WM_BUTTON1DBLCLK";
4172 case 0x0074: return "WM_BUTTON2DOWN";
4173 case 0x0075: return "WM_BUTTON2UP";
4174 case 0x0076: return "WM_BUTTON2DBLCLK";
4175 case 0x0077: return "WM_BUTTON3DOWN";
4176 case 0x0078: return "WM_BUTTON3UP";
4177 case 0x0079: return "WM_BUTTON3DBLCLK";
4178 case 0x007D: return "WM_MOUSEMAP";
4179 case 0x007E: return "WM_VRNDISABLED";
4180 case 0x007F: return "WM_VRNENABLED";
4181 case 0x0410: return "WM_CHORD";
4182 case 0x0411: return "WM_BUTTON1MOTIONSTART";
4183 case 0x0412: return "WM_BUTTON1MOTIONEND";
4184 case 0x0413: return "WM_BUTTON1CLICK";
4185 case 0x0414: return "WM_BUTTON2MOTIONSTART";
4186 case 0x0415: return "WM_BUTTON2MOTIONEND";
4187 case 0x0416: return "WM_BUTTON2CLICK";
4188 case 0x0417: return "WM_BUTTON3MOTIONSTART";
4189 case 0x0418: return "WM_BUTTON3MOTIONEND";
4190 case 0x0419: return "WM_BUTTON3CLICK";
4191 case 0x0420: return "WM_BEGINDRAG";
4192 case 0x0421: return "WM_ENDDRAG";
4193 case 0x0422: return "WM_SINGLESELECT";
4194 case 0x0423: return "WM_OPEN";
4195 case 0x0424: return "WM_CONTEXTMENU";
4196 case 0x0425: return "WM_CONTEXTHELP";
4197 case 0x0426: return "WM_TEXTEDIT";
4198 case 0x0427: return "WM_BEGINSELECT";
4199 case 0x0228: return "WM_ENDSELECT";
4200 case 0x0429: return "WM_PICKUP";
4201 case 0x04C0: return "WM_PENFIRST";
4202 case 0x04FF: return "WM_PENLAST";
4203 case 0x0500: return "WM_MMPMFIRST";
4204 case 0x05FF: return "WM_MMPMLAST";
4205 case 0x0600: return "WM_STDDLGFIRST";
4206 case 0x06FF: return "WM_STDDLGLAST";
4207 case 0x0BD0: return "WM_BIDI_FIRST";
4208 case 0x0BFF: return "WM_BIDI_LAST";
4209 // keyboard input
4210 case 0x007A: return "WM_CHAR";
4211 case 0x007B: return "WM_VIOCHAR";
4212 // DDE messages
4213 case 0x00A0: return "WM_DDE_INITIATE";
4214 case 0x00A1: return "WM_DDE_REQUEST";
4215 case 0x00A2: return "WM_DDE_ACK";
4216 case 0x00A3: return "WM_DDE_DATA";
4217 case 0x00A4: return "WM_DDE_ADVISE";
4218 case 0x00A5: return "WM_DDE_UNADVISE";
4219 case 0x00A6: return "WM_DDE_POKE";
4220 case 0x00A7: return "WM_DDE_EXECUTE";
4221 case 0x00A8: return "WM_DDE_TERMINATE";
4222 case 0x00A9: return "WM_DDE_INITIATEACK";
4223 case 0x00AF: return "WM_DDE_LAST";
4224 // Buttons
4225 case 0x0120: return "BM_CLICK";
4226 case 0x0121: return "BM_QUERYCHECKINDEX";
4227 case 0x0122: return "BM_QUERYHILITE";
4228 case 0x0123: return "BM_SETHILITE";
4229 case 0x0124: return "BM_QUERYCHECK";
4230 case 0x0125: return "BM_SETCHECK";
4231 case 0x0126: return "BM_SETDEFAULT";
4232 case 0x0128: return "BM_AUTOSIZE";
4233 // Combo boxes
4234 case 0x029A: return "CBID_LIST";
4235 case 0x029B: return "CBID_EDIT";
4236 case 0x0170: return "CBM_SHOWLIST";
4237 case 0x0171: return "CBM_HILITE";
4238 case 0x0172: return "CBM_ISLISTSHOWING";
4239 // Edit fields
4240 case 0x0140: return "EM_QUERYCHANGED";
4241 case 0x0141: return "EM_QUERYSEL";
4242 case 0x0142: return "EM_SETSEL";
4243 case 0x0143: return "EM_SETTEXTLIMIT";
4244 case 0x0144: return "EM_CUT";
4245 case 0x0145: return "EM_COPY";
4246 case 0x0146: return "EM_CLEAR";
4247 case 0x0147: return "EM_PASTE";
4248 case 0x0148: return "EM_QUERYFIRSTCHAR";
4249 case 0x0149: return "EM_SETFIRSTCHAR";
4250 case 0x014A: return "EM_QUERYREADONLY";
4251 case 0x014B: return "EM_SETREADONLY";
4252 case 0x014C: return "EM_SETINSERTMODE";
4253 // Listboxes
4254 case 0x0160: return "LM_QUERYITEMCOUNT";
4255 case 0x0161: return "LM_INSERTITEM";
4256 case 0x0162: return "LM_SETOPENINDEX";
4257 case 0x0163: return "LM_DELETEITEM";
4258 case 0x0164: return "LM_SELECTITEM";
4259 case 0x0165: return "LM_QUERYSELECTION";
4260 case 0x0166: return "LM_SETITEMTEXT";
4261 case 0x0167: return "LM_QUERYITEMTEXTLENGTH";
4262 case 0x0168: return "LM_QUERYITEMTEXT";
4263 case 0x0169: return "LM_SETITEMHANDLE";
4264 case 0x016A: return "LM_QUERYITEMHANDLE";
4265 case 0x016B: return "LM_SEARCHSTRING";
4266 case 0x016C: return "LM_SETITEMHEIGHT";
4267 case 0x016D: return "LM_QUERYTOPINDEX";
4268 case 0x016E: return "LM_DELETEALL";
4269 case 0x016F: return "LM_INSERTMULITEMS";
4270 case 0x0660: return "LM_SETITEMWIDTH";
4271 // Menus
4272 case 0x0180: return "MM_INSERTITEM";
4273 case 0x0181: return "MM_DELETEITEM";
4274 case 0x0182: return "MM_QUERYITEM";
4275 case 0x0183: return "MM_SETITEM";
4276 case 0x0184: return "MM_QUERYITEMCOUNT";
4277 case 0x0185: return "MM_STARTMENUMODE";
4278 case 0x0186: return "MM_ENDMENUMODE";
4279 case 0x0188: return "MM_REMOVEITEM";
4280 case 0x0189: return "MM_SELECTITEM";
4281 case 0x018A: return "MM_QUERYSELITEMID";
4282 case 0x018B: return "MM_QUERYITEMTEXT";
4283 case 0x018C: return "MM_QUERYITEMTEXTLENGTH";
4284 case 0x018D: return "MM_SETITEMHANDLE";
4285 case 0x018E: return "MM_SETITEMTEXT";
4286 case 0x018F: return "MM_ITEMPOSITIONFROMID";
4287 case 0x0190: return "MM_ITEMIDFROMPOSITION";
4288 case 0x0191: return "MM_QUERYITEMATTR";
4289 case 0x0192: return "MM_SETITEMATTR";
4290 case 0x0193: return "MM_ISITEMVALID";
4291 case 0x0194: return "MM_QUERYITEMRECT";
4292 case 0x0431: return "MM_QUERYDEFAULTITEMID";
4293 case 0x0432: return "MM_SETDEFAULTITEMID";
4294 // Scrollbars
4295 case 0x01A0: return "SBM_SETSCROLLBAR";
4296 case 0x01A1: return "SBM_SETPOS";
4297 case 0x01A2: return "SBM_QUERYPOS";
4298 case 0x01A3: return "SBM_QUERYRANGE";
4299 case 0x01A6: return "SBM_SETTHUMBSIZE";
4300
4301 // Help messages
4302 case 0x0F00: return "WM_HELPBASE";
4303 case 0x0FFF: return "WM_HELPTOP";
4304 // Beginning of user defined messages
4305 case 0x1000: return "WM_USER";
4306
4307 // wxWindows user defined types
cdf1e714
DW
4308
4309 // listview
61243a51 4310 // case 0x1000 + 0: return "LVM_GETBKCOLOR";
cdf1e714
DW
4311 case 0x1000 + 1: return "LVM_SETBKCOLOR";
4312 case 0x1000 + 2: return "LVM_GETIMAGELIST";
4313 case 0x1000 + 3: return "LVM_SETIMAGELIST";
4314 case 0x1000 + 4: return "LVM_GETITEMCOUNT";
4315 case 0x1000 + 5: return "LVM_GETITEMA";
4316 case 0x1000 + 75: return "LVM_GETITEMW";
4317 case 0x1000 + 6: return "LVM_SETITEMA";
4318 case 0x1000 + 76: return "LVM_SETITEMW";
4319 case 0x1000 + 7: return "LVM_INSERTITEMA";
4320 case 0x1000 + 77: return "LVM_INSERTITEMW";
4321 case 0x1000 + 8: return "LVM_DELETEITEM";
4322 case 0x1000 + 9: return "LVM_DELETEALLITEMS";
4323 case 0x1000 + 10: return "LVM_GETCALLBACKMASK";
4324 case 0x1000 + 11: return "LVM_SETCALLBACKMASK";
4325 case 0x1000 + 12: return "LVM_GETNEXTITEM";
4326 case 0x1000 + 13: return "LVM_FINDITEMA";
4327 case 0x1000 + 83: return "LVM_FINDITEMW";
4328 case 0x1000 + 14: return "LVM_GETITEMRECT";
4329 case 0x1000 + 15: return "LVM_SETITEMPOSITION";
4330 case 0x1000 + 16: return "LVM_GETITEMPOSITION";
4331 case 0x1000 + 17: return "LVM_GETSTRINGWIDTHA";
4332 case 0x1000 + 87: return "LVM_GETSTRINGWIDTHW";
4333 case 0x1000 + 18: return "LVM_HITTEST";
4334 case 0x1000 + 19: return "LVM_ENSUREVISIBLE";
4335 case 0x1000 + 20: return "LVM_SCROLL";
4336 case 0x1000 + 21: return "LVM_REDRAWITEMS";
4337 case 0x1000 + 22: return "LVM_ARRANGE";
4338 case 0x1000 + 23: return "LVM_EDITLABELA";
4339 case 0x1000 + 118: return "LVM_EDITLABELW";
4340 case 0x1000 + 24: return "LVM_GETEDITCONTROL";
4341 case 0x1000 + 25: return "LVM_GETCOLUMNA";
4342 case 0x1000 + 95: return "LVM_GETCOLUMNW";
4343 case 0x1000 + 26: return "LVM_SETCOLUMNA";
4344 case 0x1000 + 96: return "LVM_SETCOLUMNW";
4345 case 0x1000 + 27: return "LVM_INSERTCOLUMNA";
4346 case 0x1000 + 97: return "LVM_INSERTCOLUMNW";
4347 case 0x1000 + 28: return "LVM_DELETECOLUMN";
4348 case 0x1000 + 29: return "LVM_GETCOLUMNWIDTH";
4349 case 0x1000 + 30: return "LVM_SETCOLUMNWIDTH";
4350 case 0x1000 + 31: return "LVM_GETHEADER";
4351 case 0x1000 + 33: return "LVM_CREATEDRAGIMAGE";
4352 case 0x1000 + 34: return "LVM_GETVIEWRECT";
4353 case 0x1000 + 35: return "LVM_GETTEXTCOLOR";
4354 case 0x1000 + 36: return "LVM_SETTEXTCOLOR";
4355 case 0x1000 + 37: return "LVM_GETTEXTBKCOLOR";
4356 case 0x1000 + 38: return "LVM_SETTEXTBKCOLOR";
4357 case 0x1000 + 39: return "LVM_GETTOPINDEX";
4358 case 0x1000 + 40: return "LVM_GETCOUNTPERPAGE";
4359 case 0x1000 + 41: return "LVM_GETORIGIN";
4360 case 0x1000 + 42: return "LVM_UPDATE";
4361 case 0x1000 + 43: return "LVM_SETITEMSTATE";
4362 case 0x1000 + 44: return "LVM_GETITEMSTATE";
4363 case 0x1000 + 45: return "LVM_GETITEMTEXTA";
4364 case 0x1000 + 115: return "LVM_GETITEMTEXTW";
4365 case 0x1000 + 46: return "LVM_SETITEMTEXTA";
4366 case 0x1000 + 116: return "LVM_SETITEMTEXTW";
4367 case 0x1000 + 47: return "LVM_SETITEMCOUNT";
4368 case 0x1000 + 48: return "LVM_SORTITEMS";
4369 case 0x1000 + 49: return "LVM_SETITEMPOSITION32";
4370 case 0x1000 + 50: return "LVM_GETSELECTEDCOUNT";
4371 case 0x1000 + 51: return "LVM_GETITEMSPACING";
4372 case 0x1000 + 52: return "LVM_GETISEARCHSTRINGA";
4373 case 0x1000 + 117: return "LVM_GETISEARCHSTRINGW";
4374 case 0x1000 + 53: return "LVM_SETICONSPACING";
4375 case 0x1000 + 54: return "LVM_SETEXTENDEDLISTVIEWSTYLE";
4376 case 0x1000 + 55: return "LVM_GETEXTENDEDLISTVIEWSTYLE";
4377 case 0x1000 + 56: return "LVM_GETSUBITEMRECT";
4378 case 0x1000 + 57: return "LVM_SUBITEMHITTEST";
4379 case 0x1000 + 58: return "LVM_SETCOLUMNORDERARRAY";
4380 case 0x1000 + 59: return "LVM_GETCOLUMNORDERARRAY";
4381 case 0x1000 + 60: return "LVM_SETHOTITEM";
4382 case 0x1000 + 61: return "LVM_GETHOTITEM";
4383 case 0x1000 + 62: return "LVM_SETHOTCURSOR";
4384 case 0x1000 + 63: return "LVM_GETHOTCURSOR";
4385 case 0x1000 + 64: return "LVM_APPROXIMATEVIEWRECT";
4386 case 0x1000 + 65: return "LVM_SETWORKAREA";
4387
4388 // tree view
4389 case 0x1100 + 0: return "TVM_INSERTITEMA";
4390 case 0x1100 + 50: return "TVM_INSERTITEMW";
4391 case 0x1100 + 1: return "TVM_DELETEITEM";
4392 case 0x1100 + 2: return "TVM_EXPAND";
4393 case 0x1100 + 4: return "TVM_GETITEMRECT";
4394 case 0x1100 + 5: return "TVM_GETCOUNT";
4395 case 0x1100 + 6: return "TVM_GETINDENT";
4396 case 0x1100 + 7: return "TVM_SETINDENT";
4397 case 0x1100 + 8: return "TVM_GETIMAGELIST";
4398 case 0x1100 + 9: return "TVM_SETIMAGELIST";
4399 case 0x1100 + 10: return "TVM_GETNEXTITEM";
4400 case 0x1100 + 11: return "TVM_SELECTITEM";
4401 case 0x1100 + 12: return "TVM_GETITEMA";
4402 case 0x1100 + 62: return "TVM_GETITEMW";
4403 case 0x1100 + 13: return "TVM_SETITEMA";
4404 case 0x1100 + 63: return "TVM_SETITEMW";
4405 case 0x1100 + 14: return "TVM_EDITLABELA";
4406 case 0x1100 + 65: return "TVM_EDITLABELW";
4407 case 0x1100 + 15: return "TVM_GETEDITCONTROL";
4408 case 0x1100 + 16: return "TVM_GETVISIBLECOUNT";
4409 case 0x1100 + 17: return "TVM_HITTEST";
4410 case 0x1100 + 18: return "TVM_CREATEDRAGIMAGE";
4411 case 0x1100 + 19: return "TVM_SORTCHILDREN";
4412 case 0x1100 + 20: return "TVM_ENSUREVISIBLE";
4413 case 0x1100 + 21: return "TVM_SORTCHILDRENCB";
4414 case 0x1100 + 22: return "TVM_ENDEDITLABELNOW";
4415 case 0x1100 + 23: return "TVM_GETISEARCHSTRINGA";
4416 case 0x1100 + 64: return "TVM_GETISEARCHSTRINGW";
4417 case 0x1100 + 24: return "TVM_SETTOOLTIPS";
4418 case 0x1100 + 25: return "TVM_GETTOOLTIPS";
4419
4420 // header
4421 case 0x1200 + 0: return "HDM_GETITEMCOUNT";
4422 case 0x1200 + 1: return "HDM_INSERTITEMA";
4423 case 0x1200 + 10: return "HDM_INSERTITEMW";
4424 case 0x1200 + 2: return "HDM_DELETEITEM";
4425 case 0x1200 + 3: return "HDM_GETITEMA";
4426 case 0x1200 + 11: return "HDM_GETITEMW";
4427 case 0x1200 + 4: return "HDM_SETITEMA";
4428 case 0x1200 + 12: return "HDM_SETITEMW";
4429 case 0x1200 + 5: return "HDM_LAYOUT";
4430 case 0x1200 + 6: return "HDM_HITTEST";
4431 case 0x1200 + 7: return "HDM_GETITEMRECT";
4432 case 0x1200 + 8: return "HDM_SETIMAGELIST";
4433 case 0x1200 + 9: return "HDM_GETIMAGELIST";
4434 case 0x1200 + 15: return "HDM_ORDERTOINDEX";
4435 case 0x1200 + 16: return "HDM_CREATEDRAGIMAGE";
4436 case 0x1200 + 17: return "HDM_GETORDERARRAY";
4437 case 0x1200 + 18: return "HDM_SETORDERARRAY";
4438 case 0x1200 + 19: return "HDM_SETHOTDIVIDER";
4439
4440 // tab control
4441 case 0x1300 + 2: return "TCM_GETIMAGELIST";
4442 case 0x1300 + 3: return "TCM_SETIMAGELIST";
4443 case 0x1300 + 4: return "TCM_GETITEMCOUNT";
4444 case 0x1300 + 5: return "TCM_GETITEMA";
4445 case 0x1300 + 60: return "TCM_GETITEMW";
4446 case 0x1300 + 6: return "TCM_SETITEMA";
4447 case 0x1300 + 61: return "TCM_SETITEMW";
4448 case 0x1300 + 7: return "TCM_INSERTITEMA";
4449 case 0x1300 + 62: return "TCM_INSERTITEMW";
4450 case 0x1300 + 8: return "TCM_DELETEITEM";
4451 case 0x1300 + 9: return "TCM_DELETEALLITEMS";
4452 case 0x1300 + 10: return "TCM_GETITEMRECT";
4453 case 0x1300 + 11: return "TCM_GETCURSEL";
4454 case 0x1300 + 12: return "TCM_SETCURSEL";
4455 case 0x1300 + 13: return "TCM_HITTEST";
4456 case 0x1300 + 14: return "TCM_SETITEMEXTRA";
4457 case 0x1300 + 40: return "TCM_ADJUSTRECT";
4458 case 0x1300 + 41: return "TCM_SETITEMSIZE";
4459 case 0x1300 + 42: return "TCM_REMOVEIMAGE";
4460 case 0x1300 + 43: return "TCM_SETPADDING";
4461 case 0x1300 + 44: return "TCM_GETROWCOUNT";
4462 case 0x1300 + 45: return "TCM_GETTOOLTIPS";
4463 case 0x1300 + 46: return "TCM_SETTOOLTIPS";
4464 case 0x1300 + 47: return "TCM_GETCURFOCUS";
4465 case 0x1300 + 48: return "TCM_SETCURFOCUS";
4466 case 0x1300 + 49: return "TCM_SETMINTABWIDTH";
4467 case 0x1300 + 50: return "TCM_DESELECTALL";
4468
4469 // toolbar
61243a51
DW
4470 case WM_USER+1000+1: return "TB_ENABLEBUTTON";
4471 case WM_USER+1000+2: return "TB_CHECKBUTTON";
4472 case WM_USER+1000+3: return "TB_PRESSBUTTON";
4473 case WM_USER+1000+4: return "TB_HIDEBUTTON";
4474 case WM_USER+1000+5: return "TB_INDETERMINATE";
4475 case WM_USER+1000+9: return "TB_ISBUTTONENABLED";
4476 case WM_USER+1000+10: return "TB_ISBUTTONCHECKED";
4477 case WM_USER+1000+11: return "TB_ISBUTTONPRESSED";
4478 case WM_USER+1000+12: return "TB_ISBUTTONHIDDEN";
4479 case WM_USER+1000+13: return "TB_ISBUTTONINDETERMINATE";
4480 case WM_USER+1000+17: return "TB_SETSTATE";
4481 case WM_USER+1000+18: return "TB_GETSTATE";
4482 case WM_USER+1000+19: return "TB_ADDBITMAP";
4483 case WM_USER+1000+20: return "TB_ADDBUTTONS";
4484 case WM_USER+1000+21: return "TB_INSERTBUTTON";
4485 case WM_USER+1000+22: return "TB_DELETEBUTTON";
4486 case WM_USER+1000+23: return "TB_GETBUTTON";
4487 case WM_USER+1000+24: return "TB_BUTTONCOUNT";
4488 case WM_USER+1000+25: return "TB_COMMANDTOINDEX";
4489 case WM_USER+1000+26: return "TB_SAVERESTOREA";
4490 case WM_USER+1000+76: return "TB_SAVERESTOREW";
4491 case WM_USER+1000+27: return "TB_CUSTOMIZE";
4492 case WM_USER+1000+28: return "TB_ADDSTRINGA";
4493 case WM_USER+1000+77: return "TB_ADDSTRINGW";
4494 case WM_USER+1000+29: return "TB_GETITEMRECT";
4495 case WM_USER+1000+30: return "TB_BUTTONSTRUCTSIZE";
4496 case WM_USER+1000+31: return "TB_SETBUTTONSIZE";
4497 case WM_USER+1000+32: return "TB_SETBITMAPSIZE";
4498 case WM_USER+1000+33: return "TB_AUTOSIZE";
4499 case WM_USER+1000+35: return "TB_GETTOOLTIPS";
4500 case WM_USER+1000+36: return "TB_SETTOOLTIPS";
4501 case WM_USER+1000+37: return "TB_SETPARENT";
4502 case WM_USER+1000+39: return "TB_SETROWS";
4503 case WM_USER+1000+40: return "TB_GETROWS";
4504 case WM_USER+1000+42: return "TB_SETCMDID";
4505 case WM_USER+1000+43: return "TB_CHANGEBITMAP";
4506 case WM_USER+1000+44: return "TB_GETBITMAP";
4507 case WM_USER+1000+45: return "TB_GETBUTTONTEXTA";
4508 case WM_USER+1000+75: return "TB_GETBUTTONTEXTW";
4509 case WM_USER+1000+46: return "TB_REPLACEBITMAP";
4510 case WM_USER+1000+47: return "TB_SETINDENT";
4511 case WM_USER+1000+48: return "TB_SETIMAGELIST";
4512 case WM_USER+1000+49: return "TB_GETIMAGELIST";
4513 case WM_USER+1000+50: return "TB_LOADIMAGES";
4514 case WM_USER+1000+51: return "TB_GETRECT";
4515 case WM_USER+1000+52: return "TB_SETHOTIMAGELIST";
4516 case WM_USER+1000+53: return "TB_GETHOTIMAGELIST";
4517 case WM_USER+1000+54: return "TB_SETDISABLEDIMAGELIST";
4518 case WM_USER+1000+55: return "TB_GETDISABLEDIMAGELIST";
4519 case WM_USER+1000+56: return "TB_SETSTYLE";
4520 case WM_USER+1000+57: return "TB_GETSTYLE";
4521 case WM_USER+1000+58: return "TB_GETBUTTONSIZE";
4522 case WM_USER+1000+59: return "TB_SETBUTTONWIDTH";
4523 case WM_USER+1000+60: return "TB_SETMAXTEXTROWS";
4524 case WM_USER+1000+61: return "TB_GETTEXTROWS";
4525 case WM_USER+1000+41: return "TB_GETBITMAPFLAGS";
cdf1e714
DW
4526
4527 default:
4528 static char s_szBuf[128];
61243a51 4529 sprintf(s_szBuf, "<unknown message = %d>", nMessage);
cdf1e714 4530 return s_szBuf;
86de7616 4531 }
cdf1e714 4532 return NULL;
61243a51 4533} // end of wxGetMessageName
86de7616 4534
11e59d47
DW
4535#endif // __WXDEBUG__
4536
19193a2c
KB
4537// Unused?
4538#if 0
61243a51
DW
4539static void TranslateKbdEventToMouse(
4540 wxWindow* pWin
4541, int* pX
4542, int* pY
4543, ULONG* pFlags
4544)
4545{
4546 //
4547 // Construct the key mask
4548 ULONG& fwKeys = *pFlags;
4549
4550 fwKeys = VK_BUTTON2;
4551 if ((::WinGetKeyState(HWND_DESKTOP, VK_CTRL) & 0x100) != 0)
4552 fwKeys |= VK_CTRL;
4553 if ((::WinGetKeyState(HWND_DESKTOP, VK_SHIFT) & 0x100) != 0)
4554 fwKeys |= VK_SHIFT;
4555
4556 //
4557 // Simulate right mouse button click
4558 //
4559 POINTL vPoint;
4560
4561 ::WinQueryMsgPos(vHabmain, &vPoint);
4562 *pX = vPoint.x;
4563 *pY = vPoint.y;
4564
4565 pWin->ScreenToClient(pX, pY);
4566} // end of TranslateKbdEventToMouse
19193a2c 4567#endif
61243a51 4568
cd212ee4
DW
4569// Find the wxWindow at the current mouse position, returning the mouse
4570// position.
f44fdfb0 4571wxWindow* wxFindWindowAtPointer(
19193a2c 4572 wxPoint& WXUNUSED(rPt)
f44fdfb0 4573)
cd212ee4
DW
4574{
4575 return wxFindWindowAtPoint(wxGetMousePosition());
4576}
4577
f44fdfb0
DW
4578wxWindow* wxFindWindowAtPoint(
4579 const wxPoint& rPt
4580)
cd212ee4 4581{
f44fdfb0 4582 POINTL vPt2;
cd212ee4 4583
f44fdfb0
DW
4584 vPt2.x = rPt.x;
4585 vPt2.y = rPt.y;
cd212ee4 4586
f44fdfb0
DW
4587 HWND hWndHit = ::WinWindowFromPoint(HWND_DESKTOP, &vPt2, FALSE);
4588 wxWindow* pWin = wxFindWinFromHandle((WXHWND)hWndHit) ;
4589 HWND hWnd = hWndHit;
4590
4591 //
cd212ee4 4592 // Try to find a window with a wxWindow associated with it
f44fdfb0
DW
4593 //
4594 while (!pWin && (hWnd != 0))
cd212ee4 4595 {
f44fdfb0
DW
4596 hWnd = ::WinQueryWindow(hWnd, QW_PARENT);
4597 pWin = wxFindWinFromHandle((WXHWND)hWnd) ;
cd212ee4 4598 }
f44fdfb0 4599 return pWin;
cd212ee4
DW
4600}
4601
f44fdfb0
DW
4602// Get the current mouse position.
4603wxPoint wxGetMousePosition()
4604{
4605 POINTL vPt;
4606
4607 ::WinQueryPointerPos(HWND_DESKTOP, &vPt);
4608 return wxPoint(vPt.x, vPt.y);
4609}
cd212ee4 4610