1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/dfb/window.cpp 
   4 // Author:      Vaclav Slavik 
   5 //              (based on GTK, MSW, MGL implementations) 
   8 // Copyright:   (c) 2006 REA Elektronik GmbH 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // =========================================================================== 
  14 // =========================================================================== 
  16 // --------------------------------------------------------------------------- 
  18 // --------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  27 #include "wx/window.h" 
  30     #include "wx/dcclient.h" 
  31     #include "wx/nonownedwnd.h" 
  35 #include "wx/dynarray.h" 
  37 #include "wx/dfb/private.h" 
  38 #include "wx/private/overlay.h" 
  40 #define TRACE_EVENTS _T("events") 
  41 #define TRACE_PAINT  _T("paint") 
  43 // =========================================================================== 
  45 // =========================================================================== 
  47 // --------------------------------------------------------------------------- 
  49 // --------------------------------------------------------------------------- 
  51 // the window that has keyboard focus: 
  52 static wxWindowDFB 
*gs_focusedWindow 
= NULL
; 
  53 // the window that is about to be focused after currently focused 
  55 static wxWindow 
*gs_toBeFocusedWindow 
= NULL
; 
  56 // the window that has mouse capture 
  57 static wxWindowDFB 
*gs_mouseCapture 
= NULL
; 
  59 // --------------------------------------------------------------------------- 
  61 // --------------------------------------------------------------------------- 
  63 WX_DEFINE_ARRAY_PTR(wxOverlayImpl
*, wxDfbOverlaysList
); 
  65 // --------------------------------------------------------------------------- 
  67 // --------------------------------------------------------------------------- 
  69 // in wxUniv this class is abstract because it doesn't have DoPopupMenu() 
  70 IMPLEMENT_ABSTRACT_CLASS(wxWindowDFB
, wxWindowBase
) 
  72 BEGIN_EVENT_TABLE(wxWindowDFB
, wxWindowBase
) 
  75 // ---------------------------------------------------------------------------- 
  76 // constructors and such 
  77 // ---------------------------------------------------------------------------- 
  79 void wxWindowDFB::Init() 
  88 wxWindowDFB::~wxWindowDFB() 
  92     m_isBeingDeleted 
= true; 
  94     if ( gs_mouseCapture 
== this ) 
  97     if ( gs_focusedWindow 
== this ) 
 103 // real construction (Init() must have been called before!) 
 104 bool wxWindowDFB::Create(wxWindow 
*parent
, 
 109                          const wxString
& name
) 
 111     if ( !m_tlw 
&& parent 
) 
 112         m_tlw 
= parent
->GetTLW(); 
 114     if ( !CreateBase(parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
) ) 
 118         parent
->AddChild(this); 
 120     // set the size to something bogus initially, in case some code tries to 
 121     // create wxWindowDC before SetSize() is called below: 
 122     m_rect
.width 
= m_rect
.height 
= 1; 
 125     x 
= pos
.x
, y 
= pos
.y
; 
 126     if ( x 
== -1  ) x 
= 0; 
 127     if ( y 
== -1 ) y 
= 0; 
 128     w 
= WidthDefault(size
.x
); 
 129     h 
= HeightDefault(size
.y
); 
 135 // --------------------------------------------------------------------------- 
 137 // --------------------------------------------------------------------------- 
 139 wxIDirectFBSurfacePtr 
wxWindowDFB::ObtainDfbSurface() const 
 141     wxCHECK_MSG( m_parent
, NULL
, _T("parentless window?") ); 
 143     wxIDirectFBSurfacePtr 
parentSurface(m_parent
->GetDfbSurface()); 
 144     wxCHECK_MSG( parentSurface
, NULL
, _T("invalid parent surface") ); 
 147     AdjustForParentClientOrigin(r
.x
, r
.y
, 0); 
 148     DFBRectangle rect 
= { r
.x
, r
.y
, r
.width
, r
.height 
}; 
 150     return parentSurface
->GetSubSurface(&rect
); 
 153 wxIDirectFBSurfacePtr 
wxWindowDFB::GetDfbSurface() 
 157         m_surface 
= ObtainDfbSurface(); 
 158         wxASSERT_MSG( m_surface
, _T("invalid DirectFB surface") ); 
 164 void wxWindowDFB::InvalidateDfbSurface() 
 168     // surfaces of the children are subsurfaces of this window's surface, 
 169     // so they must be invalidated as well: 
 170     wxWindowList
& children 
= GetChildren(); 
 171     for ( wxWindowList::iterator i 
= children
.begin(); i 
!= children
.end(); ++i 
) 
 173         (*i
)->InvalidateDfbSurface(); 
 177 // --------------------------------------------------------------------------- 
 179 // --------------------------------------------------------------------------- 
 181 void wxWindowDFB::SetFocus() 
 183     if ( gs_focusedWindow 
== this ) 
 184         return; // nothing to do, focused already 
 186     wxWindowDFB 
