1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/dfb/toplevel.cpp
3 // Purpose: Top level window, abstraction of wxFrame and wxDialog
4 // Author: Vaclav Slavik
7 // Copyright: (c) 2006 REA Elektronik GmbH
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 // For compilers that support precompilation, includes "wx.h".
12 #include "wx/wxprec.h"
14 #include "wx/toplevel.h"
20 #include "wx/hashmap.h"
21 #include "wx/evtloop.h"
22 #include "wx/dfb/private.h"
24 #define TRACE_EVENTS _T("events")
25 #define TRACE_PAINT _T("paint")
27 // ============================================================================
29 // ============================================================================
31 // mapping of DirectFB windows to wxTLWs:
32 WX_DECLARE_HASH_MAP(DFBWindowID
, wxTopLevelWindowDFB
*,
33 wxIntegerHash
, wxIntegerEqual
,
35 static wxDfbWindowsMap gs_dfbWindowsMap
;
37 // ============================================================================
39 // ============================================================================
41 // Queue of paint requests
42 class wxDfbQueuedPaintRequests
45 ~wxDfbQueuedPaintRequests() { Clear(); }
47 // Adds paint request to the queue
48 void Add(const wxRect
& rect
)
50 // We use a simple implementation here for now: all refresh requests
51 // are merged together into single rectangle that is superset of
52 // all the requested rectangles. This wastes some blitting and painting
53 // time, but OTOH, EVT_PAINT handler is called only once per window.
54 m_invalidated
.Union(rect
);
57 // Is the queue empty?
58 bool IsEmpty() const { return m_invalidated
.IsEmpty(); }
61 void Clear() { m_invalidated
= wxRect(); }
63 // Gets the next request in the queue, returns true if there was one,
64 // false if the queue was empty
65 bool GetNext(wxRect
& rect
)
67 if ( m_invalidated
.IsEmpty() )
71 Clear(); // there's only one item in the queue
76 // currently invalidated region
80 // ============================================================================
81 // wxTopLevelWindowDFB
82 // ============================================================================
84 // ----------------------------------------------------------------------------
85 // creation & destruction
86 // ----------------------------------------------------------------------------
88 void wxTopLevelWindowDFB::Init()
91 m_isMaximized
= false;
92 m_fsIsShowing
= false;
95 m_toPaint
= new wxDfbQueuedPaintRequests
;
99 bool wxTopLevelWindowDFB::Create(wxWindow
*parent
,
101 const wxString
& title
,
102 const wxPoint
& posOrig
,
103 const wxSize
& sizeOrig
,
105 const wxString
&name
)
109 // always create a frame of some reasonable, even if arbitrary, size (at
110 // least for MSW compatibility)
111 wxSize
size(sizeOrig
);
112 if ( size
.x
== wxDefaultCoord
|| size
.y
== wxDefaultCoord
)
114 wxSize sizeDefault
= GetDefaultSize();
115 if ( size
.x
== wxDefaultCoord
)
116 size
.x
= sizeDefault
.x
;
117 if ( size
.y
== wxDefaultCoord
)
118 size
.y
= sizeDefault
.y
;
121 wxPoint
pos(posOrig
);
122 if ( pos
.x
== wxDefaultCoord
)
124 if ( pos
.y
== wxDefaultCoord
)
127 // create DirectFB window:
128 wxIDirectFBDisplayLayerPtr
layer(wxIDirectFB::Get()->GetDisplayLayer());
129 wxCHECK_MSG( layer
, false, _T("no display layer") );
131 DFBWindowDescription desc
;
132 desc
.flags
= (DFBWindowDescriptionFlags
)
134 DWDESC_WIDTH
| DWDESC_HEIGHT
| DWDESC_POSX
| DWDESC_POSY
);
135 desc
.caps
= DWCAPS_DOUBLEBUFFER
;
139 desc
.height
= size
.y
;
140 m_dfbwin
= layer
->CreateWindow(&desc
);
144 // add the new TLW to DFBWindowID->wxTLW map:
146 if ( !m_dfbwin
->GetID(&winid
) )
148 gs_dfbWindowsMap
[winid
] = this;
150 // TLWs are created initially hidden:
151 if ( !m_dfbwin
->SetOpacity(wxALPHA_TRANSPARENT
) )
154 if ( !wxWindow::Create(NULL
, id
, pos
, size
, style
, name
) )
159 parent
->AddChild(this);
161 wxTopLevelWindows
.Append(this);
164 if ( style
& (wxSTAY_ON_TOP
| wxPOPUP_WINDOW
) )
166 m_dfbwin
->SetStackingClass(DWSC_UPPER
);
169 // direct events in this window to the global event buffer:
170 m_dfbwin
->AttachEventBuffer(wxEventLoop::GetDirectFBEventBuffer());
175 wxTopLevelWindowDFB::~wxTopLevelWindowDFB()
177 m_isBeingDeleted
= true;
179 wxTopLevelWindows
.DeleteObject(this);
181 if ( wxTheApp
->GetTopWindow() == this )
182 wxTheApp
->SetTopWindow(NULL
);
184 if ( wxTopLevelWindows
.empty() && wxTheApp
->GetExitOnFrameDelete() )
186 wxTheApp
->ExitMainLoop();
194 // remove the TLW from DFBWindowID->wxTLW map:
196 if ( m_dfbwin
->GetID(&winid
) )
197 gs_dfbWindowsMap
.erase(winid
);
203 // ----------------------------------------------------------------------------
204 // window size & position
205 // ----------------------------------------------------------------------------
207 void wxTopLevelWindowDFB::DoGetPosition(int *x
, int *y
) const
209 m_dfbwin
->GetPosition(x
, y
);
212 void wxTopLevelWindowDFB::DoGetSize(int *width
, int *height
) const
214 m_dfbwin
->GetSize(width
, height
);
217 void wxTopLevelWindowDFB::DoMoveWindow(int x
, int y
, int width
, int height
)
219 wxPoint curpos
= GetPosition();
220 if ( curpos
.x
!= x
|| curpos
.y
!= y
)
222 m_dfbwin
->MoveTo(x
, y
);
225 wxSize cursize
= GetSize();
226 if ( cursize
.x
!= width
|| cursize
.y
!= height
)
228 // changing window's size changes its surface:
229 InvalidateDfbSurface();
231 m_dfbwin
->Resize(width
, height
);
233 // we must repaint the window after it changed size:
239 // ----------------------------------------------------------------------------
240 // showing and hiding
241 // ----------------------------------------------------------------------------
243 #warning "FIXME: the rest of this file is almost same as for MGL, merge it"
244 bool wxTopLevelWindowDFB::ShowFullScreen(bool show
, long style
)
246 if (show
== m_fsIsShowing
) return false; // return what?
248 m_fsIsShowing
= show
;
252 m_fsSaveStyle
= m_windowStyle
;
253 m_fsSaveFlag
= style
;
254 GetPosition(&m_fsSaveFrame
.x
, &m_fsSaveFrame
.y
);
255 GetSize(&m_fsSaveFrame
.width
, &m_fsSaveFrame
.height
);
257 if ( style
& wxFULLSCREEN_NOCAPTION
)
258 m_windowStyle
&= ~wxCAPTION
;
259 if ( style
& wxFULLSCREEN_NOBORDER
)
260 m_windowStyle
= wxSIMPLE_BORDER
;
263 wxDisplaySize(&x
, &y
);
268 m_windowStyle
= m_fsSaveStyle
;
269 SetSize(m_fsSaveFrame
.x
, m_fsSaveFrame
.y
,
270 m_fsSaveFrame
.width
, m_fsSaveFrame
.height
);
276 bool wxTopLevelWindowDFB::Show(bool show
)
278 if ( !wxTopLevelWindowBase::Show(show
) )
281 // hide/show the window by setting its opacity to 0/full:
282 m_dfbwin
->SetOpacity(show
? m_opacity
: 0);
284 // If this is the first time Show was called, send size event,
285 // so that the frame can adjust itself (think auto layout or single child)
289 wxSizeEvent
event(GetSize(), GetId());
290 event
.SetEventObject(this);
291 GetEventHandler()->ProcessEvent(event
);
296 wxWindow
*focused
= wxWindow::FindFocus();
297 if ( focused
&& focused
->GetTLW() == this )
301 else if ( AcceptsFocus() )
303 // FIXME: we should probably always call SetDfbFocus instead
304 // and call SetFocus() from wxActivateEvent/DWET_GOTFOCUS
313 bool wxTopLevelWindowDFB::SetTransparent(wxByte alpha
)
317 if ( !m_dfbwin
->SetOpacity(alpha
) )
325 // ----------------------------------------------------------------------------
326 // maximize, minimize etc.
327 // ----------------------------------------------------------------------------
329 void wxTopLevelWindowDFB::Maximize(bool maximize
)
332 wxClientDisplayRect(&x
, &y
, &w
, &h
);
334 if ( maximize
&& !m_isMaximized
)
336 m_isMaximized
= true;
338 GetPosition(&m_savedFrame
.x
, &m_savedFrame
.y
);
339 GetSize(&m_savedFrame
.width
, &m_savedFrame
.height
);
343 else if ( !maximize
&& m_isMaximized
)
345 m_isMaximized
= false;
346 SetSize(m_savedFrame
.x
, m_savedFrame
.y
,
347 m_savedFrame
.width
, m_savedFrame
.height
);
351 bool wxTopLevelWindowDFB::IsMaximized() const
353 return m_isMaximized
;
356 void wxTopLevelWindowDFB::Restore()
364 void wxTopLevelWindowDFB::Iconize(bool WXUNUSED(iconize
))
366 wxFAIL_MSG(wxT("Iconize not supported under wxDFB"));
369 bool wxTopLevelWindowDFB::IsIconized() const
375 // ----------------------------------------------------------------------------
376 // surfaces and painting
377 // ----------------------------------------------------------------------------
379 wxIDirectFBSurfacePtr
wxTopLevelWindowDFB::ObtainDfbSurface() const
381 return m_dfbwin
->GetSurface();
384 void wxTopLevelWindowDFB::HandleQueuedPaintRequests()
386 if ( m_toPaint
->IsEmpty() )
387 return; // nothing to do
389 if ( IsFrozen() || !IsShown() )
391 // nothing to do if the window is frozen or hidden; clear the queue
392 // and return (note that it's OK to clear the queue even if the window
393 // is frozen, because Thaw() calls Refresh()):
398 // process queued paint requests:
399 wxRect
winRect(wxPoint(0, 0), GetSize());
402 // important note: all DCs created from now until m_isPainting is reset to
403 // false will not update the front buffer as this flag indicates that we'll
404 // blit the entire back buffer to front soon
408 int requestsCount
= 0;
412 while ( m_toPaint
->GetNext(request
) )
417 wxRect
clipped(request
);
418 clipped
.Intersect(winRect
);
419 if ( clipped
.IsEmpty() )
420 continue; // nothing to refresh
422 wxLogTrace(TRACE_PAINT
,
423 _T("%p ('%s'): processing paint request [%i,%i,%i,%i]"),
424 this, GetName().c_str(),
425 clipped
.x
, clipped
.y
, clipped
.GetRight(), clipped
.GetBottom());
427 PaintWindow(clipped
);
429 // remember rectangle covering all repainted areas:
430 if ( paintedRect
.IsEmpty() )
431 paintedRect
= clipped
;
433 paintedRect
.Union(clipped
);
436 m_isPainting
= false;
440 if ( paintedRect
.IsEmpty() )
441 return; // no painting occurred, no need to flip
443 // Flip the surface to make the changes visible. Note that the rectangle we
444 // flip is *superset* of the union of repainted rectangles (created as
445 // "rectangles union" by wxRect::Union) and so some parts of the back
446 // buffer that we didn't touch in this HandleQueuedPaintRequests call will
447 // be copied to the front buffer as well. This is safe/correct thing to do
448 // *only* because wx always use wxIDirectFBSurface::FlipToFront() and so
449 // the back and front buffers contain the same data.
451 // Note that we do _not_ split m_toPaint into disjoint rectangles and
452 // do FlipToFront() for each of them, because that could result in visible
453 // updating of the screen; instead, we prefer to flip everything at once.
455 DFBRegion r
= {paintedRect
.GetLeft(), paintedRect
.GetTop(),
456 paintedRect
.GetRight(), paintedRect
.GetBottom()};
457 DFBRegion
*rptr
= (winRect
== paintedRect
) ? NULL
: &r
;
459 GetDfbSurface()->FlipToFront(rptr
);
461 wxLogTrace(TRACE_PAINT
,
462 _T("%p ('%s'): processed %i paint requests, flipped surface: [%i,%i,%i,%i]"),
463 this, GetName().c_str(),
465 paintedRect
.x
, paintedRect
.y
,
466 paintedRect
.GetRight(), paintedRect
.GetBottom());
469 void wxTopLevelWindowDFB::DoRefreshRect(const wxRect
& rect
)
471 // don't overlap outside of the window (NB: 'rect' is in window coords):
473 r
.Intersect(wxRect(GetSize()));
477 wxLogTrace(TRACE_PAINT
,
478 _T("%p ('%s'): [TLW] refresh rect [%i,%i,%i,%i]"),
479 this, GetName().c_str(),
480 rect
.x
, rect
.y
, rect
.GetRight(), rect
.GetBottom());
482 // defer painting until idle time or until Update() is called:
483 m_toPaint
->Add(rect
);
486 void wxTopLevelWindowDFB::Update()
488 HandleQueuedPaintRequests();
491 // ---------------------------------------------------------------------------
493 // ---------------------------------------------------------------------------
495 void wxTopLevelWindowDFB::SetDfbFocus()
497 wxCHECK_RET( IsShown(), _T("cannot set focus to hidden window") );
498 wxASSERT_MSG( FindFocus() && FindFocus()->GetTLW() == this,
499 _T("setting DirectFB focus to unexpected window") );
501 GetDirectFBWindow()->RequestFocus();
505 void wxTopLevelWindowDFB::HandleDFBWindowEvent(const wxDFBWindowEvent
& event_
)
507 const DFBWindowEvent
& event
= event_
;
509 if ( gs_dfbWindowsMap
.find(event
.window_id
) == gs_dfbWindowsMap
.end() )
511 wxLogTrace(TRACE_EVENTS
,
512 _T("received event for unknown DirectFB window, ignoring"));
516 wxTopLevelWindowDFB
*tlw
= gs_dfbWindowsMap
[event
.window_id
];
517 wxWindow
*recipient
= NULL
;
518 void (wxWindow::*handlerFunc
)(const wxDFBWindowEvent
&) = NULL
;
520 switch ( event
.type
)
525 recipient
= wxWindow::FindFocus();
526 handlerFunc
= &wxWindowDFB::HandleKeyEvent
;
533 wxFAIL_MSG( _T("invalid event type") );
538 // we're not interested in them here
544 wxLogTrace(TRACE_EVENTS
, _T("ignoring event: no recipient window"));
548 wxCHECK_RET( recipient
&& recipient
->GetTLW() == tlw
,
549 _T("event recipient not in TLW which received the event") );
551 // process the event:
552 (recipient
->*handlerFunc
)(event_
);
555 // ---------------------------------------------------------------------------
556 // idle events processing
557 // ---------------------------------------------------------------------------
559 void wxTopLevelWindowDFB::OnInternalIdle()
561 wxTopLevelWindowBase::OnInternalIdle();
562 HandleQueuedPaintRequests();