1 /////////////////////////////////////////////////////////////////////////////
2 // Name: samples/drawing/drawing.cpp
3 // Purpose: shows and tests wxDC features
4 // Author: Robert Roebling
8 // Copyright: (c) Robert Roebling
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx/wx.h".
21 #include "wx/wxprec.h"
27 // for all others, include the necessary headers (this file is usually all you
28 // need because it includes almost all "standard" wxWidgets headers
33 #include "wx/colordlg.h"
35 #include "wx/artprov.h"
37 #define wxTEST_GRAPHICS 1
40 #include "wx/graphics.h"
41 #if wxUSE_GRAPHICS_CONTEXT == 0
42 #undef wxTEST_GRAPHICS
43 #define wxTEST_GRAPHICS 0
46 #undef wxUSE_GRAPHICS_CONTEXT
47 #define wxUSE_GRAPHICS_CONTEXT 0
50 // ----------------------------------------------------------------------------
52 // ----------------------------------------------------------------------------
54 // the application icon
55 #if defined(__WXGTK__) || defined(__WXMOTIF__) || defined(__WXMAC__) || defined(__WXMGL__) || defined(__WXX11__)
56 #include "mondrian.xpm"
59 // ----------------------------------------------------------------------------
61 // ----------------------------------------------------------------------------
63 // what do we show on screen (there are too many shapes to put them all on
64 // screen simultaneously)
77 #if wxUSE_GRAPHICS_CONTEXT
84 // ----------------------------------------------------------------------------
86 // ----------------------------------------------------------------------------
88 static wxBitmap
*gs_bmpNoMask
= NULL
,
89 *gs_bmpWithColMask
= NULL
,
91 *gs_bmpWithMask
= NULL
,
96 // ----------------------------------------------------------------------------
98 // ----------------------------------------------------------------------------
100 // Define a new application type, each program should derive a class from wxApp
101 class MyApp
: public wxApp
104 // override base class virtuals
105 // ----------------------------
107 // this one is called on application startup and is a good place for the app
108 // initialization (doing it here and not in the ctor allows to have an error
109 // return: if OnInit() returns false, the application terminates)
110 virtual bool OnInit();
112 virtual int OnExit() { DeleteBitmaps(); return 0; }
115 void DeleteBitmaps();
122 // Define a new frame type: this is going to be our main frame
123 class MyFrame
: public wxFrame
127 MyFrame(const wxString
& title
, const wxPoint
& pos
, const wxSize
& size
);
129 // event handlers (these functions should _not_ be virtual)
130 void OnQuit(wxCommandEvent
& event
);
131 void OnAbout(wxCommandEvent
& event
);
132 void OnClip(wxCommandEvent
& event
);
133 #if wxUSE_GRAPHICS_CONTEXT
134 void OnGraphicContext(wxCommandEvent
& event
);
136 void OnShow(wxCommandEvent
&event
);
137 void OnOption(wxCommandEvent
&event
);
140 wxColour
SelectColour();
141 #endif // wxUSE_COLOURDLG
142 void PrepareDC(wxDC
& dc
);
144 int m_backgroundMode
;
145 int m_textureBackground
;
149 int m_xLogicalOrigin
;
150 int m_yLogicalOrigin
;
151 bool m_xAxisReversed
,
153 wxColour m_colourForeground
, // these are _text_ colours
155 wxBrush m_backgroundBrush
;
159 // any class wishing to process wxWidgets events must use this macro
160 DECLARE_EVENT_TABLE()
163 // define a scrollable canvas for drawing onto
164 class MyCanvas
: public wxScrolledWindow
167 MyCanvas( MyFrame
*parent
);
169 void OnPaint(wxPaintEvent
&event
);
170 void OnMouseMove(wxMouseEvent
&event
);
172 void ToShow(ScreenToShow show
) { m_show
= show
; Refresh(); }
174 // set or remove the clipping region
175 void Clip(bool clip
) { m_clip
= clip
; Refresh(); }
176 #if wxUSE_GRAPHICS_CONTEXT
177 void UseGraphicContext(bool use
) { m_useContext
= use
; Refresh(); }
181 void DrawTestLines( int x
, int y
, int width
, wxDC
&dc
);
182 void DrawTestPoly(wxDC
& dc
);
183 void DrawTestBrushes(wxDC
& dc
);
184 void DrawText(wxDC
& dc
);
185 void DrawImages(wxDC
& dc
);
186 void DrawWithLogicalOps(wxDC
& dc
);
187 #if wxUSE_GRAPHICS_CONTEXT
188 void DrawAlpha(wxDC
& dc
);
190 void DrawRegions(wxDC
& dc
);
191 void DrawCircles(wxDC
& dc
);
192 void DrawSplines(wxDC
& dc
);
193 void DrawDefault(wxDC
& dc
);
194 void DrawGradients(wxDC
& dc
);
196 void DrawRegionsHelper(wxDC
& dc
, wxCoord x
, bool firstTime
);
202 wxBitmap m_smile_bmp
;
205 #if wxUSE_GRAPHICS_CONTEXT
209 DECLARE_EVENT_TABLE()
212 // ----------------------------------------------------------------------------
214 // ----------------------------------------------------------------------------
216 // IDs for the controls and the menu commands
220 File_Quit
= wxID_EXIT
,
221 File_About
= wxID_ABOUT
,
223 MenuShow_First
= wxID_HIGHEST
,
224 File_ShowDefault
= MenuShow_First
,
234 #if wxUSE_GRAPHICS_CONTEXT
238 MenuShow_Last
= File_ShowGradients
,
241 #if wxUSE_GRAPHICS_CONTEXT
247 MapMode_Text
= MenuOption_First
,
253 UserScale_StretchHoriz
,
254 UserScale_ShrinkHoriz
,
255 UserScale_StretchVertic
,
256 UserScale_ShrinkVertic
,
262 LogicalOrigin_MoveDown
,
263 LogicalOrigin_MoveUp
,
264 LogicalOrigin_MoveLeft
,
265 LogicalOrigin_MoveRight
,
267 LogicalOrigin_Restore
,
270 Colour_TextForeground
,
271 Colour_TextBackground
,
273 #endif // wxUSE_COLOURDLG
274 Colour_BackgroundMode
,
275 Colour_TextureBackgound
,
277 MenuOption_Last
= Colour_TextureBackgound
280 // ----------------------------------------------------------------------------
281 // event tables and other macros for wxWidgets
282 // ----------------------------------------------------------------------------
285 // Create a new application object: this macro will allow wxWidgets to create
286 // the application object during program execution (it's better than using a
287 // static object for many reasons) and also declares the accessor function
288 // wxGetApp() which will return the reference of the right type (i.e. MyApp and
292 // ============================================================================
294 // ============================================================================
296 // ----------------------------------------------------------------------------
297 // the application class
298 // ----------------------------------------------------------------------------
300 bool MyApp::LoadImages()
302 gs_bmpNoMask
= new wxBitmap
;
303 gs_bmpWithColMask
= new wxBitmap
;
304 gs_bmpMask
= new wxBitmap
;
305 gs_bmpWithMask
= new wxBitmap
;
306 gs_bmp4
= new wxBitmap
;
307 gs_bmp4_mono
= new wxBitmap
;
308 gs_bmp36
= new wxBitmap
;
311 pathList
.Add(_T("."));
312 pathList
.Add(_T(".."));
314 wxString path
= pathList
.FindValidPath(_T("pat4.bmp"));
318 /* 4 colour bitmap */
319 gs_bmp4
->LoadFile(path
, wxBITMAP_TYPE_BMP
);
320 /* turn into mono-bitmap */
321 gs_bmp4_mono
->LoadFile(path
, wxBITMAP_TYPE_BMP
);
322 wxMask
* mask4
= new wxMask(*gs_bmp4_mono
, *wxBLACK
);
323 gs_bmp4_mono
->SetMask(mask4
);
325 path
= pathList
.FindValidPath(_T("pat36.bmp"));
328 gs_bmp36
->LoadFile(path
, wxBITMAP_TYPE_BMP
);
329 wxMask
* mask36
= new wxMask(*gs_bmp36
, *wxBLACK
);
330 gs_bmp36
->SetMask(mask36
);
332 path
= pathList
.FindValidPath(_T("image.bmp"));
335 gs_bmpNoMask
->LoadFile(path
, wxBITMAP_TYPE_BMP
);
336 gs_bmpWithMask
->LoadFile(path
, wxBITMAP_TYPE_BMP
);
337 gs_bmpWithColMask
->LoadFile(path
, wxBITMAP_TYPE_BMP
);
339 path
= pathList
.FindValidPath(_T("mask.bmp"));
342 gs_bmpMask
->LoadFile(path
, wxBITMAP_TYPE_BMP
);
344 wxMask
*mask
= new wxMask(*gs_bmpMask
, *wxBLACK
);
345 gs_bmpWithMask
->SetMask(mask
);
347 mask
= new wxMask(*gs_bmpWithColMask
, *wxWHITE
);
348 gs_bmpWithColMask
->SetMask(mask
);
353 // `Main program' equivalent: the program execution "starts" here
356 // Create the main application window
357 MyFrame
*frame
= new MyFrame(_T("Drawing sample"),
358 wxPoint(50, 50), wxSize(550, 340));
360 // Show it and tell the application that it's our main window
366 wxLogError(wxT("Can't load one of the bitmap files needed ")
367 wxT("for this sample from the current or parent ")
368 wxT("directory, please copy them there."));
380 void MyApp::DeleteBitmaps()
383 delete gs_bmpWithColMask
;
385 delete gs_bmpWithMask
;
391 gs_bmpWithColMask
= NULL
;
393 gs_bmpWithMask
= NULL
;
399 // ----------------------------------------------------------------------------
401 // ----------------------------------------------------------------------------
403 // the event tables connect the wxWidgets events with the functions (event
404 // handlers) which process them.
405 BEGIN_EVENT_TABLE(MyCanvas
, wxScrolledWindow
)
406 EVT_PAINT (MyCanvas::OnPaint
)
407 EVT_MOTION (MyCanvas::OnMouseMove
)
412 MyCanvas::MyCanvas(MyFrame
*parent
)
413 : wxScrolledWindow(parent
, wxID_ANY
, wxDefaultPosition
, wxDefaultSize
,
414 wxHSCROLL
| wxVSCROLL
| wxNO_FULL_REPAINT_ON_RESIZE
)
417 m_show
= Show_Default
;
418 m_smile_bmp
= wxBitmap(smile_xpm
);
419 m_std_icon
= wxArtProvider::GetIcon(wxART_INFORMATION
);
421 #if wxUSE_GRAPHICS_CONTEXT
422 m_useContext
= false;
426 void MyCanvas::DrawTestBrushes(wxDC
& dc
)
428 static const wxCoord WIDTH
= 200;
429 static const wxCoord HEIGHT
= 80;
434 dc
.SetBrush(wxBrush(*wxGREEN
, wxSOLID
));
435 dc
.DrawRectangle(x
, y
, WIDTH
, HEIGHT
);
436 dc
.DrawText(_T("Solid green"), x
+ 10, y
+ 10);
439 dc
.SetBrush(wxBrush(*wxRED
, wxCROSSDIAG_HATCH
));
440 dc
.DrawRectangle(x
, y
, WIDTH
, HEIGHT
);
441 dc
.DrawText(_T("Hatched red"), x
+ 10, y
+ 10);
444 dc
.SetBrush(wxBrush(*gs_bmpMask
));
445 dc
.DrawRectangle(x
, y
, WIDTH
, HEIGHT
);
446 dc
.DrawText(_T("Stipple mono"), x
+ 10, y
+ 10);
449 dc
.SetBrush(wxBrush(*gs_bmpNoMask
));
450 dc
.DrawRectangle(x
, y
, WIDTH
, HEIGHT
);
451 dc
.DrawText(_T("Stipple colour"), x
+ 10, y
+ 10);
454 void MyCanvas::DrawTestPoly(wxDC
& dc
)
456 wxBrush
brushHatch(*wxRED
, wxFDIAGONAL_HATCH
);
457 dc
.SetBrush(brushHatch
);
460 star
[0] = wxPoint(100, 60);
461 star
[1] = wxPoint(60, 150);
462 star
[2] = wxPoint(160, 100);
463 star
[3] = wxPoint(40, 100);
464 star
[4] = wxPoint(140, 150);
466 dc
.DrawText(_T("You should see two (irregular) stars below, the left one ")
467 _T("hatched"), 10, 10);
468 dc
.DrawText(_T("except for the central region and the right ")
469 _T("one entirely hatched"), 10, 30);
470 dc
.DrawText(_T("The third star only has a hatched outline"), 10, 50);
472 dc
.DrawPolygon(WXSIZEOF(star
), star
, 0, 30);
473 dc
.DrawPolygon(WXSIZEOF(star
), star
, 160, 30, wxWINDING_RULE
);
476 star2
[0] = wxPoint(0, 100);
477 star2
[1] = wxPoint(-59, -81);
478 star2
[2] = wxPoint(95, 31);
479 star2
[3] = wxPoint(-95, 31);
480 star2
[4] = wxPoint(59, -81);
481 star2
[5] = wxPoint(0, 80);
482 star2
[6] = wxPoint(-47, -64);
483 star2
[7] = wxPoint(76, 24);
484 star2
[8] = wxPoint(-76, 24);
485 star2
[9] = wxPoint(47, -64);
486 int count
[2] = {5, 5};
488 dc
.DrawPolyPolygon(WXSIZEOF(count
), count
, star2
, 450, 150);
491 void MyCanvas::DrawTestLines( int x
, int y
, int width
, wxDC
&dc
)
493 dc
.SetPen( wxPen( wxT("black"), width
, wxSOLID
) );
494 dc
.SetBrush( *wxRED_BRUSH
);
495 dc
.DrawText(wxString::Format(wxT("Testing lines of width %d"), width
), x
+ 10, y
- 10);
496 dc
.DrawRectangle( x
+10, y
+10, 100, 190 );
498 dc
.DrawText(_T("Solid/dot/short dash/long dash/dot dash"), x
+ 150, y
+ 10);
499 dc
.SetPen( wxPen( wxT("black"), width
, wxSOLID
) );
500 dc
.DrawLine( x
+20, y
+20, 100, y
+20 );
501 dc
.SetPen( wxPen( wxT("black"), width
, wxDOT
) );
502 dc
.DrawLine( x
+20, y
+30, 100, y
+30 );
503 dc
.SetPen( wxPen( wxT("black"), width
, wxSHORT_DASH
) );
504 dc
.DrawLine( x
+20, y
+40, 100, y
+40 );
505 dc
.SetPen( wxPen( wxT("black"), width
, wxLONG_DASH
) );
506 dc
.DrawLine( x
+20, y
+50, 100, y
+50 );
507 dc
.SetPen( wxPen( wxT("black"), width
, wxDOT_DASH
) );
508 dc
.DrawLine( x
+20, y
+60, 100, y
+60 );
510 dc
.DrawText(_T("Misc hatches"), x
+ 150, y
+ 70);
511 dc
.SetPen( wxPen( wxT("black"), width
, wxBDIAGONAL_HATCH
) );
512 dc
.DrawLine( x
+20, y
+70, 100, y
+70 );
513 dc
.SetPen( wxPen( wxT("black"), width
, wxCROSSDIAG_HATCH
) );
514 dc
.DrawLine( x
+20, y
+80, 100, y
+80 );
515 dc
.SetPen( wxPen( wxT("black"), width
, wxFDIAGONAL_HATCH
) );
516 dc
.DrawLine( x
+20, y
+90, 100, y
+90 );
517 dc
.SetPen( wxPen( wxT("black"), width
, wxCROSS_HATCH
) );
518 dc
.DrawLine( x
+20, y
+100, 100, y
+100 );
519 dc
.SetPen( wxPen( wxT("black"), width
, wxHORIZONTAL_HATCH
) );
520 dc
.DrawLine( x
+20, y
+110, 100, y
+110 );
521 dc
.SetPen( wxPen( wxT("black"), width
, wxVERTICAL_HATCH
) );
522 dc
.DrawLine( x
+20, y
+120, 100, y
+120 );
524 dc
.DrawText(_T("User dash"), x
+ 150, y
+ 140);
525 wxPen
ud( wxT("black"), width
, wxUSER_DASH
);
527 dash1
[0] = 8; // Long dash <---------+
528 dash1
[1] = 2; // Short gap |
529 dash1
[2] = 3; // Short dash |
530 dash1
[3] = 2; // Short gap |
531 dash1
[4] = 3; // Short dash |
532 dash1
[5] = 2; // Short gap and repeat +
533 ud
.SetDashes( 6, dash1
);
535 dc
.DrawLine( x
+20, y
+140, 100, y
+140 );
536 dash1
[0] = 5; // Make first dash shorter
537 ud
.SetDashes( 6, dash1
);
539 dc
.DrawLine( x
+20, y
+150, 100, y
+150 );
540 dash1
[2] = 5; // Make second dash longer
541 ud
.SetDashes( 6, dash1
);
543 dc
.DrawLine( x
+20, y
+160, 100, y
+160 );
544 dash1
[4] = 5; // Make third dash longer
545 ud
.SetDashes( 6, dash1
);
547 dc
.DrawLine( x
+20, y
+170, 100, y
+170 );
550 void MyCanvas::DrawDefault(wxDC
& dc
)
553 dc
.DrawCircle(0, 0, 10);
555 #if !defined(wxMAC_USE_CORE_GRAPHICS) || !wxMAC_USE_CORE_GRAPHICS
556 // GetPixel and FloodFill not supported by Mac OS X CoreGraphics
557 // (FloodFill uses Blit from a non-wxMemoryDC)
558 //flood fill using brush, starting at 1,1 and replacing whatever colour we find there
559 dc
.SetBrush(wxBrush(wxColour(128,128,0), wxSOLID
));
562 dc
.GetPixel(1,1, &tmpColour
);
563 dc
.FloodFill(1,1, tmpColour
, wxFLOOD_SURFACE
);
566 dc
.DrawCheckMark(5, 80, 15, 15);
567 dc
.DrawCheckMark(25, 80, 30, 30);
568 dc
.DrawCheckMark(60, 80, 60, 60);
570 // this is the test for "blitting bitmap into DC damages selected brush" bug
571 wxCoord rectSize
= m_std_icon
.GetWidth() + 10;
573 dc
.SetPen(*wxTRANSPARENT_PEN
);
574 dc
.SetBrush( *wxGREEN_BRUSH
);
575 dc
.DrawRectangle(x
, 10, rectSize
, rectSize
);
576 dc
.DrawBitmap(m_std_icon
, x
+ 5, 15, true);
578 dc
.DrawRectangle(x
, 10, rectSize
, rectSize
);
579 dc
.DrawIcon(m_std_icon
, x
+ 5, 15);
581 dc
.DrawRectangle(x
, 10, rectSize
, rectSize
);
583 // test for "transparent" bitmap drawing (it intersects with the last
585 //dc.SetBrush( *wxTRANSPARENT_BRUSH );
587 if (m_smile_bmp
.Ok())
588 dc
.DrawBitmap(m_smile_bmp
, x
+ rectSize
- 20, rectSize
- 10, true);
590 dc
.SetBrush( *wxBLACK_BRUSH
);
591 dc
.DrawRectangle( 0, 160, 1000, 300 );
594 wxBitmap
bitmap(20,70);
596 memdc
.SelectObject( bitmap
);
597 memdc
.SetBrush( *wxBLACK_BRUSH
);
598 memdc
.SetPen( *wxWHITE_PEN
);
599 memdc
.DrawRectangle(0,0,20,70);
600 memdc
.DrawLine( 10,0,10,70 );
603 wxPen pen
= *wxRED_PEN
;
605 memdc
.DrawLine( 10, 5,10, 5 );
606 memdc
.DrawLine( 10,10,11,10 );
607 memdc
.DrawLine( 10,15,12,15 );
608 memdc
.DrawLine( 10,20,13,20 );
611 memdc.SetPen(*wxRED_PEN);
612 memdc.DrawLine( 12, 5,12, 5 );
613 memdc.DrawLine( 12,10,13,10 );
614 memdc.DrawLine( 12,15,14,15 );
615 memdc.DrawLine( 12,20,15,20 );
619 memdc
.DrawLine( 10,25,10,25 );
620 memdc
.DrawLine( 10,30, 9,30 );
621 memdc
.DrawLine( 10,35, 8,35 );
622 memdc
.DrawLine( 10,40, 7,40 );
625 dc
.SetPen(*wxWHITE_PEN
);
626 memdc
.SetLogicalFunction( wxINVERT
);
627 memdc
.SetPen( *wxWHITE_PEN
);
628 memdc
.DrawLine( 10,50,10,50 );
629 memdc
.DrawLine( 10,55,11,55 );
630 memdc
.DrawLine( 10,60,12,60 );
631 memdc
.DrawLine( 10,65,13,65 );
633 memdc
.DrawLine( 12,50,12,50 );
634 memdc
.DrawLine( 12,55,13,55 );
635 memdc
.DrawLine( 12,60,14,60 );
636 memdc
.DrawLine( 12,65,15,65 );
638 memdc
.SelectObject( wxNullBitmap
);
639 dc
.DrawBitmap( bitmap
, 10, 170 );
640 wxImage image
= bitmap
.ConvertToImage();
641 image
.Rescale( 60,210 );
642 bitmap
= wxBitmap(image
);
643 dc
.DrawBitmap( bitmap
, 50, 170 );
645 // test the rectangle outline drawing - there should be one pixel between
646 // the rect and the lines
647 dc
.SetPen(*wxWHITE_PEN
);
648 dc
.SetBrush( *wxTRANSPARENT_BRUSH
);
649 dc
.DrawRectangle(150, 170, 49, 29);
650 dc
.DrawRectangle(200, 170, 49, 29);
651 dc
.SetPen(*wxWHITE_PEN
);
652 dc
.DrawLine(250, 210, 250, 170);
653 dc
.DrawLine(260, 200, 150, 200);
655 // test the rectangle filled drawing - there should be one pixel between
656 // the rect and the lines
657 dc
.SetPen(*wxTRANSPARENT_PEN
);
658 dc
.SetBrush( *wxWHITE_BRUSH
);
659 dc
.DrawRectangle(300, 170, 49, 29);
660 dc
.DrawRectangle(350, 170, 49, 29);
661 dc
.SetPen(*wxWHITE_PEN
);
662 dc
.DrawLine(400, 170, 400, 210);
663 dc
.DrawLine(300, 200, 410, 200);
665 // a few more tests of this kind
666 dc
.SetPen(*wxRED_PEN
);
667 dc
.SetBrush( *wxWHITE_BRUSH
);
668 dc
.DrawRectangle(300, 220, 1, 1);
669 dc
.DrawRectangle(310, 220, 2, 2);
670 dc
.DrawRectangle(320, 220, 3, 3);
671 dc
.DrawRectangle(330, 220, 4, 4);
673 dc
.SetPen(*wxTRANSPARENT_PEN
);
674 dc
.SetBrush( *wxWHITE_BRUSH
);
675 dc
.DrawRectangle(300, 230, 1, 1);
676 dc
.DrawRectangle(310, 230, 2, 2);
677 dc
.DrawRectangle(320, 230, 3, 3);
678 dc
.DrawRectangle(330, 230, 4, 4);
680 // and now for filled rect with outline
681 dc
.SetPen(*wxRED_PEN
);
682 dc
.SetBrush( *wxWHITE_BRUSH
);
683 dc
.DrawRectangle(500, 170, 49, 29);
684 dc
.DrawRectangle(550, 170, 49, 29);
685 dc
.SetPen(*wxWHITE_PEN
);
686 dc
.DrawLine(600, 170, 600, 210);
687 dc
.DrawLine(500, 200, 610, 200);
689 // test the rectangle outline drawing - there should be one pixel between
690 // the rect and the lines
691 dc
.SetPen(*wxWHITE_PEN
);
692 dc
.SetBrush( *wxTRANSPARENT_BRUSH
);
693 dc
.DrawRoundedRectangle(150, 270, 49, 29, 6);
694 dc
.DrawRoundedRectangle(200, 270, 49, 29, 6);
695 dc
.SetPen(*wxWHITE_PEN
);
696 dc
.DrawLine(250, 270, 250, 310);
697 dc
.DrawLine(150, 300, 260, 300);
699 // test the rectangle filled drawing - there should be one pixel between
700 // the rect and the lines
701 dc
.SetPen(*wxTRANSPARENT_PEN
);
702 dc
.SetBrush( *wxWHITE_BRUSH
);
703 dc
.DrawRoundedRectangle(300, 270, 49, 29, 6);
704 dc
.DrawRoundedRectangle(350, 270, 49, 29, 6);
705 dc
.SetPen(*wxWHITE_PEN
);
706 dc
.DrawLine(400, 270, 400, 310);
707 dc
.DrawLine(300, 300, 410, 300);
709 // Added by JACS to demonstrate bizarre behaviour.
710 // With a size of 70, we get a missing red RHS,
711 // and the height is too small, so we get yellow
712 // showing. With a size of 40, it draws as expected:
713 // it just shows a white rectangle with red outline.
715 int totalHeight
= 70;
716 wxBitmap
bitmap2(totalWidth
, totalHeight
);
719 memdc2
.SelectObject(bitmap2
);
721 wxColour
clr(255, 255, 0);
722 wxBrush
yellowBrush(clr
, wxSOLID
);
723 memdc2
.SetBackground(yellowBrush
);
726 wxPen
yellowPen(clr
, 1, wxSOLID
);
728 // Now draw a white rectangle with red outline. It should
729 // entirely eclipse the yellow background.
730 memdc2
.SetPen(*wxRED_PEN
);
731 memdc2
.SetBrush(*wxWHITE_BRUSH
);
733 memdc2
.DrawRectangle(0, 0, totalWidth
, totalHeight
);
735 memdc2
.SetPen(wxNullPen
);
736 memdc2
.SetBrush(wxNullBrush
);
737 memdc2
.SelectObject(wxNullBitmap
);
739 dc
.DrawBitmap(bitmap2
, 500, 270);
741 // Repeat, but draw directly on dc
742 // Draw a yellow rectangle filling the bitmap
744 x
= 600; int y
= 270;
745 dc
.SetPen(yellowPen
);
746 dc
.SetBrush(yellowBrush
);
747 dc
.DrawRectangle(x
, y
, totalWidth
, totalHeight
);
749 // Now draw a white rectangle with red outline. It should
750 // entirely eclipse the yellow background.
751 dc
.SetPen(*wxRED_PEN
);
752 dc
.SetBrush(*wxWHITE_BRUSH
);
754 dc
.DrawRectangle(x
, y
, totalWidth
, totalHeight
);
757 void MyCanvas::DrawText(wxDC
& dc
)
759 // set underlined font for testing
760 dc
.SetFont( wxFont(12, wxMODERN
, wxNORMAL
, wxNORMAL
, true) );
761 dc
.DrawText( _T("This is text"), 110, 10 );
762 dc
.DrawRotatedText( _T("That is text"), 20, 10, -45 );
764 // use wxSWISS_FONT and not wxNORMAL_FONT as the latter can't be rotated
765 // under Win9x (it is not TrueType)
766 dc
.SetFont( *wxSWISS_FONT
);
769 dc
.SetBackgroundMode(wxTRANSPARENT
);
771 for ( int n
= -180; n
< 180; n
+= 30 )
773 text
.Printf(wxT(" %d rotated text"), n
);
774 dc
.DrawRotatedText(text
, 400, 400, n
);
777 dc
.SetFont( wxFont( 18, wxSWISS
, wxNORMAL
, wxNORMAL
) );
779 dc
.DrawText( _T("This is Swiss 18pt text."), 110, 40 );
784 dc
.GetTextExtent( _T("This is Swiss 18pt text."), &length
, &height
, &descent
);
785 text
.Printf( wxT("Dimensions are length %ld, height %ld, descent %ld"), length
, height
, descent
);
786 dc
.DrawText( text
, 110, 80 );
788 text
.Printf( wxT("CharHeight() returns: %d"), dc
.GetCharHeight() );
789 dc
.DrawText( text
, 110, 120 );
791 dc
.DrawRectangle( 100, 40, 4, height
);
793 // test the logical function effect
795 dc
.SetLogicalFunction(wxINVERT
);
796 dc
.DrawText( _T("There should be no text below"), 110, 150 );
797 dc
.DrawRectangle( 110, y
, 100, height
);
799 // twice drawn inverted should result in invisible
801 dc
.DrawText( _T("Invisible text"), 110, y
);
802 dc
.DrawRectangle( 110, y
, 100, height
);
803 dc
.DrawText( _T("Invisible text"), 110, y
);
804 dc
.DrawRectangle( 110, y
, 100, height
);
805 dc
.SetLogicalFunction(wxCOPY
);
808 dc
.DrawRectangle( 110, y
, 100, height
);
809 dc
.DrawText( _T("Visible text"), 110, y
);
816 } rasterOperations
[] =
818 { wxT("wxAND"), wxAND
},
819 { wxT("wxAND_INVERT"), wxAND_INVERT
},
820 { wxT("wxAND_REVERSE"), wxAND_REVERSE
},
821 { wxT("wxCLEAR"), wxCLEAR
},
822 { wxT("wxCOPY"), wxCOPY
},
823 { wxT("wxEQUIV"), wxEQUIV
},
824 { wxT("wxINVERT"), wxINVERT
},
825 { wxT("wxNAND"), wxNAND
},
826 { wxT("wxNO_OP"), wxNO_OP
},
827 { wxT("wxOR"), wxOR
},
828 { wxT("wxOR_INVERT"), wxOR_INVERT
},
829 { wxT("wxOR_REVERSE"), wxOR_REVERSE
},
830 { wxT("wxSET"), wxSET
},
831 { wxT("wxSRC_INVERT"), wxSRC_INVERT
},
832 { wxT("wxXOR"), wxXOR
},
835 void MyCanvas::DrawImages(wxDC
& dc
)
837 dc
.DrawText(_T("original image"), 0, 0);
838 dc
.DrawBitmap(*gs_bmpNoMask
, 0, 20, 0);
839 dc
.DrawText(_T("with colour mask"), 0, 100);
840 dc
.DrawBitmap(*gs_bmpWithColMask
, 0, 120, true);
841 dc
.DrawText(_T("the mask image"), 0, 200);
842 dc
.DrawBitmap(*gs_bmpMask
, 0, 220, 0);
843 dc
.DrawText(_T("masked image"), 0, 300);
844 dc
.DrawBitmap(*gs_bmpWithMask
, 0, 320, true);
846 int cx
= gs_bmpWithColMask
->GetWidth(),
847 cy
= gs_bmpWithColMask
->GetHeight();
850 for ( size_t n
= 0; n
< WXSIZEOF(rasterOperations
); n
++ )
852 wxCoord x
= 120 + 150*(n%4
),
855 dc
.DrawText(rasterOperations
[n
].name
, x
, y
- 20);
856 memDC
.SelectObject(*gs_bmpWithColMask
);
857 dc
.Blit(x
, y
, cx
, cy
, &memDC
, 0, 0, rasterOperations
[n
].rop
, true);
861 void MyCanvas::DrawWithLogicalOps(wxDC
& dc
)
863 static const wxCoord w
= 60;
864 static const wxCoord h
= 60;
866 // reuse the text colour here
867 dc
.SetPen(wxPen(m_owner
->m_colourForeground
, 1, wxSOLID
));
868 dc
.SetBrush(*wxTRANSPARENT_BRUSH
);
871 for ( n
= 0; n
< WXSIZEOF(rasterOperations
); n
++ )
873 wxCoord x
= 20 + 150*(n%4
),
876 dc
.DrawText(rasterOperations
[n
].name
, x
, y
- 20);
877 dc
.SetLogicalFunction(rasterOperations
[n
].rop
);
878 dc
.DrawRectangle(x
, y
, w
, h
);
879 dc
.DrawLine(x
, y
, x
+ w
, y
+ h
);
880 dc
.DrawLine(x
+ w
, y
, x
, y
+ h
);
883 // now some filled rectangles
884 dc
.SetBrush(wxBrush(m_owner
->m_colourForeground
, wxSOLID
));
886 for ( n
= 0; n
< WXSIZEOF(rasterOperations
); n
++ )
888 wxCoord x
= 20 + 150*(n%4
),
891 dc
.DrawText(rasterOperations
[n
].name
, x
, y
- 20);
892 dc
.SetLogicalFunction(rasterOperations
[n
].rop
);
893 dc
.DrawRectangle(x
, y
, w
, h
);
897 #if wxUSE_GRAPHICS_CONTEXT
899 void MyCanvas::DrawAlpha(wxDC
& no_dc
)
901 void MyCanvas::DrawAlpha(wxDC
& dc
)
909 wxDouble margin
= 20 ;
910 wxDouble width
= 180 ;
911 wxDouble radius
= 30 ;
913 dc
.SetPen( wxPen( wxColour( 128, 0, 0, 255 ),12, wxSOLID
));
914 dc
.SetBrush( wxBrush( wxColour( 255, 0, 0, 255),wxSOLID
));
916 wxRect
r(margin
,margin
+width
*0.66,width
,width
) ;
918 dc
.DrawRoundedRectangle( r
.x
, r
.y
, r
.width
, r
.width
, radius
) ;
920 dc
.SetPen( wxPen( wxColour( 0, 0, 128, 255 ),12, wxSOLID
));
921 dc
.SetBrush( wxBrush( wxColour( 0, 0, 255, 255),wxSOLID
));
923 r
.Offset( width
* 0.8 , - width
* 0.66 ) ;
925 dc
.DrawRoundedRectangle( r
.x
, r
.y
, r
.width
, r
.width
, radius
) ;
927 dc
.SetPen( wxPen( wxColour( 128, 128, 0, 255 ),12, wxSOLID
));
928 dc
.SetBrush( wxBrush( wxColour( 192, 192, 0, 255),wxSOLID
));
930 r
.Offset( width
* 0.8 , width
*0.5 ) ;
932 dc
.DrawRoundedRectangle( r
.x
, r
.y
, r
.width
, r
.width
, radius
) ;
934 dc
.SetPen( *wxTRANSPARENT_PEN
) ;
935 dc
.SetBrush( wxBrush( wxColour(255,255,128,128) ) );
936 dc
.DrawRoundedRectangle( 0 , margin
+ width
/ 2 , width
* 3 , 100 , radius
) ;
938 dc
.SetTextForeground( wxColour(255,255,0,128) );
939 dc
.SetFont( wxFont( 40, wxFONTFAMILY_SWISS
, wxFONTSTYLE_ITALIC
, wxFONTWEIGHT_NORMAL
) );
940 dc
.DrawText( wxT("Hello!"), 120, 80 );
945 void MyCanvas::DrawCircles(wxDC
& dc
)
951 dc
.SetPen( *wxRED_PEN
);
952 dc
.SetBrush( *wxGREEN_BRUSH
);
954 dc
.DrawText(_T("Some circles"), 0, y
);
955 dc
.DrawCircle(x
, y
, r
);
956 dc
.DrawCircle(x
+ 2*r
, y
, r
);
957 dc
.DrawCircle(x
+ 4*r
, y
, r
);
960 dc
.DrawText(_T("And ellipses"), 0, y
);
961 dc
.DrawEllipse(x
- r
, y
, 2*r
, r
);
962 dc
.DrawEllipse(x
+ r
, y
, 2*r
, r
);
963 dc
.DrawEllipse(x
+ 3*r
, y
, 2*r
, r
);
966 dc
.DrawText(_T("And arcs"), 0, y
);
967 dc
.DrawArc(x
- r
, y
, x
+ r
, y
, x
, y
);
968 dc
.DrawArc(x
+ 4*r
, y
, x
+ 2*r
, y
, x
+ 3*r
, y
);
969 dc
.DrawArc(x
+ 5*r
, y
, x
+ 5*r
, y
, x
+ 6*r
, y
);
972 dc
.DrawEllipticArc(x
- r
, y
, 2*r
, r
, 0, 90);
973 dc
.DrawEllipticArc(x
+ r
, y
, 2*r
, r
, 90, 180);
974 dc
.DrawEllipticArc(x
+ 3*r
, y
, 2*r
, r
, 180, 270);
975 dc
.DrawEllipticArc(x
+ 5*r
, y
, 2*r
, r
, 270, 360);
977 // same as above, just transparent brush
979 dc
.SetPen( *wxRED_PEN
);
980 dc
.SetBrush( *wxTRANSPARENT_BRUSH
);
983 dc
.DrawText(_T("Some circles"), 0, y
);
984 dc
.DrawCircle(x
, y
, r
);
985 dc
.DrawCircle(x
+ 2*r
, y
, r
);
986 dc
.DrawCircle(x
+ 4*r
, y
, r
);
989 dc
.DrawText(_T("And ellipses"), 0, y
);
990 dc
.DrawEllipse(x
- r
, y
, 2*r
, r
);
991 dc
.DrawEllipse(x
+ r
, y
, 2*r
, r
);
992 dc
.DrawEllipse(x
+ 3*r
, y
, 2*r
, r
);
995 dc
.DrawText(_T("And arcs"), 0, y
);
996 dc
.DrawArc(x
- r
, y
, x
+ r
, y
, x
, y
);
997 dc
.DrawArc(x
+ 4*r
, y
, x
+ 2*r
, y
, x
+ 3*r
, y
);
998 dc
.DrawArc(x
+ 5*r
, y
, x
+ 5*r
, y
, x
+ 6*r
, y
);
1001 dc
.DrawEllipticArc(x
- r
, y
, 2*r
, r
, 0, 90);
1002 dc
.DrawEllipticArc(x
+ r
, y
, 2*r
, r
, 90, 180);
1003 dc
.DrawEllipticArc(x
+ 3*r
, y
, 2*r
, r
, 180, 270);
1004 dc
.DrawEllipticArc(x
+ 5*r
, y
, 2*r
, r
, 270, 360);
1008 void MyCanvas::DrawSplines(wxDC
& dc
)
1011 dc
.DrawText(_T("Some splines"), 10, 5);
1013 // values are hardcoded rather than randomly generated
1014 // so the output can be compared between native
1015 // implementations on platforms with different random
1019 const wxPoint
center( R
+ 20, R
+ 20 );
1020 const int angles
[7] = { 0, 10, 33, 77, 13, 145, 90 };
1021 const int radii
[5] = { 100 , 59, 85, 33, 90 };
1025 // background spline calculation
1026 unsigned int radius_pos
= 0;
1027 unsigned int angle_pos
= 0;
1029 for ( int i
= 0; i
< n
; i
++ )
1031 angle
+= angles
[ angle_pos
];
1032 int r
= R
* radii
[ radius_pos
] / 100;
1033 pts
[ i
].x
= center
.x
+ (wxCoord
)( r
* cos( M_PI
* angle
/ 180.0) );
1034 pts
[ i
].y
= center
.y
+ (wxCoord
)( r
* sin( M_PI
* angle
/ 180.0) );
1037 if ( angle_pos
>= WXSIZEOF(angles
) ) angle_pos
= 0;
1040 if ( radius_pos
>= WXSIZEOF(radii
) ) radius_pos
= 0;
1043 // background spline drawing
1044 dc
.SetPen(*wxRED_PEN
);
1045 dc
.DrawSpline(WXSIZEOF(pts
), pts
);
1047 // less detailed spline calculation
1048 wxPoint letters
[4][5];
1050 letters
[0][0] = wxPoint( 0,1); // O O
1051 letters
[0][1] = wxPoint( 1,3); // * *
1052 letters
[0][2] = wxPoint( 2,2); // * O *
1053 letters
[0][3] = wxPoint( 3,3); // * * * *
1054 letters
[0][4] = wxPoint( 4,1); // O O
1056 letters
[1][0] = wxPoint( 5,1); // O*O
1057 letters
[1][1] = wxPoint( 6,1); // *
1058 letters
[1][2] = wxPoint( 7,2); // O
1059 letters
[1][3] = wxPoint( 8,3); // *
1060 letters
[1][4] = wxPoint( 9,3); // O*O
1062 letters
[2][0] = wxPoint( 5,3); // O*O
1063 letters
[2][1] = wxPoint( 6,3); // *
1064 letters
[2][2] = wxPoint( 7,2); // O
1065 letters
[2][3] = wxPoint( 8,1); // *
1066 letters
[2][4] = wxPoint( 9,1); // O*O
1068 letters
[3][0] = wxPoint(10,0); // O O
1069 letters
[3][1] = wxPoint(11,3); // * *
1070 letters
[3][2] = wxPoint(12,1); // * O *
1071 letters
[3][3] = wxPoint(13,3); // * * * *
1072 letters
[3][4] = wxPoint(14,0); // O O
1074 const int dx
= 2 * R
/ letters
[3][4].x
;
1075 const int h
[4] = { -R
/2, 0, R
/4, R
/2 };
1077 for ( int m
= 0; m
< 4; m
++ )
1079 for ( int n
= 0; n
< 5; n
++ )
1081 letters
[m
][n
].x
= center
.x
- R
+ letters
[m
][n
].x
* dx
;
1082 letters
[m
][n
].y
= center
.y
+ h
[ letters
[m
][n
].y
];
1085 dc
.SetPen( wxPen( wxT("blue"), 1, wxDOT
) );
1086 dc
.DrawLines(5, letters
[m
]);
1087 dc
.SetPen( wxPen( wxT("black"), 4, wxSOLID
) );
1088 dc
.DrawSpline(5, letters
[m
]);
1092 dc
.DrawText(_T("Splines not supported."), 10, 5);
1096 void MyCanvas::DrawGradients(wxDC
& dc
)
1099 wxRect
r(10, 10, 100, 100);
1100 dc
.GradientFillLinear(r
, *wxWHITE
, *wxBLUE
, wxRIGHT
);
1103 dc
.GradientFillLinear(r
, *wxWHITE
, *wxBLUE
, wxLEFT
);
1106 dc
.GradientFillLinear(r
, *wxWHITE
, *wxBLUE
, wxDOWN
);
1109 dc
.GradientFillLinear(r
, *wxWHITE
, *wxBLUE
, wxUP
);
1113 r
= wxRect(200, 10, 100, 100);
1114 dc
.GradientFillConcentric(r
, *wxBLUE
, *wxWHITE
);
1117 dc
.GradientFillConcentric(r
, *wxWHITE
, *wxBLUE
);
1120 dc
.GradientFillConcentric(r
, *wxBLUE
, *wxWHITE
, wxPoint(0, 0));
1123 dc
.GradientFillConcentric(r
, *wxBLUE
, *wxWHITE
, wxPoint(100, 100));
1126 void MyCanvas::DrawRegions(wxDC
& dc
)
1128 dc
.DrawText(_T("You should see a red rect partly covered by a cyan one ")
1129 _T("on the left"), 10, 5);
1130 dc
.DrawText(_T("and 5 smileys from which 4 are partially clipped on the right"),
1131 10, 5 + dc
.GetCharHeight());
1132 dc
.DrawText(_T("The second copy should be identical but right part of it ")
1133 _T("should be offset by 10 pixels."),
1134 10, 5 + 2*dc
.GetCharHeight());
1136 DrawRegionsHelper(dc
, 10, true);
1137 DrawRegionsHelper(dc
, 350, false);
1140 void MyCanvas::DrawRegionsHelper(wxDC
& dc
, wxCoord x
, bool firstTime
)
1144 dc
.DestroyClippingRegion();
1145 dc
.SetBrush( *wxWHITE_BRUSH
);
1146 dc
.SetPen( *wxTRANSPARENT_PEN
);
1147 dc
.DrawRectangle( x
, y
, 310, 310 );
1149 dc
.SetClippingRegion( x
+ 10, y
+ 10, 100, 270 );
1151 dc
.SetBrush( *wxRED_BRUSH
);
1152 dc
.DrawRectangle( x
, y
, 310, 310 );
1154 dc
.SetClippingRegion( x
+ 10, y
+ 10, 100, 100 );
1156 dc
.SetBrush( *wxCYAN_BRUSH
);
1157 dc
.DrawRectangle( x
, y
, 310, 310 );
1159 dc
.DestroyClippingRegion();
1161 wxRegion
region(x
+ 110, y
+ 20, 100, 270);
1162 #if !defined(__WXMOTIF__) && !defined(__WXMAC__)
1164 region
.Offset(10, 10);
1166 dc
.SetClippingRegion(region
);
1168 dc
.SetBrush( *wxGREY_BRUSH
);
1169 dc
.DrawRectangle( x
, y
, 310, 310 );
1171 if (m_smile_bmp
.Ok())
1173 dc
.DrawBitmap( m_smile_bmp
, x
+ 150, y
+ 150, true );
1174 dc
.DrawBitmap( m_smile_bmp
, x
+ 130, y
+ 10, true );
1175 dc
.DrawBitmap( m_smile_bmp
, x
+ 130, y
+ 280, true );
1176 dc
.DrawBitmap( m_smile_bmp
, x
+ 100, y
+ 70, true );
1177 dc
.DrawBitmap( m_smile_bmp
, x
+ 200, y
+ 70, true );
1181 void MyCanvas::OnPaint(wxPaintEvent
&WXUNUSED(event
))
1183 wxPaintDC
pdc(this);
1185 #if wxUSE_GRAPHICS_CONTEXT
1187 wxDC
&dc
= m_useContext
? (wxDC
&) gdc
: (wxDC
&) pdc
;
1194 m_owner
->PrepareDC(dc
);
1196 dc
.SetBackgroundMode( m_owner
->m_backgroundMode
);
1197 if ( m_owner
->m_backgroundBrush
.Ok() )
1198 dc
.SetBackground( m_owner
->m_backgroundBrush
);
1199 if ( m_owner
->m_colourForeground
.Ok() )
1200 dc
.SetTextForeground( m_owner
->m_colourForeground
);
1201 if ( m_owner
->m_colourBackground
.Ok() )
1202 dc
.SetTextBackground( m_owner
->m_colourBackground
);
1204 if ( m_owner
->m_textureBackground
) {
1205 if ( ! m_owner
->m_backgroundBrush
.Ok() ) {
1206 wxColour
clr(0,128,0);
1207 wxBrush
b(clr
, wxSOLID
);
1208 dc
.SetBackground(b
);
1213 dc
.SetClippingRegion(100, 100, 100, 100);
1217 if ( m_owner
->m_textureBackground
)
1219 dc
.SetPen(*wxMEDIUM_GREY_PEN
);
1220 for ( int i
= 0; i
< 200; i
++ )
1221 dc
.DrawLine(0, i
*10, i
*10, 0);
1247 DrawTestLines( 0, 100, 0, dc
);
1248 DrawTestLines( 0, 320, 1, dc
);
1249 DrawTestLines( 0, 540, 2, dc
);
1250 DrawTestLines( 0, 760, 6, dc
);
1254 DrawTestBrushes(dc
);
1266 DrawWithLogicalOps(dc
);
1269 #if wxUSE_GRAPHICS_CONTEXT
1284 void MyCanvas::OnMouseMove(wxMouseEvent
&event
)
1287 wxClientDC
dc(this);
1289 m_owner
->PrepareDC(dc
);
1291 wxPoint pos
= event
.GetPosition();
1292 long x
= dc
.DeviceToLogicalX( pos
.x
);
1293 long y
= dc
.DeviceToLogicalY( pos
.y
);
1295 str
.Printf( wxT("Current mouse position: %d,%d"), (int)x
, (int)y
);
1296 m_owner
->SetStatusText( str
);
1299 #endif // wxUSE_STATUSBAR
1302 // ----------------------------------------------------------------------------
1304 // ----------------------------------------------------------------------------
1306 // the event tables connect the wxWidgets events with the functions (event
1307 // handlers) which process them. It can be also done at run-time, but for the
1308 // simple menu events like this the static method is much simpler.
1309 BEGIN_EVENT_TABLE(MyFrame
, wxFrame
)
1310 EVT_MENU (File_Quit
, MyFrame::OnQuit
)
1311 EVT_MENU (File_About
, MyFrame::OnAbout
)
1312 EVT_MENU (File_Clip
, MyFrame::OnClip
)
1313 #if wxUSE_GRAPHICS_CONTEXT
1314 EVT_MENU (File_GraphicContext
, MyFrame::OnGraphicContext
)
1317 EVT_MENU_RANGE(MenuShow_First
, MenuShow_Last
, MyFrame::OnShow
)
1319 EVT_MENU_RANGE(MenuOption_First
, MenuOption_Last
, MyFrame::OnOption
)
1322 // frame constructor
1323 MyFrame::MyFrame(const wxString
& title
, const wxPoint
& pos
, const wxSize
& size
)
1324 : wxFrame((wxFrame
*)NULL
, wxID_ANY
, title
, pos
, size
,
1325 wxDEFAULT_FRAME_STYLE
| wxNO_FULL_REPAINT_ON_RESIZE
)
1327 // set the frame icon
1328 SetIcon(wxICON(mondrian
));
1330 wxMenu
*menuFile
= new wxMenu
;
1331 menuFile
->Append(File_ShowDefault
, _T("&Default screen\tF1"));
1332 menuFile
->Append(File_ShowText
, _T("&Text screen\tF2"));
1333 menuFile
->Append(File_ShowLines
, _T("&Lines screen\tF3"));
1334 menuFile
->Append(File_ShowBrushes
, _T("&Brushes screen\tF4"));
1335 menuFile
->Append(File_ShowPolygons
, _T("&Polygons screen\tF5"));
1336 menuFile
->Append(File_ShowMask
, _T("&Mask screen\tF6"));
1337 menuFile
->Append(File_ShowOps
, _T("&ROP screen\tF7"));
1338 menuFile
->Append(File_ShowRegions
, _T("Re&gions screen\tF8"));
1339 menuFile
->Append(File_ShowCircles
, _T("&Circles screen\tF9"));
1340 #if wxUSE_GRAPHICS_CONTEXT
1341 menuFile
->Append(File_ShowAlpha
, _T("&Alpha screen\tF10"));
1343 menuFile
->Append(File_ShowSplines
, _T("&Splines screen\tF11"));
1344 menuFile
->Append(File_ShowGradients
, _T("&Gradients screen\tF12"));
1345 menuFile
->AppendSeparator();
1346 menuFile
->AppendCheckItem(File_Clip
, _T("&Clip\tCtrl-C"), _T("Clip/unclip drawing"));
1347 #if wxUSE_GRAPHICS_CONTEXT
1348 menuFile
->AppendCheckItem(File_GraphicContext
, _T("&Use GraphicContext\tCtrl-Y"), _T("Use GraphicContext"));
1350 menuFile
->AppendSeparator();
1351 menuFile
->Append(File_About
, _T("&About...\tCtrl-A"), _T("Show about dialog"));
1352 menuFile
->AppendSeparator();
1353 menuFile
->Append(File_Quit
, _T("E&xit\tAlt-X"), _T("Quit this program"));
1355 wxMenu
*menuMapMode
= new wxMenu
;
1356 menuMapMode
->Append( MapMode_Text
, _T("&TEXT map mode") );
1357 menuMapMode
->Append( MapMode_Lometric
, _T("&LOMETRIC map mode") );
1358 menuMapMode
->Append( MapMode_Twips
, _T("T&WIPS map mode") );
1359 menuMapMode
->Append( MapMode_Points
, _T("&POINTS map mode") );
1360 menuMapMode
->Append( MapMode_Metric
, _T("&METRIC map mode") );
1362 wxMenu
*menuUserScale
= new wxMenu
;
1363 menuUserScale
->Append( UserScale_StretchHoriz
, _T("Stretch &horizontally\tCtrl-H") );
1364 menuUserScale
->Append( UserScale_ShrinkHoriz
, _T("Shrin&k horizontally\tCtrl-G") );
1365 menuUserScale
->Append( UserScale_StretchVertic
, _T("Stretch &vertically\tCtrl-V") );
1366 menuUserScale
->Append( UserScale_ShrinkVertic
, _T("&Shrink vertically\tCtrl-W") );
1367 menuUserScale
->AppendSeparator();
1368 menuUserScale
->Append( UserScale_Restore
, _T("&Restore to normal\tCtrl-0") );
1370 wxMenu
*menuAxis
= new wxMenu
;
1371 menuAxis
->AppendCheckItem( AxisMirror_Horiz
, _T("Mirror horizontally\tCtrl-M") );
1372 menuAxis
->AppendCheckItem( AxisMirror_Vertic
, _T("Mirror vertically\tCtrl-N") );
1374 wxMenu
*menuLogical
= new wxMenu
;
1375 menuLogical
->Append( LogicalOrigin_MoveDown
, _T("Move &down\tCtrl-D") );
1376 menuLogical
->Append( LogicalOrigin_MoveUp
, _T("Move &up\tCtrl-U") );
1377 menuLogical
->Append( LogicalOrigin_MoveLeft
, _T("Move &right\tCtrl-L") );
1378 menuLogical
->Append( LogicalOrigin_MoveRight
, _T("Move &left\tCtrl-R") );
1379 menuLogical
->AppendSeparator();
1380 menuLogical
->Append( LogicalOrigin_Set
, _T("Set to (&100, 100)\tShift-Ctrl-1") );
1381 menuLogical
->Append( LogicalOrigin_Restore
, _T("&Restore to normal\tShift-Ctrl-0") );
1383 wxMenu
*menuColour
= new wxMenu
;
1385 menuColour
->Append( Colour_TextForeground
, _T("Text &foreground...") );
1386 menuColour
->Append( Colour_TextBackground
, _T("Text &background...") );
1387 menuColour
->Append( Colour_Background
, _T("Background &colour...") );
1388 #endif // wxUSE_COLOURDLG
1389 menuColour
->AppendCheckItem( Colour_BackgroundMode
, _T("&Opaque/transparent\tCtrl-B") );
1390 menuColour
->AppendCheckItem( Colour_TextureBackgound
, _T("Draw textured back&ground\tCtrl-T") );
1392 // now append the freshly created menu to the menu bar...
1393 wxMenuBar
*menuBar
= new wxMenuBar
;
1394 menuBar
->Append(menuFile
, _T("&File"));
1395 menuBar
->Append(menuMapMode
, _T("&Mode"));
1396 menuBar
->Append(menuUserScale
, _T("&Scale"));
1397 menuBar
->Append(menuAxis
, _T("&Axis"));
1398 menuBar
->Append(menuLogical
, _T("&Origin"));
1399 menuBar
->Append(menuColour
, _T("&Colours"));
1401 // ... and attach this menu bar to the frame
1402 SetMenuBar(menuBar
);
1406 SetStatusText(_T("Welcome to wxWidgets!"));
1407 #endif // wxUSE_STATUSBAR
1409 m_mapMode
= wxMM_TEXT
;
1412 m_xLogicalOrigin
= 0;
1413 m_yLogicalOrigin
= 0;
1415 m_yAxisReversed
= false;
1416 m_backgroundMode
= wxSOLID
;
1417 m_colourForeground
= *wxRED
;
1418 m_colourBackground
= *wxBLUE
;
1419 m_textureBackground
= false;
1421 m_canvas
= new MyCanvas( this );
1422 m_canvas
->SetScrollbars( 10, 10, 100, 240 );
1427 void MyFrame::OnQuit(wxCommandEvent
& WXUNUSED(event
))
1429 // true is to force the frame to close
1433 void MyFrame::OnAbout(wxCommandEvent
& WXUNUSED(event
))
1436 msg
.Printf( wxT("This is the about dialog of the drawing sample.\n")
1437 wxT("This sample tests various primitive drawing functions\n")
1438 wxT("(without any attempts to prevent flicker).\n")
1439 wxT("Copyright (c) Robert Roebling 1999")
1442 wxMessageBox(msg
, _T("About Drawing"), wxOK
| wxICON_INFORMATION
, this);
1445 void MyFrame::OnClip(wxCommandEvent
& event
)
1447 m_canvas
->Clip(event
.IsChecked());
1450 #if wxUSE_GRAPHICS_CONTEXT
1451 void MyFrame::OnGraphicContext(wxCommandEvent
& event
)
1453 m_canvas
->UseGraphicContext(event
.IsChecked());
1457 void MyFrame::OnShow(wxCommandEvent
& event
)
1459 m_canvas
->ToShow((ScreenToShow
)(event
.GetId() - MenuShow_First
));
1462 void MyFrame::OnOption(wxCommandEvent
& event
)
1464 switch (event
.GetId())
1467 m_mapMode
= wxMM_TEXT
;
1469 case MapMode_Lometric
:
1470 m_mapMode
= wxMM_LOMETRIC
;
1473 m_mapMode
= wxMM_TWIPS
;
1475 case MapMode_Points
:
1476 m_mapMode
= wxMM_POINTS
;
1478 case MapMode_Metric
:
1479 m_mapMode
= wxMM_METRIC
;
1482 case LogicalOrigin_MoveDown
:
1483 m_yLogicalOrigin
+= 10;
1485 case LogicalOrigin_MoveUp
:
1486 m_yLogicalOrigin
-= 10;
1488 case LogicalOrigin_MoveLeft
:
1489 m_xLogicalOrigin
+= 10;
1491 case LogicalOrigin_MoveRight
:
1492 m_xLogicalOrigin
-= 10;
1494 case LogicalOrigin_Set
:
1496 m_yLogicalOrigin
= -100;
1498 case LogicalOrigin_Restore
:
1500 m_yLogicalOrigin
= 0;
1503 case UserScale_StretchHoriz
:
1504 m_xUserScale
*= 1.10;
1506 case UserScale_ShrinkHoriz
:
1507 m_xUserScale
/= 1.10;
1509 case UserScale_StretchVertic
:
1510 m_yUserScale
*= 1.10;
1512 case UserScale_ShrinkVertic
:
1513 m_yUserScale
/= 1.10;
1515 case UserScale_Restore
:
1520 case AxisMirror_Vertic
:
1521 m_yAxisReversed
= !m_yAxisReversed
;
1523 case AxisMirror_Horiz
:
1524 m_xAxisReversed
= !m_xAxisReversed
;
1528 case Colour_TextForeground
:
1529 m_colourForeground
= SelectColour();
1531 case Colour_TextBackground
:
1532 m_colourBackground
= SelectColour();
1534 case Colour_Background
:
1536 wxColour col
= SelectColour();
1539 m_backgroundBrush
.SetColour(col
);
1543 #endif // wxUSE_COLOURDLG
1545 case Colour_BackgroundMode
:
1546 m_backgroundMode
= m_backgroundMode
== wxSOLID
? wxTRANSPARENT
1550 case Colour_TextureBackgound
:
1551 m_textureBackground
= ! m_textureBackground
;
1559 m_canvas
->Refresh();
1562 void MyFrame::PrepareDC(wxDC
& dc
)
1564 dc
.SetLogicalOrigin( m_xLogicalOrigin
, m_yLogicalOrigin
);
1565 dc
.SetAxisOrientation( !m_xAxisReversed
, m_yAxisReversed
);
1566 dc
.SetUserScale( m_xUserScale
, m_yUserScale
);
1567 dc
.SetMapMode( m_mapMode
);
1571 wxColour
MyFrame::SelectColour()
1575 wxColourDialog
dialog(this, &data
);
1577 if ( dialog
.ShowModal() == wxID_OK
)
1579 col
= dialog
.GetColourData().GetColour();
1584 #endif // wxUSE_COLOURDLG