*oldFocusedWindow 
= gs_focusedWindow
; 
 188     if ( gs_focusedWindow 
) 
 190         gs_toBeFocusedWindow 
= (wxWindow
*)this; 
 191         gs_focusedWindow
->DFBKillFocus(); 
 192         gs_toBeFocusedWindow 
= NULL
; 
 195     gs_focusedWindow 
= this; 
 197     if ( IsShownOnScreen() ) 
 199         m_tlw
->SetDfbFocus(); 
 201     // else: do nothing, because DirectFB windows cannot have focus if they 
 202     //       are hidden; when the TLW becomes visible, it will set the focus 
 203     //       to use from wxTLW::Show() 
 205     // notify the parent keeping track of focus for the kbd navigation 
 206     // purposes that we got it 
 207     wxChildFocusEvent 
eventFocus((wxWindow
*)this); 
 208     GetEventHandler()->ProcessEvent(eventFocus
); 
 210     wxFocusEvent 
event(wxEVT_SET_FOCUS
, GetId()); 
 211     event
.SetEventObject(this); 
 212     event
.SetWindow((wxWindow
*)oldFocusedWindow
); 
 213     GetEventHandler()->ProcessEvent(event
); 
 216     // caret needs to be informed about focus change 
 217     wxCaret 
*caret 
= GetCaret(); 
 220 #endif // wxUSE_CARET 
 223 void wxWindowDFB::DFBKillFocus() 
 225     wxCHECK_RET( gs_focusedWindow 
== this, 
 226                  _T("killing focus on window that doesn't have it") ); 
 228     gs_focusedWindow 
= NULL
; 
 230     if ( m_isBeingDeleted 
) 
 231         return; // don't send any events from dtor 
 234     // caret needs to be informed about focus change 
 235     wxCaret 
*caret 
= GetCaret(); 
 237         caret
->OnKillFocus(); 
 238 #endif // wxUSE_CARET 
 240     wxFocusEvent 
event(wxEVT_KILL_FOCUS
, GetId()); 
 241     event
.SetEventObject(this); 
 242     event
.SetWindow(gs_toBeFocusedWindow
); 
 243     GetEventHandler()->ProcessEvent(event
); 
 246 // ---------------------------------------------------------------------------- 
 247 // this wxWindowBase function is implemented here (in platform-specific file) 
 248 // because it is static and so couldn't be made virtual 
 249 // ---------------------------------------------------------------------------- 
 250 wxWindow 
*wxWindowBase::DoFindFocus() 
 252     return (wxWindow
*)gs_focusedWindow
; 
 255 bool wxWindowDFB::Show(bool show
) 
 257     if ( !wxWindowBase::Show(show
) ) 
 260     // Unlike Refresh(), DoRefreshWindow() doesn't check visibility, so 
 261     // call it to force refresh of either this window (if showing) or its 
 262     // parent area at the place of this window (if hiding): 
 268 // Raise the window to the top of the Z order 
 269 void wxWindowDFB::Raise() 
 271     wxFAIL_MSG( _T("Raise() not implemented") ); 
 274 // Lower the window to the bottom of the Z order 
 275 void wxWindowDFB::Lower() 
 277     wxFAIL_MSG( _T("Lower() not implemented") ); 
 280 void wxWindowDFB::DoCaptureMouse() 
 282 #warning "implement this" 
 284     if ( gs_mouseCapture 
) 
 285         DFB_wmUncaptureEvents(gs_mouseCapture
->m_wnd
, wxDFB_CAPTURE_MOUSE
); 
 287     gs_mouseCapture 
= this; 
 289     DFB_wmCaptureEvents(m_wnd
, EVT_MOUSEEVT
, wxDFB_CAPTURE_MOUSE
); 
 293 void wxWindowDFB::DoReleaseMouse() 
 295     wxASSERT_MSG( gs_mouseCapture 
== this, wxT("attempt to release mouse, but this window hasn't captured it") ); 
 297 #warning "implement this" 
 299     DFB_wmUncaptureEvents(m_wnd
, wxDFB_CAPTURE_MOUSE
); 
 301     gs_mouseCapture 
= NULL
; 
 304 /* static */ wxWindow 
