1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "window.h"
27 #include "wx/dcclient.h"
31 #include "wx/layout.h"
32 #include "wx/dialog.h"
33 #include "wx/listbox.h"
34 #include "wx/button.h"
35 #include "wx/settings.h"
36 #include "wx/msgdlg.h"
38 #include "wx/scrolwin.h"
39 #include "wx/module.h"
40 #include "wx/menuitem.h"
43 #if wxUSE_DRAG_AND_DROP
47 #include "wx/x11/private.h"
51 // ----------------------------------------------------------------------------
53 // ----------------------------------------------------------------------------
55 static const int SCROLL_MARGIN
= 4;
57 // ----------------------------------------------------------------------------
58 // global variables for this module
59 // ----------------------------------------------------------------------------
61 extern wxHashTable
*wxWidgetHashTable
;
62 static wxWindow
* g_captureWindow
= NULL
;
64 // ----------------------------------------------------------------------------
66 // ----------------------------------------------------------------------------
68 #define event_left_is_down(x) ((x)->xbutton.state & Button1Mask)
69 #define event_middle_is_down(x) ((x)->xbutton.state & Button2Mask)
70 #define event_right_is_down(x) ((x)->xbutton.state & Button3Mask)
72 // ----------------------------------------------------------------------------
74 // ----------------------------------------------------------------------------
76 IMPLEMENT_DYNAMIC_CLASS(wxWindowX11
, wxWindowBase
)
78 BEGIN_EVENT_TABLE(wxWindowX11
, wxWindowBase
)
79 EVT_SYS_COLOUR_CHANGED(wxWindowX11::OnSysColourChanged
)
80 EVT_IDLE(wxWindowX11::OnIdle
)
83 // ============================================================================
85 // ============================================================================
87 // ----------------------------------------------------------------------------
89 // ----------------------------------------------------------------------------
91 // ----------------------------------------------------------------------------
93 // ----------------------------------------------------------------------------
95 void wxWindowX11::Init()
97 // generic initializations first
101 m_needsRefresh
= TRUE
;
102 m_mainWidget
= (WXWidget
) 0;
106 m_button3Pressed
= FALSE
;
108 m_winCaptured
= FALSE
;
111 m_isBeingDeleted
= FALSE
;
117 m_drawingArea
= (WXWidget
) 0;
125 m_backingPixmap
= (WXPixmap
) 0;
134 m_canAddEventHandler
= FALSE
;
137 // real construction (Init() must have been called before!)
138 bool wxWindowX11::Create(wxWindow
*parent
, wxWindowID id
,
142 const wxString
& name
)
144 wxCHECK_MSG( parent
, FALSE
, "can't create wxWindow without parent" );
146 CreateBase(parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
);
148 parent
->AddChild(this);
150 m_backgroundColour
= wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
);
151 m_foregroundColour
= *wxBLACK
;
153 if (style
& wxSIMPLE_BORDER
)
155 } else if (style
& wxSUNKEN_BORDER
)
157 } else if (style
& wxRAISED_BORDER
)
161 // TODO: create XWindow
164 wxAddWindowToTable((Window
) m_drawingArea
, this);
165 wxAddWindowToTable((Window
) m_scrolledWindow
, this);
168 // Without this, the cursor may not be restored properly (e.g. in splitter
170 SetCursor(*wxSTANDARD_CURSOR
);
171 SetFont(wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT
));
172 SetSize(pos
.x
, pos
.y
, size
.x
, size
.y
);
178 wxWindowX11::~wxWindow()
180 if (g_captureWindow
== this)
181 g_captureWindow
= NULL
;
183 m_isBeingDeleted
= TRUE
;
185 // Motif-specific actions first
186 WXWindow wMain
= GetMainWindow();
189 // Removes event handlers
190 //DetachWidget(wMain);
196 m_parent
->RemoveChild( this );
201 // If m_drawingArea, we're a fully-fledged window with drawing area,
202 // scrollbars etc. (what wxCanvas used to be)
205 // Destroy children before destroying self
209 XFreePixmap (XtDisplay ((Widget
) GetMainWidget()), (Pixmap
) m_backingPixmap
);
211 Widget w
= (Widget
) m_drawingArea
;
212 wxDeleteWindowFromTable(w
);
217 m_drawingArea
= (WXWidget
) 0;
220 // Only if we're _really_ a canvas (not a dialog box/panel)
221 if (m_scrolledWindow
)
223 wxDeleteWindowFromTable((Widget
) m_scrolledWindow
);
228 wxDeleteWindowFromTable((Widget
) m_hScrollBar
);
229 XtUnmanageChild((Widget
) m_hScrollBar
);
233 wxDeleteWindowFromTable((Widget
) m_vScrollBar
);
234 XtUnmanageChild((Widget
) m_vScrollBar
);
238 XtDestroyWidget((Widget
) m_hScrollBar
);
240 XtDestroyWidget((Widget
) m_vScrollBar
);
242 UnmanageAndDestroy(m_scrolledWindow
);
246 XtDestroyWidget ((Widget
) m_borderWidget
);
247 m_borderWidget
= (WXWidget
) 0;
250 else // Why wasn't this here before? JACS 8/3/2000
255 // Destroy the window
259 // XtDestroyWidget((Widget) GetMainWidget());
260 SetMainWindow((WXWindow
) NULL
);
264 // ----------------------------------------------------------------------------
265 // scrollbar management
266 // ----------------------------------------------------------------------------
269 void wxWindowX11::CreateScrollbar(wxOrientation orientation
)
273 wxCHECK_RET( m_drawingArea
, "this window can't have scrollbars" );
275 XtVaSetValues((Widget
) m_scrolledWindow
, XmNresizePolicy
, XmRESIZE_NONE
, NULL
);
277 // Add scrollbars if required
278 if (orientation
== wxHORIZONTAL
)
280 Widget hScrollBar
= XtVaCreateManagedWidget ("hsb",
281 xmScrollBarWidgetClass
, (Widget
) m_scrolledWindow
,
282 XmNorientation
, XmHORIZONTAL
,
284 XtAddCallback (hScrollBar
, XmNvalueChangedCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
285 XtAddCallback (hScrollBar
, XmNdragCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
286 XtAddCallback (hScrollBar
, XmNincrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
287 XtAddCallback (hScrollBar
, XmNdecrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
288 XtAddCallback (hScrollBar
, XmNpageIncrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
289 XtAddCallback (hScrollBar
, XmNpageDecrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
290 XtAddCallback (hScrollBar
, XmNtoTopCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
291 XtAddCallback (hScrollBar
, XmNtoBottomCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmHORIZONTAL
);
293 XtVaSetValues (hScrollBar
,
298 m_hScrollBar
= (WXWidget
) hScrollBar
;
300 wxColour backgroundColour
= wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
);
301 DoChangeBackgroundColour(m_hScrollBar
, backgroundColour
, TRUE
);
303 XtRealizeWidget(hScrollBar
);
305 XtVaSetValues((Widget
) m_scrolledWindow
,
306 XmNhorizontalScrollBar
, (Widget
) m_hScrollBar
,
311 wxAddWindowToTable( hScrollBar
, this );
314 if (orientation
== wxVERTICAL
)
316 Widget vScrollBar
= XtVaCreateManagedWidget ("vsb",
317 xmScrollBarWidgetClass
, (Widget
) m_scrolledWindow
,
318 XmNorientation
, XmVERTICAL
,
320 XtAddCallback (vScrollBar
, XmNvalueChangedCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
321 XtAddCallback (vScrollBar
, XmNdragCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
322 XtAddCallback (vScrollBar
, XmNincrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
323 XtAddCallback (vScrollBar
, XmNdecrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
324 XtAddCallback (vScrollBar
, XmNpageIncrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
325 XtAddCallback (vScrollBar
, XmNpageDecrementCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
326 XtAddCallback (vScrollBar
, XmNtoTopCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
327 XtAddCallback (vScrollBar
, XmNtoBottomCallback
, (XtCallbackProc
) wxScrollBarCallback
, (XtPointer
) XmVERTICAL
);
329 XtVaSetValues (vScrollBar
,
334 m_vScrollBar
= (WXWidget
) vScrollBar
;
335 wxColour backgroundColour
= wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
);
336 DoChangeBackgroundColour(m_vScrollBar
, backgroundColour
, TRUE
);
338 XtRealizeWidget(vScrollBar
);
340 XtVaSetValues((Widget
) m_scrolledWindow
,
341 XmNverticalScrollBar
, (Widget
) m_vScrollBar
,
346 wxAddWindowToTable( vScrollBar
, this );
349 XtVaSetValues((Widget
) m_scrolledWindow
, XmNresizePolicy
, XmRESIZE_ANY
, NULL
);
353 void wxWindowX11::DestroyScrollbar(wxOrientation orientation
)
357 wxCHECK_RET( m_drawingArea
, "this window can't have scrollbars" );
359 XtVaSetValues((Widget
) m_scrolledWindow
, XmNresizePolicy
, XmRESIZE_NONE
, NULL
);
360 // Add scrollbars if required
361 if (orientation
== wxHORIZONTAL
)
365 wxDeleteWindowFromTable((Widget
)m_hScrollBar
);
366 XtDestroyWidget((Widget
) m_hScrollBar
);
368 m_hScrollBar
= (WXWidget
) 0;
371 XtVaSetValues((Widget
) m_scrolledWindow
,
372 XmNhorizontalScrollBar
, (Widget
) 0,
377 if (orientation
== wxVERTICAL
)
381 wxDeleteWindowFromTable((Widget
)m_vScrollBar
);
382 XtDestroyWidget((Widget
) m_vScrollBar
);
384 m_vScrollBar
= (WXWidget
) 0;
387 XtVaSetValues((Widget
) m_scrolledWindow
,
388 XmNverticalScrollBar
, (Widget
) 0,
392 XtVaSetValues((Widget
) m_scrolledWindow
, XmNresizePolicy
, XmRESIZE_ANY
, NULL
);
396 // ---------------------------------------------------------------------------
398 // ---------------------------------------------------------------------------
400 void wxWindowX11::SetFocus()
404 Widget wMain
= (Widget
) GetMainWidget();
405 XmProcessTraversal(wMain
, XmTRAVERSE_CURRENT
);
406 XmProcessTraversal((Widget
) GetMainWidget(), XmTRAVERSE_CURRENT
);
410 // Get the window with the focus
411 wxWindow
*wxWindowBase::FindFocus()
418 // (1) Can there be multiple focussed widgets in an application?
419 // In which case we need to find the top-level window that's
421 // (2) The widget with the focus may not be in the widget table
422 // depending on which widgets I put in the table
423 wxWindow
*winFocus
= (wxWindow
*)NULL
;
424 for ( wxWindowList::Node
*node
= wxTopLevelWindows
.GetFirst();
426 node
= node
->GetNext() )
428 wxWindow
*win
= node
->GetData();
430 Widget w
= XmGetFocusWidget ((Widget
) win
->GetTopWidget());
432 if (w
!= (Widget
) NULL
)
434 winFocus
= wxGetWindowFromTable(w
);
444 bool wxWindowX11::Enable(bool enable
)
446 if ( !wxWindowBase::Enable(enable
) )
451 Widget wMain
= (Widget
)GetMainWidget();
454 XtSetSensitive(wMain
, enable
);
455 XmUpdateDisplay(wMain
);
462 bool wxWindowX11::Show(bool show
)
464 if ( !wxWindowBase::Show(show
) )
467 Window xwin
= (Window
) GetXWindow();
468 Display
*xdisp
= (Display
*) GetXDisplay();
470 XMapWindow(xdisp
, xwin
);
472 XUnmapWindow(xdisp
, xwin
);
477 // Raise the window to the top of the Z order
478 void wxWindowX11::Raise()
480 Window window
= GetTopWindow();
482 XRaiseWindow(wxGetDisplay(), window
);
485 // Lower the window to the bottom of the Z order
486 void wxWindowX11::Lower()
488 Window window
= GetTopWindow();
490 XLowerWindow(wxGetDisplay(), window
);
493 void wxWindowX11::SetTitle(const wxString
& title
)
496 // XtVaSetValues((Widget)GetMainWidget(), XmNtitle, title.c_str(), NULL);
499 wxString
wxWindowX11::GetTitle() const
502 return wxEmptyString
;
505 XtVaGetValues((Widget
)GetMainWidget(), XmNtitle
, &title
, NULL
);
507 return wxString(title
);
511 void wxWindowX11::DoCaptureMouse()
513 g_captureWindow
= this;
519 Widget wMain
= (Widget
)GetMainWidget();
521 XtAddGrab(wMain
, TRUE
, FALSE
);
524 m_winCaptured
= TRUE
;
527 void wxWindowX11::DoReleaseMouse()
529 g_captureWindow
= NULL
;
530 if ( !m_winCaptured
)
535 Widget wMain
= (Widget
)GetMainWidget();
540 m_winCaptured
= FALSE
;
543 bool wxWindowX11::SetFont(const wxFont
& font
)
545 if ( !wxWindowBase::SetFont(font
) )
556 bool wxWindowX11::SetCursor(const wxCursor
& cursor
)
558 if ( !wxWindowBase::SetCursor(cursor
) )
564 wxCursor
* cursor2
= NULL
;
566 cursor2
= & m_cursor
;
568 cursor2
= wxSTANDARD_CURSOR
;
570 WXDisplay
*dpy
= GetXDisplay();
571 WXCursor x_cursor
= cursor2
->GetXCursor(dpy
);
573 Window win
= (Window
) GetMainWindow();
574 XDefineCursor((Display
*) dpy
, win
, (Cursor
) x_cursor
);
579 // Coordinates relative to the window
580 void wxWindowX11::WarpPointer (int x
, int y
)
582 Window wClient
= (Window
) GetClientWindow();
584 XWarpPointer(wxGetDisplay(), None
, wClient
, 0, 0, 0, 0, x
, y
);
587 // ---------------------------------------------------------------------------
589 // ---------------------------------------------------------------------------
591 int wxWindowX11::GetScrollPos(int orient
) const
593 if (orient
== wxHORIZONTAL
)
599 // This now returns the whole range, not just the number of positions that we
601 int wxWindowX11::GetScrollRange(int WXUNUSED(orient
)) const
606 Widget scrollBar
= (Widget
)GetScrollbar((wxOrientation
)orient
);
607 wxCHECK_MSG( scrollBar
, 0, "no such scrollbar" );
610 XtVaGetValues(scrollBar
, XmNmaximum
, &range
, NULL
);
615 int wxWindowX11::GetScrollThumb(int orient
) const
621 Widget scrollBar
= (Widget
)GetScrollbar((wxOrientation
)orient
);
622 wxCHECK_MSG( scrollBar
, 0, "no such scrollbar" );
625 XtVaGetValues(scrollBar
, XmNsliderSize
, &thumb
, NULL
);
630 void wxWindowX11::SetScrollPos(int WXUNUSED(orient
), int WXUNUSED(pos
), bool WXUNUSED(refresh
))
635 Widget scrollBar
= (Widget
)GetScrollbar((wxOrientation
)orient
);
639 XtVaSetValues (scrollBar
, XmNvalue
, pos
, NULL
);
642 SetInternalScrollPos((wxOrientation
)orient
, pos
);
645 // New function that will replace some of the above.
646 void wxWindowX11::SetScrollbar(int WXUNUSED(orient
), int WXUNUSED(pos
), int WXUNUSED(thumbVisible
),
647 int WXUNUSED(range
), bool WXUNUSED(refresh
))
652 GetSize(& oldW
, & oldH
);
656 if (thumbVisible
== 0)
659 if (thumbVisible
> range
)
660 thumbVisible
= range
;
662 // Save the old state to see if it changed
663 WXWidget oldScrollBar
= GetScrollbar((wxOrientation
)orient
);
665 if (orient
== wxHORIZONTAL
)
667 if (thumbVisible
== range
)
670 DestroyScrollbar(wxHORIZONTAL
);
675 CreateScrollbar(wxHORIZONTAL
);
678 if (orient
== wxVERTICAL
)
680 if (thumbVisible
== range
)
683 DestroyScrollbar(wxVERTICAL
);
688 CreateScrollbar(wxVERTICAL
);
691 WXWidget newScrollBar
= GetScrollbar((wxOrientation
)orient
);
693 if (oldScrollBar
!= newScrollBar
)
695 // This is important! Without it, scrollbars misbehave badly.
696 XtUnrealizeWidget((Widget
) m_scrolledWindow
);
697 XmScrolledWindowSetAreas ((Widget
) m_scrolledWindow
, (Widget
) m_hScrollBar
, (Widget
) m_vScrollBar
, (Widget
) m_drawingArea
);
698 XtRealizeWidget((Widget
) m_scrolledWindow
);
699 XtManageChild((Widget
) m_scrolledWindow
);
704 XtVaSetValues((Widget
) newScrollBar
,
708 XmNsliderSize
, thumbVisible
,
712 SetInternalScrollPos((wxOrientation
)orient
, pos
);
715 GetSize(& newW
, & newH
);
717 // Adjusting scrollbars can resize the canvas accidentally
718 if (newW
!= oldW
|| newH
!= oldH
)
719 SetSize(-1, -1, oldW
, oldH
);
723 // Does a physical scroll
724 void wxWindowX11::ScrollWindow(int dx
, int dy
, const wxRect
*rect
)
729 // Use specified rectangle
730 x
= rect
->x
; y
= rect
->y
; w
= rect
->width
; h
= rect
->height
;
734 // Use whole client area
736 GetClientSize(& w
, & h
);
739 wxNode
*cnode
= m_children
.First();
742 wxWindow
*child
= (wxWindow
*) cnode
->Data();
745 child
->GetSize( &sx
, &sy
);
746 wxPoint
pos( child
->GetPosition() );
747 child
->SetSize( pos
.x
+ dx
, pos
.y
+ dy
, sx
, sy
, wxSIZE_ALLOW_MINUS_ONE
);
748 cnode
= cnode
->Next();
751 int x1
= (dx
>= 0) ? x
: x
- dx
;
752 int y1
= (dy
>= 0) ? y
: y
- dy
;
753 int w1
= w
- abs(dx
);
754 int h1
= h
- abs(dy
);
755 int x2
= (dx
>= 0) ? x
+ dx
: x
;
756 int y2
= (dy
>= 0) ? y
+ dy
: y
;
760 dc
.SetLogicalFunction (wxCOPY
);
762 Window window
= (Window
) GetMainWindow();
763 Display
* display
= wxGetDisplay();
765 XCopyArea(display
, window
, window
, (GC
) dc
.GetGC(),
766 x1
, y1
, w1
, h1
, x2
, y2
);
768 dc
.SetAutoSetting(TRUE
);
769 wxBrush
brush(GetBackgroundColour(), wxSOLID
);
770 dc
.SetBrush(brush
); // FIXME: needed?
772 // We'll add rectangles to the list of update rectangles according to which
773 // bits we've exposed.
778 wxRect
*rect
= new wxRect
;
784 XFillRectangle(display
, window
,
785 (GC
) dc
.GetGC(), rect
->x
, rect
->y
, rect
->width
, rect
->height
);
789 rect
->width
= rect
->width
;
790 rect
->height
= rect
->height
;
792 updateRects
.Append((wxObject
*) rect
);
796 wxRect
*rect
= new wxRect
;
798 rect
->x
= x
+ w
+ dx
;
803 XFillRectangle(display
, window
,
804 (GC
) dc
.GetGC(), rect
->x
, rect
->y
, rect
->width
,
809 rect
->width
= rect
->width
;
810 rect
->height
= rect
->height
;
812 updateRects
.Append((wxObject
*) rect
);
816 wxRect
*rect
= new wxRect
;
823 XFillRectangle(display
, window
,
824 (GC
) dc
.GetGC(), rect
->x
, rect
->y
, rect
->width
, rect
->height
);
828 rect
->width
= rect
->width
;
829 rect
->height
= rect
->height
;
831 updateRects
.Append((wxObject
*) rect
);
835 wxRect
*rect
= new wxRect
;
838 rect
->y
= y
+ h
+ dy
;
842 XFillRectangle(display
, window
,
843 (GC
) dc
.GetGC(), rect
->x
, rect
->y
, rect
->width
, rect
->height
);
847 rect
->width
= rect
->width
;
848 rect
->height
= rect
->height
;
850 updateRects
.Append((wxObject
*) rect
);
852 dc
.SetBrush(wxNullBrush
);
854 // Now send expose events
856 wxNode
* node
= updateRects
.First();
859 wxRect
* rect
= (wxRect
*) node
->Data();
863 event
.display
= display
;
864 event
.send_event
= True
;
865 event
.window
= window
;
869 event
.width
= rect
->width
;
870 event
.height
= rect
->height
;
874 XSendEvent(display
, window
, False
, ExposureMask
, (XEvent
*)&event
);
880 // Delete the update rects
881 node
= updateRects
.First();
884 wxRect
* rect
= (wxRect
*) node
->Data();
891 // XmUpdateDisplay((Widget) GetMainWidget());
894 // ---------------------------------------------------------------------------
896 // ---------------------------------------------------------------------------
898 #if wxUSE_DRAG_AND_DROP
900 void wxWindowX11::SetDropTarget(wxDropTarget
* WXUNUSED(pDropTarget
))
907 // Old style file-manager drag&drop
908 void wxWindowX11::DragAcceptFiles(bool WXUNUSED(accept
))
913 // ----------------------------------------------------------------------------
915 // ----------------------------------------------------------------------------
919 void wxWindowX11::DoSetToolTip(wxToolTip
* WXUNUSED(tooltip
))
924 #endif // wxUSE_TOOLTIPS
926 // ---------------------------------------------------------------------------
927 // moving and resizing
928 // ---------------------------------------------------------------------------
930 bool wxWindowX11::PreResize()
936 void wxWindowX11::DoGetSize(int *x
, int *y
) const
946 Widget widget
= (Widget
) GetTopWidget();
948 XtVaGetValues(widget
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
949 if(x
) *x
= xx
; if(y
) *y
= yy
;
953 void wxWindowX11::DoGetPosition(int *x
, int *y
) const
959 CanvasGetPosition(x
, y
);
962 Widget widget
= (Widget
) GetTopWidget();
964 XtVaGetValues(widget
, XmNx
, &xx
, XmNy
, &yy
, NULL
);
966 // We may be faking the client origin. So a window that's really at (0, 30)
967 // may appear (to wxWin apps) to be at (0, 0).
970 wxPoint
pt(GetParent()->GetClientAreaOrigin());
975 if(x
) *x
= xx
; if(y
) *y
= yy
;
979 void wxWindowX11::DoScreenToClient(int *x
, int *y
) const
981 Display
*display
= wxGetDisplay();
982 Window rootWindow
= RootWindowOfScreen(DefaultScreenOfDisplay(display
));
983 Window thisWindow
= (Window
) GetClientWindow();
988 XTranslateCoordinates(display
, rootWindow
, thisWindow
, xx
, yy
, x
, y
, &childWindow
);
991 void wxWindowX11::DoClientToScreen(int *x
, int *y
) const
993 Display
*display
= wxGetDisplay();
994 Window rootWindow
= RootWindowOfScreen(DefaultScreenOfDisplay(display
));
995 Window thisWindow
= (Window
) GetClientWindow();
1000 XTranslateCoordinates(display
, thisWindow
, rootWindow
, xx
, yy
, x
, y
, &childWindow
);
1004 // Get size *available for subwindows* i.e. excluding menu bar etc.
1005 void wxWindowX11::DoGetClientSize(int *x
, int *y
) const
1009 Widget widget
= (Widget
) GetClientWidget();
1011 XtVaGetValues(widget
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
1012 if(x
) *x
= xx
; if(y
) *y
= yy
;
1016 void wxWindowX11::DoSetSize(int x
, int y
, int width
, int height
, int sizeFlags
)
1020 // A bit of optimization to help sort out the flickers.
1021 int oldX
, oldY
, oldW
, oldH
;
1022 GetSize(& oldW
, & oldH
);
1023 GetPosition(& oldX
, & oldY
);
1025 if ( !(sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
) )
1038 bool nothingChanged
= (x
== oldX
) && (y
== oldY
) &&
1039 (width
== oldW
) && (height
== oldH
);
1041 if (!wxNoOptimize::CanOptimize())
1043 nothingChanged
= FALSE
;
1046 if ( !nothingChanged
)
1050 CanvasSetSize(x
, y
, width
, height
, sizeFlags
);
1054 Widget widget
= (Widget
) GetTopWidget();
1058 bool managed
= XtIsManaged( widget
);
1060 XtUnmanageChild(widget
);
1064 AdjustForParentClientOrigin(xx
, yy
, sizeFlags
);
1066 DoMoveWindow(xx
, yy
, width
, height
);
1069 XtManageChild(widget
);
1075 void wxWindowX11::DoSetClientSize(int width
, int height
)
1081 CanvasSetClientSize(width
, height
);
1085 Widget widget
= (Widget
) GetTopWidget();
1088 XtVaSetValues(widget
, XmNwidth
, width
, NULL
);
1090 XtVaSetValues(widget
, XmNheight
, height
, NULL
);
1092 wxSizeEvent
sizeEvent(wxSize(width
, height
), GetId());
1093 sizeEvent
.SetEventObject(this);
1095 GetEventHandler()->ProcessEvent(sizeEvent
);
1099 // For implementation purposes - sometimes decorations make the client area
1101 wxPoint
wxWindowX11::GetClientAreaOrigin() const
1103 return wxPoint(0, 0);
1106 // Makes an adjustment to the window position (for example, a frame that has
1107 // a toolbar that it manages itself).
1108 void wxWindowX11::AdjustForParentClientOrigin(int& x
, int& y
, int sizeFlags
)
1110 if (((sizeFlags
& wxSIZE_NO_ADJUSTMENTS
) == 0) && GetParent())
1112 wxPoint
pt(GetParent()->GetClientAreaOrigin());
1113 x
+= pt
.x
; y
+= pt
.y
;
1117 void wxWindowX11::SetSizeHints(int minW
, int minH
, int maxW
, int maxH
, int incW
, int incH
)
1126 wxFrame
*frame
= wxDynamicCast(this, wxFrame
);
1129 // TODO what about dialogs?
1133 Widget widget
= (Widget
) frame
->GetShellWidget();
1136 XtVaSetValues(widget
, XmNminWidth
, minW
, NULL
);
1138 XtVaSetValues(widget
, XmNminHeight
, minH
, NULL
);
1140 XtVaSetValues(widget
, XmNmaxWidth
, maxW
, NULL
);
1142 XtVaSetValues(widget
, XmNmaxHeight
, maxH
, NULL
);
1144 XtVaSetValues(widget
, XmNwidthInc
, incW
, NULL
);
1146 XtVaSetValues(widget
, XmNheightInc
, incH
, NULL
);
1150 void wxWindowX11::DoMoveWindow(int x
, int y
, int width
, int height
)
1154 XtVaSetValues((Widget
)GetTopWidget(),
1163 // ---------------------------------------------------------------------------
1165 // ---------------------------------------------------------------------------
1167 int wxWindowX11::GetCharHeight() const
1169 wxCHECK_MSG( m_font
.Ok(), 0, "valid window font needed" );
1171 WXFontStructPtr pFontStruct
= m_font
.GetFontStruct(1.0, GetXDisplay());
1173 int direction
, ascent
, descent
;
1174 XCharStruct overall
;
1175 XTextExtents ((XFontStruct
*) pFontStruct
, "x", 1, &direction
, &ascent
,
1176 &descent
, &overall
);
1178 // return (overall.ascent + overall.descent);
1179 return (ascent
+ descent
);
1182 int wxWindowX11::GetCharWidth() const
1184 wxCHECK_MSG( m_font
.Ok(), 0, "valid window font needed" );
1186 WXFontStructPtr pFontStruct
= m_font
.GetFontStruct(1.0, GetXDisplay());
1188 int direction
, ascent
, descent
;
1189 XCharStruct overall
;
1190 XTextExtents ((XFontStruct
*) pFontStruct
, "x", 1, &direction
, &ascent
,
1191 &descent
, &overall
);
1193 return overall
.width
;
1196 void wxWindowX11::GetTextExtent(const wxString
& string
,
1198 int *descent
, int *externalLeading
,
1199 const wxFont
*theFont
) const
1201 wxFont
*fontToUse
= (wxFont
*)theFont
;
1203 fontToUse
= (wxFont
*) & m_font
;
1205 wxCHECK_RET( fontToUse
->Ok(), "valid window font needed" );
1207 WXFontStructPtr pFontStruct
= theFont
->GetFontStruct(1.0, GetXDisplay());
1209 int direction
, ascent
, descent2
;
1210 XCharStruct overall
;
1211 int slen
= string
.Len();
1215 XTextExtents16((XFontStruct
*) pFontStruct
, (XChar2b
*) (char*) (const char*) string
, slen
, &direction
,
1216 &ascent
, &descent2
, &overall
);
1219 XTextExtents((XFontStruct
*) pFontStruct
, string
, slen
,
1220 &direction
, &ascent
, &descent2
, &overall
);
1223 *x
= (overall
.width
);
1225 *y
= (ascent
+ descent2
);
1227 *descent
= descent2
;
1228 if (externalLeading
)
1229 *externalLeading
= 0;
1233 // ----------------------------------------------------------------------------
1235 // ----------------------------------------------------------------------------
1237 void wxWindowX11::Refresh(bool eraseBack
, const wxRect
*rect
)
1239 m_needsRefresh
= TRUE
;
1240 Display
*display
= wxGetDisplay();
1241 Window thisWindow
= (Widget
) GetMainWindow();
1243 XExposeEvent dummyEvent
;
1245 GetSize(&width
, &height
);
1247 dummyEvent
.type
= Expose
;
1248 dummyEvent
.display
= display
;
1249 dummyEvent
.send_event
= True
;
1250 dummyEvent
.window
= thisWindow
;
1253 dummyEvent
.x
= rect
->x
;
1254 dummyEvent
.y
= rect
->y
;
1255 dummyEvent
.width
= rect
->width
;
1256 dummyEvent
.height
= rect
->height
;
1262 dummyEvent
.width
= width
;
1263 dummyEvent
.height
= height
;
1265 dummyEvent
.count
= 0;
1269 wxClientDC
dc(this);
1270 wxBrush
backgroundBrush(GetBackgroundColour(), wxSOLID
);
1271 dc
.SetBackground(backgroundBrush
);
1278 XSendEvent(display
, thisWindow
, False
, ExposureMask
, (XEvent
*)&dummyEvent
);
1281 void wxWindowX11::Clear()
1283 wxClientDC
dc(this);
1284 wxBrush
brush(GetBackgroundColour(), wxSOLID
);
1285 dc
.SetBackground(brush
);
1289 void wxWindowX11::ClearUpdateRects()
1291 wxRectList::Node
* node
= m_updateRects
.GetFirst();
1294 wxRect
* rect
= node
->GetData();
1296 node
= node
->GetNext();
1299 m_updateRects
.Clear();
1302 void wxWindowX11::DoPaint()
1304 // Set an erase event first
1305 wxEraseEvent
eraseEvent(GetId());
1306 eraseEvent
.SetEventObject(this);
1307 GetEventHandler()->ProcessEvent(eraseEvent
);
1309 wxPaintEvent
event(GetId());
1310 event
.SetEventObject(this);
1311 GetEventHandler()->ProcessEvent(event
);
1313 m_needsRefresh
= FALSE
;
1316 // ----------------------------------------------------------------------------
1318 // ----------------------------------------------------------------------------
1320 // Responds to colour changes: passes event on to children.
1321 void wxWindowX11::OnSysColourChanged(wxSysColourChangedEvent
& event
)
1323 wxWindowList::Node
*node
= GetChildren().GetFirst();
1326 // Only propagate to non-top-level windows
1327 wxWindow
*win
= node
->GetData();
1328 if ( win
->GetParent() )
1330 wxSysColourChangedEvent event2
;
1331 event
.m_eventObject
= win
;
1332 win
->GetEventHandler()->ProcessEvent(event2
);
1335 node
= node
->GetNext();
1339 void wxWindowX11::OnIdle(wxIdleEvent
& WXUNUSED(event
))
1341 // This calls the UI-update mechanism (querying windows for
1342 // menu/toolbar/control state information)
1346 // ----------------------------------------------------------------------------
1348 // ----------------------------------------------------------------------------
1350 bool wxWindowX11::ProcessAccelerator(wxKeyEvent
& event
)
1352 if (!m_acceleratorTable
.Ok())
1355 int count
= m_acceleratorTable
.GetCount();
1356 wxAcceleratorEntry
* entries
= m_acceleratorTable
.GetEntries();
1358 for (i
= 0; i
< count
; i
++)
1360 wxAcceleratorEntry
* entry
= & (entries
[i
]);
1361 if (entry
->MatchesEvent(event
))
1363 // Bingo, we have a match. Now find a control that matches the
1364 // entry command id.
1366 // Need to go up to the top of the window hierarchy, since it might
1367 // be e.g. a menu item
1368 wxWindow
* parent
= this;
1369 while ( parent
&& !parent
->IsTopLevel() )
1370 parent
= parent
->GetParent();
1375 wxFrame
* frame
= wxDynamicCast(parent
, wxFrame
);
1378 // Try for a menu command
1379 if (frame
->GetMenuBar())
1381 wxMenuItem
* item
= frame
->GetMenuBar()->FindItem(entry
->GetCommand());
1384 wxCommandEvent
commandEvent(wxEVT_COMMAND_MENU_SELECTED
, entry
->GetCommand());
1385 commandEvent
.SetEventObject(frame
);
1387 // If ProcessEvent returns TRUE (it was handled), then
1388 // the calling code will skip the event handling.
1389 return frame
->GetEventHandler()->ProcessEvent(commandEvent
);
1394 // Find a child matching the command id
1395 wxWindow
* child
= parent
->FindWindow(entry
->GetCommand());
1401 // Now we process those kinds of windows that we can.
1402 // For now, only buttons.
1403 if ( wxDynamicCast(child
, wxButton
) )
1405 wxCommandEvent
commandEvent (wxEVT_COMMAND_BUTTON_CLICKED
, child
->GetId());
1406 commandEvent
.SetEventObject(child
);
1407 return child
->GetEventHandler()->ProcessEvent(commandEvent
);
1414 // We didn't match the key event against an accelerator.
1418 // ============================================================================
1419 // X11-specific stuff from here on
1420 // ============================================================================
1422 // ----------------------------------------------------------------------------
1423 // function which maintain the global hash table mapping Widgets to wxWindows
1424 // ----------------------------------------------------------------------------
1426 bool wxAddWindowToTable(Window w
, wxWindow
*win
)
1428 wxWindow
*oldItem
= NULL
;
1429 if ((oldItem
= (wxWindow
*)wxWidgetHashTable
->Get ((long) w
)))
1431 wxLogDebug("Widget table clash: new widget is %ld, %s",
1432 (long)w
, win
->GetClassInfo()->GetClassName());
1436 wxWidgetHashTable
->Put((long) w
, win
);
1438 wxLogTrace("widget", "XWindow 0x%08x <-> window %p (%s)",
1439 w
, win
, win
->GetClassInfo()->GetClassName());
1444 wxWindow
*wxGetWindowFromTable(Window w
)
1446 return (wxWindow
*)wxWidgetHashTable
->Get((long) w
);
1449 void wxDeleteWindowFromTable(Window w
)
1451 wxWidgetHashTable
->Delete((long)w
);
1454 // ----------------------------------------------------------------------------
1455 // add/remove window from the table
1456 // ----------------------------------------------------------------------------
1458 // Add to hash table, add event handler
1459 bool wxWindowX11::AttachWidget (wxWindow
* WXUNUSED(parent
), WXWindow mainWidget
,
1460 int x
, int y
, int width
, int height
)
1462 wxAddWindowToTable((Window
) mainWidget
, this);
1466 if (CanAddEventHandler())
1468 XtAddEventHandler((Widget
) mainWidget
,
1469 ButtonPressMask
| ButtonReleaseMask
| PointerMotionMask
, // | KeyPressMask,
1471 wxPanelItemEventHandler
,
1478 XtOverrideTranslations ((Widget
) mainWidget
,
1479 ptr
= XtParseTranslationTable ("<Configure>: resize()"));
1480 XtFree ((char *) ptr
);
1483 // Some widgets have a parent form widget, e.g. wxRadioBox
1486 if (!wxAddWindowToTable((Widget
) formWidget
, this))
1490 XtOverrideTranslations ((Widget
) formWidget
,
1491 ptr
= XtParseTranslationTable ("<Configure>: resize()"));
1492 XtFree ((char *) ptr
);
1500 SetSize (x
, y
, width
, height
);
1505 // Remove event handler, remove from hash table
1506 bool wxWindowX11::DetachWidget(WXWindow widget
)
1510 if (CanAddEventHandler())
1512 XtRemoveEventHandler((Widget
) widget
,
1513 ButtonPressMask
| ButtonReleaseMask
| PointerMotionMask
, // | KeyPressMask,
1515 wxPanelItemEventHandler
,
1520 wxDeleteWindowFromTable((Window
) widget
);
1524 // ----------------------------------------------------------------------------
1525 // X11-specific accessors
1526 // ----------------------------------------------------------------------------
1528 // Get the underlying X window
1529 WXWindow
wxWindowX11::GetXWindow() const
1531 return GetMainWindow();
1534 // Get the underlying X display
1535 WXDisplay
*wxWindowX11::GetXDisplay() const
1537 return wxGetDisplay();
1540 WXWindow
wxWindowX11::GetMainWindow() const
1543 return m_drawingArea
;
1545 return m_mainWidget
;
1548 WXWindow
wxWindowX11::GetClientWidget() const
1550 if (m_drawingArea
!= (WXWindow
) 0)
1551 return m_drawingArea
;
1553 return GetMainWindow();
1556 WXWindow
wxWindowX11::GetTopWindow() const
1558 return GetMainWindow();
1561 WXWindow
wxWindowX11::GetLabelWindow() const
1563 return GetMainWindow();
1566 // ----------------------------------------------------------------------------
1568 // ----------------------------------------------------------------------------
1570 // TODO: implement wxWindow scrollbar, presumably using wxScrollBar
1572 static void wxScrollBarCallback(Widget scrollbar
,
1573 XtPointer clientData
,
1574 XmScrollBarCallbackStruct
*cbs
)
1576 wxWindow
*win
= wxGetWindowFromTable(scrollbar
);
1577 int orientation
= (int) clientData
;
1579 wxEventType eventType
= wxEVT_NULL
;
1580 switch (cbs
->reason
)
1582 case XmCR_INCREMENT
:
1584 eventType
= wxEVT_SCROLLWIN_LINEDOWN
;
1587 case XmCR_DECREMENT
:
1589 eventType
= wxEVT_SCROLLWIN_LINEUP
;
1594 eventType
= wxEVT_SCROLLWIN_THUMBTRACK
;
1597 case XmCR_VALUE_CHANGED
:
1599 eventType
= wxEVT_SCROLLWIN_THUMBRELEASE
;
1602 case XmCR_PAGE_INCREMENT
:
1604 eventType
= wxEVT_SCROLLWIN_PAGEDOWN
;
1607 case XmCR_PAGE_DECREMENT
:
1609 eventType
= wxEVT_SCROLLWIN_PAGEUP
;
1614 eventType
= wxEVT_SCROLLWIN_TOP
;
1617 case XmCR_TO_BOTTOM
:
1619 eventType
= wxEVT_SCROLLWIN_BOTTOM
;
1624 // Should never get here
1625 wxFAIL_MSG("Unknown scroll event.");
1630 wxScrollWinEvent
event(eventType
,
1632 ((orientation
== XmHORIZONTAL
) ?
1633 wxHORIZONTAL
: wxVERTICAL
));
1634 event
.SetEventObject( win
);
1635 win
->GetEventHandler()->ProcessEvent(event
);
1640 // ----------------------------------------------------------------------------
1641 // CanvaseXXXSize() functions
1642 // ----------------------------------------------------------------------------
1644 // SetSize, but as per old wxCanvas (with drawing widget etc.)
1645 void wxWindowX11::CanvasSetSize (int x
, int y
, int w
, int h
, int sizeFlags
)
1649 // A bit of optimization to help sort out the flickers.
1650 int oldX
, oldY
, oldW
, oldH
;
1651 GetSize(& oldW
, & oldH
);
1652 GetPosition(& oldX
, & oldY
);
1654 bool useOldPos
= FALSE
;
1655 bool useOldSize
= FALSE
;
1657 if ((x
== -1) && (x
== -1) && ((sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
) == 0))
1659 else if (x
== oldX
&& y
== oldY
)
1662 if ((w
== -1) && (h
== -1))
1664 else if (w
== oldW
&& h
== oldH
)
1667 if (!wxNoOptimize::CanOptimize())
1669 useOldSize
= FALSE
; useOldPos
= FALSE
;
1672 if (useOldPos
&& useOldSize
)
1675 Widget drawingArea
= (Widget
) m_drawingArea
;
1676 bool managed
= XtIsManaged(m_borderWidget
? (Widget
) m_borderWidget
: (Widget
) m_scrolledWindow
);
1679 XtUnmanageChild (m_borderWidget
? (Widget
) m_borderWidget
: (Widget
) m_scrolledWindow
);
1680 XtVaSetValues(drawingArea
, XmNresizePolicy
, XmRESIZE_ANY
, NULL
);
1682 int xx
= x
; int yy
= y
;
1683 AdjustForParentClientOrigin(xx
, yy
, sizeFlags
);
1687 if (x
> -1 || (sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
1689 XtVaSetValues (m_borderWidget
? (Widget
) m_borderWidget
: (Widget
) m_scrolledWindow
,
1693 if (y
> -1 || (sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
1695 XtVaSetValues (m_borderWidget
? (Widget
) m_borderWidget
: (Widget
) m_scrolledWindow
,
1707 XtVaSetValues ((Widget
) m_borderWidget
, XmNwidth
, w
, NULL
);
1708 short thick
, margin
;
1709 XtVaGetValues ((Widget
) m_borderWidget
,
1710 XmNshadowThickness
, &thick
,
1711 XmNmarginWidth
, &margin
,
1713 w
-= 2 * (thick
+ margin
);
1716 XtVaSetValues ((Widget
) m_scrolledWindow
, XmNwidth
, w
, NULL
);
1720 XtVaGetValues ((Widget
) m_scrolledWindow
,
1721 XmNspacing
, &spacing
,
1722 XmNverticalScrollBar
, &sbar
,
1726 XtVaGetValues (sbar
, XmNwidth
, &wsbar
, NULL
);
1730 w
-= (spacing
+ wsbar
);
1733 XtVaSetValues(drawingArea
, XmNwidth
, w
, NULL
);
1740 XtVaSetValues ((Widget
) m_borderWidget
, XmNheight
, h
, NULL
);
1741 short thick
, margin
;
1742 XtVaGetValues ((Widget
) m_borderWidget
,
1743 XmNshadowThickness
, &thick
,
1744 XmNmarginHeight
, &margin
,
1746 h
-= 2 * (thick
+ margin
);
1749 XtVaSetValues ((Widget
) m_scrolledWindow
, XmNheight
, h
, NULL
);
1753 XtVaGetValues ((Widget
) m_scrolledWindow
,
1754 XmNspacing
, &spacing
,
1755 XmNhorizontalScrollBar
, &sbar
,
1759 XtVaGetValues (sbar
, XmNheight
, &wsbar
, NULL
);
1763 h
-= (spacing
+ wsbar
);
1766 XtVaSetValues(drawingArea
, XmNheight
, h
, NULL
);
1772 XtManageChild (m_borderWidget
? (Widget
) m_borderWidget
: (Widget
) m_scrolledWindow
);
1773 XtVaSetValues(drawingArea
, XmNresizePolicy
, XmRESIZE_NONE
, NULL
);
1778 void wxWindowX11::CanvasSetClientSize (int w
, int h
)
1782 Widget drawingArea
= (Widget
) m_drawingArea
;
1784 XtVaSetValues(drawingArea
, XmNresizePolicy
, XmRESIZE_ANY
, NULL
);
1787 XtVaSetValues(drawingArea
, XmNwidth
, w
, NULL
);
1789 XtVaSetValues(drawingArea
, XmNheight
, h
, NULL
);
1791 XtVaSetValues(drawingArea
, XmNresizePolicy
, XmRESIZE_NONE
, NULL
);
1795 void wxWindowX11::CanvasGetClientSize (int *w
, int *h
) const
1799 // Must return the same thing that was set via SetClientSize
1801 XtVaGetValues ((Widget
) m_drawingArea
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
1807 void wxWindowX11::CanvasGetSize (int *w
, int *h
) const
1812 if ((Widget
) m_borderWidget
)
1813 XtVaGetValues ((Widget
) m_borderWidget
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
1814 else if ((Widget
) m_scrolledWindow
)
1815 XtVaGetValues ((Widget
) m_scrolledWindow
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
1817 XtVaGetValues ((Widget
) m_drawingArea
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
1824 void wxWindowX11::CanvasGetPosition (int *x
, int *y
) const
1829 XtVaGetValues (m_borderWidget
? (Widget
) m_borderWidget
: (Widget
) m_scrolledWindow
, XmNx
, &xx
, XmNy
, &yy
, NULL
);
1831 // We may be faking the client origin.
1832 // So a window that's really at (0, 30) may appear
1833 // (to wxWin apps) to be at (0, 0).
1836 wxPoint
pt(GetParent()->GetClientAreaOrigin());
1846 // ----------------------------------------------------------------------------
1847 // TranslateXXXEvent() functions
1848 // ----------------------------------------------------------------------------
1850 bool wxTranslateMouseEvent(wxMouseEvent
& wxevent
, wxWindow
*win
, Window window
, XEvent
*xevent
)
1852 switch (xevent
->xany
.type
)
1860 wxEventType eventType
= wxEVT_NULL
;
1862 if (xevent
->xany
.type
== EnterNotify
)
1864 //if (local_event.xcrossing.mode!=NotifyNormal)
1865 // return ; // Ignore grab events
1866 eventType
= wxEVT_ENTER_WINDOW
;
1867 // canvas->GetEventHandler()->OnSetFocus();
1869 else if (xevent
->xany
.type
== LeaveNotify
)
1871 //if (local_event.xcrossingr.mode!=NotifyNormal)
1872 // return ; // Ignore grab events
1873 eventType
= wxEVT_LEAVE_WINDOW
;
1874 // canvas->GetEventHandler()->OnKillFocus();
1876 else if (xevent
->xany
.type
== MotionNotify
)
1878 eventType
= wxEVT_MOTION
;
1880 else if (xevent
->xany
.type
== ButtonPress
)
1882 wxevent
.SetTimestamp(xevent
->xbutton
.time
);
1884 if (xevent
->xbutton
.button
== Button1
)
1886 eventType
= wxEVT_LEFT_DOWN
;
1887 win
->SetButton1(TRUE
);
1890 else if (xevent
->xbutton
.button
== Button2
)
1892 eventType
= wxEVT_MIDDLE_DOWN
;
1893 win
->SetButton2(TRUE
);
1896 else if (xevent
->xbutton
.button
== Button3
)
1898 eventType
= wxEVT_RIGHT_DOWN
;
1899 win
->SetButton3(TRUE
);
1903 // check for a double click
1904 // TODO: where can we get this value from?
1905 //long dclickTime = XtGetMultiClickTime((Display*) wxGetDisplay());
1906 long dClickTime
= 200;
1907 long ts
= wxevent
.GetTimestamp();
1909 int buttonLast
= win
->GetLastClickedButton();
1910 long lastTS
= win
->GetLastClickTime();
1911 if ( buttonLast
&& buttonLast
== button
&& (ts
- lastTS
) < dclickTime
)
1914 win
->SetLastClick(0, ts
);
1915 if ( eventType
== wxEVT_LEFT_DOWN
)
1916 eventType
= wxEVT_LEFT_DCLICK
;
1917 else if ( eventType
== wxEVT_MIDDLE_DOWN
)
1918 eventType
= wxEVT_MIDDLE_DCLICK
;
1919 else if ( eventType
== wxEVT_RIGHT_DOWN
)
1920 eventType
= wxEVT_RIGHT_DCLICK
;
1924 // not fast enough or different button
1925 win
->SetLastClick(button
, ts
);
1928 else if (xevent
->xany
.type
== ButtonRelease
)
1930 if (xevent
->xbutton
.button
== Button1
)
1932 eventType
= wxEVT_LEFT_UP
;
1933 win
->SetButton1(FALSE
);
1935 else if (xevent
->xbutton
.button
== Button2
)
1937 eventType
= wxEVT_MIDDLE_UP
;
1938 win
->SetButton2(FALSE
);
1940 else if (xevent
->xbutton
.button
== Button3
)
1942 eventType
= wxEVT_RIGHT_UP
;
1943 win
->SetButton3(FALSE
);
1952 wxevent
.SetEventType(eventType
);
1954 wxevent
.m_x
= xevent
->xbutton
.x
;
1955 wxevent
.m_y
= xevent
->xbutton
.y
;
1957 wxevent
.m_leftDown
= ((eventType
== wxEVT_LEFT_DOWN
)
1958 || (event_left_is_down (xevent
)
1959 && (eventType
!= wxEVT_LEFT_UP
)));
1960 wxevent
.m_middleDown
= ((eventType
== wxEVT_MIDDLE_DOWN
)
1961 || (event_middle_is_down (xevent
)
1962 && (eventType
!= wxEVT_MIDDLE_UP
)));
1963 wxevent
.m_rightDown
= ((eventType
== wxEVT_RIGHT_DOWN
)
1964 || (event_right_is_down (xevent
)
1965 && (eventType
!= wxEVT_RIGHT_UP
)));
1967 wxevent
.m_shiftDown
= xevent
->xbutton
.state
& ShiftMask
;
1968 wxevent
.m_controlDown
= xevent
->xbutton
.state
& ControlMask
;
1969 wxevent
.m_altDown
= xevent
->xbutton
.state
& Mod3Mask
;
1970 wxevent
.m_metaDown
= xevent
->xbutton
.state
& Mod1Mask
;
1972 wxevent
.SetId(win
->GetId());
1973 wxevent
.SetEventObject(win
);
1981 bool wxTranslateKeyEvent(wxKeyEvent
& wxevent
, wxWindow
*win
, Window
WXUNUSED(win
), XEvent
*xevent
)
1983 switch (xevent
->xany
.type
)
1991 (void) XLookupString ((XKeyEvent
*) xevent
, buf
, 20, &keySym
, NULL
);
1992 int id
= wxCharCodeXToWX (keySym
);
1994 if (xevent
->xkey
.state
& ShiftMask
)
1995 wxevent
.m_shiftDown
= TRUE
;
1996 if (xevent
->xkey
.state
& ControlMask
)
1997 wxevent
.m_controlDown
= TRUE
;
1998 if (xevent
->xkey
.state
& Mod3Mask
)
1999 wxevent
.m_altDown
= TRUE
;
2000 if (xevent
->xkey
.state
& Mod1Mask
)
2001 wxevent
.m_metaDown
= TRUE
;
2002 wxevent
.SetEventObject(win
);
2003 wxevent
.m_keyCode
= id
;
2004 wxevent
.SetTimestamp(xevent
->xkey
.time
);
2006 wxevent
.m_x
= xevent
->xbutton
.x
;
2007 wxevent
.m_y
= xevent
->xbutton
.y
;
2021 // ----------------------------------------------------------------------------
2023 // ----------------------------------------------------------------------------
2027 #define YAllocColor XAllocColor
2028 XColor g_itemColors
[5];
2029 int wxComputeColours (Display
*display
, wxColour
* back
, wxColour
* fore
)
2032 static XmColorProc colorProc
;
2034 result
= wxNO_COLORS
;
2038 g_itemColors
[0].red
= (((long) back
->Red ()) << 8);
2039 g_itemColors
[0].green
= (((long) back
->Green ()) << 8);
2040 g_itemColors
[0].blue
= (((long) back
->Blue ()) << 8);
2041 g_itemColors
[0].flags
= DoRed
| DoGreen
| DoBlue
;
2042 if (colorProc
== (XmColorProc
) NULL
)
2044 // Get a ptr to the actual function
2045 colorProc
= XmSetColorCalculation ((XmColorProc
) NULL
);
2046 // And set it back to motif.
2047 XmSetColorCalculation (colorProc
);
2049 (*colorProc
) (&g_itemColors
[wxBACK_INDEX
],
2050 &g_itemColors
[wxFORE_INDEX
],
2051 &g_itemColors
[wxSELE_INDEX
],
2052 &g_itemColors
[wxTOPS_INDEX
],
2053 &g_itemColors
[wxBOTS_INDEX
]);
2054 result
= wxBACK_COLORS
;
2058 g_itemColors
[wxFORE_INDEX
].red
= (((long) fore
->Red ()) << 8);
2059 g_itemColors
[wxFORE_INDEX
].green
= (((long) fore
->Green ()) << 8);
2060 g_itemColors
[wxFORE_INDEX
].blue
= (((long) fore
->Blue ()) << 8);
2061 g_itemColors
[wxFORE_INDEX
].flags
= DoRed
| DoGreen
| DoBlue
;
2062 if (result
== wxNO_COLORS
)
2063 result
= wxFORE_COLORS
;
2066 Display
*dpy
= display
;
2067 Colormap cmap
= (Colormap
) wxTheApp
->GetMainColormap((WXDisplay
*) dpy
);
2071 /* 5 Colours to allocate */
2072 for (int i
= 0; i
< 5; i
++)
2073 if (!YAllocColor (dpy
, cmap
, &g_itemColors
[i
]))
2074 result
= wxNO_COLORS
;
2078 /* Only 1 colour to allocate */
2079 if (!YAllocColor (dpy
, cmap
, &g_itemColors
[wxFORE_INDEX
]))
2080 result
= wxNO_COLORS
;
2088 // Changes the foreground and background colours to be derived from the current
2089 // background colour. To change the foreground colour, you must call
2090 // SetForegroundColour explicitly.
2091 void wxWindowX11::ChangeBackgroundColour()
2095 WXWidget mainWidget
= GetMainWidget();
2097 DoChangeBackgroundColour(mainWidget
, m_backgroundColour
);
2101 void wxWindowX11::ChangeForegroundColour()
2105 WXWidget mainWidget
= GetMainWidget();
2107 DoChangeForegroundColour(mainWidget
, m_foregroundColour
);
2108 if ( m_scrolledWindow
&& mainWidget
!= m_scrolledWindow
)
2109 DoChangeForegroundColour(m_scrolledWindow
, m_foregroundColour
);
2113 // Change a widget's foreground and background colours.
2114 void wxWindowX11::DoChangeForegroundColour(WXWindow widget
, wxColour
& foregroundColour
)
2118 // When should we specify the foreground, if it's calculated
2119 // by wxComputeColours?
2120 // Solution: say we start with the default (computed) foreground colour.
2121 // If we call SetForegroundColour explicitly for a control or window,
2122 // then the foreground is changed.
2123 // Therefore SetBackgroundColour computes the foreground colour, and
2124 // SetForegroundColour changes the foreground colour. The ordering is
2127 Widget w
= (Widget
)widget
;
2130 XmNforeground
, foregroundColour
.AllocColour(XtDisplay(w
)),
2136 void wxWindowX11::DoChangeBackgroundColour(WXWindow widget
, wxColour
& backgroundColour
, bool changeArmColour
)
2140 wxComputeColours (XtDisplay((Widget
) widget
), & backgroundColour
,
2143 XtVaSetValues ((Widget
) widget
,
2144 XmNbackground
, g_itemColors
[wxBACK_INDEX
].pixel
,
2145 XmNtopShadowColor
, g_itemColors
[wxTOPS_INDEX
].pixel
,
2146 XmNbottomShadowColor
, g_itemColors
[wxBOTS_INDEX
].pixel
,
2147 XmNforeground
, g_itemColors
[wxFORE_INDEX
].pixel
,
2150 if (changeArmColour
)
2151 XtVaSetValues ((Widget
) widget
,
2152 XmNarmColor
, g_itemColors
[wxSELE_INDEX
].pixel
,
2157 bool wxWindowX11::SetBackgroundColour(const wxColour
& col
)
2159 if ( !wxWindowBase::SetBackgroundColour(col
) )
2162 ChangeBackgroundColour();
2167 bool wxWindowX11::SetForegroundColour(const wxColour
& col
)
2169 if ( !wxWindowBase::SetForegroundColour(col
) )
2172 ChangeForegroundColour();
2177 void wxWindowX11::ChangeFont(bool keepOriginalSize
)
2181 // Note that this causes the widget to be resized back
2182 // to its original size! We therefore have to set the size
2183 // back again. TODO: a better way in Motif?
2184 Widget w
= (Widget
) GetLabelWidget(); // Usually the main widget
2185 if (w
&& m_font
.Ok())
2187 int width
, height
, width1
, height1
;
2188 GetSize(& width
, & height
);
2190 // lesstif 0.87 hangs here
2191 #ifndef LESSTIF_VERSION
2193 XmNfontList
, (XmFontList
) m_font
.GetFontList(1.0, XtDisplay(w
)),
2197 GetSize(& width1
, & height1
);
2198 if (keepOriginalSize
&& (width
!= width1
|| height
!= height1
))
2200 SetSize(-1, -1, width
, height
);
2206 // ----------------------------------------------------------------------------
2208 // ----------------------------------------------------------------------------
2210 wxWindow
*wxGetActiveWindow()
2213 wxFAIL_MSG("Not implemented");
2218 wxWindow
*wxWindowBase::GetCapture()
2220 return (wxWindow
*)g_captureWindow
;
2224 // Find the wxWindow at the current mouse position, returning the mouse
2226 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
)
2228 return wxFindWindowAtPoint(wxGetMousePosition());
2231 // Get the current mouse position.
2232 wxPoint
wxGetMousePosition()
2234 Display
*display
= (Display
*) wxGetDisplay();
2235 Window rootWindow
= RootWindowOfScreen (DefaultScreenOfDisplay(display
));
2236 Window rootReturn
, childReturn
;
2237 int rootX
, rootY
, winX
, winY
;
2238 unsigned int maskReturn
;
2240 XQueryPointer (display
,
2244 &rootX
, &rootY
, &winX
, &winY
, &maskReturn
);
2245 return wxPoint(rootX
, rootY
);
2249 // ----------------------------------------------------------------------------
2250 // wxNoOptimize: switch off size optimization
2251 // ----------------------------------------------------------------------------
2253 int wxNoOptimize::ms_count
= 0;