]> git.saurik.com Git - wxWidgets.git/blame - src/aui/floatpane.cpp
Refactor children traversal in wxWindow::TransferData{To,From}Window().
[wxWidgets.git] / src / aui / floatpane.cpp
CommitLineData
50acee04 1///////////////////////////////////////////////////////////////////////////////
be66f18e 2// Name: src/aui/floatpane.cpp
50acee04
JS
3// Purpose: wxaui: wx advanced user interface - docking window manager
4// Author: Benjamin I. Williams
5// Modified by:
6// Created: 2005-05-17
be66f18e 7// RCS-ID: $Id$
50acee04
JS
8// Copyright: (C) Copyright 2005-2006, Kirix Corporation, All Rights Reserved
9// Licence: wxWindows Library Licence, Version 3.1
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20#include "wx/wxprec.h"
21
22#ifdef __BORLANDC__
23 #pragma hdrstop
24#endif
25
26#if wxUSE_AUI
27
28#include "wx/aui/framemanager.h"
29#include "wx/aui/floatpane.h"
30#include "wx/aui/dockart.h"
31
32#ifndef WX_PRECOMP
50acee04
JS
33#endif
34
20d64475
RR
35#ifdef __WXMSW__
36#include "wx/msw/private.h"
37#endif
38
4dc79cfb 39IMPLEMENT_CLASS(wxAuiFloatingFrame, wxAuiFloatingFrameBaseClass)
e5ab82d3 40
00c1c94c 41wxAuiFloatingFrame::wxAuiFloatingFrame(wxWindow* parent,
a3a5df9d
BW
42 wxAuiManager* owner_mgr,
43 const wxAuiPaneInfo& pane,
bc429ce0
RR
44 wxWindowID id /*= wxID_ANY*/,
45 long style /*=wxRESIZE_BORDER | wxSYSTEM_MENU | wxCAPTION |
cedd7b22 46 wxFRAME_NO_TASKBAR | wxFRAME_FLOAT_ON_PARENT |
bc429ce0
RR
47 wxCLIP_CHILDREN
48 */)
00c1c94c 49 : wxAuiFloatingFrameBaseClass(parent, id, wxEmptyString,
e5ab82d3 50 pane.floating_pos, pane.floating_size,
cedd7b22 51 style |
96743140 52 (pane.HasCloseButton()?wxCLOSE_BOX:0) |
74a94478 53 (pane.HasMaximizeButton()?wxMAXIMIZE_BOX:0) |
e5ab82d3
AB
54 (pane.IsFixed()?0:wxRESIZE_BORDER)
55 )
50acee04 56{
9a29fe70 57 m_ownerMgr = owner_mgr;
50acee04 58 m_moving = false;
461125ea 59 m_mgr.SetManagedWindow(this);
9a29fe70 60 m_solidDrag = true;
cedd7b22 61
22fec94a
BW
62 // find out if the system supports solid window drag.
63 // on non-msw systems, this is assumed to be the case
20d64475 64#ifdef __WXMSW__
bc119dd5 65 BOOL b = TRUE;
22fec94a 66 SystemParametersInfo(38 /*SPI_GETDRAGFULLWINDOWS*/, 0, &b, 0);
9a29fe70 67 m_solidDrag = b ? true : false;
20d64475 68#endif
cedd7b22 69
322c5ec4 70 SetExtraStyle(wxWS_EX_PROCESS_IDLE);
50acee04
JS
71}
72
00c1c94c 73wxAuiFloatingFrame::~wxAuiFloatingFrame()
50acee04 74{
20d64475 75 // if we do not do this, then we can crash...
9a29fe70 76 if (m_ownerMgr && m_ownerMgr->m_actionWindow == this)
4dc79cfb 77 {
9a29fe70 78 m_ownerMgr->m_actionWindow = NULL;
20d64475 79 }
601398b9 80
50acee04
JS
81 m_mgr.UnInit();
82}
83
00c1c94c 84void wxAuiFloatingFrame::SetPaneWindow(const wxAuiPaneInfo& pane)
50acee04 85{
9a29fe70
VZ
86 m_paneWindow = pane.window;
87 m_paneWindow->Reparent(this);
be66f18e 88
a3a5df9d 89 wxAuiPaneInfo contained_pane = pane;
50acee04
JS
90 contained_pane.Dock().Center().Show().
91 CaptionVisible(false).
92 PaneBorder(false).
93 Layer(0).Row(0).Position(0);
be66f18e 94
5c62cb6c 95 // Carry over the minimum size
bc07ab17 96 wxSize pane_min_size = pane.window->GetMinSize();
cedd7b22 97
bc07ab17
RD
98 // if the frame window's max size is greater than the min size
99 // then set the max size to the min size as well
100 wxSize cur_max_size = GetMaxSize();
1dc6ec2c
BW
101 if (cur_max_size.IsFullySpecified() &&
102 (cur_max_size.x < pane.min_size.x ||
103 cur_max_size.y < pane.min_size.y)
104 )
bc07ab17
RD
105 {
106 SetMaxSize(pane_min_size);
107 }
cedd7b22 108
5c62cb6c
AB
109 SetMinSize(pane.window->GetMinSize());
110
9a29fe70 111 m_mgr.AddPane(m_paneWindow, contained_pane);
be66f18e 112 m_mgr.Update();
50acee04
JS
113
114 if (pane.min_size.IsFullySpecified())
115 {
116 // because SetSizeHints() calls Fit() too (which sets the window
117 // size to its minimum allowed), we keep the size before calling
118 // SetSizeHints() and reset it afterwards...
119 wxSize tmp = GetSize();
120 GetSizer()->SetSizeHints(this);
121 SetSize(tmp);
122 }
be66f18e 123
50acee04
JS
124 SetTitle(pane.caption);
125
efd516c2
VZ
126 // This code is slightly awkward because we need to reset wxRESIZE_BORDER
127 // before calling SetClientSize() below as doing it after setting the
128 // client size would actually change it, at least under MSW, where the
129 // total window size doesn't change and hence, as the borders size changes,
130 // the client size does change.
131 //
132 // So we must call it first but doing it generates a size event and updates
133 // pane.floating_size from inside it so we must also record its original
134 // value before doing it.
135 const bool hasFloatingSize = pane.floating_size != wxDefaultSize;
136 if (pane.IsFixed())
137 {
138 SetWindowStyleFlag(GetWindowStyleFlag() & ~wxRESIZE_BORDER);
139 }
140
141 if ( hasFloatingSize )
50acee04
JS
142 {
143 SetSize(pane.floating_size);
144 }
cedd7b22 145 else
50acee04
JS
146 {
147 wxSize size = pane.best_size;
148 if (size == wxDefaultSize)
149 size = pane.min_size;
150 if (size == wxDefaultSize)
9a29fe70
VZ
151 size = m_paneWindow->GetSize();
152 if (m_ownerMgr && pane.HasGripper())
50acee04
JS
153 {
154 if (pane.HasGripperTop())
9a29fe70 155 size.y += m_ownerMgr->m_art->GetMetric(wxAUI_DOCKART_GRIPPER_SIZE);
50acee04 156 else
9a29fe70 157 size.x += m_ownerMgr->m_art->GetMetric(wxAUI_DOCKART_GRIPPER_SIZE);
50acee04 158 }
be66f18e 159
50acee04
JS
160 SetClientSize(size);
161 }
162}
163
4dc79cfb
BW
164wxAuiManager* wxAuiFloatingFrame::GetOwnerManager() const
165{
9a29fe70 166 return m_ownerMgr;
4dc79cfb
BW
167}
168
169
76e1326a 170void wxAuiFloatingFrame::OnSize(wxSizeEvent& WXUNUSED(event))
50acee04 171{
9a29fe70 172 if (m_ownerMgr)
601398b9 173 {
9a29fe70 174 m_ownerMgr->OnFloatingPaneResized(m_paneWindow, GetRect());
601398b9 175 }
50acee04
JS
176}
177
00c1c94c 178void wxAuiFloatingFrame::OnClose(wxCloseEvent& evt)
50acee04 179{
9a29fe70 180 if (m_ownerMgr)
601398b9 181 {
9a29fe70 182 m_ownerMgr->OnFloatingPaneClosed(m_paneWindow, evt);
601398b9
BP
183 }
184 if (!evt.GetVeto())
185 {
9a29fe70 186 m_mgr.DetachPane(m_paneWindow);
c68038f3 187 Destroy();
a23717e5 188 }
50acee04
JS
189}
190
00c1c94c 191void wxAuiFloatingFrame::OnMoveEvent(wxMoveEvent& event)
50acee04 192{
9a29fe70 193 if (!m_solidDrag)
22fec94a
BW
194 {
195 // systems without solid window dragging need to be
196 // handled slightly differently, due to the lack of
197 // the constant stream of EVT_MOVING events
198 if (!isMouseDown())
199 return;
200 OnMoveStart();
201 OnMoving(event.GetRect(), wxNORTH);
202 m_moving = true;
203 return;
204 }
cedd7b22
PC
205
206
9a29fe70 207 wxRect winRect = GetRect();
50acee04 208
9a29fe70 209 if (winRect == m_lastRect)
322c5ec4
RR
210 return;
211
50acee04 212 // skip the first move event
9a29fe70 213 if (m_lastRect.IsEmpty())
50acee04 214 {
9a29fe70 215 m_lastRect = winRect;
50acee04
JS
216 return;
217 }
218
1e9e4113
SC
219 // as on OSX moving windows are not getting all move events, only sporadically, this difference
220 // is almost always big on OSX, so avoid this early exit opportunity
221#ifndef __WXOSX__
cedd7b22 222 // skip if moving too fast to avoid massive redraws and
322c5ec4 223 // jumping hint windows
9a29fe70
VZ
224 if ((abs(winRect.x - m_lastRect.x) > 3) ||
225 (abs(winRect.y - m_lastRect.y) > 3))
0ae3bace 226 {
9a29fe70
VZ
227 m_last3Rect = m_last2Rect;
228 m_last2Rect = m_lastRect;
229 m_lastRect = winRect;
c180fd77
VZ
230
231 // However still update the internally stored position to avoid
232 // snapping back to the old one later.
9a29fe70 233 if (m_ownerMgr)
c180fd77 234 {
9a29fe70
VZ
235 m_ownerMgr->GetPane(m_paneWindow).
236 floating_pos = winRect.GetPosition();
c180fd77
VZ
237 }
238
0ae3bace
RR
239 return;
240 }
1e9e4113 241#endif
0ae3bace 242
50acee04 243 // prevent frame redocking during resize
9a29fe70 244 if (m_lastRect.GetSize() != winRect.GetSize())
50acee04 245 {
9a29fe70
VZ
246 m_last3Rect = m_last2Rect;
247 m_last2Rect = m_lastRect;
248 m_lastRect = winRect;
50acee04
JS
249 return;
250 }
251
322c5ec4 252 wxDirection dir = wxALL;
cedd7b22 253
9a29fe70
VZ
254 int horiz_dist = abs(winRect.x - m_last3Rect.x);
255 int vert_dist = abs(winRect.y - m_last3Rect.y);
cedd7b22 256
322c5ec4
RR
257 if (vert_dist >= horiz_dist)
258 {
9a29fe70 259 if (winRect.y < m_last3Rect.y)
322c5ec4
RR
260 dir = wxNORTH;
261 else
262 dir = wxSOUTH;
263 }
264 else
265 {
9a29fe70 266 if (winRect.x < m_last3Rect.x)
322c5ec4
RR
267 dir = wxWEST;
268 else
269 dir = wxEAST;
270 }
cedd7b22 271
9a29fe70
VZ
272 m_last3Rect = m_last2Rect;
273 m_last2Rect = m_lastRect;
274 m_lastRect = winRect;
be66f18e 275
50acee04
JS
276 if (!isMouseDown())
277 return;
278
279 if (!m_moving)
280 {
281 OnMoveStart();
282 m_moving = true;
283 }
284
9a29fe70 285 if (m_last3Rect.IsEmpty())
322c5ec4 286 return;
cedd7b22 287
1e9e4113
SC
288 if ( event.GetEventType() == wxEVT_MOVING )
289 OnMoving(event.GetRect(), dir);
c81f528f 290 else
1e9e4113 291 OnMoving(wxRect(event.GetPosition(),GetSize()), dir);
50acee04
JS
292}
293
00c1c94c 294void wxAuiFloatingFrame::OnIdle(wxIdleEvent& event)
50acee04
JS
295{
296 if (m_moving)
297 {
298 if (!isMouseDown())
299 {
300 m_moving = false;
301 OnMoveFinished();
302 }
cedd7b22 303 else
50acee04
JS
304 {
305 event.RequestMore();
306 }
307 }
308}
309
00c1c94c 310void wxAuiFloatingFrame::OnMoveStart()
50acee04
JS
311{
312 // notify the owner manager that the pane has started to move
9a29fe70 313 if (m_ownerMgr)
601398b9 314 {
9a29fe70 315 m_ownerMgr->OnFloatingPaneMoveStart(m_paneWindow);
601398b9 316 }
50acee04
JS
317}
318
00c1c94c 319void wxAuiFloatingFrame::OnMoving(const wxRect& WXUNUSED(window_rect), wxDirection dir)
50acee04
JS
320{
321 // notify the owner manager that the pane is moving
9a29fe70 322 if (m_ownerMgr)
601398b9 323 {
9a29fe70 324 m_ownerMgr->OnFloatingPaneMoving(m_paneWindow, dir);
601398b9 325 }
322c5ec4 326 m_lastDirection = dir;
50acee04
JS
327}
328
00c1c94c 329void wxAuiFloatingFrame::OnMoveFinished()
50acee04
JS
330{
331 // notify the owner manager that the pane has finished moving
9a29fe70 332 if (m_ownerMgr)
601398b9 333 {
9a29fe70 334 m_ownerMgr->OnFloatingPaneMoved(m_paneWindow, m_lastDirection);
601398b9 335 }
50acee04
JS
336}
337
00c1c94c 338void wxAuiFloatingFrame::OnActivate(wxActivateEvent& event)
50acee04 339{
9a29fe70 340 if (m_ownerMgr && event.GetActive())
50acee04 341 {
9a29fe70 342 m_ownerMgr->OnFloatingPaneActivated(m_paneWindow);
50acee04
JS
343 }
344}
345
346// utility function which determines the state of the mouse button
c81f528f 347// (independent of having a wxMouseEvent handy) - utimately a better
50acee04
JS
348// mechanism for this should be found (possibly by adding the
349// functionality to wxWidgets itself)
00c1c94c 350bool wxAuiFloatingFrame::isMouseDown()
50acee04 351{
45496810 352 return wxGetMouseState().LeftIsDown();
50acee04
JS
353}
354
355
00c1c94c
BW
356BEGIN_EVENT_TABLE(wxAuiFloatingFrame, wxAuiFloatingFrameBaseClass)
357 EVT_SIZE(wxAuiFloatingFrame::OnSize)
358 EVT_MOVE(wxAuiFloatingFrame::OnMoveEvent)
359 EVT_MOVING(wxAuiFloatingFrame::OnMoveEvent)
360 EVT_CLOSE(wxAuiFloatingFrame::OnClose)
361 EVT_IDLE(wxAuiFloatingFrame::OnIdle)
362 EVT_ACTIVATE(wxAuiFloatingFrame::OnActivate)
50acee04
JS
363END_EVENT_TABLE()
364
365
366#endif // wxUSE_AUI