*wxWindowBase::GetCapture() 
 306     return (wxWindow
*)gs_mouseCapture
; 
 309 bool wxWindowDFB::SetCursor(const wxCursor
& cursor
) 
 311     if ( !wxWindowBase::SetCursor(cursor
) ) 
 317 #warning "implement this" 
 320         DFB_wmSetWindowCursor(m_wnd
, *m_cursor
.GetDFBCursor()); 
 322         DFB_wmSetWindowCursor(m_wnd
, *wxSTANDARD_CURSOR
->GetDFBCursor()); 
 328 void wxWindowDFB::WarpPointer(int x
, int y
) 
 331     wxDisplaySize(&w
, &h
); 
 333     ClientToScreen(&x
, &y
); 
 336     if ( x 
>= w 
) x 
= w
-1; 
 337     if ( y 
>= h 
) y 
= h
-1; 
 339     wxIDirectFBDisplayLayerPtr 
layer(wxIDirectFB::Get()->GetDisplayLayer()); 
 340     wxCHECK_RET( layer
, _T("no display layer") ); 
 342     layer
->WarpCursor(x
, y
); 
 345 // Set this window to be the child of 'parent'. 
 346 bool wxWindowDFB::Reparent(wxWindowBase 
*parent
) 
 348     if ( !wxWindowBase::Reparent(parent
) ) 
 351 #warning "implement this" 
 352     wxFAIL_MSG( _T("reparenting not yet implemented") ); 
 357 // --------------------------------------------------------------------------- 
 358 // moving and resizing 
 359 // --------------------------------------------------------------------------- 
 362 void wxWindowDFB::DoGetSize(int *x
, int *y
) const 
 364     if (x
) *x 
= m_rect
.width
; 
 365     if (y
) *y 
= m_rect
.height
; 
 368 void wxWindowDFB::DoGetPosition(int *x
, int *y
) const 
 370     if (x
) *x 
= m_rect
.x
; 
 371     if (y
) *y 
= m_rect
.y
; 
 374 static wxPoint 
GetScreenPosOfClientOrigin(const wxWindowDFB 
*win
) 
 376     wxCHECK_MSG( win
, wxPoint(0, 0), _T("no window provided") ); 
 378     wxPoint 
pt(win
->GetPosition() + win
->GetClientAreaOrigin()); 
 380     if ( !win
->IsTopLevel() ) 
 381         pt 
+= GetScreenPosOfClientOrigin(win
->GetParent()); 
 386 void wxWindowDFB::DoScreenToClient(int *x
, int *y
) const 
 388     wxPoint o 
= GetScreenPosOfClientOrigin(this); 
 394 void wxWindowDFB::DoClientToScreen(int *x
, int *y
) const 
 396     wxPoint o 
= GetScreenPosOfClientOrigin(this); 
 402 // Get size *available for subwindows* i.e. excluding menu bar etc. 
 403 void wxWindowDFB::DoGetClientSize(int *x
, int *y
) const 
 408 void wxWindowDFB::DoMoveWindow(int x
, int y
, int width
, int height
) 
 410     // NB: [x,y] arguments are in (parent's) window coordinates, while 
 411     //     m_rect.{x,y} are in (parent's) client coordinates. That's why we 
 412     //     offset by parentOrigin in some places below 
 414     wxPoint 
parentOrigin(0, 0); 
 415     AdjustForParentClientOrigin(parentOrigin
.x
, parentOrigin
.y
); 
 417     wxRect 
oldpos(m_rect
); 
 418     oldpos
.Offset(parentOrigin
); 
 420     wxRect 
newpos(x
, y
, width
, height
); 
 422     // input [x,y] is in window coords, but we store client coords in m_rect: 
 424     m_rect
.Offset(-parentOrigin
); 
 426     // window's position+size changed and so did the subsurface that covers it 
 427     InvalidateDfbSurface(); 
 431         // queue both former and new position of the window for repainting: 
 432         wxWindow 
*parent 
= GetParent(); 
 434         // only refresh the visible parts: 
 435         if ( !CanBeOutsideClientArea() ) 
 437             wxRect 
parentClient(parent
->GetClientSize()); 
 438             oldpos
.Intersect(parentClient
); 
 439             newpos
.Intersect(parentClient
); 
 442         parent
->RefreshRect(oldpos
); 
 443         parent
->RefreshRect(newpos
); 
 447 // set the size of the window: if the dimensions are positive, just use them, 
 448 // but if any of them is equal to -1, it means that we must find the value for 
 449 // it ourselves (unless sizeFlags contains wxSIZE_ALLOW_MINUS_ONE flag, in 
 450 // which case -1 is a valid value for x and y) 
 452 // If sizeFlags contains wxSIZE_AUTO_WIDTH/HEIGHT flags (default), we calculate 
 453 // the width/height to best suit our contents, otherwise we reuse the current 
 455 void wxWindowDFB::DoSetSize(int x
, int y
, int width
, int height
, int sizeFlags
) 
 457     // get the current size and position... 
 458     int currentX
