X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/8907154c1a8a6882c6797d1f16393ddfb23e7f3a..3214ac5ad1d515d917b8273c2d2af4c4287b7a8b:/samples/shaped/shaped.cpp diff --git a/samples/shaped/shaped.cpp b/samples/shaped/shaped.cpp index ea00e88566..df4afac07e 100644 --- a/samples/shaped/shaped.cpp +++ b/samples/shaped/shaped.cpp @@ -40,6 +40,25 @@ #include "wx/dcclient.h" #include "wx/image.h" +// ---------------------------------------------------------------------------- +// constants +// ---------------------------------------------------------------------------- + +// menu ids +enum +{ + Show_Shaped = 100, + Show_Transparent, + + // must be consecutive and in the same order as wxShowEffect enum elements + Show_Effect_First, + Show_Effect_Roll = Show_Effect_First, + Show_Effect_Slide, + Show_Effect_Blend, + Show_Effect_Expand, + Show_Effect_Last = Show_Effect_Expand +}; + // ---------------------------------------------------------------------------- // private classes // ---------------------------------------------------------------------------- @@ -58,12 +77,27 @@ public: }; -// Define a new frame type: this is going to be our main frame +// Main frame just contains the menu items invoking the other tests +class MainFrame : public wxFrame +{ +public: + MainFrame(); + +private: + void OnShowShaped(wxCommandEvent& event); + void OnShowTransparent(wxCommandEvent& event); + void OnShowEffect(wxCommandEvent& event); + + DECLARE_EVENT_TABLE() +}; + +// Define a new frame type: this is going to the frame showing the +// effect of wxFRAME_SHAPED class ShapedFrame : public wxFrame { public: // ctor(s) - ShapedFrame(); + ShapedFrame(wxFrame *parent); void SetWindowShape(); // event handlers (these functions should _not_ be virtual) @@ -84,35 +118,87 @@ private: DECLARE_EVENT_TABLE() }; +// Define a new frame type: this is going to the frame showing the +// effect of wxWindow::SetTransparent and of +// wxWindow::SetBackgroundStyle(wxBG_STYLE_TRANSPARENT) +class SeeThroughFrame : public wxFrame +{ +public: + // ctor(s) + SeeThroughFrame(); -// ---------------------------------------------------------------------------- -// event tables and other macros for wxWidgets -// ---------------------------------------------------------------------------- + // event handlers (these functions should _not_ be virtual) + void OnDoubleClick(wxMouseEvent& evt); + void OnPaint(wxPaintEvent& evt); -// the event tables connect the wxWidgets events with the functions (event -// handlers) which process them. It can be also done at run-time, but for the -// simple menu events like this the static method is much simpler. -BEGIN_EVENT_TABLE(ShapedFrame, wxFrame) - EVT_LEFT_DCLICK(ShapedFrame::OnDoubleClick) - EVT_LEFT_DOWN(ShapedFrame::OnLeftDown) - EVT_LEFT_UP(ShapedFrame::OnLeftUp) - EVT_MOTION(ShapedFrame::OnMouseMove) - EVT_RIGHT_UP(ShapedFrame::OnExit) +private: + enum State + { + STATE_SEETHROUGH, + STATE_TRANSPARENT, + STATE_OPAQUE, + STATE_MAX + }; - EVT_PAINT(ShapedFrame::OnPaint) + State m_currentState; -#ifdef __WXGTK__ - EVT_WINDOW_CREATE(ShapedFrame::OnWindowCreate) -#endif -END_EVENT_TABLE() + // any class wishing to process wxWidgets events must use this macro + DECLARE_EVENT_TABLE() +}; +class EffectFrame : public wxFrame +{ +public: + EffectFrame(wxWindow *parent, + wxShowEffect effect, + // TODO: add menu command to the main frame to allow changing + // these parameters + unsigned timeout = 1000, + wxDirection dir = wxBOTTOM) + : wxFrame(parent, wxID_ANY, + wxString::Format("Frame shown with %s effect", + GetEffectName(effect)), + wxDefaultPosition, wxSize(450, 300)), + m_effect(effect), + m_timeout(timeout), + m_dir(dir) + { + new wxStaticText(this, wxID_ANY, + wxString::Format("Effect: %s", GetEffectName(effect)), + wxPoint(20, 20)); + new wxStaticText(this, wxID_ANY, + wxString::Format("Timeout: %ums", m_timeout), + wxPoint(20, 60)); -// Create a new application object: this macro will allow wxWidgets to create -// the application object during program execution (it's better than using a -// static object for many reasons) and also declares the accessor function -// wxGetApp() which will return the reference of the right type (i.e. MyApp and -// not wxApp) -IMPLEMENT_APP(MyApp) + ShowWithEffect(m_effect, m_timeout, m_dir); + + Connect(wxEVT_CLOSE_WINDOW, wxCloseEventHandler(EffectFrame::OnClose)); + } + +private: + static const char *GetEffectName(wxShowEffect effect) + { + static const char *names[] = + { + "roll", "slide", "fade", "expand", + }; + wxCOMPILE_TIME_ASSERT( WXSIZEOF(names) == wxSHOW_EFFECT_MAX, + EffectNamesMismatch ); + + return names[effect]; + } + + void OnClose(wxCloseEvent& event) + { + HideWithEffect(m_effect, m_timeout, m_dir); + + event.Skip(); + } + + wxShowEffect m_effect; + unsigned m_timeout; + wxDirection m_dir; +}; // ============================================================================ // implementation @@ -122,15 +208,17 @@ IMPLEMENT_APP(MyApp) // the application class // ---------------------------------------------------------------------------- +IMPLEMENT_APP(MyApp) + // `Main program' equivalent: the program execution "starts" here bool MyApp::OnInit() { + if ( !wxApp::OnInit() ) + return false; + wxInitAllImageHandlers(); - // Create the main application window - ShapedFrame *frame = new ShapedFrame(); - frame->Show(true); - SetTopWindow(frame); + new MainFrame; // success: wxApp::OnRun() will be called which will enter the main message // loop and the application will run. If we returned false here, the @@ -142,10 +230,79 @@ bool MyApp::OnInit() // main frame // ---------------------------------------------------------------------------- +BEGIN_EVENT_TABLE(MainFrame, wxFrame) + EVT_MENU(Show_Shaped, MainFrame::OnShowShaped) + EVT_MENU(Show_Transparent, MainFrame::OnShowTransparent) + EVT_MENU_RANGE(Show_Effect_First, Show_Effect_Last, MainFrame::OnShowEffect) +END_EVENT_TABLE() + +MainFrame::MainFrame() + : wxFrame(NULL, wxID_ANY, "wxWidgets Shaped Sample", + wxDefaultPosition, wxSize(200, 100)) +{ + wxMenuBar * const mbar = new wxMenuBar; + wxMenu * const menuFrames = new wxMenu; + menuFrames->Append(Show_Shaped, "Show &shaped window\tCtrl-S"); + menuFrames->Append(Show_Transparent, "Show &transparent window\tCtrl-T"); + menuFrames->AppendSeparator(); + menuFrames->Append(Show_Effect_Roll, "Show &rolled effect\tCtrl-R"); + menuFrames->Append(Show_Effect_Slide, "Show s&lide effect\tCtrl-L"); + menuFrames->Append(Show_Effect_Blend, "Show &fade effect\tCtrl-F"); + menuFrames->Append(Show_Effect_Expand, "Show &expand effect\tCtrl-E"); + menuFrames->AppendSeparator(); + menuFrames->Append(wxID_EXIT, "E&xit"); + + mbar->Append(menuFrames, "&Show"); + SetMenuBar(mbar); + + Show(); +} + +void MainFrame::OnShowShaped(wxCommandEvent& WXUNUSED(event)) +{ + ShapedFrame *shapedFrame = new ShapedFrame(this); + shapedFrame->Show(true); +} + +void MainFrame::OnShowTransparent(wxCommandEvent& WXUNUSED(event)) +{ + SeeThroughFrame *seeThroughFrame = new SeeThroughFrame(); + seeThroughFrame->Show(true); +} + +void MainFrame::OnShowEffect(wxCommandEvent& event) +{ + int effect = wxSHOW_EFFECT_ROLL + event.GetId() - Show_Effect_Roll; + static wxDirection direction = wxLEFT; + direction = (wxDirection)(((int)direction)<< 1); + if ( direction > wxDOWN ) + direction = wxLEFT ; + new EffectFrame(this, wx_static_cast(wxShowEffect, effect),1000,direction); +} + +// ---------------------------------------------------------------------------- +// shaped frame +// ---------------------------------------------------------------------------- + +BEGIN_EVENT_TABLE(ShapedFrame, wxFrame) + EVT_LEFT_DCLICK(ShapedFrame::OnDoubleClick) + EVT_LEFT_DOWN(ShapedFrame::OnLeftDown) + EVT_LEFT_UP(ShapedFrame::OnLeftUp) + EVT_MOTION(ShapedFrame::OnMouseMove) + EVT_RIGHT_UP(ShapedFrame::OnExit) + + EVT_PAINT(ShapedFrame::OnPaint) + +#ifdef __WXGTK__ + EVT_WINDOW_CREATE(ShapedFrame::OnWindowCreate) +#endif +END_EVENT_TABLE() + + // frame constructor -ShapedFrame::ShapedFrame() - : wxFrame((wxFrame *)NULL, wxID_ANY, wxEmptyString, - wxDefaultPosition, wxSize(100, 100), //wxDefaultSize, +ShapedFrame::ShapedFrame(wxFrame *parent) + : wxFrame(parent, wxID_ANY, wxEmptyString, + wxDefaultPosition, wxSize(100, 100), 0 | wxFRAME_SHAPED | wxSIMPLE_BORDER @@ -156,7 +313,7 @@ ShapedFrame::ShapedFrame() m_hasShape = false; m_bmp = wxBitmap(_T("star.png"), wxBITMAP_TYPE_PNG); SetSize(wxSize(m_bmp.GetWidth(), m_bmp.GetHeight())); - SetToolTip(wxT("Right-click to exit")); + SetToolTip(wxT("Right-click to close")); #ifndef __WXGTK__ // On wxGTK we can't do this yet because the window hasn't been created @@ -188,7 +345,6 @@ void ShapedFrame::OnDoubleClick(wxMouseEvent& WXUNUSED(evt)) void ShapedFrame::OnLeftDown(wxMouseEvent& evt) { CaptureMouse(); - //printf("Mouse captured\n"); wxPoint pos = ClientToScreen(evt.GetPosition()); wxPoint origin = GetPosition(); int dx = pos.x - origin.x; @@ -201,14 +357,12 @@ void ShapedFrame::OnLeftUp(wxMouseEvent& WXUNUSED(evt)) if (HasCapture()) { ReleaseMouse(); - //printf("Mouse released\n"); } } void ShapedFrame::OnMouseMove(wxMouseEvent& evt) { wxPoint pt = evt.GetPosition(); - //printf("x:%d y:%d\n", pt.x, pt.y); if (evt.Dragging() && evt.LeftIsDown()) { wxPoint pos = ClientToScreen(pt); @@ -232,3 +386,87 @@ void ShapedFrame::OnWindowCreate(wxWindowCreateEvent& WXUNUSED(evt)) SetWindowShape(); } +// ---------------------------------------------------------------------------- +// see-through frame +// ---------------------------------------------------------------------------- + +BEGIN_EVENT_TABLE(SeeThroughFrame, wxFrame) + EVT_LEFT_DCLICK(SeeThroughFrame::OnDoubleClick) + EVT_PAINT(SeeThroughFrame::OnPaint) +END_EVENT_TABLE() + +SeeThroughFrame::SeeThroughFrame() + : wxFrame(NULL, wxID_ANY, "Transparency test: double click here", + wxPoint(100, 30), wxSize(300, 300), + wxDEFAULT_FRAME_STYLE | + wxFULL_REPAINT_ON_RESIZE | + wxSTAY_ON_TOP), + m_currentState(STATE_SEETHROUGH) +{ + SetBackgroundColour(wxColour(255, 255, 255, 255)); + SetBackgroundStyle(wxBG_STYLE_TRANSPARENT); +} + +// Paints a grid of varying hue and alpha +void SeeThroughFrame::OnPaint(wxPaintEvent& WXUNUSED(evt)) +{ + wxPaintDC dc(this); + dc.SetPen(wxNullPen); + + int xcount = 8; + int ycount = 8; + + float xstep = 1. / xcount; + float ystep = 1. / ycount; + + int width = GetClientSize().GetWidth(); + int height = GetClientSize().GetHeight(); + + for ( float x = 0.; x < 1.; x += xstep ) + { + for ( float y = 0.; y < 1.; y += ystep ) + { + wxImage::RGBValue v = wxImage::HSVtoRGB(wxImage::HSVValue(x, 1., 1.)); + dc.SetBrush(wxBrush(wxColour(v.red, v.green, v.blue, + (int)(255*(1. - y))))); + int x1 = (int)(x * width); + int y1 = (int)(y * height); + int x2 = (int)((x + xstep) * width); + int y2 = (int)((y + ystep) * height); + dc.DrawRectangle(x1, y1, x2 - x1, y2 - y1); + } + } +} + +// Switches between colour and transparent background on doubleclick +void SeeThroughFrame::OnDoubleClick(wxMouseEvent& WXUNUSED(evt)) +{ + m_currentState = (State)((m_currentState + 1) % STATE_MAX); + + switch ( m_currentState ) + { + case STATE_OPAQUE: + SetBackgroundStyle(wxBG_STYLE_COLOUR); + SetTransparent(255); + SetTitle("Opaque"); + break; + + case STATE_SEETHROUGH: + SetBackgroundStyle(wxBG_STYLE_TRANSPARENT); + SetTransparent(255); + SetTitle("See through"); + break; + + case STATE_TRANSPARENT: + SetBackgroundStyle(wxBG_STYLE_COLOUR); + SetTransparent(128); + SetTitle("Semi-transparent"); + break; + + case STATE_MAX: + wxFAIL_MSG( "unreachable" ); + } + + Refresh(); +} +