--- /dev/null
+///////////////////////////////////////////////////////////////////////////////
+// Name: src/aui/notebook.cpp
+// Purpose: wxaui: wx advanced user interface - notebook
+// Author: Benjamin I. Williams
+// Modified by:
+// Created: 2006-06-28
+// Copyright: (C) Copyright 2006, Kirix Corporation, All Rights Reserved
+// Licence: wxWindows Library Licence, Version 3.1
+///////////////////////////////////////////////////////////////////////////////
+
+// ----------------------------------------------------------------------------
+// headers
+// ----------------------------------------------------------------------------
+
+#include "wx/wxprec.h"
+
+#ifdef __BORLANDC__
+ #pragma hdrstop
+#endif
+
+#if wxUSE_AUI
+
+#include "wx/aui/notebook.h"
+#include "wx/aui/tabmdi.h"
+#include "wx/dcbuffer.h"
+
+#ifndef WX_PRECOMP
+#endif
+
+#include "wx/arrimpl.cpp"
+WX_DEFINE_OBJARRAY(wxAuiNotebookPageArray)
+WX_DEFINE_OBJARRAY(wxAuiTabContainerButtonArray)
+
+DEFINE_EVENT_TYPE(wxEVT_COMMAND_AUINOTEBOOK_PAGE_CHANGING)
+DEFINE_EVENT_TYPE(wxEVT_COMMAND_AUINOTEBOOK_PAGE_CHANGED)
+DEFINE_EVENT_TYPE(wxEVT_COMMAND_AUINOTEBOOK_BUTTON)
+DEFINE_EVENT_TYPE(wxEVT_COMMAND_AUINOTEBOOK_BEGIN_DRAG)
+DEFINE_EVENT_TYPE(wxEVT_COMMAND_AUINOTEBOOK_END_DRAG)
+DEFINE_EVENT_TYPE(wxEVT_COMMAND_AUINOTEBOOK_DRAG_MOTION)
+
+
+
+IMPLEMENT_DYNAMIC_CLASS(wxAuiNotebookEvent, wxEvent)
+
+// -- wxAuiTabContainer class implementation --
+
+
+// wxAuiTabContainer is a class which contains information about each
+// tab. It also can render an entire tab control to a specified DC.
+// It's not a window class itself, because this code will be used by
+// the wxFrameMananger, where it is disadvantageous to have separate
+// windows for each tab control in the case of "docked tabs"
+
+// A derived class, wxAuiTabCtrl, is an actual wxWindow-derived window
+// which can be used as a tab control in the normal sense.
+
+
+// This functions are here for this proof of concept
+// and will be factored out later. See dockart.cpp
+static wxColor StepColour(const wxColor& c, int percent)
+{
+ int r = c.Red(), g = c.Green(), b = c.Blue();
+ return wxColour((unsigned char)wxMin((r*percent)/100,255),
+ (unsigned char)wxMin((g*percent)/100,255),
+ (unsigned char)wxMin((b*percent)/100,255));
+}
+
+// This functions are here for this proof of concept
+// and will be factored out later. See dockart.cpp
+static wxBitmap BitmapFromBits(const unsigned char bits[], int w, int h,
+ const wxColour& color)
+{
+ wxImage img = wxBitmap((const char*)bits, w, h).ConvertToImage();
+ img.Replace(255,255,255,123,123,123);
+ img.Replace(0,0,0,color.Red(),color.Green(),color.Blue());
+ img.SetMaskColour(123,123,123);
+ return wxBitmap(img);
+}
+
+static void DrawButton(wxDC& dc,
+ const wxRect& _rect,
+ const wxBitmap& bmp,
+ const wxColour& bkcolour,
+ int button_state)
+{
+ wxRect rect = _rect;
+
+ if (button_state == wxAUI_BUTTON_STATE_PRESSED)
+ {
+ rect.x++;
+ rect.y++;
+ }
+
+ if (button_state == wxAUI_BUTTON_STATE_HOVER ||
+ button_state == wxAUI_BUTTON_STATE_PRESSED)
+ {
+ dc.SetBrush(wxBrush(StepColour(bkcolour, 120)));
+ dc.SetPen(wxPen(StepColour(bkcolour, 70)));
+
+ // draw the background behind the button
+ dc.DrawRectangle(rect.x, rect.y, 15, 15);
+ }
+
+ // draw the button itself
+ dc.DrawBitmap(bmp, rect.x, rect.y, true);
+}
+
+
+
+
+wxAuiTabContainer::wxAuiTabContainer()
+{
+ m_normal_font = *wxNORMAL_FONT;
+ m_selected_font = *wxNORMAL_FONT;
+ m_selected_font.SetWeight(wxBOLD);
+ m_measuring_font = m_selected_font;
+
+ wxColour base_colour = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE);
+
+ wxColour background_colour = StepColour(base_colour, 95);
+ wxColour normaltab_colour = base_colour;
+ wxColour selectedtab_colour = *wxWHITE;
+
+ m_bkbrush = wxBrush(background_colour);
+ m_normal_bkbrush = wxBrush(normaltab_colour);
+ m_normal_bkpen = wxPen(normaltab_colour);
+ m_selected_bkbrush = wxBrush(selectedtab_colour);
+ m_selected_bkpen = wxPen(selectedtab_colour);
+}
+
+void wxAuiTabContainer::SetNormalFont(const wxFont& font)
+{
+ m_normal_font = font;
+}
+
+void wxAuiTabContainer::SetSelectedFont(const wxFont& font)
+{
+ m_selected_font = font;
+}
+
+void wxAuiTabContainer::SetMeasuringFont(const wxFont& font)
+{
+ m_measuring_font = font;
+}
+
+void wxAuiTabContainer::SetRect(const wxRect& rect)
+{
+ m_rect = rect;
+}
+
+bool wxAuiTabContainer::AddPage(wxWindow* page,
+ const wxAuiNotebookPage& info)
+{
+ wxAuiNotebookPage page_info;
+ page_info = info;
+ page_info.window = page;
+
+ m_pages.Add(page_info);
+
+ return true;
+}
+
+bool wxAuiTabContainer::InsertPage(wxWindow* page,
+ const wxAuiNotebookPage& info,
+ size_t idx)
+{
+ wxAuiNotebookPage page_info;
+ page_info = info;
+ page_info.window = page;
+
+ if (idx >= m_pages.GetCount())
+ m_pages.Add(page_info);
+ else
+ m_pages.Insert(page_info, idx);
+
+ return true;
+}
+
+bool wxAuiTabContainer::RemovePage(wxWindow* wnd)
+{
+ size_t i, page_count = m_pages.GetCount();
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = m_pages.Item(i);
+ if (page.window == wnd)
+ {
+ m_pages.RemoveAt(i);
+ return true;
+ }
+ }
+
+ return false;
+}
+
+bool wxAuiTabContainer::SetActivePage(wxWindow* wnd)
+{
+ bool found = false;
+
+ size_t i, page_count = m_pages.GetCount();
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = m_pages.Item(i);
+ if (page.window == wnd)
+ {
+ page.active = true;
+ found = true;
+ }
+ else
+ {
+ page.active = false;
+ }
+ }
+
+ return found;
+}
+
+void wxAuiTabContainer::SetNoneActive()
+{
+ size_t i, page_count = m_pages.GetCount();
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = m_pages.Item(i);
+ page.active = false;
+ }
+}
+
+bool wxAuiTabContainer::SetActivePage(size_t page)
+{
+ if (page >= m_pages.GetCount())
+ return false;
+
+ return SetActivePage(m_pages.Item(page).window);
+}
+
+int wxAuiTabContainer::GetActivePage() const
+{
+ size_t i, page_count = m_pages.GetCount();
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = m_pages.Item(i);
+ if (page.active)
+ return i;
+ }
+
+ return -1;
+}
+
+wxWindow* wxAuiTabContainer::GetWindowFromIdx(size_t idx) const
+{
+ if (idx >= m_pages.GetCount())
+ return NULL;
+
+ return m_pages[idx].window;
+}
+
+int wxAuiTabContainer::GetIdxFromWindow(wxWindow* wnd) const
+{
+ size_t i, page_count = m_pages.GetCount();
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = m_pages.Item(i);
+ if (page.window == wnd)
+ return i;
+ }
+ return -1;
+}
+
+wxAuiNotebookPage& wxAuiTabContainer::GetPage(size_t idx)
+{
+ wxASSERT_MSG(idx < m_pages.GetCount(), wxT("Invalid Page index"));
+
+ return m_pages[idx];
+}
+
+wxAuiNotebookPageArray& wxAuiTabContainer::GetPages()
+{
+ return m_pages;
+}
+
+size_t wxAuiTabContainer::GetPageCount() const
+{
+ return m_pages.GetCount();
+}
+
+void wxAuiTabContainer::AddButton(int id, const wxBitmap& bmp)
+{
+ wxAuiTabContainerButton button;
+ button.id = id;
+ button.bitmap = bmp;
+ button.cur_state = wxAUI_BUTTON_STATE_NORMAL;
+
+ m_buttons.Add(button);
+}
+
+
+
+// DrawTab() draws an individual tab.
+// As it is virtual it may be overridden.
+//
+// dc - output dc
+// in_rect - rectangle the tab should be confined to
+// caption - tab's caption
+// active - whether or not the tab is active
+// out_rect - actual output rectangle
+// x_extent - the advance x; where the next tab should start
+
+void wxAuiTabContainer::DrawTab(wxDC* dc,
+ const wxRect& in_rect,
+ const wxString& caption,
+ bool active,
+ wxRect* out_rect,
+ int* x_extent)
+{
+ wxCoord normal_textx, normal_texty;
+ wxCoord selected_textx, selected_texty;
+ wxCoord measured_textx, measured_texty;
+ wxCoord textx, texty;
+
+
+ // measure text
+ dc->SetFont(m_measuring_font);
+ dc->GetTextExtent(caption, &measured_textx, &measured_texty);
+
+ dc->SetFont(m_selected_font);
+ dc->GetTextExtent(caption, &selected_textx, &selected_texty);
+
+ dc->SetFont(m_normal_font);
+ dc->GetTextExtent(caption, &normal_textx, &normal_texty);
+
+
+ wxCoord tab_height = measured_texty + 4;
+ wxCoord tab_width = measured_textx + tab_height + 5;
+ wxCoord tab_x = in_rect.x;
+ wxCoord tab_y = in_rect.y + in_rect.height - tab_height;
+
+
+ // select pen, brush and font for the tab to be drawn
+
+ if (active)
+ {
+ dc->SetPen(m_selected_bkpen);
+ dc->SetBrush(m_selected_bkbrush);
+ dc->SetFont(m_selected_font);
+ textx = selected_textx;
+ texty = selected_texty;
+ }
+ else
+ {
+ dc->SetPen(m_normal_bkpen);
+ dc->SetBrush(m_normal_bkbrush);
+ dc->SetFont(m_normal_font);
+ textx = normal_textx;
+ texty = normal_texty;
+ }
+
+
+ // -- draw line --
+
+ wxPoint points[7];
+ points[0].x = tab_x;
+ points[0].y = tab_y + tab_height - 1;
+ points[1].x = tab_x + tab_height - 3;
+ points[1].y = tab_y + 2;
+ points[2].x = tab_x + tab_height + 3;
+ points[2].y = tab_y;
+ points[3].x = tab_x + tab_width - 2;
+ points[3].y = tab_y;
+ points[4].x = tab_x + tab_width;
+ points[4].y = tab_y + 2;
+ points[5].x = tab_x + tab_width;
+ points[5].y = tab_y + tab_height - 1;
+ points[6] = points[0];
+
+
+ dc->DrawPolygon(6, points);
+
+ dc->SetPen(*wxGREY_PEN);
+
+ //dc->DrawLines(active ? 6 : 7, points);
+ dc->DrawLines(7, points);
+
+ // -- draw text --
+
+ dc->DrawText(caption,
+ tab_x + (tab_height/3) + (tab_width/2) - (textx/2),
+ tab_y + tab_height - texty - 2);
+
+ *out_rect = wxRect(tab_x, tab_y, tab_width, tab_height);
+ *x_extent = tab_width - (tab_height/2) - 1;
+}
+
+
+// Render() renders the tab catalog to the specified DC
+// It is a virtual function and can be overridden to
+// provide custom drawing capabilities
+void wxAuiTabContainer::Render(wxDC* raw_dc)
+{
+ wxMemoryDC dc;
+ wxBitmap bmp;
+ bmp.Create(m_rect.GetWidth(), m_rect.GetHeight());
+ dc.SelectObject(bmp);
+
+ // draw background
+ dc.SetBrush(m_bkbrush);
+ dc.SetPen(*wxTRANSPARENT_PEN);
+ dc.DrawRectangle(-1, -1, m_rect.GetWidth()+2, m_rect.GetHeight()+2);
+
+ // draw base line
+ dc.SetPen(*wxGREY_PEN);
+ dc.DrawLine(0, m_rect.GetHeight()-1, m_rect.GetWidth(), m_rect.GetHeight()-1);
+
+
+ size_t i, page_count = m_pages.GetCount();
+ int offset = 0;
+
+ size_t active = 999;
+ int active_offset = 0;
+
+ int x_extent = 0;
+ wxRect rect = m_rect;
+ rect.y = 0;
+ rect.width = 1000;
+ rect.height = m_rect.height;
+
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = m_pages.Item(i);
+
+ rect.x = offset;
+
+ DrawTab(&dc,
+ rect,
+ page.caption,
+ page.active,
+ &page.rect,
+ &x_extent);
+
+ if (page.active)
+ {
+ active = i;
+ active_offset = offset;
+ }
+
+ offset += x_extent;
+ }
+
+ // draw the active tab again so it stands in the foreground
+ if (active < m_pages.GetCount())
+ {
+ wxAuiNotebookPage& page = m_pages.Item(active);
+
+ rect.x = active_offset;
+ DrawTab(&dc,
+ rect,
+ page.caption,
+ page.active,
+ &page.rect,
+ &x_extent);
+ }
+
+ // draw the buttons
+ offset = m_rect.x + m_rect.width;
+ size_t button_count = m_buttons.GetCount();
+ for (i = 0; i < button_count; ++i)
+ {
+ wxAuiTabContainerButton& button = m_buttons.Item(button_count - i - 1);
+
+ wxRect button_rect(offset - button.bitmap.GetWidth(), 1,
+ button.bitmap.GetWidth(), button.bitmap.GetHeight());
+
+ button.rect = button_rect;
+
+ DrawButton(dc, button.rect, button.bitmap,
+ m_bkbrush.GetColour(),
+ button.cur_state);
+
+ offset -= button.bitmap.GetWidth();
+ }
+
+
+ raw_dc->Blit(m_rect.x, m_rect.y, m_rect.GetWidth(), m_rect.GetHeight(), &dc, 0, 0);
+}
+
+
+// TabHitTest() tests if a tab was hit, passing the window pointer
+// back if that condition was fulfilled. The function returns
+// true if a tab was hit, otherwise false
+bool wxAuiTabContainer::TabHitTest(int x, int y, wxWindow** hit) const
+{
+ if (!m_rect.Inside(x,y))
+ return false;
+
+ size_t i, page_count = m_pages.GetCount();
+
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = m_pages.Item(i);
+ if (page.rect.Inside(x,y))
+ {
+ *hit = page.window;
+ return true;
+ }
+ }
+
+ return false;
+}
+
+// ButtonHitTest() tests if a button was hit. The function returns
+// true if a button was hit, otherwise false
+bool wxAuiTabContainer::ButtonHitTest(int x, int y,
+ wxAuiTabContainerButton** hit) const
+{
+ if (!m_rect.Inside(x,y))
+ return false;
+
+ size_t i, button_count = m_buttons.GetCount();
+
+ for (i = 0; i < button_count; ++i)
+ {
+ wxAuiTabContainerButton& button = m_buttons.Item(i);
+ if (button.rect.Inside(x,y))
+ {
+ *hit = &button;
+ return true;
+ }
+ }
+
+ return false;
+}
+
+
+
+// the utility function ShowWnd() is the same as show,
+// except it handles wxTabMDIChildFrame windows as well,
+// as the Show() method on this class is "unplugged"
+static void ShowWnd(wxWindow* wnd, bool show)
+{
+ if (wnd->IsKindOf(CLASSINFO(wxTabMDIChildFrame)))
+ {
+ wxTabMDIChildFrame* cf = (wxTabMDIChildFrame*)wnd;
+ cf->DoShow(show);
+ }
+ else
+ {
+ wnd->Show(show);
+ }
+}
+
+
+// DoShowHide() this function shows the active window, then
+// hides all of the other windows (in that order)
+void wxAuiTabContainer::DoShowHide()
+{
+ wxAuiNotebookPageArray& pages = GetPages();
+ size_t i, page_count = pages.GetCount();
+
+ // show new active page first
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = pages.Item(i);
+ if (page.active)
+ {
+ ShowWnd(page.window, true);
+ break;
+ }
+ }
+
+ // hide all other pages
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = pages.Item(i);
+ ShowWnd(page.window, page.active);
+ }
+}
+
+
+
+
+
+
+// -- wxAuiTabCtrl class implementation --
+
+
+const int wxAuiButtonClose = 101;
+
+BEGIN_EVENT_TABLE(wxAuiTabCtrl, wxControl)
+ EVT_PAINT(wxAuiTabCtrl::OnPaint)
+ EVT_ERASE_BACKGROUND(wxAuiTabCtrl::OnEraseBackground)
+ EVT_SIZE(wxAuiTabCtrl::OnSize)
+ EVT_LEFT_DOWN(wxAuiTabCtrl::OnLeftDown)
+ EVT_LEFT_UP(wxAuiTabCtrl::OnLeftUp)
+ EVT_MOTION(wxAuiTabCtrl::OnMotion)
+ EVT_LEAVE_WINDOW(wxAuiTabCtrl::OnLeaveWindow)
+END_EVENT_TABLE()
+
+
+wxAuiTabCtrl::wxAuiTabCtrl(wxWindow* parent,
+ wxWindowID id,
+ const wxPoint& pos,
+ const wxSize& size,
+ long style) : wxControl(parent, id, pos, size, style)
+{
+ m_click_pt = wxDefaultPosition;
+ m_is_dragging = false;
+ m_hover_button = NULL;
+
+ // copied from dockart-- needs to put in a common place
+ static unsigned char close_bits[]={
+ 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xef,0xfb,0xcf,0xf9,
+ 0x9f,0xfc,0x3f,0xfe,0x3f,0xfe,0x9f,0xfc,0xcf,0xf9,0xef,0xfb,
+ 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff};
+
+ AddButton(101, BitmapFromBits(close_bits, 16, 16, *wxBLACK));
+}
+
+
+void wxAuiTabCtrl::OnPaint(wxPaintEvent&)
+{
+ wxPaintDC dc(this);
+
+ dc.SetFont(GetFont());
+
+ if (GetPageCount() > 0)
+ Render(&dc);
+}
+
+void wxAuiTabCtrl::OnEraseBackground(wxEraseEvent& WXUNUSED(evt))
+{
+}
+
+void wxAuiTabCtrl::OnSize(wxSizeEvent& evt)
+{
+ wxSize s = evt.GetSize();
+ wxRect r(0, 0, s.GetWidth(), s.GetHeight());
+ SetRect(r);
+}
+
+void wxAuiTabCtrl::OnLeftDown(wxMouseEvent& evt)
+{
+ CaptureMouse();
+ m_click_pt = wxDefaultPosition;
+ m_is_dragging = false;
+ m_click_tab = -1;
+
+ wxWindow* wnd;
+ if (TabHitTest(evt.m_x, evt.m_y, &wnd))
+ {
+ wxAuiNotebookEvent e(wxEVT_COMMAND_AUINOTEBOOK_PAGE_CHANGING, m_windowId);
+ e.SetSelection(GetIdxFromWindow(wnd));
+ e.SetOldSelection(GetActivePage());
+ e.SetEventObject(this);
+ GetEventHandler()->ProcessEvent(e);
+
+ m_click_pt.x = evt.m_x;
+ m_click_pt.y = evt.m_y;
+ m_click_tab = e.GetSelection();
+ }
+
+ if (m_hover_button)
+ {
+ m_hover_button->cur_state = wxAUI_BUTTON_STATE_PRESSED;
+ Refresh();
+ Update();
+ }
+}
+
+void wxAuiTabCtrl::OnLeftUp(wxMouseEvent&)
+{
+ if (GetCapture() == this)
+ ReleaseMouse();
+
+ if (m_is_dragging)
+ {
+ wxAuiNotebookEvent evt(wxEVT_COMMAND_AUINOTEBOOK_END_DRAG, m_windowId);
+ evt.SetSelection(m_click_tab);
+ evt.SetOldSelection(m_click_tab);
+ evt.SetEventObject(this);
+ GetEventHandler()->ProcessEvent(evt);
+ return;
+ }
+
+ if (m_hover_button)
+ {
+ m_hover_button->cur_state = wxAUI_BUTTON_STATE_HOVER;
+ Refresh();
+ Update();
+
+ wxAuiNotebookEvent evt(wxEVT_COMMAND_AUINOTEBOOK_BUTTON, m_windowId);
+ evt.SetInt(m_hover_button->id);
+ evt.SetEventObject(this);
+ GetEventHandler()->ProcessEvent(evt);
+ }
+
+ m_click_pt = wxDefaultPosition;
+ m_is_dragging = false;
+ m_click_tab = -1;
+}
+
+void wxAuiTabCtrl::OnMotion(wxMouseEvent& evt)
+{
+ wxPoint pos = evt.GetPosition();
+
+ // check if the mouse is hovering above a button
+ wxAuiTabContainerButton* button;
+ if (ButtonHitTest(pos.x, pos.y, &button))
+ {
+ if (button->cur_state != wxAUI_BUTTON_STATE_HOVER)
+ {
+ button->cur_state = wxAUI_BUTTON_STATE_HOVER;
+ Refresh();
+ Update();
+ m_hover_button = button;
+ return;
+ }
+ }
+ else
+ {
+ if (m_hover_button)
+ {
+ m_hover_button->cur_state = wxAUI_BUTTON_STATE_NORMAL;
+ m_hover_button = NULL;
+ Refresh();
+ Update();
+ }
+ }
+
+
+ if (!evt.LeftIsDown() || m_click_pt == wxDefaultPosition)
+ return;
+
+ if (m_is_dragging)
+ {
+ wxAuiNotebookEvent evt(wxEVT_COMMAND_AUINOTEBOOK_DRAG_MOTION, m_windowId);
+ evt.SetSelection(m_click_tab);
+ evt.SetOldSelection(m_click_tab);
+ evt.SetEventObject(this);
+ GetEventHandler()->ProcessEvent(evt);
+ return;
+ }
+
+
+ int drag_x_threshold = wxSystemSettings::GetMetric(wxSYS_DRAG_X);
+ int drag_y_threshold = wxSystemSettings::GetMetric(wxSYS_DRAG_Y);
+
+ if (abs(pos.x - m_click_pt.x) > drag_x_threshold ||
+ abs(pos.y - m_click_pt.y) > drag_y_threshold)
+ {
+ wxAuiNotebookEvent evt(wxEVT_COMMAND_AUINOTEBOOK_BEGIN_DRAG, m_windowId);
+ evt.SetSelection(m_click_tab);
+ evt.SetOldSelection(m_click_tab);
+ evt.SetEventObject(this);
+ GetEventHandler()->ProcessEvent(evt);
+
+ m_is_dragging = true;
+ }
+}
+
+void wxAuiTabCtrl::OnLeaveWindow(wxMouseEvent& WXUNUSED(event))
+{
+ if (m_hover_button)
+ {
+ m_hover_button->cur_state = wxAUI_BUTTON_STATE_NORMAL;
+ m_hover_button = NULL;
+ Refresh();
+ Update();
+ }
+}
+
+
+// wxTabFrame is an interesting case. It's important that all child pages
+// of the multi-notebook control are all actually children of that control
+// (and not grandchildren). wxTabFrame facilitates this. There is one
+// instance of wxTabFrame for each tab control inside the multi-notebook.
+// It's important to know that wxTabFrame is not a real window, but it merely
+// used to capture the dimensions/positioning of the internal tab control and
+// it's managed page windows
+
+class wxTabFrame : public wxWindow
+{
+public:
+
+ wxTabFrame()
+ {
+ m_tabs = NULL;
+ m_rect = wxRect(0,0,200,200);
+ }
+
+ void DoSetSize(int x, int y,
+ int width, int height,
+ int WXUNUSED(sizeFlags = wxSIZE_AUTO))
+ {
+ m_rect = wxRect(x, y, width, height);
+ DoSizing();
+ }
+
+ void DoGetClientSize(int* x, int* y) const
+ {
+ *x = m_rect.width;
+ *y = m_rect.height;
+ }
+
+ void DoSizing()
+ {
+ if (!m_tabs)
+ return;
+
+ int tab_height = wxMin(m_rect.height, 19);
+ m_tab_rect = wxRect(m_rect.x, m_rect.y, m_rect.width, tab_height);
+ m_tabs->SetSize(m_rect.x, m_rect.y, m_rect.width, tab_height);
+ m_tabs->SetRect(wxRect(0, 0, m_rect.width, tab_height));
+ m_tabs->Refresh();
+
+ wxAuiNotebookPageArray& pages = m_tabs->GetPages();
+ size_t i, page_count = pages.GetCount();
+
+ for (i = 0; i < page_count; ++i)
+ {
+ wxAuiNotebookPage& page = pages.Item(i);
+ page.window->SetSize(m_rect.x, m_rect.y+tab_height, m_rect.width, m_rect.height-tab_height);
+
+ if (page.window->IsKindOf(CLASSINFO(wxTabMDIChildFrame)))
+ {
+ wxTabMDIChildFrame* wnd = (wxTabMDIChildFrame*)page.window;
+ wnd->ApplyMDIChildFrameRect();
+ }
+ }
+ }
+
+public:
+
+ wxRect m_rect;
+ wxRect m_tab_rect;
+ wxAuiTabCtrl* m_tabs;
+};
+
+
+
+
+
+// -- wxAuiMultiNotebook class implementation --
+
+BEGIN_EVENT_TABLE(wxAuiMultiNotebook, wxControl)
+ //EVT_ERASE_BACKGROUND(wxAuiMultiNotebook::OnEraseBackground)
+ //EVT_SIZE(wxAuiMultiNotebook::OnSize)
+ //EVT_LEFT_DOWN(wxAuiMultiNotebook::OnLeftDown)
+ EVT_CHILD_FOCUS(wxAuiMultiNotebook::OnChildFocus)
+ EVT_COMMAND_RANGE(10000, 10100,
+ wxEVT_COMMAND_AUINOTEBOOK_PAGE_CHANGING,
+ wxAuiMultiNotebook::OnTabClicked)
+ EVT_COMMAND_RANGE(10000, 10100,
+ wxEVT_COMMAND_AUINOTEBOOK_BEGIN_DRAG,
+ wxAuiMultiNotebook::OnTabBeginDrag)
+ EVT_COMMAND_RANGE(10000, 10100,
+ wxEVT_COMMAND_AUINOTEBOOK_END_DRAG,
+ wxAuiMultiNotebook::OnTabEndDrag)
+ EVT_COMMAND_RANGE(10000, 10100,
+ wxEVT_COMMAND_AUINOTEBOOK_DRAG_MOTION,
+ wxAuiMultiNotebook::OnTabDragMotion)
+ EVT_COMMAND_RANGE(10000, 10100,
+ wxEVT_COMMAND_AUINOTEBOOK_BUTTON,
+ wxAuiMultiNotebook::OnTabButton)
+END_EVENT_TABLE()
+
+wxAuiMultiNotebook::wxAuiMultiNotebook()
+{
+ m_curpage = -1;
+ m_tab_id_counter = 10000;
+ m_dummy_wnd = NULL;
+}
+
+wxAuiMultiNotebook::wxAuiMultiNotebook(wxWindow *parent,
+ wxWindowID id,
+ const wxPoint& pos,
+ const wxSize& size,
+ long style) : wxControl(parent, id, pos, size, style)
+{
+ InitNotebook();
+}
+
+bool wxAuiMultiNotebook::Create(wxWindow* parent,
+ wxWindowID id,
+ const wxPoint& pos,
+ const wxSize& size,
+ long style)
+{
+ if (!wxControl::Create(parent, id, pos, size, style))
+ return false;
+
+ InitNotebook();
+
+ return true;
+}
+
+// InitNotebook() contains common initialization
+// code called by all constructors
+void wxAuiMultiNotebook::InitNotebook()
+{
+ m_normal_font = *wxNORMAL_FONT;
+ m_selected_font = *wxNORMAL_FONT;
+ m_selected_font.SetWeight(wxBOLD);
+
+
+ m_tab_id_counter = 10000;
+
+ m_dummy_wnd = new wxWindow(this, -1, wxPoint(0,0), wxSize(0,0));
+ m_dummy_wnd->SetSize(200, 200);
+ m_dummy_wnd->Show(false);
+
+ m_mgr.SetManagedWindow(this);
+
+ m_mgr.AddPane(m_dummy_wnd,
+ wxPaneInfo().Name(wxT("dummy")).Bottom().Show(false));
+
+ m_mgr.Update();
+}
+
+wxAuiMultiNotebook::~wxAuiMultiNotebook()
+{
+ m_mgr.UnInit();
+}
+
+bool wxAuiMultiNotebook::AddPage(wxWindow* page,
+ const wxString& caption,
+ bool select,
+ const wxBitmap& bitmap)
+{
+ return InsertPage(GetPageCount(), page, caption, select, bitmap);
+}
+
+bool wxAuiMultiNotebook::InsertPage(size_t page_idx,
+ wxWindow* page,
+ const wxString& caption,
+ bool select,
+ const wxBitmap& bitmap)
+{
+ wxAuiNotebookPage info;
+ info.window = page;
+ info.caption = caption;
+ info.bitmap = bitmap;
+ info.active = false;
+
+ // if there are currently no tabs, the first added
+ // tab must be active
+ if (m_tabs.GetPageCount() == 0)
+ info.active = true;
+
+ m_tabs.InsertPage(page, info, page_idx);
+
+ wxAuiTabCtrl* active_tabctrl = GetActiveTabCtrl();
+ if (page_idx >= active_tabctrl->GetPageCount())
+ active_tabctrl->AddPage(page, info);
+ else
+ active_tabctrl->InsertPage(page, info, page_idx);
+
+ DoSizing();
+ active_tabctrl->DoShowHide();
+
+ if (select)
+ {
+ int idx = m_tabs.GetIdxFromWindow(page);
+ wxASSERT_MSG(idx != -1, wxT("Invalid Page index returned on wxAuiMultiNotebook::InsertPage()"));
+
+ SetSelection(idx);
+ }
+
+ return true;
+}
+
+
+// DeletePage() removes a tab from the multi-notebook,
+// and destroys the window as well
+bool wxAuiMultiNotebook::DeletePage(size_t page_idx)
+{
+ wxWindow* wnd = m_tabs.GetWindowFromIdx(page_idx);
+
+
+ // find out which onscreen tab ctrl owns this tab
+ wxAuiTabCtrl* ctrl;
+ int ctrl_idx;
+ if (!FindTab(wnd, &ctrl, &ctrl_idx))
+ return false;
+
+ // find a new page and set it as active
+ int new_idx = ctrl_idx+1;
+ if (new_idx >= (int)ctrl->GetPageCount())
+ new_idx = ctrl_idx-1;
+
+ if (new_idx >= 0 && new_idx < (int)ctrl->GetPageCount())
+ {
+ wxWindow* new_wnd = ctrl->GetWindowFromIdx(new_idx);
+ int main_idx = m_tabs.GetIdxFromWindow(new_wnd);
+ wxASSERT(main_idx != -1);
+ SetSelection(main_idx);
+ }
+ else
+ {
+ // set the active page to the first page that
+ // isn't the one being deleted
+ bool found = false;
+ size_t i, page_count = m_tabs.GetPageCount();
+ for (i = 0; i < page_count; ++i)
+ {
+ wxWindow* w = m_tabs.GetWindowFromIdx(i);
+ if (wnd != w)
+ {
+ found = true;
+ SetSelection(i);
+ break;
+ }
+ }
+
+ if (!found)
+ m_curpage = -1;
+ }
+
+
+ // remove the tab from main catalog
+ if (!m_tabs.RemovePage(wnd))
+ return false;
+
+ // remove the tab from the onscreen tab ctrl
+ ctrl->RemovePage(wnd);
+
+ // actually destroy the window now
+ if (wnd->IsKindOf(CLASSINFO(wxTabMDIChildFrame)))
+ {
+ // delete the child frame with pending delete, as is
+ // customary with frame windows
+ if (!wxPendingDelete.Member(wnd))
+ wxPendingDelete.Append(wnd);
+ }
+ else
+ {
+ wnd->Destroy();
+ }
+
+ RemoveEmptyTabFrames();
+
+ return true;
+}
+
+
+
+// RemovePage() removes a tab from the multi-notebook,
+// but does not destroy the window
+bool wxAuiMultiNotebook::RemovePage(size_t page_idx)
+{
+ // remove the tab from our own catalog
+ wxWindow* wnd = m_tabs.GetWindowFromIdx(page_idx);
+ if (!m_tabs.RemovePage(wnd))
+ return false;
+
+ // remove the tab from the onscreen tab ctrl
+ wxAuiTabCtrl* ctrl;
+ int ctrl_idx;
+ if (FindTab(wnd, &ctrl, &ctrl_idx))
+ {
+ ctrl->RemovePage(wnd);
+ return true;
+ }
+
+ return false;
+}
+
+// SetPageText() changes the tab caption of the specified page
+bool wxAuiMultiNotebook::SetPageText(size_t page_idx, const wxString& text)
+{
+ if (page_idx >= m_tabs.GetPageCount())
+ return false;
+
+ // update our own tab catalog
+ wxAuiNotebookPage& page_info = m_tabs.GetPage(page_idx);
+ page_info.caption = text;
+
+ // update what's on screen
+ wxAuiTabCtrl* ctrl;
+ int ctrl_idx;
+ if (FindTab(page_info.window, &ctrl, &ctrl_idx))
+ {
+ wxAuiNotebookPage& info = ctrl->GetPage(ctrl_idx);
+ info.caption = text;
+ ctrl->Refresh();
+ }
+
+
+ return true;
+}
+
+// GetSelection() returns the index of the currently active page
+int wxAuiMultiNotebook::GetSelection() const
+{
+ return m_curpage;
+}
+
+// SetSelection() sets the currently active page
+size_t wxAuiMultiNotebook::SetSelection(size_t new_page)
+{
+ wxWindow* wnd = m_tabs.GetWindowFromIdx(new_page);
+ if (!wnd)
+ return m_curpage;
+
+ wxAuiNotebookEvent evt(wxEVT_COMMAND_AUINOTEBOOK_PAGE_CHANGING, m_windowId);
+ evt.SetSelection(new_page);
+ evt.SetOldSelection(m_curpage);
+ evt.SetEventObject(this);
+ if (!GetEventHandler()->ProcessEvent(evt) || evt.IsAllowed())
+ {
+ // program allows the page change
+ evt.SetEventType(wxEVT_COMMAND_AUINOTEBOOK_PAGE_CHANGED);
+ (void)GetEventHandler()->ProcessEvent(evt);
+
+
+
+ wxAuiTabCtrl* ctrl;
+ int ctrl_idx;
+ if (FindTab(wnd, &ctrl, &ctrl_idx))
+ {
+ m_tabs.SetActivePage(wnd);
+
+ ctrl->SetActivePage(ctrl_idx);
+ DoSizing();
+ ctrl->DoShowHide();
+
+ int old_curpage = m_curpage;
+ m_curpage = new_page;
+
+
+ // set fonts
+ wxPaneInfoArray& all_panes = m_mgr.GetAllPanes();
+ size_t i, pane_count = all_panes.GetCount();
+ for (i = 0; i < pane_count; ++i)
+ {
+ wxPaneInfo& pane = all_panes.Item(i);
+ if (pane.name == wxT("dummy"))
+ continue;
+ wxAuiTabCtrl* tabctrl = ((wxTabFrame*)pane.window)->m_tabs;
+ if (tabctrl != ctrl)
+ tabctrl->SetSelectedFont(m_normal_font);
+ else
+ tabctrl->SetSelectedFont(m_selected_font);
+ tabctrl->Refresh();
+ }
+
+ wnd->SetFocus();
+
+ return old_curpage;
+ }
+ }
+
+ return m_curpage;
+}
+
+// GetPageCount() returns the total number of
+// pages managed by the multi-notebook
+size_t wxAuiMultiNotebook::GetPageCount() const
+{
+ return m_tabs.GetPageCount();
+}
+
+// GetPage() returns the wxWindow pointer of the
+// specified page
+wxWindow* wxAuiMultiNotebook::GetPage(size_t page_idx) const
+{
+ wxASSERT(page_idx < m_tabs.GetPageCount());
+
+ return m_tabs.GetWindowFromIdx(page_idx);
+}
+
+// DoSizing() performs all sizing operations in each tab control
+void wxAuiMultiNotebook::DoSizing()
+{
+ wxPaneInfoArray& all_panes = m_mgr.GetAllPanes();
+ size_t i, pane_count = all_panes.GetCount();
+ for (i = 0; i < pane_count; ++i)
+ {
+ if (all_panes.Item(i).name == wxT("dummy"))
+ continue;
+
+ wxTabFrame* tabframe = (wxTabFrame*)all_panes.Item(i).window;
+ tabframe->DoSizing();
+ }
+}
+
+// GetActiveTabCtrl() returns the active tab control. It is
+// called to determine which control gets new windows being added
+wxAuiTabCtrl* wxAuiMultiNotebook::GetActiveTabCtrl()
+{
+ if (m_curpage >= 0 && m_curpage < (int)m_tabs.GetPageCount())
+ {
+ wxAuiTabCtrl* ctrl;
+ int idx;
+
+ // find the tab ctrl with the current page
+ if (FindTab(m_tabs.GetPage(m_curpage).window,
+ &ctrl, &idx))
+ {
+ return ctrl;
+ }
+ }
+
+ // no current page, just find the first tab ctrl
+ wxPaneInfoArray& all_panes = m_mgr.GetAllPanes();
+ size_t i, pane_count = all_panes.GetCount();
+ for (i = 0; i < pane_count; ++i)
+ {
+ if (all_panes.Item(i).name == wxT("dummy"))
+ continue;
+
+ wxTabFrame* tabframe = (wxTabFrame*)all_panes.Item(i).window;
+ return tabframe->m_tabs;
+ }
+
+ // If there is no tabframe at all, create one
+ wxTabFrame* tabframe = new wxTabFrame;
+ tabframe->m_tabs = new wxAuiTabCtrl(this,
+ m_tab_id_counter++,
+ wxDefaultPosition,
+ wxDefaultSize,
+ wxNO_BORDER);
+ m_mgr.AddPane(tabframe,
+ wxPaneInfo().Center().CaptionVisible(false));
+
+ m_mgr.Update();
+
+ return tabframe->m_tabs;
+}
+
+// FindTab() finds the tab control that currently contains the window as well
+// as the index of the window in the tab control. It returns true if the
+// window was found, otherwise false.
+bool wxAuiMultiNotebook::FindTab(wxWindow* page, wxAuiTabCtrl** ctrl, int* idx)
+{
+ wxPaneInfoArray& all_panes = m_mgr.GetAllPanes();
+ size_t i, pane_count = all_panes.GetCount();
+ for (i = 0; i < pane_count; ++i)
+ {
+ if (all_panes.Item(i).name == wxT("dummy"))
+ continue;
+
+ wxTabFrame* tabframe = (wxTabFrame*)all_panes.Item(i).window;
+
+ int page_idx = tabframe->m_tabs->GetIdxFromWindow(page);
+ if (page_idx != -1)
+ {
+ *ctrl = tabframe->m_tabs;
+ *idx = page_idx;
+ return true;
+ }
+ }
+
+ return false;
+}
+
+
+void wxAuiMultiNotebook::OnEraseBackground(wxEraseEvent&)
+{
+}
+
+void wxAuiMultiNotebook::OnSize(wxSizeEvent&)
+{
+}
+
+void wxAuiMultiNotebook::OnTabClicked(wxCommandEvent& command_evt)
+{
+ wxAuiNotebookEvent& evt = (wxAuiNotebookEvent&)command_evt;
+
+ wxAuiTabCtrl* ctrl = (wxAuiTabCtrl*)evt.GetEventObject();
+ wxASSERT(ctrl != NULL);
+
+ wxWindow* wnd = ctrl->GetWindowFromIdx(evt.GetSelection());
+ wxASSERT(wnd != NULL);
+
+ int idx = m_tabs.GetIdxFromWindow(wnd);
+ wxASSERT(idx != -1);
+
+ SetSelection(idx);
+}
+
+void wxAuiMultiNotebook::OnTabBeginDrag(wxCommandEvent&)
+{
+}
+
+void wxAuiMultiNotebook::OnTabDragMotion(wxCommandEvent& evt)
+{
+ wxPoint screen_pt = ::wxGetMousePosition();
+ wxPoint client_pt = ScreenToClient(screen_pt);
+ wxPoint zero(0,0);
+
+ wxAuiTabCtrl* src_tabs = (wxAuiTabCtrl*)evt.GetEventObject();
+
+ wxAuiTabCtrl* tab_ctrl = GetTabCtrlFromPoint(client_pt);
+ if (tab_ctrl == src_tabs)
+ {
+ // inner-tabctrl dragging is not yet implemented
+ m_mgr.HideHint();
+ return;
+ }
+
+ if (tab_ctrl)
+ {
+ wxRect hint_rect = tab_ctrl->GetRect();
+ ClientToScreen(&hint_rect.x, &hint_rect.y);
+ m_mgr.ShowHint(hint_rect);
+ }
+ else
+ {
+ m_mgr.DrawHintRect(m_dummy_wnd, client_pt, zero);
+ }
+}
+
+
+
+void wxAuiMultiNotebook::OnTabEndDrag(wxCommandEvent& command_evt)
+{
+ wxAuiNotebookEvent& evt = (wxAuiNotebookEvent&)command_evt;
+
+ m_mgr.HideHint();
+
+
+ // get the mouse position, which will be used to determine the drop point
+ wxPoint mouse_screen_pt = ::wxGetMousePosition();
+ wxPoint mouse_client_pt = ScreenToClient(mouse_screen_pt);
+
+
+ // the src tab control is the control that fired this event
+ wxAuiTabCtrl* src_tabs = (wxAuiTabCtrl*)evt.GetEventObject();
+ wxAuiTabCtrl* dest_tabs = NULL;
+
+
+ // If the pointer is in an existing tab frame, do a tab insert
+ wxWindow* hit_wnd = ::wxFindWindowAtPoint(mouse_screen_pt);
+ wxTabFrame* tab_frame = (wxTabFrame*)GetTabFrameFromTabCtrl(hit_wnd);
+ if (tab_frame)
+ {
+ dest_tabs = tab_frame->m_tabs;
+
+ if (dest_tabs == src_tabs)
+ return;
+ }
+ else
+ {
+ // If there is no tabframe at all, create one
+ wxTabFrame* new_tabs = new wxTabFrame;
+ new_tabs->m_tabs = new wxAuiTabCtrl(this,
+ m_tab_id_counter++,
+ wxDefaultPosition,
+ wxDefaultSize,
+ wxNO_BORDER);
+ m_mgr.AddPane(new_tabs,
+ wxPaneInfo().Bottom().CaptionVisible(false),
+ mouse_client_pt);
+ m_mgr.Update();
+ dest_tabs = new_tabs->m_tabs;
+ }
+
+
+
+ // remove the page from the source tabs
+ wxAuiNotebookPage page_info = src_tabs->GetPage(evt.GetSelection());
+ page_info.active = false;
+ src_tabs->RemovePage(page_info.window);
+ if (src_tabs->GetPageCount() > 0)
+ {
+ src_tabs->SetActivePage((size_t)0);
+ src_tabs->DoShowHide();
+ src_tabs->Refresh();
+ }
+
+
+
+ // add the page to the destination tabs
+ dest_tabs->AddPage(page_info.window, page_info);
+
+ if (src_tabs->GetPageCount() == 0)
+ {
+ RemoveEmptyTabFrames();
+ }
+
+ DoSizing();
+ dest_tabs->DoShowHide();
+ dest_tabs->Refresh();
+
+ SetSelection(m_tabs.GetIdxFromWindow(page_info.window));
+}
+
+wxAuiTabCtrl* wxAuiMultiNotebook::GetTabCtrlFromPoint(const wxPoint& pt)
+{
+ // if we've just removed the last tab from the source
+ // tab set, the remove the tab control completely
+ wxPaneInfoArray& all_panes = m_mgr.GetAllPanes();
+ size_t i, pane_count = all_panes.GetCount();
+ for (i = 0; i < pane_count; ++i)
+ {
+ if (all_panes.Item(i).name == wxT("dummy"))
+ continue;
+
+ wxTabFrame* tabframe = (wxTabFrame*)all_panes.Item(i).window;
+ if (tabframe->m_tab_rect.Inside(pt))
+ return tabframe->m_tabs;
+ }
+
+ return NULL;
+}
+
+wxWindow* wxAuiMultiNotebook::GetTabFrameFromTabCtrl(wxWindow* tab_ctrl)
+{
+ // if we've just removed the last tab from the source
+ // tab set, the remove the tab control completely
+ wxPaneInfoArray& all_panes = m_mgr.GetAllPanes();
+ size_t i, pane_count = all_panes.GetCount();
+ for (i = 0; i < pane_count; ++i)
+ {
+ if (all_panes.Item(i).name == wxT("dummy"))
+ continue;
+
+ wxTabFrame* tabframe = (wxTabFrame*)all_panes.Item(i).window;
+ if (tabframe->m_tabs == tab_ctrl)
+ {
+ return tabframe;
+ }
+ }
+
+ return NULL;
+}
+
+void wxAuiMultiNotebook::RemoveEmptyTabFrames()
+{
+ bool must_update = false;
+
+ // if we've just removed the last tab from the source
+ // tab set, the remove the tab control completely
+ wxPaneInfoArray all_panes = m_mgr.GetAllPanes();
+ size_t i, pane_count = all_panes.GetCount();
+ for (i = 0; i < pane_count; ++i)
+ {
+ if (all_panes.Item(i).name == wxT("dummy"))
+ continue;
+
+ wxTabFrame* tab_frame = (wxTabFrame*)all_panes.Item(i).window;
+ if (tab_frame->m_tabs->GetPageCount() == 0)
+ {
+ m_mgr.DetachPane(tab_frame);
+
+ // use pending delete because sometimes during
+ // window closing, refreshs are pending
+ if (!wxPendingDelete.Member(tab_frame->m_tabs))
+ wxPendingDelete.Append(tab_frame->m_tabs);
+ //tab_frame->m_tabs->Destroy();
+
+ delete tab_frame;
+ must_update = true;
+ }
+ }
+
+
+ // check to see if there is still a center pane;
+ // if there isn't, make a frame the center pane
+ wxPaneInfoArray panes = m_mgr.GetAllPanes();
+ pane_count = panes.GetCount();
+ wxWindow* first_good = NULL;
+ bool center_found = false;
+ for (i = 0; i < pane_count; ++i)
+ {
+ if (panes.Item(i).name == wxT("dummy"))
+ continue;
+ if (panes.Item(i).dock_direction == wxAUI_DOCK_CENTRE)
+ center_found = true;
+ if (!first_good)
+ first_good = panes.Item(i).window;
+ }
+
+ if (!center_found && first_good)
+ {
+ m_mgr.GetPane(first_good).Centre();
+ must_update = true;
+ }
+
+ m_mgr.Update();
+}
+
+void wxAuiMultiNotebook::OnChildFocus(wxChildFocusEvent& evt)
+{
+ int idx = m_tabs.GetIdxFromWindow(evt.GetWindow());
+ if (idx != -1 && idx != m_curpage)
+ {
+ SetSelection(idx);
+ }
+}
+
+
+void wxAuiMultiNotebook::OnTabButton(wxCommandEvent& command_evt)
+{
+ wxAuiNotebookEvent& evt = (wxAuiNotebookEvent&)command_evt;
+ wxAuiTabCtrl* tabs = (wxAuiTabCtrl*)evt.GetEventObject();
+
+ int button_id = evt.GetInt();
+
+ if (button_id == wxAuiButtonClose)
+ {
+ int selection = tabs->GetActivePage();
+
+ if (selection != -1)
+ {
+ wxWindow* close_wnd = tabs->GetWindowFromIdx(selection);
+
+ if (close_wnd->IsKindOf(CLASSINFO(wxTabMDIChildFrame)))
+ {
+ close_wnd->Close();
+ }
+ else
+ {
+ int main_idx = m_tabs.GetIdxFromWindow(close_wnd);
+ DeletePage(main_idx);
+ }
+ }
+ }
+}
+
+
+
+
+#endif // wxUSE_AUI
--- /dev/null
+/////////////////////////////////////////////////////////////////////////////
+// Name: src/generic/mdig.cpp
+// Purpose: Generic MDI (Multiple Document Interface) classes
+// Author: Hans Van Leemputten
+// Modified by: Benjamin I. Williams / Kirix Corporation
+// Created: 29/07/2002
+// RCS-ID: $Id$
+// Copyright: (c) Hans Van Leemputten
+// Licence: wxWindows licence
+/////////////////////////////////////////////////////////////////////////////
+
+// ===========================================================================
+// declarations
+// ===========================================================================
+
+// ---------------------------------------------------------------------------
+// headers
+// ---------------------------------------------------------------------------
+
+// For compilers that support precompilation, includes "wx.h".
+#include "wx/wxprec.h"
+
+#ifdef __BORLANDC__
+ #pragma hdrstop
+#endif
+
+#if wxUSE_MDI
+
+#include "wx/aui/tabmdi.h"
+
+#ifndef WX_PRECOMP
+ #include "wx/panel.h"
+ #include "wx/menu.h"
+ #include "wx/intl.h"
+ #include "wx/log.h"
+#endif //WX_PRECOMP
+
+#include "wx/stockitem.h"
+
+enum MDI_MENU_ID
+{
+ wxWINDOWCLOSE = 4001,
+ wxWINDOWCLOSEALL,
+ wxWINDOWNEXT,
+ wxWINDOWPREV
+};
+
+//-----------------------------------------------------------------------------
+// wxTabMDIParentFrame
+//-----------------------------------------------------------------------------
+
+IMPLEMENT_DYNAMIC_CLASS(wxTabMDIParentFrame, wxFrame)
+
+BEGIN_EVENT_TABLE(wxTabMDIParentFrame, wxFrame)
+#if wxUSE_MENUS
+ EVT_MENU (wxID_ANY, wxTabMDIParentFrame::DoHandleMenu)
+#endif
+END_EVENT_TABLE()
+
+wxTabMDIParentFrame::wxTabMDIParentFrame()
+{
+ Init();
+}
+
+wxTabMDIParentFrame::wxTabMDIParentFrame(wxWindow *parent,
+ wxWindowID id,
+ const wxString& title,
+ const wxPoint& pos,
+ const wxSize& size,
+ long style,
+ const wxString& name)
+{
+ Init();
+ (void)Create(parent, id, title, pos, size, style, name);
+}
+
+wxTabMDIParentFrame::~wxTabMDIParentFrame()
+{
+ // Make sure the client window is destructed before the menu bars are!
+ wxDELETE(m_pClientWindow);
+
+#if wxUSE_MENUS
+ RemoveWindowMenu(GetMenuBar());
+ delete m_pWindowMenu;
+#endif // wxUSE_MENUS
+}
+
+bool wxTabMDIParentFrame::Create(wxWindow *parent,
+ wxWindowID id,
+ const wxString& title,
+ const wxPoint& pos,
+ const wxSize& size,
+ long style,
+ const wxString& name)
+{
+#if wxUSE_MENUS
+ // this style can be used to prevent a window from having the standard MDI
+ // "Window" menu
+ if (!(style & wxFRAME_NO_WINDOW_MENU))
+ {
+ m_pWindowMenu = new wxMenu;
+ m_pWindowMenu->Append(wxWINDOWCLOSE, _("Cl&ose"));
+ m_pWindowMenu->Append(wxWINDOWCLOSEALL, _("Close All"));
+ m_pWindowMenu->AppendSeparator();
+ m_pWindowMenu->Append(wxWINDOWNEXT, _("&Next"));
+ m_pWindowMenu->Append(wxWINDOWPREV, _("&Previous"));
+ }
+#endif // wxUSE_MENUS
+
+ wxFrame::Create(parent, id, title, pos, size, style, name);
+ OnCreateClient();
+ return true;
+}
+
+#if wxUSE_MENUS
+void wxTabMDIParentFrame::SetWindowMenu(wxMenu* pMenu)
+{
+ // Replace the window menu from the currently loaded menu bar.
+ wxMenuBar *pMenuBar = GetMenuBar();
+
+ if (m_pWindowMenu)
+ {
+ RemoveWindowMenu(pMenuBar);
+ wxDELETE(m_pWindowMenu);
+ }
+
+ if (pMenu)
+ {
+ m_pWindowMenu = pMenu;
+ AddWindowMenu(pMenuBar);
+ }
+}
+
+void wxTabMDIParentFrame::SetMenuBar(wxMenuBar *pMenuBar)
+{
+ // Remove the Window menu from the old menu bar
+ RemoveWindowMenu(GetMenuBar());
+
+ // Add the Window menu to the new menu bar.
+ AddWindowMenu(pMenuBar);
+
+ wxFrame::SetMenuBar(pMenuBar);
+ m_pMyMenuBar = GetMenuBar();
+}
+#endif // wxUSE_MENUS
+
+void wxTabMDIParentFrame::SetChildMenuBar(wxTabMDIChildFrame* pChild)
+{
+#if wxUSE_MENUS
+ if (!pChild)
+ {
+ // No Child, set Our menu bar back.
+ SetMenuBar(m_pMyMenuBar);
+
+ // Make sure we know our menu bar is in use
+ m_pMyMenuBar = NULL;
+ }
+ else
+ {
+ if (pChild->GetMenuBar() == NULL)
+ return;
+
+ // Do we need to save the current bar?
+ if (m_pMyMenuBar == NULL)
+ m_pMyMenuBar = GetMenuBar();
+
+ SetMenuBar(pChild->GetMenuBar());
+ }
+#endif // wxUSE_MENUS
+}
+
+bool wxTabMDIParentFrame::ProcessEvent(wxEvent& event)
+{
+ // Stops the same event being processed repeatedly
+ static wxEventType inEvent = wxEVT_NULL;
+ if (inEvent == event.GetEventType())
+ return false;
+
+ inEvent = event.GetEventType();
+
+ // Let the active child (if any) process the event first.
+ bool res = false;
+ if (m_pActiveChild &&
+ event.IsCommandEvent() &&
+ event.GetEventObject() != m_pClientWindow &&
+ !(event.GetEventType() == wxEVT_ACTIVATE ||
+ event.GetEventType() == wxEVT_SET_FOCUS ||
+ event.GetEventType() == wxEVT_KILL_FOCUS ||
+ event.GetEventType() == wxEVT_CHILD_FOCUS ||
+ event.GetEventType() == wxEVT_COMMAND_SET_FOCUS ||
+ event.GetEventType() == wxEVT_COMMAND_KILL_FOCUS )
+ )
+ {
+ res = m_pActiveChild->GetEventHandler()->ProcessEvent(event);
+ }
+
+ // If the event was not handled this frame will handle it!
+ if (!res)
+ {
+ //res = GetEventHandler()->ProcessEvent(event);
+ res = wxEvtHandler::ProcessEvent(event);
+ }
+
+ inEvent = wxEVT_NULL;
+
+ return res;
+}
+
+wxTabMDIChildFrame *wxTabMDIParentFrame::GetActiveChild() const
+{
+ return m_pActiveChild;
+}
+
+void wxTabMDIParentFrame::SetActiveChild(wxTabMDIChildFrame* pChildFrame)
+{
+ m_pActiveChild = pChildFrame;
+}
+
+wxTabMDIClientWindow *wxTabMDIParentFrame::GetClientWindow() const
+{
+ return m_pClientWindow;
+}
+
+wxTabMDIClientWindow *wxTabMDIParentFrame::OnCreateClient()
+{
+ m_pClientWindow = new wxTabMDIClientWindow( this );
+ return m_pClientWindow;
+}
+
+void wxTabMDIParentFrame::ActivateNext()
+{
+ if (m_pClientWindow && m_pClientWindow->GetSelection() != -1)
+ {
+ size_t active = m_pClientWindow->GetSelection() + 1;
+ if (active >= m_pClientWindow->GetPageCount())
+ active = 0;
+
+ m_pClientWindow->SetSelection(active);
+ }
+}
+
+void wxTabMDIParentFrame::ActivatePrevious()
+{
+ if (m_pClientWindow && m_pClientWindow->GetSelection() != -1)
+ {
+ int active = m_pClientWindow->GetSelection() - 1;
+ if (active < 0)
+ active = m_pClientWindow->GetPageCount() - 1;
+
+ m_pClientWindow->SetSelection(active);
+ }
+}
+
+void wxTabMDIParentFrame::Init()
+{
+ m_pClientWindow = NULL;
+ m_pActiveChild = NULL;
+#if wxUSE_MENUS
+ m_pWindowMenu = NULL;
+ m_pMyMenuBar = NULL;
+#endif // wxUSE_MENUS
+}
+
+#if wxUSE_MENUS
+void wxTabMDIParentFrame::RemoveWindowMenu(wxMenuBar* pMenuBar)
+{
+ if (pMenuBar && m_pWindowMenu)
+ {
+ // Remove old window menu
+ int pos = pMenuBar->FindMenu(_("&Window"));
+ if (pos != wxNOT_FOUND)
+ {
+ // DBG:: We're going to delete the wrong menu!!!
+ wxASSERT(m_pWindowMenu == pMenuBar->GetMenu(pos));
+ pMenuBar->Remove(pos);
+ }
+ }
+}
+
+void wxTabMDIParentFrame::AddWindowMenu(wxMenuBar *pMenuBar)
+{
+ if (pMenuBar && m_pWindowMenu)
+ {
+ int pos = pMenuBar->FindMenu(wxGetStockLabel(wxID_HELP,false));
+ if (pos == wxNOT_FOUND)
+ pMenuBar->Append(m_pWindowMenu, _("&Window"));
+ else
+ pMenuBar->Insert(pos, m_pWindowMenu, _("&Window"));
+ }
+}
+
+void wxTabMDIParentFrame::DoHandleMenu(wxCommandEvent& event)
+{
+ switch (event.GetId())
+ {
+ case wxWINDOWCLOSE:
+ if (m_pActiveChild)
+ m_pActiveChild->Close();
+ break;
+ case wxWINDOWCLOSEALL:
+ while (m_pActiveChild)
+ {
+ if (!m_pActiveChild->Close())
+ {
+ return; // failure
+ }
+ else
+ {
+ delete m_pActiveChild;
+ m_pActiveChild = NULL;
+ }
+ }
+ break;
+ case wxWINDOWNEXT:
+ ActivateNext();
+ break;
+ case wxWINDOWPREV:
+ ActivatePrevious();
+ break;
+ default:
+ event.Skip();
+ }
+}
+#endif // wxUSE_MENUS
+
+void wxTabMDIParentFrame::DoGetClientSize(int* width, int* height) const
+{
+ wxFrame::DoGetClientSize(width, height);
+}
+
+//-----------------------------------------------------------------------------
+// wxTabMDIChildFrame
+//-----------------------------------------------------------------------------
+
+IMPLEMENT_DYNAMIC_CLASS(wxTabMDIChildFrame, wxPanel)
+
+BEGIN_EVENT_TABLE(wxTabMDIChildFrame, wxPanel)
+ EVT_MENU_HIGHLIGHT_ALL(wxTabMDIChildFrame::OnMenuHighlight)
+ EVT_ACTIVATE(wxTabMDIChildFrame::OnActivate)
+ EVT_CLOSE(wxTabMDIChildFrame::OnCloseWindow)
+END_EVENT_TABLE()
+
+wxTabMDIChildFrame::wxTabMDIChildFrame()
+{
+ Init();
+}
+
+wxTabMDIChildFrame::wxTabMDIChildFrame(wxTabMDIParentFrame *parent,
+ wxWindowID id,
+ const wxString& title,
+ const wxPoint& WXUNUSED(pos),
+ const wxSize& size,
+ long style,
+ const wxString& name)
+{
+ Init();
+ Create(parent, id, title, wxDefaultPosition, size, style, name);
+}
+
+wxTabMDIChildFrame::~wxTabMDIChildFrame()
+{
+#if wxUSE_MENUS
+ wxDELETE(m_pMenuBar);
+#endif // wxUSE_MENUS
+}
+
+bool wxTabMDIChildFrame::Create(wxTabMDIParentFrame* parent,
+ wxWindowID id,
+ const wxString& title,
+ const wxPoint& WXUNUSED(pos),
+ const wxSize& size,
+ long style,
+ const wxString& name)
+{
+ wxTabMDIClientWindow* pClientWindow = parent->GetClientWindow();
+ wxASSERT_MSG((pClientWindow != (wxWindow*) NULL), wxT("Missing MDI client window."));
+
+ wxPanel::Create(pClientWindow, id, wxDefaultPosition, size, style|wxNO_BORDER, name);
+
+ SetMDIParentFrame(parent);
+
+ // this is the currently active child
+ parent->SetActiveChild(this);
+
+ m_title = title;
+
+ pClientWindow->AddPage(this, title, true);
+ pClientWindow->Refresh();
+
+ return true;
+}
+
+bool wxTabMDIChildFrame::Destroy()
+{
+ wxTabMDIParentFrame* pParentFrame = GetMDIParentFrame();
+ wxASSERT_MSG(pParentFrame, wxT("Missing MDI Parent Frame"));
+
+ wxTabMDIClientWindow* pClientWindow = pParentFrame->GetClientWindow();
+ wxASSERT_MSG(pClientWindow, wxT("Missing MDI Client Window"));
+
+ bool bActive = false;
+ if (pParentFrame->GetActiveChild() == this)
+ {
+ pParentFrame->SetActiveChild(NULL);
+ pParentFrame->SetChildMenuBar(NULL);
+ bActive = true;
+ }
+
+ size_t pos, page_count = pClientWindow->GetPageCount();
+ for (pos = 0; pos < page_count; pos++)
+ {
+ if (pClientWindow->GetPage(pos) == this)
+ return pClientWindow->DeletePage(pos);
+ }
+
+ return false;
+}
+
+/*
+ wxTabMDIParentFrame* pParentFrame = GetMDIParentFrame();
+ wxASSERT_MSG(pParentFrame, wxT("Missing MDI Parent Frame"));
+
+ bool bActive = false;
+ if (pParentFrame->GetActiveChild() == this)
+ {
+ pParentFrame->SetActiveChild(NULL);
+ pParentFrame->SetChildMenuBar(NULL);
+ bActive = true;
+ }
+
+ wxTabMDIClientWindow* pClientWindow = pParentFrame->GetClientWindow();
+
+ // remove page if it is still there
+ size_t pos;
+ for (pos = 0; pos < pClientWindow->GetPageCount(); pos++)
+ {
+ if (pClientWindow->GetPage(pos) == this)
+ {
+ if (pClientWindow->RemovePage(pos))
+ pClientWindow->Refresh();
+ break;
+ }
+ }
+
+ if (bActive)
+ {
+ // Set the new selection to the a remaining page
+ if (pos < pClientWindow->GetPageCount())
+ {
+ pClientWindow->SetSelection(pos);
+ }
+ else
+ {
+ if (pClientWindow->GetPageCount() >= 1)
+ pClientWindow->SetSelection(pClientWindow->GetPageCount() - 1);
+ }
+ }
+
+ // delete the child frame with pending delete, as is
+ // customary with frame windows
+ if (!wxPendingDelete.Member(this))
+ wxPendingDelete.Append(this);
+
+ return true;
+*/
+
+
+#if wxUSE_MENUS
+void wxTabMDIChildFrame::SetMenuBar(wxMenuBar *menu_bar)
+{
+ wxMenuBar *pOldMenuBar = m_pMenuBar;
+ m_pMenuBar = menu_bar;
+
+ if (m_pMenuBar)
+ {
+ wxTabMDIParentFrame* pParentFrame = GetMDIParentFrame();
+ wxASSERT_MSG(pParentFrame, wxT("Missing MDI Parent Frame"));
+
+ m_pMenuBar->SetParent(pParentFrame);
+ if (pParentFrame->GetActiveChild() == this)
+ {
+ // replace current menu bars
+ if (pOldMenuBar)
+ pParentFrame->SetChildMenuBar(NULL);
+ pParentFrame->SetChildMenuBar(this);
+ }
+ }
+}
+
+wxMenuBar *wxTabMDIChildFrame::GetMenuBar() const
+{
+ return m_pMenuBar;
+}
+#endif // wxUSE_MENUS
+
+void wxTabMDIChildFrame::SetTitle(const wxString& title)
+{
+ m_title = title;
+
+ wxTabMDIParentFrame* pParentFrame = GetMDIParentFrame();
+ wxASSERT_MSG(pParentFrame, wxT("Missing MDI Parent Frame"));
+
+ wxTabMDIClientWindow* pClientWindow = pParentFrame->GetClientWindow();
+ if (pClientWindow != NULL)
+ {
+ size_t pos;
+ for (pos = 0; pos < pClientWindow->GetPageCount(); pos++)
+ {
+ if (pClientWindow->GetPage(pos) == this)
+ {
+ pClientWindow->SetPageText(pos, m_title);
+ break;
+ }
+ }
+ }
+}
+
+wxString wxTabMDIChildFrame::GetTitle() const
+{
+ return m_title;
+}
+
+void wxTabMDIChildFrame::Activate()
+{
+ wxTabMDIParentFrame* pParentFrame = GetMDIParentFrame();
+ wxASSERT_MSG(pParentFrame, wxT("Missing MDI Parent Frame"));
+
+ wxTabMDIClientWindow* pClientWindow = pParentFrame->GetClientWindow();
+
+ if (pClientWindow != NULL)
+ {
+ size_t pos;
+ for (pos = 0; pos < pClientWindow->GetPageCount(); pos++)
+ {
+ if (pClientWindow->GetPage(pos) == this)
+ {
+ pClientWindow->SetSelection(pos);
+ break;
+ }
+ }
+ }
+}
+
+void wxTabMDIChildFrame::OnMenuHighlight(wxMenuEvent& event)
+{
+#if wxUSE_STATUSBAR
+ if (m_pMDIParentFrame)
+ {
+ // we don't have any help text for this item,
+ // but may be the MDI frame does?
+ m_pMDIParentFrame->OnMenuHighlight(event);
+ }
+#else
+ wxUnusedVar(event);
+#endif // wxUSE_STATUSBAR
+}
+
+void wxTabMDIChildFrame::OnActivate(wxActivateEvent& WXUNUSED(event))
+{
+ // do nothing
+}
+
+void wxTabMDIChildFrame::OnCloseWindow(wxCloseEvent& WXUNUSED(event))
+{
+ Destroy();
+}
+
+void wxTabMDIChildFrame::SetMDIParentFrame(wxTabMDIParentFrame* parentFrame)
+{
+ m_pMDIParentFrame = parentFrame;
+}
+
+wxTabMDIParentFrame* wxTabMDIChildFrame::GetMDIParentFrame() const
+{
+ return m_pMDIParentFrame;
+}
+
+void wxTabMDIChildFrame::Init()
+{
+ m_pMDIParentFrame = NULL;
+#if wxUSE_MENUS
+ m_pMenuBar = NULL;
+#endif // wxUSE_MENUS
+}
+
+bool wxTabMDIChildFrame::Show(bool WXUNUSED(show))
+{
+ // do nothing
+ return true;
+}
+
+void wxTabMDIChildFrame::DoShow(bool show)
+{
+ wxWindow::Show(show);
+}
+
+void wxTabMDIChildFrame::DoSetSize(int x, int y, int width, int height, int WXUNUSED(sizeFlags))
+{
+ m_mdi_newrect = wxRect(x, y, width, height);
+}
+
+void wxTabMDIChildFrame::DoMoveWindow(int x, int y, int width, int height)
+{
+ m_mdi_newrect = wxRect(x, y, width, height);
+}
+
+void wxTabMDIChildFrame::ApplyMDIChildFrameRect()
+{
+ if (m_mdi_currect != m_mdi_newrect)
+ {
+ wxPanel::DoMoveWindow(m_mdi_newrect.x, m_mdi_newrect.y,
+ m_mdi_newrect.width, m_mdi_newrect.height);
+ m_mdi_currect = m_mdi_newrect;
+ }
+}
+
+
+//-----------------------------------------------------------------------------
+// wxTabMDIClientWindow
+//-----------------------------------------------------------------------------
+
+IMPLEMENT_DYNAMIC_CLASS(wxTabMDIClientWindow, wxAuiMultiNotebook)
+
+BEGIN_EVENT_TABLE(wxTabMDIClientWindow, wxAuiMultiNotebook)
+ EVT_AUINOTEBOOK_PAGE_CHANGED(-1, wxTabMDIClientWindow::OnPageChanged)
+ EVT_SIZE(wxTabMDIClientWindow::OnSize)
+END_EVENT_TABLE()
+
+wxTabMDIClientWindow::wxTabMDIClientWindow()
+{
+}
+
+wxTabMDIClientWindow::wxTabMDIClientWindow(wxTabMDIParentFrame* parent, long style)
+{
+ CreateClient(parent, style);
+}
+
+wxTabMDIClientWindow::~wxTabMDIClientWindow()
+{
+ DestroyChildren();
+}
+
+bool wxTabMDIClientWindow::CreateClient(wxTabMDIParentFrame* parent, long style)
+{
+ SetWindowStyleFlag(style);
+
+ if (!wxAuiMultiNotebook::Create(parent,
+ -1,
+ wxPoint(0,0),
+ wxSize(100, 100),
+ wxNO_BORDER))
+ {
+ return false;
+ }
+
+ wxColour bkcolour = wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE);
+ SetBackgroundColour(bkcolour);
+
+ m_mgr.GetArtProvider()->SetColour(wxAUI_ART_BACKGROUND_COLOUR, bkcolour);
+
+ return true;
+}
+
+int wxTabMDIClientWindow::SetSelection(size_t nPage)
+{
+ return wxAuiMultiNotebook::SetSelection(nPage);
+}
+
+void wxTabMDIClientWindow::PageChanged(int old_selection, int new_selection)
+{
+ // don't do anything if the page doesn't actually change
+ if (old_selection == new_selection)
+ return;
+
+ // don't do anything if the new page is already active
+ if (new_selection != -1)
+ {
+ wxTabMDIChildFrame* child = (wxTabMDIChildFrame*)GetPage(new_selection);
+ if (child->GetMDIParentFrame()->GetActiveChild() == child)
+ return;
+ }
+
+ // notify old active child that it has been deactivated
+ if (old_selection != -1)
+ {
+ wxTabMDIChildFrame* old_child = (wxTabMDIChildFrame*)GetPage(old_selection);
+ wxASSERT_MSG(old_child, wxT("wxTabMDIClientWindow::PageChanged - null page pointer"));
+
+ wxActivateEvent event(wxEVT_ACTIVATE, false, old_child->GetId());
+ event.SetEventObject(old_child);
+ old_child->GetEventHandler()->ProcessEvent(event);
+ }
+
+ // notify new active child that it has been activated
+ if (new_selection != -1)
+ {
+ wxTabMDIChildFrame* active_child = (wxTabMDIChildFrame*)GetPage(new_selection);
+ wxASSERT_MSG(active_child, wxT("wxTabMDIClientWindow::PageChanged - null page pointer"));
+
+ wxActivateEvent event(wxEVT_ACTIVATE, true, active_child->GetId());
+ event.SetEventObject(active_child);
+ active_child->GetEventHandler()->ProcessEvent(event);
+
+ if (active_child->GetMDIParentFrame())
+ {
+ active_child->GetMDIParentFrame()->SetActiveChild(active_child);
+ active_child->GetMDIParentFrame()->SetChildMenuBar(active_child);
+ }
+ }
+}
+
+void wxTabMDIClientWindow::OnPageChanged(wxAuiNotebookEvent& evt)
+{
+ PageChanged(evt.GetOldSelection(), evt.GetSelection());
+ evt.Skip();
+}
+
+void wxTabMDIClientWindow::OnSize(wxSizeEvent& evt)
+{
+ wxAuiMultiNotebook::OnSize(evt);
+
+ for (size_t pos = 0; pos < GetPageCount(); pos++)
+ ((wxTabMDIChildFrame *)GetPage(pos))->ApplyMDIChildFrameRect();
+}
+
+#endif // wxUSE_MDI