, currentY
; 
 459     GetPosition(¤tX
, ¤tY
); 
 460     int currentW
,currentH
; 
 461     GetSize(¤tW
, ¤tH
); 
 463     if ( x 
== -1 && !(sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
) ) 
 465     if ( y 
== -1 && !(sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
) ) 
 468     // ... and don't do anything (avoiding flicker) if it's already ok 
 469     if ( x 
== currentX 
&& y 
== currentY 
&& 
 470          width 
== currentW 
&& height 
== currentH 
) 
 478         if ( sizeFlags 
& wxSIZE_AUTO_WIDTH 
) 
 480             size 
= DoGetBestSize(); 
 485             // just take the current one 
 492         if ( sizeFlags 
& wxSIZE_AUTO_HEIGHT 
) 
 496                 size 
= DoGetBestSize(); 
 498             //else: already called DoGetBestSize() above 
 504             // just take the current one 
 509     int maxWidth 
= GetMaxWidth(), 
 510         minWidth 
= GetMinWidth(), 
 511         maxHeight 
= GetMaxHeight(), 
 512         minHeight 
= GetMinHeight(); 
 514     if ( minWidth 
!= -1 && width 
< minWidth 
) width 
= minWidth
; 
 515     if ( maxWidth 
!= -1 && width 
> maxWidth 
) width 
= maxWidth
; 
 516     if ( minHeight 
!= -1 && height 
< minHeight 
) height 
= minHeight
; 
 517     if ( maxHeight 
!= -1 && height 
> maxHeight 
) height 
= maxHeight
; 
 519     if ( m_rect
.x 
!= x 
|| m_rect
.y 
!= y 
|| 
 520          m_rect
.width 
!= width 
|| m_rect
.height 
!= height 
) 
 522         AdjustForParentClientOrigin(x
, y
, sizeFlags
); 
 523         DoMoveWindow(x
, y
, width
, height
); 
 525         wxSize 
newSize(width
, height
); 
 526         wxSizeEvent 
event(newSize
, GetId()); 
 527         event
.SetEventObject(this); 
 528         GetEventHandler()->ProcessEvent(event
); 
 532 void wxWindowDFB::DoSetClientSize(int width
, int height
) 
 534     SetSize(width
, height
); 
 537 // --------------------------------------------------------------------------- 
 539 // --------------------------------------------------------------------------- 
 541 int wxWindowDFB::GetCharHeight() const 
 543     wxWindowDC 
dc((wxWindow
*)this); 
 544     return dc
.GetCharHeight(); 
 547 int wxWindowDFB::GetCharWidth() const 
 549     wxWindowDC 
dc((wxWindow
*)this); 
 550     return dc
.GetCharWidth(); 
 553 void wxWindowDFB::GetTextExtent(const wxString
& string
, 
 555                              int *descent
, int *externalLeading
, 
 556                              const wxFont 
*theFont
) const 
 558     wxWindowDC 
dc((wxWindow
*)this); 
 559     dc
.GetTextExtent(string
, x
, y
, descent
, externalLeading
, (wxFont
*)theFont
); 
 563 // --------------------------------------------------------------------------- 
 565 // --------------------------------------------------------------------------- 
 567 void wxWindowDFB::Clear() 
 569     wxClientDC 
dc((wxWindow 
*)this); 
 570     wxBrush 
brush(GetBackgroundColour(), wxSOLID
); 
 571     dc
.SetBackground(brush
); 
 575 void wxWindowDFB::Refresh(bool WXUNUSED(eraseBack
), const wxRect 
*rect
) 
 577     if ( !IsShown() || IsFrozen() ) 
 580     // NB[1]: We intentionally ignore the eraseBack argument here. This is 
 581     //        because of the way wxDFB's painting is implemented: the refresh 
 582     //        request is propagated up to wxTLW, which is then painted in 
 583     //        top-down order. This means that this window's area is first 
 584     //        painted by its parent and this window is then painted over it, so 
 585     //        it's not safe to not paint this window's background even if 
 587     // NB[2]: wxWindow::Refresh() takes the rectangle in client coords, but 
 588     //        wxUniv translates it to window coords before passing it to 
 589     //        wxWindowDFB::Refresh(), so we can directly pass the rect to 
 590     //        DoRefreshRect (which takes window, not client, coords) here. 
 592         DoRefreshRect(*rect
); 
 597 void wxWindowDFB::RefreshWindowRect(const wxRect
& rect
) 
 599     if ( !IsShown() || IsFrozen() ) 
 605 void wxWindowDFB::DoRefreshWindow() 
 607     // NB: DoRefreshRect() takes window coords, not client, so this is correct 
 608     DoRefreshRect(wxRect(GetSize())); 
 611 void wxWindowDFB::DoRefreshRect(const wxRect
& rect
) 
 613     wxWindow 
*parent 
= GetParent(); 
 614     wxCHECK_RET( parent
, _T("no parent") ); 
 616     // don't overlap outside of the window (NB: 'rect' is in window coords): 
 618     r
.Intersect(wxRect(GetSize())); 
 622     wxLogTrace(TRACE_PAINT
, 
 623                _T("%p ('%s'): refresh rect [%i,%i,%i,%i]"), 
 624                this, GetName().c_str(), 
 625                rect
.x
, rect
.y
, rect
.GetRight(), rect
.GetBottom()); 
 627     // convert the refresh rectangle to parent's coordinates and 
 628     // recursively refresh the parent: 
 629     r
.Offset(GetPosition()); 
 630     r
.Offset(parent
->GetClientAreaOrigin()); 
 632     // normal windows cannot extend out of its parent's client area, so don't 
 633     // refresh any hidden parts: 
 634     if ( !CanBeOutsideClientArea() ) 
 635         r
.Intersect(parent
->GetClientRect()); 
 637     parent
->DoRefreshRect(r
); 
 640 void wxWindowDFB::Update() 
 642     if ( !IsShown() || IsFrozen() ) 
 645     GetParent()->Update(); 
 648 void wxWindowDFB::Freeze() 
 653 void wxWindowDFB::Thaw() 
 655     wxASSERT_MSG( IsFrozen(), _T("Thaw() without matching Freeze()") ); 
 657     if ( --m_frozenness 
== 0 ) 
 664 void wxWindowDFB::PaintWindow(const wxRect
& rect
) 
 666     wxCHECK_RET( !IsFrozen() && IsShown(), _T("shouldn't be called") ); 
 668     wxLogTrace(TRACE_PAINT
, 
 669                _T("%p ('%s'): painting region [%i,%i,%i,%i]"), 
 670                this, GetName().c_str(), 
 671                rect
.x
, rect
.y
, rect
.GetRight(), rect
.GetBottom()); 
 673     m_updateRegion 
= rect
; 
 675     // FIXME_DFB: don't waste time rendering the area if it's fully covered 
 676     //            by some children, go directly to rendering the children 
 678     // NB: unconditionally send wxEraseEvent, because our implementation of 
 679     //     wxWindow::Refresh() ignores the eraseBack argument 
 680     wxWindowDC 
dc((wxWindow
*)this); 
 681     wxEraseEvent 
eventEr(m_windowId
, &dc
); 
 682     eventEr
.SetEventObject(this); 
 683     GetEventHandler()->ProcessEvent(eventEr
); 
 685     wxRect 
clientRect(GetClientRect()); 
 687     // only send wxNcPaintEvent if drawing at least part of nonclient area: 
 688     if ( !clientRect
.Contains(rect
) ) 
 690         wxNcPaintEvent 
eventNc(GetId()); 
 691         eventNc
.SetEventObject(this); 
 692         GetEventHandler()->ProcessEvent(eventNc
); 
 696         wxLogTrace(TRACE_PAINT
, _T("%p ('%s'): not sending wxNcPaintEvent"), 
 697                    this, GetName().c_str()); 
 700     // only send wxPaintEvent if drawing at least part of client area: 
 701     if ( rect
.Intersects(clientRect
) ) 
 703         wxPaintEvent 
eventPt(GetId()); 
 704         eventPt
.SetEventObject(this); 
 705         GetEventHandler()->ProcessEvent(eventPt
); 
 709         wxLogTrace(TRACE_PAINT
, _T("%p ('%s'): not sending wxPaintEvent"), 
 710                    this, GetName().c_str()); 
 713     // draw window's overlays on top of the painted window, if we have any: 
 716     m_updateRegion
.Clear(); 
 718     // client area portion of 'rect': 
 719     wxRect 
rectClientOnly(rect
); 
 720     rectClientOnly
.Intersect(clientRect
); 
 722     // paint the children: 
 723     wxPoint origin 
= GetClientAreaOrigin(); 
 724     wxWindowList
& children 
= GetChildren(); 
 725     for ( wxWindowList::iterator i 
= children
.begin(); 
 726           i 
!= children
.end(); ++i 
) 
 728         wxWindow 
*child 
= *i
; 
 730         if ( child
->IsFrozen() || !child
->IsShown() ) 
 731             continue; // don't paint anything if the window is frozen or hidden 
 733         // compute child's area to repaint 
 734         wxRect 
childrect(child
->GetRect()); 
 735         childrect
.Offset(origin
); 
 737         if ( child
->CanBeOutsideClientArea() ) 
 738             childrect
.Intersect(rect
); 
 740             childrect
.Intersect(rectClientOnly
); 
 742         if ( childrect
.IsEmpty() ) 
 746         childrect
.Offset(-child
->GetPosition()); 
 747         childrect
.Offset(-origin
); 
 748         child
->PaintWindow(childrect
); 
 752 void wxWindowDFB::PaintOverlays(const wxRect
& rect
) 
 757     for ( wxDfbOverlaysList::const_iterator i 
= m_overlays
->begin(); 
 758           i 
!= m_overlays
->end(); ++i 
) 
 760         wxOverlayImpl 
*overlay 
= *i
; 
 762         wxRect 
orectOrig(overlay
->GetRect()); 
 763         wxRect 
orect(orectOrig
); 
 764         orect
.Intersect(rect
); 
 765         if ( orect
.IsEmpty() ) 
 768         if ( overlay
->IsEmpty() ) 
 769             continue; // nothing to paint 
 771         DFBRectangle dfbRect 
= { orect
.x 
- orectOrig
.x
, orect
.y 
- orectOrig
.y
, 
 772                                  orect
.width
, orect
.height 
}; 
 773         GetDfbSurface()->Blit
 
 775                            overlay
->GetDirectFBSurface(), 
 782 void wxWindowDFB::AddOverlay(wxOverlayImpl 
*overlay
) 
 785         m_overlays 
= new wxDfbOverlaysList
; 
 787     m_overlays
->Add(overlay
); 
 790 void wxWindowDFB::RemoveOverlay(wxOverlayImpl 
*overlay
) 
 792     wxCHECK_RET( m_overlays
, _T("no overlays to remove") ); 
 794     m_overlays
->Remove(overlay
); 
 796     if ( m_overlays
->empty() ) 
 798         wxDELETE(m_overlays
); 
 801     if ( !m_isBeingDeleted 
) 
 802         RefreshWindowRect(overlay
->GetRect()); 
 806 // --------------------------------------------------------------------------- 
 808 // --------------------------------------------------------------------------- 
 810 #define KEY(dfb, wx)                                                \ 
 812           wxLogTrace(TRACE_EVENTS,                                  \ 
 813                      _T("key " #dfb " mapped to " #wx));            \ 
 816 // returns translated keycode, i.e. the one for KEYUP/KEYDOWN where 'a'..'z' is 
 817 // translated to 'A'..'Z' 
 818 static long GetTranslatedKeyCode(DFBInputDeviceKeyIdentifier key_id
) 
 822         KEY(DIKI_UNKNOWN
,           0); 
 862         KEY(DIKI_F1
,                WXK_F1
); 
 863         KEY(DIKI_F2
,                WXK_F2
); 
 864         KEY(DIKI_F3
,                WXK_F3
); 
 865         KEY(DIKI_F4
,                WXK_F4
); 
 866         KEY(DIKI_F5
,                WXK_F5
); 
 867         KEY(DIKI_F6
,                WXK_F6
); 
 868         KEY(DIKI_F7
,                WXK_F7
); 
 869         KEY(DIKI_F8
,                WXK_F8
); 
 870         KEY(DIKI_F9
,                WXK_F9
); 
 871         KEY(DIKI_F10
,               WXK_F10
); 
 872         KEY(DIKI_F11
,               WXK_F11
); 
 873         KEY(DIKI_F12
,               WXK_F12
); 
 875         KEY(DIKI_SHIFT_L
,           WXK_SHIFT
); 
 876         KEY(DIKI_SHIFT_R
,           WXK_SHIFT
); 
 877         KEY(DIKI_CONTROL_L
,         WXK_CONTROL
); 
 878         KEY(DIKI_CONTROL_R
,         WXK_CONTROL
); 
 879         KEY(DIKI_ALT_L
,             WXK_ALT
); 
 880         KEY(DIKI_ALT_R
,             WXK_ALT
); 
 881         // this key was removed in 0.9.25 but include it for previous versions 
 882         // just to avoid gcc warnings about unhandled enum value in switch 
 883 #if !wxCHECK_DFB_VERSION(0, 9, 24) 
 888         KEY(DIKI_SUPER_L
,           0); 
 889         KEY(DIKI_SUPER_R
,           0); 
 890         KEY(DIKI_HYPER_L
,           0); 
 891         KEY(DIKI_HYPER_R
,           0); 
 893         KEY(DIKI_CAPS_LOCK
,         0); 
 894         KEY(DIKI_NUM_LOCK
,          WXK_NUMLOCK
); 
 895         KEY(DIKI_SCROLL_LOCK
,       0); 
 897         KEY(DIKI_ESCAPE
,            WXK_ESCAPE
); 
 898         KEY(DIKI_LEFT
,              WXK_LEFT
); 
 899         KEY(DIKI_RIGHT
,             WXK_RIGHT
); 
 900         KEY(DIKI_UP
,                WXK_UP
); 
 901         KEY(DIKI_DOWN
,              WXK_DOWN
); 
 902         KEY(DIKI_TAB
,               WXK_TAB
); 
 903         KEY(DIKI_ENTER
,             WXK_RETURN
); 
 904         KEY(DIKI_SPACE
,             WXK_SPACE
); 
 905         KEY(DIKI_BACKSPACE
,         WXK_BACK
); 
 906         KEY(DIKI_INSERT
,            WXK_INSERT
); 
 907         KEY(DIKI_DELETE
,            WXK_DELETE
); 
 908         KEY(DIKI_HOME
,              WXK_HOME
); 
 909         KEY(DIKI_END
,               WXK_END
); 
 910         KEY(DIKI_PAGE_UP
,           WXK_PAGEUP
); 
 911         KEY(DIKI_PAGE_DOWN
,         WXK_PAGEDOWN
); 
 912         KEY(DIKI_PRINT
,             WXK_PRINT
); 
 913         KEY(DIKI_PAUSE
,             WXK_PAUSE
); 
 915         KEY(DIKI_QUOTE_LEFT
,        '`'); 
 916         KEY(DIKI_MINUS_SIGN
,        '-'); 
 917         KEY(DIKI_EQUALS_SIGN
,       '='); 
 918         KEY(DIKI_BRACKET_LEFT
,      '['); 
 919         KEY(DIKI_BRACKET_RIGHT
,     ']'); 
 920         KEY(DIKI_BACKSLASH
,         '\\'); 
 921         KEY(DIKI_SEMICOLON
,         ';'); 
 922         KEY(DIKI_QUOTE_RIGHT
,       '\''); 
 923         KEY(DIKI_COMMA
,             ','); 
 924         KEY(DIKI_PERIOD
,            '.'); 
 925         KEY(DIKI_SLASH
,             '/'); 
 927         KEY(DIKI_LESS_SIGN
,         '<'); 
 929         KEY(DIKI_KP_DIV
,            WXK_NUMPAD_DIVIDE
); 
 930         KEY(DIKI_KP_MULT
,           WXK_NUMPAD_MULTIPLY
); 
 931         KEY(DIKI_KP_MINUS
,          WXK_NUMPAD_SUBTRACT
); 
 932         KEY(DIKI_KP_PLUS
,           WXK_NUMPAD_ADD
); 
 933         KEY(DIKI_KP_ENTER
,          WXK_NUMPAD_ENTER
); 
 934         KEY(DIKI_KP_SPACE
,          WXK_NUMPAD_SPACE
); 
 935         KEY(DIKI_KP_TAB
,            WXK_NUMPAD_TAB
); 
 936         KEY(DIKI_KP_F1
,             WXK_NUMPAD_F1
); 
 937         KEY(DIKI_KP_F2
,             WXK_NUMPAD_F2
); 
 938         KEY(DIKI_KP_F3
,             WXK_NUMPAD_F3
); 
 939         KEY(DIKI_KP_F4
,             WXK_NUMPAD_F4
); 
 940         KEY(DIKI_KP_EQUAL
,          WXK_NUMPAD_EQUAL
); 
 941         KEY(DIKI_KP_SEPARATOR
,      WXK_NUMPAD_SEPARATOR
); 
 943         KEY(DIKI_KP_DECIMAL
,        WXK_NUMPAD_DECIMAL
); 
 944         KEY(DIKI_KP_0
,              WXK_NUMPAD0
); 
 945         KEY(DIKI_KP_1
,              WXK_NUMPAD1
); 
 946         KEY(DIKI_KP_2
,              WXK_NUMPAD2
); 
 947         KEY(DIKI_KP_3
,              WXK_NUMPAD3
); 
 948         KEY(DIKI_KP_4
,              WXK_NUMPAD4
); 
 949         KEY(DIKI_KP_5
,              WXK_NUMPAD5
); 
 950         KEY(DIKI_KP_6
,              WXK_NUMPAD6
); 
 951         KEY(DIKI_KP_7
,              WXK_NUMPAD7
); 
 952         KEY(DIKI_KP_8
,              WXK_NUMPAD8
); 
 953         KEY(DIKI_KP_9
,              WXK_NUMPAD9
); 
 955         case DIKI_KEYDEF_END
: 
 956         case DIKI_NUMBER_OF_KEYS
: 
 957             wxFAIL_MSG( _T("invalid key_id value") ); 
 961     return 0; // silence compiler warnings 
 964 // returns untranslated keycode, i.e. for EVT_CHAR, where characters are left in 
 965 // the form they were entered (lowercase, diacritics etc.) 
 966 static long GetUntraslatedKeyCode(DFBInputDeviceKeyIdentifier key_id
, 
 967                                   DFBInputDeviceKeySymbol key_symbol
) 
 969     switch ( DFB_KEY_TYPE(key_symbol
) ) 
 975             if ( key_symbol 
< 128 ) 
 980                 wchar_t chr 
= key_symbol
; 
 981                 wxCharBuffer 
buf(wxConvUI
->cWC2MB(&chr
, 1, NULL
)); 
 983                     return *buf
; // may be 0 if failed 
 985 #endif // wxUSE_WCHAR_T 
 991             return GetTranslatedKeyCode(key_id
); 
 997 void wxWindowDFB::HandleKeyEvent(const wxDFBWindowEvent
& event_
) 
1002     const DFBWindowEvent
& e 
= event_
; 
1004     wxLogTrace(TRACE_EVENTS
, 
1005                _T("handling key %s event for window %p ('%s')"), 
1006                e
.type 
== DWET_KEYUP 
? _T("up") : _T("down"), 
1007                this, GetName().c_str()); 
1009     // fill in wxKeyEvent fields: 
1011     event
.SetEventObject(this); 
1012     event
.SetTimestamp(wxDFB_EVENT_TIMESTAMP(e
)); 
1013     event
.m_rawCode 
= e
.key_code
; 
1014     event
.m_keyCode 
= GetTranslatedKeyCode(e
.key_id
); 
1015     event
.m_scanCode 
= 0; // not used by wx at all 
1017     event
.m_uniChar 
= e
.key_symbol
; 
1019     event
.m_shiftDown 
= ( e
.modifiers 
& DIMM_SHIFT 
) != 0; 
1020     event
.m_controlDown 
= ( e
.modifiers 
& DIMM_CONTROL 
) != 0; 
1021     event
.m_altDown 
= ( e
.modifiers 
& DIMM_ALT 
) != 0; 
1022     event
.m_metaDown 
= ( e
.modifiers 
& DIMM_META 
) != 0; 
1024     // translate coordinates from TLW-relative to this window-relative: 
1027     GetTLW()->ClientToScreen(&event
.m_x
, &event
.m_y
); 
1028     this->ScreenToClient(&event
.m_x
, &event
.m_y
); 
1030     if ( e
.type 
== DWET_KEYUP 
) 
1032         event
.SetEventType(wxEVT_KEY_UP
); 
1033         GetEventHandler()->ProcessEvent(event
); 
1037         bool isTab 
= (event
.m_keyCode 
== WXK_TAB
); 
1039         event
.SetEventType(wxEVT_KEY_DOWN
); 
1041         if ( GetEventHandler()->ProcessEvent(event
) ) 
1044         // only send wxEVT_CHAR event if not processed yet: 
1045         event
.m_keyCode 
= GetUntraslatedKeyCode(e
.key_id
, e
.key_symbol
); 
1046         if ( event
.m_keyCode 
!= 0 ) 
1048             event
.SetEventType(wxEVT_CHAR
); 
1049             if ( GetEventHandler()->ProcessEvent(event
) ) 
1053         // Synthetize navigation key event, but do it only if the TAB key 
1054         // wasn't handled yet: 
1055         if ( isTab 
&& GetParent() && GetParent()->HasFlag(wxTAB_TRAVERSAL
) ) 
1057             wxNavigationKeyEvent navEvent
; 
1058             navEvent
.SetEventObject(GetParent()); 
1059             // Shift-TAB goes in reverse direction: 
1060             navEvent
.SetDirection(!event
.m_shiftDown
); 
1061             // Ctrl-TAB changes the (parent) window, i.e. switch notebook page: 
1062             navEvent
.SetWindowChange(event
.m_controlDown
); 
1063             navEvent
.SetCurrentFocus(wxStaticCast(this, wxWindow
)); 
1064             GetParent()->GetEventHandler()->ProcessEvent(navEvent
); 
1069 // --------------------------------------------------------------------------- 
1070 // idle events processing 
1071 // --------------------------------------------------------------------------- 
1073 void wxWindowDFB::OnInternalIdle() 
1075     if (wxUpdateUIEvent::CanUpdate(this)) 
1076         UpdateWindowUI(wxUPDATE_UI_FROMIDLE
); 
1080 // Find the wxWindow at the current mouse position, returning the mouse 
1082 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
) 
1084     return wxFindWindowAtPoint(pt 
= wxGetMousePosition()); 
1087 wxWindow
* wxFindWindowAtPoint(const wxPoint
& pt
) 
1089     wxFAIL_MSG( _T("wxFindWindowAtPoint not implemented") );