1 /////////////////////////////////////////////////////////////////////////////
3 // Author: Robert Roebling
5 // Copyright: 2000 (c) Robert Roebling
6 // Licence: wxWindows Licence
7 /////////////////////////////////////////////////////////////////////////////
10 #pragma implementation "canvas.cpp"
13 // For compilers that support precompilation, includes "wx/wx.h".
14 #include "wx/wxprec.h"
20 #include "wx/canvas/canvas.h"
21 #include "wx/canvas/polygon.h"
22 #include "wx/canvas/liner.h"
26 #include <gdk/gdkrgb.h>
27 #include "wx/gtk/win_gtk.h"
30 #ifndef wxUSE_FREETYPE
31 #define wxUSE_FREETYPE 0
35 #include <freetype/freetype.h>
40 //----------------------------------------------------------------------------
42 //----------------------------------------------------------------------------
44 const double pi
= 3.1415926535;
47 FT_Library g_freetypeLibrary
;
50 //----------------------------------------------------------------------------
52 //----------------------------------------------------------------------------
53 wxCanvasObject::wxCanvasObject()
55 // the default event handler is just this object
62 m_dragmode
= DRAG_ONTOP
;
63 // handy when debugging
64 // m_dragmode = DRAG_RECTANGLE;
68 bool wxCanvasObject::ProcessCanvasObjectEvent(wxEvent
& event
)
70 return m_eventHandler
->ProcessEvent(event
);
73 void wxCanvasObject::PushEventHandler(wxEvtHandler
*handler
)
75 handler
->SetNextHandler(GetEventHandler());
76 m_eventHandler
=handler
;
79 wxEvtHandler
*wxCanvasObject::PopEventHandler(bool deleteHandler
)
81 wxEvtHandler
*handlerA
= m_eventHandler
;
84 wxEvtHandler
*handlerB
= handlerA
->GetNextHandler();
85 handlerA
->SetNextHandler((wxEvtHandler
*)NULL
);
86 m_eventHandler
=handlerB
;
90 handlerA
= (wxEvtHandler
*)NULL
;
97 void wxCanvasObject::AppendEventHandler(wxEvtHandler
*handler
)
99 GetEventHandler()->SetNextHandler(handler
);
102 wxEvtHandler
*wxCanvasObject::RemoveLastEventHandler(bool deleteHandler
)
104 //always the first in the row
105 wxEvtHandler
*handlerA
= m_eventHandler
;
106 wxEvtHandler
*handlerB
=handlerA
;
108 while ( handlerA
->GetNextHandler() )
111 handlerA
= handlerA
->GetNextHandler();
114 handlerB
->SetNextHandler((wxEvtHandler
*)NULL
);
120 return GetEventHandler();
123 wxRect
wxCanvasObject::GetAbsoluteArea(const wxTransformMatrix
& cworld
)
126 wxBoundingBox tmp
=m_bbox
;
130 int x1
= m_admin
->LogicalToDeviceX( tmp
.GetMinX() );
131 int y1
= m_admin
->LogicalToDeviceY( tmp
.GetMinY() );
132 int x2
= m_admin
->LogicalToDeviceX( tmp
.GetMaxX() );
133 int y2
= m_admin
->LogicalToDeviceY( tmp
.GetMaxY() );
151 tmparea
.width
= x2
-x1
;
152 tmparea
.height
= y2
-y1
;
157 void wxCanvasObject::MoveAbsolute( double x
, double y
)
159 //save old position of boundingbox
160 double oldx
= GetXMin();
161 double oldy
= GetYMin();
162 double w
= m_bbox
.GetWidth();
163 double h
= m_bbox
.GetHeight();
167 double newx
=GetXMin();
168 double newy
=GetYMin();
170 double leftu
,rightu
,bottomu
,topu
;
171 leftu
= wxMin (oldx
, newx
) ;
172 rightu
= wxMax (oldx
+ w
, newx
+ w
) ;
173 topu
= wxMin (oldy
, newy
) ;
174 bottomu
= wxMax (oldy
+ h
, newy
+ h
) ;
176 if ( rightu
- leftu
< 2*w
&& bottomu
- topu
< 2*h
)
178 m_admin
->Update( this,leftu
, topu
, rightu
- leftu
, bottomu
- topu
);
182 m_admin
->Update( this, oldx
, oldy
, w
, h
);
183 m_admin
->Update( this, newx
, newy
, w
, h
);
187 void wxCanvasObject::MoveRelative( double x
, double y
)
189 //save old position of boundingbox
190 double oldx
= GetXMin();
191 double oldy
= GetYMin();
192 double w
= m_bbox
.GetWidth();
193 double h
= m_bbox
.GetHeight();
197 double newx
=GetXMin();
198 double newy
=GetYMin();
200 double leftu
,rightu
,bottomu
,topu
;
201 leftu
= wxMin (oldx
, newx
) ;
202 rightu
= wxMax (oldx
+ w
, newx
+ w
) ;
203 topu
= wxMin (oldy
, newy
) ;
204 bottomu
= wxMax (oldy
+ h
, newy
+ h
) ;
206 if ( rightu
- leftu
< 2*w
&& bottomu
- topu
< 2*h
)
208 m_admin
->Update( this,leftu
, topu
, rightu
- leftu
, bottomu
- topu
);
212 m_admin
->Update( this, oldx
, oldy
, w
, h
);
213 m_admin
->Update( this, newx
, newy
, w
, h
);
218 void wxCanvasObject::DragStart()
222 if (m_dragmode
== DRAG_RECTANGLE
)
224 this->SetVisible(FALSE
);
225 wxTransformMatrix help
;
226 double x
= GetXMin();
227 double y
= GetYMin();
228 double w
= m_bbox
.GetWidth();
229 double h
= m_bbox
.GetHeight();
230 m_admin
->Update( this, x
, y
, w
, h
);
231 m_admin
->UpdateNow();
233 wxRect recold
=GetAbsoluteArea(help
);
234 wxClientDC
dc(m_admin
->GetActive());
235 dc
.SetPen(*wxBLACK_PEN
);
236 dc
.SetBrush(*wxTRANSPARENT_BRUSH
);
237 dc
.SetLogicalFunction(wxINVERT
);
238 dc
.DrawRectangle(recold
);
239 dc
.SetBrush(wxNullBrush
);
240 dc
.SetPen(wxNullPen
);
244 this->SetVisible(FALSE
);
245 wxTransformMatrix help
;
246 double x
= GetXMin();
247 double y
= GetYMin();
248 double w
= m_bbox
.GetWidth();
249 double h
= m_bbox
.GetHeight();
251 wxRect recnew
=GetAbsoluteArea(help
);
253 //redraw in buffer what should be there without this object
254 m_admin
->Update( this, x
, y
, w
, h
);
255 m_admin
->GetActive()->Freeze();
257 //save the drawing (without the object itself to a bitmap)
258 m_atnewpos
= wxBitmap(recnew
.width
,recnew
.height
);
260 dcm
.SelectObject(*m_admin
->GetActive()->GetBuffer());
262 tmp
.SelectObject(m_atnewpos
);
263 tmp
.Blit(0,0,recnew
.width
,recnew
.height
,&dcm
,recnew
.x
,recnew
.y
,wxCOPY
,FALSE
);
264 tmp
.SelectObject(wxNullBitmap
);
265 dcm
.SelectObject(wxNullBitmap
);
271 void wxCanvasObject::DragRelative( double x
, double y
)
275 if (m_dragmode
== DRAG_RECTANGLE
)
277 wxTransformMatrix help
;
279 wxRect recold
=GetAbsoluteArea(help
);
283 wxRect recnew
=GetAbsoluteArea(help
);
285 wxClientDC
dc(m_admin
->GetActive());
286 dc
.SetPen(*wxBLACK_PEN
);
287 dc
.SetBrush(*wxTRANSPARENT_BRUSH
);
288 dc
.SetLogicalFunction(wxINVERT
);
289 dc
.DrawRectangle(recold
);
290 dc
.DrawRectangle(recnew
);
291 dc
.SetBrush(wxNullBrush
);
292 dc
.SetPen(wxNullPen
);
296 wxClientDC
dc(m_admin
->GetActive());
299 wxTransformMatrix help
;
300 wxRect recold
=GetAbsoluteArea(help
);
302 //restore what was there (without the object itself)
304 dcm
.SelectObject(*m_admin
->GetActive()->GetBuffer());
305 tmp
.SelectObject(m_atnewpos
);
306 dcm
.Blit(recold
.x
,recold
.y
,recold
.width
,recold
.height
,&tmp
,0,0,wxCOPY
,FALSE
);
310 wxRect recnew
=GetAbsoluteArea(help
);
312 //save the contents of the buffer at the new position
313 tmp
.Blit(0,0,recnew
.width
,recnew
.height
,&dcm
,recnew
.x
,recnew
.y
,wxCOPY
,FALSE
);
314 tmp
.SelectObject(wxNullBitmap
);
316 //m_atnewpos = m_admin->GetActive()->GetBuffer()->GetSubBitmap( recnew );
318 this->SetVisible(TRUE
);
319 //redraw object into the buffer
320 m_admin
->GetActive()->SetDC(&dcm
);
321 Render(&help
,recnew
.x
,recnew
.y
,recnew
.width
,recnew
.height
);
323 //draw the union or seperate to the canvas
324 double leftu
,rightu
,bottomu
,topu
;
325 leftu
= wxMin (recold
.x
, recnew
.x
) ;
326 rightu
= wxMax (recold
.x
+ recold
.width
, recnew
.x
+ recnew
.width
) ;
327 topu
= wxMin (recold
.y
, recnew
.y
) ;
328 bottomu
= wxMax (recold
.y
+ recold
.height
, recnew
.y
+ recnew
.height
) ;
330 if ( rightu
- leftu
< 2*recold
.width
&& bottomu
- topu
< 2*recold
.height
)
332 dc
.Blit(leftu
,topu
,rightu
- leftu
,bottomu
- topu
,&dcm
,leftu
,topu
,wxCOPY
,FALSE
);
337 //first redraw what should be at the old position in the canvas
338 dc
.Blit(recold
.x
,recold
.y
,recold
.width
,recold
.height
,&dcm
,recold
.x
,recold
.y
,wxCOPY
,FALSE
);
339 //blit the new position of the object to the canvas
340 dc
.Blit(recnew
.x
,recnew
.y
,recnew
.width
,recnew
.height
,&dcm
,recnew
.x
,recnew
.y
,wxCOPY
,FALSE
);
342 dcm
.SelectObject(wxNullBitmap
);
343 this->SetVisible(FALSE
);
349 void wxCanvasObject::DragEnd()
353 m_atnewpos
= wxBitmap(0,0);
354 m_admin
->GetActive()->Thaw();
355 this->SetVisible(TRUE
);
356 double x
= GetXMin();
357 double y
= GetYMin();
358 double w
= m_bbox
.GetWidth();
359 double h
= m_bbox
.GetHeight();
360 m_admin
->Update( this, x
, y
, w
, h
);
361 m_admin
->UpdateNow();
365 wxCanvasObject
* wxCanvasObject::IsHitWorld( double x
, double y
, double margin
)
367 if ((x
>= m_bbox
.GetMinX()-margin
) &&
368 (x
<= m_bbox
.GetMaxX()+margin
) &&
369 (y
>= m_bbox
.GetMinY()-margin
) &&
370 (y
<= m_bbox
.GetMaxY()+margin
)
373 return (wxCanvasObject
*) NULL
;
376 wxCanvasObject
* wxCanvasObject::Contains( wxCanvasObject
* obj
)
380 return (wxCanvasObject
*) NULL
;
383 void wxCanvasObject::CaptureMouse()
385 m_admin
->GetActive()->SetCaptureMouse( this );
388 void wxCanvasObject::ReleaseMouse()
390 m_admin
->GetActive()->SetCaptureMouse( NULL
);
393 bool wxCanvasObject::IsCapturedMouse()
395 return m_admin
->GetActive()->GetCaptured()==this;
399 void wxCanvasObject::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
403 void wxCanvasObject::CalcBoundingBox()
407 void wxCanvasObject::WriteSVG( wxTextOutputStream
&stream
)
411 //----------------------------------------------------------------------------
412 // wxCanvasObjectGroup
413 //----------------------------------------------------------------------------
415 wxCanvasObjectGroup::wxCanvasObjectGroup(double x
, double y
)
417 lworld
.Translate(x
,y
);
418 //no objects make the bounding box the x,y and take care of it later
422 wxCanvasObjectGroup::~wxCanvasObjectGroup()
426 void wxCanvasObjectGroup::PushEventHandler(wxEvtHandler
*handler
)
428 wxCanvasObject::PushEventHandler(handler
);
429 wxNode
*node
= m_objects
.First();
432 wxCanvasObject
*obj
= (wxCanvasObject
*) node
->Data();
434 obj
->PushEventHandler(handler
);
440 wxEvtHandler
*wxCanvasObjectGroup::PopEventHandler(bool deleteHandler
)
442 wxNode
*node
= m_objects
.First();
445 wxCanvasObject
*obj
= (wxCanvasObject
*) node
->Data();
447 obj
->PopEventHandler(deleteHandler
);
451 return wxCanvasObject::PopEventHandler(deleteHandler
);
454 void wxCanvasObjectGroup::AppendEventHandler(wxEvtHandler
*handler
)
456 wxCanvasObject::AppendEventHandler(handler
);
457 wxNode
*node
= m_objects
.First();
460 wxCanvasObject
*obj
= (wxCanvasObject
*) node
->Data();
462 obj
->AppendEventHandler(handler
);
468 wxEvtHandler
*wxCanvasObjectGroup::RemoveLastEventHandler(bool deleteHandler
)
470 wxNode
*node
= m_objects
.First();
473 wxCanvasObject
*obj
= (wxCanvasObject
*) node
->Data();
475 obj
->RemoveLastEventHandler(deleteHandler
);
479 return wxCanvasObject::RemoveLastEventHandler(deleteHandler
);
482 void wxCanvasObjectGroup::TransLate( double x
, double y
)
484 lworld
.Translate(x
,y
);
488 void wxCanvasObjectGroup::SetAdmin(wxCanvasAdmin
* admin
)
491 wxNode
*node
= m_objects
.First();
494 wxCanvasObject
*obj
= (wxCanvasObject
*) node
->Data();
496 obj
->SetAdmin(admin
);
502 void wxCanvasObjectGroup::DeleteContents( bool flag
)
504 m_objects
.DeleteContents( flag
);
505 m_bbox
.SetValid(FALSE
);
509 void wxCanvasObjectGroup::Prepend( wxCanvasObject
* obj
)
511 m_objects
.Insert( obj
);
512 if (m_objects
.First())
513 m_bbox
.Expand(obj
->GetBbox());
516 m_bbox
.SetValid(FALSE
);
521 void wxCanvasObjectGroup::Append( wxCanvasObject
* obj
)
523 m_objects
.Append( obj
);
524 if (m_objects
.First())
525 m_bbox
.Expand(obj
->GetBbox());
528 m_bbox
.SetValid(FALSE
);
533 void wxCanvasObjectGroup::Insert( size_t before
, wxCanvasObject
* obj
)
535 m_objects
.Insert( before
, obj
);
536 m_bbox
.SetValid(FALSE
);
537 if (m_objects
.First())
538 m_bbox
.Expand(obj
->GetBbox());
541 m_bbox
.SetValid(FALSE
);
546 void wxCanvasObjectGroup::Remove( wxCanvasObject
* obj
)
548 m_objects
.DeleteObject( obj
);
549 m_bbox
.SetValid(FALSE
);
553 void wxCanvasObjectGroup::CalcBoundingBox()
555 m_bbox
.SetValid(FALSE
);
556 wxNode
*node
= m_objects
.First();
559 wxCanvasObject
*obj
= (wxCanvasObject
*) node
->Data();
562 obj
->CalcBoundingBox();
567 m_bbox
.Expand( tmp
);
572 void wxCanvasObjectGroup::Render(wxTransformMatrix
* cworld
, int x
, int y
, int width
, int height
)
574 if (!m_visible
) return;
576 wxTransformMatrix backup
= *cworld
;
579 wxNode
*node
= m_objects
.First();
585 wxRect absarea
=GetAbsoluteArea(*cworld
);
586 wxDC
*dc
= m_admin
->GetActive()->GetDC();
587 dc
->SetPen(*wxBLACK_PEN
);
588 dc
->SetBrush(*wxTRANSPARENT_BRUSH
);
589 dc
->DrawRectangle( absarea
.x
, absarea
.y
, absarea
.width
, absarea
.height
);
590 dc
->SetBrush(wxNullBrush
);
591 dc
->SetPen(wxNullPen
);
593 //TODO the next will only work if all boundingboxes stay up to date (problem when mowing currently
595 if (! ( wxMax(absarea.x, x) < wxMin(absarea.x + absarea.width , x + width ) &&
596 wxMax(absarea.y, y) < wxMin(absarea.y + absarea.height , y + height )
602 // cycle through all objects
605 wxCanvasObject
*obj
= (wxCanvasObject
*) node
->Data();
607 if (!obj
->IsControl() && obj
->GetVisible())
610 //get area at the absolute position
611 wxRect absareaobject
=obj
->GetAbsoluteArea(*cworld
);
613 // If we have 10.000 objects, we will go through
614 // this 10.000 times for each update, so we have
615 // to optimise carefully.
616 int clip_x
= absareaobject
.x
;
617 int clip_width
= absareaobject
.width
;
620 clip_width
-= x
-clip_x
;
625 if (clip_x
+ clip_width
> x
+ width
)
626 clip_width
= x
+width
-clip_x
;
630 int clip_y
= absareaobject
.y
;
631 int clip_height
= absareaobject
.height
;
634 clip_height
-= y
-clip_y
;
639 if (clip_y
+ clip_height
> y
+ height
)
640 clip_height
= y
+height
-clip_y
;
644 obj
->Render(cworld
, clip_x
, clip_y
, clip_width
, clip_height
);
656 void wxCanvasObjectGroup::WriteSVG( wxTextOutputStream
&stream
)
660 wxCanvasObject
* wxCanvasObjectGroup::IsHitWorld( double x
, double y
, double margin
)
662 //KKKfirst check if within bbox
663 //will only work if they are always uptodate
664 //if (!m_bbox.PointInBox(x,y,margin))
665 // return (wxCanvasObject*) NULL;
667 wxTransformMatrix inverse
= lworld
;
670 inverse
.TransformPoint(x
,y
,xh
,yh
);
672 wxCanvasObject
*obj
=0;
673 wxNode
*node
= m_objects
.Last();
676 obj
=(wxCanvasObject
*) node
->Data();
678 if (!obj
->IsControl() )
680 if (obj
->IsHitWorld(x
,y
,margin
))
685 node
= node
->Previous();
688 return (wxCanvasObject
*) NULL
;
691 wxCanvasObject
* wxCanvasObjectGroup::Contains( wxCanvasObject
* obj
)
693 wxCanvasObject
* cobj
;
694 wxNode
*node
= m_objects
.First();
697 cobj
=(wxCanvasObject
*) node
->Data();
699 if (cobj
->Contains(obj
))
706 return (wxCanvasObject
*) NULL
;
709 int wxCanvasObjectGroup::IndexOf( wxCanvasObject
* obj
)
711 return m_objects
.IndexOf( obj
);
716 //----------------------------------------------------------------------------
718 //----------------------------------------------------------------------------
720 wxCanvasObjectRef::wxCanvasObjectRef(double x
, double y
, wxCanvasObject
* obj
)
723 lworld
.Translate(x
,y
);
726 m_bbox
.SetValid(FALSE
);
730 m_bbox
.Expand( tmp
);
733 void wxCanvasObjectRef::PushEventHandler(wxEvtHandler
*handler
)
735 wxCanvasObject::PushEventHandler(handler
);
736 m_obj
->PushEventHandler(handler
);
739 wxEvtHandler
*wxCanvasObjectRef::PopEventHandler(bool deleteHandler
)
741 m_obj
->PopEventHandler(deleteHandler
);
742 return wxCanvasObject::PopEventHandler(deleteHandler
);
745 void wxCanvasObjectRef::AppendEventHandler(wxEvtHandler
*handler
)
747 wxCanvasObject::AppendEventHandler(handler
);
748 m_obj
->AppendEventHandler(handler
);
751 wxEvtHandler
*wxCanvasObjectRef::RemoveLastEventHandler(bool deleteHandler
)
753 m_obj
->RemoveLastEventHandler(deleteHandler
);
754 return wxCanvasObject::RemoveLastEventHandler(deleteHandler
);
757 void wxCanvasObjectRef::TransLate( double x
, double y
)
759 lworld
.Translate(x
,y
);
763 wxCanvasObject
* wxCanvasObjectRef::Contains( wxCanvasObject
* obj
)
765 if (obj
== this || m_obj
->Contains(obj
))
768 return (wxCanvasObject
*) NULL
;
772 void wxCanvasObjectRef::SetRotation(double rotation
)
774 lworld
.SetRotation(rotation
);
778 void wxCanvasObjectRef::SetScale(double scalex
,double scaley
)
780 lworld
.Scale(scalex
,scaley
,lworld
.GetValue(2,0),lworld
.GetValue(2,1));
784 void wxCanvasObjectRef::SetAdmin(wxCanvasAdmin
* admin
)
787 m_obj
->SetAdmin(admin
);
790 void wxCanvasObjectRef::CalcBoundingBox()
792 m_bbox
.SetValid(FALSE
);
793 m_obj
->CalcBoundingBox();
796 tmp
=m_obj
->GetBbox();
798 m_bbox
.Expand( tmp
);
801 void wxCanvasObjectRef::Render(wxTransformMatrix
* cworld
, int x
, int y
, int width
, int height
)
803 if (!m_visible
) return;
805 //get the absolute area (without the local matrix included)
806 //the boundingbox is relative to the parent.
807 wxRect absarea
=GetAbsoluteArea(*cworld
);
809 wxTransformMatrix backup
= *cworld
;
813 wxDC
*dc
= m_admin
->GetActive()->GetDC();
814 dc
->SetPen(*wxBLACK_PEN
);
815 dc
->SetBrush(*wxTRANSPARENT_BRUSH
);
816 dc
->DrawRectangle( absarea
.x
, absarea
.y
, absarea
.width
, absarea
.height
);
817 dc
->SetBrush(wxNullBrush
);
818 dc
->SetPen(wxNullPen
);
821 int clip_x
= absarea
.x
;
822 int clip_width
= absarea
.width
;
825 clip_width
-= x
-clip_x
;
830 if (clip_x
+ clip_width
> x
+ width
)
831 clip_width
= x
+width
-clip_x
;
835 int clip_y
= absarea
.y
;
836 int clip_height
= absarea
.height
;
839 clip_height
-= y
-clip_y
;
844 if (clip_y
+ clip_height
> y
+ height
)
845 clip_height
= y
+height
-clip_y
;
848 m_obj
->Render(cworld
, clip_x
, clip_y
, clip_width
, clip_height
);
856 void wxCanvasObjectRef::WriteSVG( wxTextOutputStream
&stream
)
860 wxCanvasObject
* wxCanvasObjectRef::IsHitWorld( double x
, double y
, double margin
)
862 //KKKfirst check if within bbox
863 //will only work if they are always uptodate
864 //if (!m_bbox.PointInBox(x,y,margin))
865 // return (wxCanvasObject*) NULL;
867 wxTransformMatrix inverse
= lworld
;
870 inverse
.TransformPoint(x
,y
,xh
,yh
);
872 if (m_obj
->IsHitWorld(xh
,yh
,margin
))
875 return (wxCanvasObject
*) NULL
;
880 //----------------------------------------------------------------------------
882 //----------------------------------------------------------------------------
884 wxCanvasRect::wxCanvasRect( double x
, double y
, double w
, double h
, double radius
)
893 m_brush
= *wxBLACK_BRUSH
;
894 m_pen
= *wxTRANSPARENT_PEN
;
898 void wxCanvasRect::TransLate( double x
, double y
)
905 void wxCanvasRect::CalcBoundingBox()
907 m_bbox
.SetMin( m_x
, m_y
);
908 m_bbox
.SetMax( m_x
+ m_width
,m_y
+ m_height
);
910 //include the pen width also
911 //KKK m_bbox.EnLarge(m_pen.GetWidth()+m_radius);
912 m_bbox
.EnLarge(m_pen
.GetWidth()/2);
915 void wxCanvasRect::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
917 if (!m_visible
) return;
920 wxImage
*image
= m_admin
->GetActive()->GetBuffer();
922 int start_y
= clip_y
;
923 int end_y
= clip_y
+clip_height
;
925 int start_x
= clip_x
;
926 int end_x
= clip_x
+clip_width
;
929 for (int y
= start_y
; y
< end_y
; y
++)
930 for (int x
= start_x
; x
< end_x
; x
++)
932 int red
=m_brush
.GetColour().Red();
933 int green
=m_brush
.GetColour().Green();
934 int blue
=m_brush
.GetColour().Blue();
935 image
->SetRGB( x
, y
, red
, green
, blue
);
938 if (cworld
->GetRotation())
940 wxPoint
*cpoints
= new wxPoint
[4];
943 cworld
->TransformPoint( m_x
, m_y
, x
, y
);
944 cpoints
[0].x
= m_admin
->LogicalToDeviceX(x
);
945 cpoints
[0].y
= m_admin
->LogicalToDeviceY(y
);
946 cworld
->TransformPoint( m_x
, m_y
+ m_height
, x
, y
);
947 cpoints
[1].x
= m_admin
->LogicalToDeviceX(x
);
948 cpoints
[1].y
= m_admin
->LogicalToDeviceY(y
);
949 cworld
->TransformPoint( m_x
+ m_width
, m_y
+ m_height
, x
, y
);
950 cpoints
[2].x
= m_admin
->LogicalToDeviceX(x
);
951 cpoints
[2].y
= m_admin
->LogicalToDeviceY(y
);
952 cworld
->TransformPoint( m_x
+ m_width
, m_y
, x
, y
);
953 cpoints
[3].x
= m_admin
->LogicalToDeviceX(x
);
954 cpoints
[3].y
= m_admin
->LogicalToDeviceY(y
);
956 wxDC
*dc
= m_admin
->GetActive()->GetDC();
957 dc
->SetClippingRegion(clip_x
, clip_y
, clip_width
, clip_height
);
958 dc
->SetBrush(m_brush
);
959 int pw
=m_pen
.GetWidth();
960 m_pen
.SetWidth(m_admin
->LogicalToDeviceXRel(pw
));
962 dc
->DrawPolygon(4, cpoints
, 0,0,wxWINDING_RULE
);
964 dc
->SetBrush(wxNullBrush
);
965 dc
->SetPen(wxNullPen
);
966 dc
->DestroyClippingRegion();
971 wxDC
*dc
= m_admin
->GetActive()->GetDC();
972 dc
->SetClippingRegion(clip_x
, clip_y
, clip_width
, clip_height
);
973 dc
->SetBrush(m_brush
);
974 int pw
=m_pen
.GetWidth();
975 m_pen
.SetWidth(m_admin
->LogicalToDeviceXRel(pw
));
977 //yes the whole not only the clipping region, because we have a pen also
978 int x
= m_admin
->LogicalToDeviceX(cworld
->GetValue(2,0) + m_x
);
979 int y
= m_admin
->LogicalToDeviceY(cworld
->GetValue(2,1) + m_y
);
980 int w
= m_admin
->LogicalToDeviceXRel( m_width
);
981 int h
= m_admin
->LogicalToDeviceYRel( m_height
);
982 int r
= m_admin
->LogicalToDeviceYRel( m_radius
);
983 if (w
> 0 && w
< 1) w
=1;
984 if (w
< 0 && w
> -1) w
=-1;
985 if (h
> 0 && h
< 1) h
=1;
986 if (h
< 0 && h
> -1) h
=-1;
988 dc
->DrawRoundedRectangle( x
,y
,w
,h
,r
);
990 dc
->DrawRectangle( x
,y
,w
,h
);
991 dc
->SetBrush(wxNullBrush
);
992 dc
->SetPen(wxNullPen
);
993 dc
->DestroyClippingRegion();
999 void wxCanvasRect::WriteSVG( wxTextOutputStream
&stream
)
1003 //----------------------------------------------------------------------------
1005 //----------------------------------------------------------------------------
1007 wxCanvasCircle::wxCanvasCircle( double x
, double y
, double radius
)
1014 m_brush
= *wxBLACK_BRUSH
;
1015 m_pen
= *wxTRANSPARENT_PEN
;
1019 void wxCanvasCircle::TransLate( double x
, double y
)
1026 void wxCanvasCircle::CalcBoundingBox()
1028 m_bbox
.SetMin( m_x
-m_radius
, m_y
-m_radius
);
1029 m_bbox
.SetMax( m_x
+m_radius
, m_y
+m_radius
);
1031 //include the pen width also
1032 m_bbox
.EnLarge(m_pen
.GetWidth()/2);
1035 void wxCanvasCircle::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
1037 if (!m_visible
) return;
1041 wxDC
*dc
= m_admin
->GetActive()->GetDC();
1042 dc
->SetClippingRegion(clip_x
, clip_y
, clip_width
, clip_height
);
1043 dc
->SetBrush(m_brush
);
1044 int pw
=m_pen
.GetWidth();
1045 m_pen
.SetWidth(m_admin
->LogicalToDeviceXRel(pw
));
1047 //yes the whole not only the clipping region, because we have a pen also
1048 //and rotation on a circle is not important so only a shift with cworld
1049 int x
= m_admin
->LogicalToDeviceX(cworld
->GetValue(2,0) + m_x
);
1050 int y
= m_admin
->LogicalToDeviceY(cworld
->GetValue(2,1) + m_y
);
1051 int radius
= m_admin
->LogicalToDeviceXRel( m_radius
);
1052 if (radius
< 1) radius
=1;
1053 dc
->DrawCircle( x
,y
,radius
);
1054 dc
->SetBrush(wxNullBrush
);
1055 dc
->SetPen(wxNullPen
);
1056 dc
->DestroyClippingRegion();
1061 void wxCanvasCircle::WriteSVG( wxTextOutputStream
&stream
)
1065 wxCanvasObject
* wxCanvasCircle::IsHitWorld( double x
, double y
, double margin
)
1067 if ((x
>= m_bbox
.GetMinX()-margin
) &&
1068 (x
<= m_bbox
.GetMaxX()+margin
) &&
1069 (y
>= m_bbox
.GetMinY()-margin
) &&
1070 (y
<= m_bbox
.GetMaxY()+margin
)
1073 if (m_radius
+m_pen
.GetWidth()/2+margin
> sqrt(pow(m_x
-x
,2)+pow(m_y
-y
,2)))
1076 return (wxCanvasObject
*) NULL
;
1078 return (wxCanvasObject
*) NULL
;
1081 //----------------------------------------------------------------------------
1083 //----------------------------------------------------------------------------
1085 wxCanvasEllipse::wxCanvasEllipse( double x
, double y
, double width
, double height
)
1093 m_brush
= *wxBLACK_BRUSH
;
1094 m_pen
= *wxTRANSPARENT_PEN
;
1098 void wxCanvasEllipse::TransLate( double x
, double y
)
1105 void wxCanvasEllipse::CalcBoundingBox()
1107 m_bbox
.SetMin( m_x
, m_y
);
1108 m_bbox
.SetMax( m_x
+m_width
, m_y
+m_height
);
1110 //include the pen width also
1111 m_bbox
.EnLarge(m_pen
.GetWidth()/2);
1114 void wxCanvasEllipse::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
1116 if (!m_visible
) return;
1120 wxDC
*dc
= m_admin
->GetActive()->GetDC();
1121 dc
->SetClippingRegion(clip_x
, clip_y
, clip_width
, clip_height
);
1122 dc
->SetBrush(m_brush
);
1123 int pw
=m_pen
.GetWidth();
1124 m_pen
.SetWidth(m_admin
->LogicalToDeviceXRel(pw
));
1126 int x
= m_admin
->LogicalToDeviceX(cworld
->GetValue(2,0) + m_x
);
1127 int y
= m_admin
->LogicalToDeviceY(cworld
->GetValue(2,1) + m_y
);
1128 int w
= m_admin
->LogicalToDeviceXRel( m_width
);
1129 int h
= m_admin
->LogicalToDeviceYRel( m_height
);
1130 if (w
> 0 && w
< 1) w
=1;
1131 if (w
< 0 && w
> -1) w
=-1;
1132 if (h
> 0 && h
< 1) h
=1;
1133 if (h
< 0 && h
> -1) h
=-1;
1134 dc
->DrawEllipse( x
,y
,w
,h
);
1135 dc
->SetBrush(wxNullBrush
);
1136 dc
->SetPen(wxNullPen
);
1137 dc
->DestroyClippingRegion();
1142 void wxCanvasEllipse::WriteSVG( wxTextOutputStream
&stream
)
1146 wxCanvasObject
* wxCanvasEllipse::IsHitWorld( double x
, double y
, double margin
)
1148 if ((x
>= m_bbox
.GetMinX()-margin
) &&
1149 (x
<= m_bbox
.GetMaxX()+margin
) &&
1150 (y
>= m_bbox
.GetMinY()-margin
) &&
1151 (y
<= m_bbox
.GetMaxY()+margin
)
1154 double a
=(m_width
+m_pen
.GetWidth())/2+margin
;
1155 double b
=(m_height
+m_pen
.GetWidth())/2+margin
;
1156 double c
=pow((m_x
+m_width
/2-x
)/a
,2)+pow((m_y
+m_height
/2-y
)/b
,2);
1160 return (wxCanvasObject
*) NULL
;
1162 return (wxCanvasObject
*) NULL
;
1165 //----------------------------------------------------------------------------
1166 // wxCanvasEllipticArc
1167 //----------------------------------------------------------------------------
1169 wxCanvasEllipticArc::wxCanvasEllipticArc( double x
, double y
, double width
, double height
, double start
, double end
)
1179 m_brush
= *wxBLACK_BRUSH
;
1180 m_pen
= *wxTRANSPARENT_PEN
;
1184 void wxCanvasEllipticArc::TransLate( double x
, double y
)
1191 void wxCanvasEllipticArc::CalcBoundingBox()
1193 m_bbox
.SetMin( m_x
, m_y
);
1194 m_bbox
.SetMax( m_x
+m_width
, m_y
+m_height
);
1196 //include the pen width also
1197 m_bbox
.EnLarge(m_pen
.GetWidth()/2);
1200 void wxCanvasEllipticArc::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
1202 if (!m_visible
) return;
1206 wxDC
*dc
= m_admin
->GetActive()->GetDC();
1207 dc
->SetClippingRegion(clip_x
, clip_y
, clip_width
, clip_height
);
1208 dc
->SetBrush(m_brush
);
1209 int pw
=m_pen
.GetWidth();
1210 m_pen
.SetWidth(m_admin
->LogicalToDeviceXRel(pw
));
1212 int x
= m_admin
->LogicalToDeviceX(cworld
->GetValue(2,0) + m_x
);
1213 int y
= m_admin
->LogicalToDeviceY(cworld
->GetValue(2,1) + m_y
);
1214 int w
= m_admin
->LogicalToDeviceXRel( m_width
);
1215 int h
= m_admin
->LogicalToDeviceYRel( m_height
);
1216 if (w
> 0 && w
< 1) w
=1;
1217 if (w
< 0 && w
> -1) w
=-1;
1218 if (h
> 0 && h
< 1) h
=1;
1219 if (h
< 0 && h
> -1) h
=-1;
1220 if (m_admin
->GetActive()->GetYaxis())
1221 dc
->DrawEllipticArc( x
,y
,w
,h
,-m_end
,-m_start
);
1223 dc
->DrawEllipticArc( x
,y
,w
,h
,m_start
,m_end
);
1224 dc
->SetBrush(wxNullBrush
);
1225 dc
->SetPen(wxNullPen
);
1226 dc
->DestroyClippingRegion();
1231 void wxCanvasEllipticArc::WriteSVG( wxTextOutputStream
&stream
)
1235 wxCanvasObject
* wxCanvasEllipticArc::IsHitWorld( double x
, double y
, double margin
)
1237 if ((x
>= m_bbox
.GetMinX()-margin
) &&
1238 (x
<= m_bbox
.GetMaxX()+margin
) &&
1239 (y
>= m_bbox
.GetMinY()-margin
) &&
1240 (y
<= m_bbox
.GetMaxY()+margin
)
1243 double a
=(m_width
+m_pen
.GetWidth())/2+margin
;
1244 double b
=(m_height
+m_pen
.GetWidth())/2+margin
;
1245 double c
=pow((m_x
+m_width
/2-x
)/a
,2)+pow((m_y
+m_height
/2-y
)/b
,2);
1249 return (wxCanvasObject
*) NULL
;
1251 return (wxCanvasObject
*) NULL
;
1254 //----------------------------------------------------------------------------
1256 //----------------------------------------------------------------------------
1258 wxCanvasLine::wxCanvasLine( double x1
, double y1
, double x2
, double y2
)
1266 m_pen
= *wxBLACK_PEN
;
1270 void wxCanvasLine::TransLate( double x
, double y
)
1279 void wxCanvasLine::CalcBoundingBox()
1281 m_bbox
.SetMin( m_x1
, m_y1
);
1282 m_bbox
.SetMax( m_x2
, m_y2
);
1284 //include the pen width also
1285 m_bbox
.EnLarge(m_pen
.GetWidth()/2);
1288 void wxCanvasLine::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
1290 if (!m_visible
) return;
1293 cworld
->TransformPoint( m_x1
, m_y1
, x1
, y1
);
1294 cworld
->TransformPoint( m_x2
, m_y2
, x2
, y2
);
1295 x1
= m_admin
->LogicalToDeviceX( x1
);
1296 y1
= m_admin
->LogicalToDeviceY( y1
);
1297 x2
= m_admin
->LogicalToDeviceX( x2
);
1298 y2
= m_admin
->LogicalToDeviceY( y2
);
1302 tmparea
.x
= m_admin
->LogicalToDeviceXRel( m_bbox
.GetMinX());
1303 tmparea
.y
= m_admin
->LogicalToDeviceYRel( m_bbox
.GetMinY());
1304 tmparea
.width
= m_admin
->LogicalToDeviceXRel( m_bbox
.GetWidth() );
1305 tmparea
.height
= m_admin
->LogicalToDeviceYRel( m_bbox
.GetHeight() );
1306 wxImage
*image
= m_admin
->GetActive()->GetBuffer();
1307 if ((tmparea
.width
== 0) && (tmparea
.height
== 0))
1309 int red
=m_pen
.GetColour().Red();
1310 int green
=m_pen
.GetColour().Green();
1311 int blue
=m_pen
.GetColour().Blue();
1312 image
->SetRGB( tmparea
.x
, tmparea
.y
, red
, green
, blue
);
1316 int red
=m_pen
.GetColour().Red();
1317 int green
=m_pen
.GetColour().Green();
1318 int blue
=m_pen
.GetColour().Blue();
1319 wxInt32 d
, ii
, jj
, di
, ai
, si
, dj
, aj
, sj
;
1322 si
= (di
< 0)? -1 : 1;
1325 sj
= (dj
< 0)? -1 : 1;
1337 if ((ii
>= clip_x
) && (ii
< clip_x
+clip_width
) &&
1338 (jj
>= clip_y
) && (jj
< clip_y
+clip_height
))
1340 image
->SetRGB( ii
, jj
, red
, blue
, green
);
1358 if ((ii
>= clip_x
) && (ii
< clip_x
+clip_width
) &&
1359 (jj
>= clip_y
) && (jj
< clip_y
+clip_height
))
1361 image
->SetRGB( ii
, jj
, red
, blue
, green
);
1374 wxDC
*dc
= m_admin
->GetActive()->GetDC();
1375 dc
->SetClippingRegion( clip_x
, clip_y
, clip_width
, clip_height
);
1376 int pw
=m_pen
.GetWidth();
1377 m_pen
.SetWidth(m_admin
->LogicalToDeviceXRel(pw
));
1378 dc
->SetPen( m_pen
);
1379 dc
->DrawLine( x1
, y1
, x2
, y2
);
1381 dc
->DestroyClippingRegion();
1386 void wxCanvasLine::WriteSVG( wxTextOutputStream
&stream
)
1391 wxCanvasObject
* wxCanvasLine::IsHitWorld( double x
, double y
, double margin
)
1393 if ((x
>= m_bbox
.GetMinX()-margin
) &&
1394 (x
<= m_bbox
.GetMaxX()+margin
) &&
1395 (y
>= m_bbox
.GetMinY()-margin
) &&
1396 (y
<= m_bbox
.GetMaxY()+margin
)
1399 wxLine
line1(m_x1
,m_y1
,m_x2
,m_y2
);
1400 wxPoint2DDouble P
=wxPoint2DDouble(x
,y
);
1402 if (line1
.PointInLine(P
,distance
,m_pen
.GetWidth()/2+margin
) == R_IN_AREA
)
1405 return (wxCanvasObject
*) NULL
;
1407 return (wxCanvasObject
*) NULL
;
1410 //----------------------------------------------------------------------------
1412 //----------------------------------------------------------------------------
1414 wxCanvasImage::wxCanvasImage( const wxImage
&image
, double x
, double y
, double w
, double h
)
1424 m_orgw
=m_image
.GetWidth();
1425 m_orgh
=m_image
.GetHeight();
1429 //KKK m_visible=TRUE;
1433 void wxCanvasImage::TransLate( double x
, double y
)
1440 void wxCanvasImage::CalcBoundingBox()
1442 m_bbox
.SetMin( m_x
, m_y
);
1443 m_bbox
.SetMax( m_x
+ m_width
, m_y
+ m_height
);
1446 void wxCanvasImage::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
1448 if (!m_visible
) return;
1452 tmparea
.x
= m_admin
->LogicalToDeviceXRel( m_bbox
.GetMinX());
1453 tmparea
.y
= m_admin
->LogicalToDeviceYRel( m_bbox
.GetMinY());
1454 tmparea
.width
= m_admin
->LogicalToDeviceXRel( m_bbox
.GetWidth() );
1455 tmparea
.height
= m_admin
->LogicalToDeviceYRel( m_bbox
.GetHeight() );
1459 cworld
->TransformPoint( m_x
, m_y
, x
, y
);
1460 x
= m_admin
->LogicalToDeviceX(x
);
1461 y
= m_admin
->LogicalToDeviceY(y
);
1465 if ((clip_x
== xabs
+ tmparea
.x
) &&
1466 (clip_y
== yabs
+ tmparea
.y
) &&
1467 (clip_width
== tmparea
.width
) &&
1468 (clip_height
== tmparea
.height
))
1470 m_admin
->GetActive()->GetBuffer()->Paste( m_tmp
, clip_x
, clip_y
);
1474 // local coordinates
1475 int start_x
= clip_x
- (xabs
+ tmparea
.x
);
1476 int start_y
= clip_y
- (yabs
+ tmparea
.y
);
1478 wxRect
rect( start_x
, start_y
, clip_width
, clip_height
);
1479 wxImage
sub_image( m_tmp
.GetSubImage( rect
) );
1480 m_admin
->GetActive()->GetBuffer()->Paste( sub_image
, clip_x
, clip_y
);
1483 if ( m_orgw
*5 < m_admin
->LogicalToDeviceXRel( m_bbox
.GetWidth() ) ||
1484 m_orgw
/5 > m_admin
->LogicalToDeviceXRel( m_bbox
.GetWidth() ) ||
1485 m_orgh
*5 < m_admin
->LogicalToDeviceYRel( m_bbox
.GetHeight() ) ||
1486 m_orgh
/5 > m_admin
->LogicalToDeviceYRel( m_bbox
.GetHeight() )
1489 wxDC
*dc
= m_admin
->GetActive()->GetDC();
1490 dc
->SetClippingRegion(clip_x
, clip_y
, clip_width
, clip_height
);
1491 dc
->SetBrush(*wxTRANSPARENT_BRUSH
);
1492 dc
->SetPen(*wxBLACK_PEN
);
1493 //yes the whole not only the clipping region, because we have a pen also
1494 int x
= m_admin
->LogicalToDeviceX(cworld
->GetValue(2,0) + m_x
);
1495 int y
= m_admin
->LogicalToDeviceY(cworld
->GetValue(2,1) + m_y
);
1496 int w
= m_admin
->LogicalToDeviceXRel( m_width
);
1497 int h
= m_admin
->LogicalToDeviceYRel( m_height
);
1500 dc
->DrawRectangle( x
,y
,w
,h
);
1501 dc
->SetBrush(wxNullBrush
);
1502 dc
->SetPen(wxNullPen
);
1503 dc
->DestroyClippingRegion();
1507 if ((m_admin
->LogicalToDeviceXRel( m_bbox
.GetWidth() ) == m_image
.GetWidth()) &&
1508 (m_admin
->LogicalToDeviceYRel( m_bbox
.GetHeight() ) == m_image
.GetHeight()))
1514 m_tmp
= m_image
.Scale( m_admin
->LogicalToDeviceXRel( m_bbox
.GetWidth()),
1515 m_admin
->LogicalToDeviceYRel( m_bbox
.GetHeight()) );
1519 // wxPoint centr(m_admin->LogicalToDeviceX(m_x),m_admin->LogicalToDeviceY(m_y));
1522 if (cworld
->GetRotation())
1524 bmp
=m_tmp
.Rotate(-cworld
->GetRotation()/180.0 * pi
,centr
, TRUE
, NULL
).ConvertToBitmap();
1528 bmp
= m_tmp
.ConvertToBitmap();
1531 wxDC
*dc
= m_admin
->GetActive()->GetDC();
1534 if (cworld
->GetRotation()> 0)
1536 centr2
.x
= (int) (x
+m_height
*sin(-cworld
->GetRotation()/180.0 * pi
));
1542 centr2
.y
= (int) (y
-m_width
*sin(-cworld
->GetRotation()/180.0 * pi
));
1545 if (cworld
->GetRotation() != 0)
1547 //TODO clipping not right
1548 dc
->DrawBitmap(bmp
,centr2
,TRUE
);
1549 // dc->DrawPoint(centr2);
1550 // dc->DrawPoint(x,y);
1554 //TODO clipping not right
1555 // dc->DrawPoint(centr2);
1556 // dc->DrawPoint(x,y);
1558 if ((clip_x
== x
) &&
1560 (clip_width
== tmparea
.width
) &&
1561 (clip_height
== tmparea
.height
))
1563 dc
->DrawBitmap( m_tmp
, clip_x
, clip_y
, TRUE
);
1567 int start_x
= clip_x
- (int)x
;
1568 int start_y
= clip_y
- (int)y
;
1570 //dc->DrawBitmap( bmp, x, y, TRUE );
1572 dcm
.SelectObject(bmp
);
1573 dc
->Blit(clip_x
, clip_y
,clip_width
, clip_height
,&dcm
,start_x
,start_y
,wxCOPY
,TRUE
);
1574 dcm
.SelectObject(wxNullBitmap
);
1580 void wxCanvasImage::WriteSVG( wxTextOutputStream
&stream
)
1585 //----------------------------------------------------------------------------
1587 //----------------------------------------------------------------------------
1589 wxCanvasControl::wxCanvasControl( wxWindow
*control
)
1593 m_control
= control
;
1597 double wxCanvasControl::GetPosX()
1600 m_control
->GetPosition( &x
, &y
);
1601 return m_admin
->DeviceToLogicalX(x
);
1604 double wxCanvasControl::GetPosY()
1607 m_control
->GetPosition( &x
, &y
);
1608 return m_admin
->DeviceToLogicalY(y
);
1611 void wxCanvasControl::SetPosXY( double x
, double y
)
1613 int xd
= m_admin
->LogicalToDeviceX(x
);
1614 int yd
= m_admin
->LogicalToDeviceY(y
);
1615 m_control
->Move(xd
,yd
);
1619 void wxCanvasControl::TransLate( double x
, double y
)
1622 m_control
->GetPosition( &xdo
, &ydo
);
1623 int xd
= m_admin
->LogicalToDeviceX(x
)-xdo
;
1624 int yd
= m_admin
->LogicalToDeviceY(y
)-ydo
;
1625 m_control
->Move(xd
,yd
);
1629 wxCanvasControl::~wxCanvasControl()
1631 m_control
->Destroy();
1634 void wxCanvasControl::CalcBoundingBox()
1638 m_control
->GetSize( &tmparea
.width
, &tmparea
.height
);
1639 m_control
->GetPosition( &tmparea
.x
, &tmparea
.y
);
1641 m_bbox
.SetMin( tmparea
.x
, tmparea
.y
);
1642 m_bbox
.SetMax( tmparea
.x
+ tmparea
.width
, tmparea
.y
+ tmparea
.height
);
1646 void wxCanvasControl::MoveRelative( double x
, double y
)
1648 m_control
->Move( m_admin
->LogicalToDeviceX(x
), m_admin
->LogicalToDeviceX(y
) );
1651 //----------------------------------------------------------------------------
1653 //----------------------------------------------------------------------------
1665 wxCanvasText::wxCanvasText( const wxString
&text
, double x
, double y
, const wxString
&fontFile
, int size
)
1669 m_fontFileName
= fontFile
;
1682 wxFaceData
*data
= new wxFaceData
;
1685 int error
= FT_New_Face( g_freetypeLibrary
,
1690 error
= FT_Set_Char_Size( data
->m_face
,
1699 wxCanvasText::~wxCanvasText()
1702 wxFaceData
*data
= (wxFaceData
*) m_faceData
;
1706 if (m_alpha
) delete [] m_alpha
;
1709 void wxCanvasText::SetRGB( unsigned char red
, unsigned char green
, unsigned char blue
)
1716 void wxCanvasText::SetFlag( int flag
)
1721 void wxCanvasText::Render(wxTransformMatrix
* cworld
, int clip_x
, int clip_y
, int clip_width
, int clip_height
)
1723 if (!m_visible
) return;
1726 tmparea
.x
= m_admin
->LogicalToDeviceX( m_bbox
.GetMinX());
1727 tmparea
.y
= m_admin
->LogicalToDeviceY( m_bbox
.GetMinY());
1728 tmparea
.width
= m_admin
->LogicalToDeviceXRel( m_bbox
.GetWidth() );
1729 tmparea
.height
= m_admin
->LogicalToDeviceYRel( m_bbox
.GetHeight() );
1731 m_alpha
= new unsigned char[tmparea
.width
*tmparea
.height
];
1732 memset( m_alpha
, 0, tmparea
.width
*tmparea
.height
);
1734 if (!m_alpha
) return;
1737 FT_Face face
= ((wxFaceData
*)m_faceData
)->m_face
;
1738 FT_GlyphSlot slot
= face
->glyph
;
1742 for (int n
= 0; n
< (int)m_text
.Len(); n
++)
1744 FT_UInt index
= FT_Get_Char_Index( face
, m_text
[(unsigned int)n
] );
1746 int error
= FT_Load_Glyph( face
, index
, FT_LOAD_DEFAULT
);
1747 if (error
) continue;
1749 error
= FT_Render_Glyph( face
->glyph
, ft_render_mode_normal
);
1750 if (error
) continue;
1752 FT_Bitmap
*bitmap
= &slot
->bitmap
;
1753 unsigned char* buffer
= bitmap
->buffer
;
1754 for (int y
= 0; y
< bitmap
->rows
; y
++)
1755 for (int x
= 0; x
< bitmap
->width
; x
++)
1757 unsigned char alpha
= buffer
[ y
*bitmap
->pitch
+ x
];
1758 if (alpha
== 0) continue;
1760 int xx
= pen_x
+ slot
->bitmap_left
+ x
;
1761 int yy
= pen_y
- slot
->bitmap_top
+ y
;
1762 m_alpha
[ yy
* tmparea
.width
+ xx
] = alpha
;
1765 pen_x
+= slot
->advance
.x
>> 6;
1766 pen_y
+= slot
->advance
.y
>> 6;
1771 wxImage
*image
= m_admin
->GetActive()->GetBuffer();
1773 // local coordinates
1774 int start_x
= clip_x
- tmparea
.x
;
1775 int end_x
= clip_width
+ start_x
;
1776 int start_y
= clip_y
- tmparea
.y
;
1777 int end_y
= clip_height
+ start_y
;
1779 for (int y
= start_y
; y
< end_y
; y
++)
1780 for (int x
= start_x
; x
< end_x
; x
++)
1782 int alpha
= m_alpha
[y
*tmparea
.width
+ x
];
1785 int image_x
= tmparea
.x
+x
;
1786 int image_y
= tmparea
.y
+y
;
1789 image
->SetRGB( image_x
, image_y
, m_red
, m_green
, m_blue
);
1792 int red1
= (m_red
* alpha
) / 255;
1793 int green1
= (m_green
* alpha
) / 255;
1794 int blue1
= (m_blue
* alpha
) / 255;
1797 int red2
= image
->GetRed( image_x
, image_y
);
1798 int green2
= image
->GetGreen( image_x
, image_y
);
1799 int blue2
= image
->GetBlue( image_x
, image_y
);
1800 red2
= (red2
* alpha
) / 255;
1801 green2
= (green2
* alpha
) / 255;
1802 blue2
= (blue2
* alpha
) / 255;
1804 image
->SetRGB( image_x
, image_y
, red1
+red2
, green1
+green2
, blue1
+blue2
);
1808 wxBitmap
*bitmap
= m_admin
->GetActive()->GetBuffer();
1809 wxRect
sub_rect( clip_x
, clip_y
, clip_width
, clip_height
);
1810 wxBitmap
sub_bitmap( bitmap
->GetSubBitmap( sub_rect
) );
1812 wxImage
image( sub_bitmap
);
1814 // local coordinates
1815 int start_x
= clip_x
- tmparea
.x
;
1816 int end_x
= clip_width
+ start_x
;
1817 int start_y
= clip_y
- tmparea
.y
;
1818 int end_y
= clip_height
+ start_y
;
1820 for (int y
= start_y
; y
< end_y
; y
++)
1821 for (int x
= start_x
; x
< end_x
; x
++)
1823 int alpha
= m_alpha
[y
*tmparea
.width
+ x
];
1826 int image_x
= x
- start_x
;
1827 int image_y
= y
- start_y
;
1830 image
.SetRGB( image_x
, image_y
, m_red
, m_green
, m_blue
);
1833 int red1
= (m_red
* alpha
) / 255;
1834 int green1
= (m_green
* alpha
) / 255;
1835 int blue1
= (m_blue
* alpha
) / 255;
1838 int red2
= image
.GetRed( image_x
, image_y
);
1839 int green2
= image
.GetGreen( image_x
, image_y
);
1840 int blue2
= image
.GetBlue( image_x
, image_y
);
1841 red2
= (red2
* alpha
) / 255;
1842 green2
= (green2
* alpha
) / 255;
1843 blue2
= (blue2
* alpha
) / 255;
1845 image
.SetRGB( image_x
, image_y
, red1
+red2
, green1
+green2
, blue1
+blue2
);
1849 sub_bitmap
= image
.ConvertToBitmap();
1851 wxDC
*dc
= m_admin
->GetActive()->GetDC();
1852 dc
->DrawBitmap( sub_bitmap
, clip_x
, clip_y
);
1856 void wxCanvasText::WriteSVG( wxTextOutputStream
&stream
)
1860 void wxCanvasText::TransLate( double x
, double y
)
1867 void wxCanvasText::CalcBoundingBox()
1869 if (m_alpha
) delete [] m_alpha
;
1871 m_bbox
.SetMin( m_x
, m_y
);
1872 m_bbox
.SetMax( m_x
+ 100 , m_y
+ m_size
+ (m_size
/2));
1877 //----------------------------------------------------------------------------
1879 //----------------------------------------------------------------------------
1881 IMPLEMENT_CLASS(wxCanvas
,wxWindow
)
1883 BEGIN_EVENT_TABLE(wxCanvas
,wxWindow
)
1884 EVT_SCROLLWIN( wxCanvas::OnScroll
)
1885 EVT_CHAR( wxCanvas::OnChar
)
1886 EVT_PAINT( wxCanvas::OnPaint
)
1887 EVT_SIZE( wxCanvas::OnSize
)
1888 EVT_IDLE( wxCanvas::OnIdle
)
1889 EVT_MOUSE_EVENTS( wxCanvas::OnMouse
)
1890 EVT_SET_FOCUS( wxCanvas::OnSetFocus
)
1891 EVT_KILL_FOCUS( wxCanvas::OnKillFocus
)
1892 EVT_ERASE_BACKGROUND( wxCanvas::OnEraseBackground
)
1895 wxCanvas::wxCanvas( wxCanvasAdmin
* admin
, wxWindow
*parent
, wxWindowID id
,
1896 const wxPoint
&position
, const wxSize
& size
, long style
) :
1897 wxWindow( parent
, id
, position
, size
, style
)
1900 m_admin
->Append(this);
1901 m_needUpdate
= FALSE
;
1902 m_background
= *wxWHITE
;
1903 m_lastMouse
= (wxCanvasObject
*)NULL
;
1904 m_captureMouse
= (wxCanvasObject
*)NULL
;
1913 wxCanvas::~wxCanvas()
1915 wxNode
*node
= m_updateRects
.First();
1918 wxRect
*rect
= (wxRect
*) node
->Data();
1920 m_updateRects
.DeleteNode( node
);
1921 node
= m_updateRects
.First();
1925 void wxCanvas::SetColour( const wxColour
& background
)
1927 m_background
=background
;
1928 SetBackgroundColour( m_background
);
1930 if (m_frozen
) return;
1933 unsigned char red
= background
.Red();
1934 unsigned char green
= background
.Green();
1935 unsigned char blue
= background
.Blue();
1937 unsigned char *data
= m_buffer
.GetData();
1939 for (int y
= 0; y
< m_buffer
.GetHeight(); y
++)
1940 for (int x
= 0; x
< m_buffer
.GetWidth(); x
++)
1951 dc
.SelectObject( m_buffer
);
1952 dc
.SetPen( *wxTRANSPARENT_PEN
);
1953 wxBrush
brush( m_background
, wxSOLID
);
1954 dc
.SetBrush( brush
);
1955 dc
.DrawRectangle( 0, 0, m_buffer
.GetWidth(), m_buffer
.GetHeight() );
1956 dc
.SelectObject( wxNullBitmap
);
1960 void wxCanvas::SetCaptureMouse( wxCanvasObject
*obj
)
1964 wxWindow::CaptureMouse();
1965 m_captureMouse
= obj
;
1969 wxWindow::ReleaseMouse();
1970 m_captureMouse
= NULL
;
1974 void wxCanvas::Freeze()
1979 void wxCanvas::Thaw()
1981 wxNode
*node
= m_updateRects
.First();
1984 wxRect
*rect
= (wxRect
*) node
->Data();
1986 m_updateRects
.DeleteNode( node
);
1987 node
= m_updateRects
.First();
1993 Update( 0,0, m_buffer
.GetWidth(), m_buffer
.GetHeight() );
1996 void wxCanvas::Update( int x
, int y
, int width
, int height
, bool blit
)
1998 m_admin
->SetActive(this);
2000 if (!m_root
) return;
2002 if (m_frozen
) return;
2010 if (width
<= 0) return;
2017 if (height
<= 0) return;
2019 if (x
+width
> m_buffer
.GetWidth())
2021 width
= m_buffer
.GetWidth() - x
;
2023 if (width
<= 0) return;
2025 if (y
+height
> m_buffer
.GetHeight())
2027 height
= m_buffer
.GetHeight() - y
;
2029 if (height
<= 0) return;
2031 // update is within the buffer
2032 m_needUpdate
= TRUE
;
2034 // has to be blitted to screen later
2037 m_updateRects
.Append(
2038 (wxObject
*) new wxRect( x
,y
,width
,height
) );
2041 wxTransformMatrix cworld
;
2044 // speed up with direct access, maybe add wxImage::Clear(x,y,w,h,r,g,b)
2046 int end_y
= y
+height
;
2048 int end_x
= x
+width
;
2049 int red
=m_background
.Red();
2050 int green
=m_background
.Green();
2051 int blue
=m_background
.Blue();
2052 for (int yy
= start_y
; yy
< end_y
; yy
++)
2053 for (int xx
= start_x
; xx
< end_x
; xx
++)
2054 m_buffer
.SetRGB( xx
, yy
, red
, green
, blue
);
2056 m_root
->Render(&cworld
, x
, y
, width
, height
);
2059 dc
.SelectObject( m_buffer
);
2061 dc
.SetPen( *wxTRANSPARENT_PEN
);
2062 wxBrush
brush( m_background
, wxSOLID
);
2063 dc
.SetBrush( brush
);
2065 if (width
!= m_buffer
.GetWidth() && height
!= m_buffer
.GetHeight())
2067 dc
.SetLogicalFunction(wxCOPY
);
2068 dc
.SetClippingRegion(x
,y
,width
,height
);
2069 dc
.DrawRectangle(x
-2,y
-2,width
+4,height
+4);
2070 dc
.DestroyClippingRegion();
2075 dc
.SetLogicalFunction(wxCOPY
);
2076 dc
.DrawRectangle(0,0,m_buffer
.GetWidth(),m_buffer
.GetHeight());
2078 dc
.SetBrush(wxNullBrush
);
2079 dc
.SetPen(wxNullPen
);
2083 m_root
->Render(&cworld
,x
, y
, width
, height
);
2086 dc
.SelectObject( wxNullBitmap
);
2090 void wxCanvas::BlitBuffer( wxDC
&dc
)
2092 wxNode
*node
= m_updateRects
.First();
2095 wxRect
*rect
= (wxRect
*) node
->Data();
2099 wxImage
sub_image( m_buffer
.GetSubImage( *rect
) );
2101 int bpp
= wxDisplayDepth();
2104 // the init code is doubled in wxImage
2105 static bool s_hasInitialized
= FALSE
;
2107 if (!s_hasInitialized
)
2110 s_hasInitialized
= TRUE
;
2113 gdk_draw_rgb_image( GTK_PIZZA(m_wxwindow
)->bin_window
,
2114 m_wxwindow
->style
->black_gc
,
2115 sub_rect
.x
, sub_rect
.y
,
2116 sub_image
.GetWidth(), sub_image
.GetHeight(),
2117 GDK_RGB_DITHER_NONE
,
2118 sub_image
.GetData(),
2119 sub_image
.GetWidth()*3 );
2123 wxBitmap
bitmap( sub_image
.ConvertToBitmap() );
2124 dc
.DrawBitmap( bitmap
, rect
->x
, rect
->y
);
2127 wxBitmap
bitmap( sub_image
.ConvertToBitmap() );
2128 dc
.DrawBitmap( bitmap
, rect
->x
, rect
->y
);
2131 #else // IMAGE_CANVAS
2133 // Maybe clipping use SetClipping() is faster than
2134 // getting the subrect first and drawing it then?
2136 // wxBitmap sub_bitmap( m_buffer.GetSubBitmap( *rect ) );
2137 // dc.DrawBitmap( sub_bitmap, rect->x, rect->y );
2140 mdc
.SelectObject( m_buffer
);
2141 dc
.Blit (rect
->x
, rect
->y
, rect
->GetWidth(), rect
->GetHeight(),&mdc
, rect
->x
, rect
->y
);
2142 mdc
.SelectObject( wxNullBitmap
);
2146 m_updateRects
.DeleteNode( node
);
2147 node
= m_updateRects
.First();
2150 m_needUpdate
= FALSE
;
2153 void wxCanvas::UpdateNow()
2155 if (m_frozen
) return;
2157 if (!m_needUpdate
) return;
2159 wxClientDC
dc( this );
2165 void wxCanvas::OnPaint(wxPaintEvent
&event
)
2170 if (!m_buffer
.Ok()) return;
2172 if (m_frozen
) return;
2174 m_needUpdate
= TRUE
;
2176 wxRegionIterator
it( GetUpdateRegion() );
2182 int w
= it
.GetWidth();
2183 int h
= it
.GetHeight();
2185 if (x
+w
> m_buffer
.GetWidth())
2186 w
= m_buffer
.GetWidth() - x
;
2187 if (y
+h
> m_buffer
.GetHeight())
2188 h
= m_buffer
.GetHeight() - y
;
2190 if ((w
> 0) && (h
> 0))
2192 m_updateRects
.Append( (wxObject
*) new wxRect( x
, y
, w
, h
) );
2201 void wxCanvas::ScrollWindow( int dx
, int dy
, const wxRect
* rect
)
2203 // If any updates are pending, do them now since they will
2204 // expect the previous m_bufferX and m_bufferY as well as
2205 // the previous device origin values.
2206 wxClientDC
dc( this );
2207 dc
.SetDeviceOrigin( m_oldDeviceX
, m_oldDeviceY
);
2212 double dyv
=DeviceToLogicalYRel(dy
);
2213 m_virt_minY
=m_virt_minY
-dyv
;
2214 m_virt_maxY
=m_virt_maxY
-dyv
;
2218 double dxv
=DeviceToLogicalXRel(dx
);
2219 m_virt_minX
=m_virt_minX
-dxv
;
2220 m_virt_maxX
=m_virt_maxX
-dxv
;
2223 m_admin
->SetActive(this);
2224 SetMappingScroll(m_virt_minX
,m_virt_minY
,m_virt_maxX
,m_virt_maxY
,FALSE
);
2227 unsigned char* data
= m_buffer
.GetData();
2233 unsigned char *source
= data
;
2234 unsigned char *dest
= data
+ (dy
* m_buffer
.GetWidth() * 3);
2235 size_t count
= (size_t) (m_buffer
.GetWidth() * 3 * (m_buffer
.GetHeight()-dy
));
2236 memmove( dest
, source
, count
);
2238 // We update the new buffer area, but there is no need to
2239 // blit (last param FALSE) since the ensuing paint event will
2241 Update( 0, 0, m_buffer
.GetWidth(), dy
, FALSE
);
2245 unsigned char *dest
= data
;
2246 unsigned char *source
= data
+ (-dy
* m_buffer
.GetWidth() * 3);
2247 size_t count
= (size_t) (m_buffer
.GetWidth() * 3 * (m_buffer
.GetHeight()+dy
));
2248 memmove( dest
, source
, count
);
2250 // We update the new buffer area, but there is no need to
2251 // blit (last param FALSE) since the ensuing paint event will
2253 Update( 0, m_buffer
.GetHeight()+dy
, m_buffer
.GetWidth(), -dy
, FALSE
);
2261 unsigned char *source
= data
;
2262 for (int y
= 0; y
< m_buffer
.GetHeight(); y
++)
2264 unsigned char *dest
= source
+ dx
*3;
2265 memmove( dest
, source
, (m_buffer
.GetWidth()-dx
) * 3 );
2266 source
+= m_buffer
.GetWidth()*3;
2269 // We update the new buffer area, but there is no need to
2270 // blit (last param FALSE) since the ensuing paint event will
2272 Update( 0,0, dx
, m_buffer
.GetHeight(), FALSE
);
2276 unsigned char *dest
= data
;
2277 for (int y
= 0; y
< m_buffer
.GetHeight(); y
++)
2279 unsigned char *source
= dest
- dx
*3;
2280 memmove( dest
, source
, (m_buffer
.GetWidth()+dx
) * 3 );
2281 dest
+= m_buffer
.GetWidth()*3;
2284 // We update the new buffer area, but there is no need to
2285 // blit (last param FALSE) since the ensuing paint event will
2287 Update( m_buffer
.GetWidth()+dx
, 0, -dx
, m_buffer
.GetHeight(), FALSE
);
2294 if (dy
> 0 && dy
< m_buffer
.GetHeight())
2296 wxRect
rect( 0, 0, m_buffer
.GetWidth(), m_buffer
.GetHeight()-dy
);
2297 wxBitmap
sub_bitmap( m_buffer
.GetSubBitmap( rect
) );
2299 dcm
.SelectObject( m_buffer
);
2300 dcm
.DrawBitmap( sub_bitmap
, 0, dy
, TRUE
);
2301 dcm
.SelectObject( wxNullBitmap
);
2303 Update( 0, 0, m_buffer
.GetWidth(), dy
, TRUE
);
2305 else if (dy
< 0 && dy
> -m_buffer
.GetHeight())
2307 wxRect
rect( 0, -dy
, m_buffer
.GetWidth(), m_buffer
.GetHeight()+dy
);
2308 wxBitmap
sub_bitmap( m_buffer
.GetSubBitmap( rect
) );
2310 dcm
.SelectObject( m_buffer
);
2311 dcm
.DrawBitmap( sub_bitmap
, 0, 0, TRUE
);
2312 dcm
.SelectObject( wxNullBitmap
);
2314 Update( 0, m_buffer
.GetHeight()+dy
, m_buffer
.GetWidth(), m_buffer
.GetHeight(), TRUE
);
2317 Update( 0, 0, m_buffer
.GetWidth(), m_buffer
.GetHeight(), TRUE
);
2322 if (dx
> 0 && dx
< m_buffer
.GetWidth())
2324 wxRect
rect( 0, 0, m_buffer
.GetWidth()-dx
, m_buffer
.GetHeight());
2325 wxBitmap
sub_bitmap( m_buffer
.GetSubBitmap( rect
) );
2327 dcm
.SelectObject( m_buffer
);
2328 dcm
.DrawBitmap( sub_bitmap
, dx
, 0, TRUE
);
2329 dcm
.SelectObject( wxNullBitmap
);
2331 Update( 0, 0, dx
, m_buffer
.GetHeight(), TRUE
);
2333 else if (dx
< 0 && dx
> -m_buffer
.GetWidth())
2335 wxRect
rect( -dx
, 0, m_buffer
.GetWidth()+dx
, m_buffer
.GetHeight());
2336 wxBitmap
sub_bitmap( m_buffer
.GetSubBitmap( rect
) );
2338 dcm
.SelectObject( m_buffer
);
2339 dcm
.DrawBitmap( sub_bitmap
, 0, 0, TRUE
);
2340 dcm
.SelectObject( wxNullBitmap
);
2342 Update( m_buffer
.GetWidth()+dx
, 0, m_buffer
.GetWidth(), m_buffer
.GetHeight(), TRUE
);
2345 Update( 0,0, m_buffer
.GetWidth(), m_buffer
.GetHeight(), TRUE
);
2348 wxWindow::ScrollWindow( dx
, dy
, rect
);
2350 //must be done now because quick repeated scrolling will prevent wxPaint
2351 //from doing it properly
2355 void wxCanvas::OnMouse(wxMouseEvent
&event
)
2357 m_admin
->SetActive(this);
2364 int x
= event
.GetX();
2365 int y
= event
.GetY();
2367 //to world coordinates to do hit test in world coordinates
2368 double xw
= DeviceToLogicalX( x
);
2369 double yw
= DeviceToLogicalY( y
);
2371 //make a select margin of 2 pixels, so also zero line thickness will be hit
2372 double margin
= DeviceToLogicalXRel( 2 );
2374 if (event
.GetEventType() == wxEVT_MOTION
)
2376 if (m_captureMouse
) //no matter what go to this one
2378 wxMouseEvent
child_event( wxEVT_MOTION
);
2379 child_event
.SetEventObject(m_captureMouse
);
2380 child_event
.m_x
= x
;
2381 child_event
.m_y
= y
;
2382 child_event
.m_leftDown
= event
.m_leftDown
;
2383 child_event
.m_rightDown
= event
.m_rightDown
;
2384 child_event
.m_middleDown
= event
.m_middleDown
;
2385 child_event
.m_controlDown
= event
.m_controlDown
;
2386 child_event
.m_shiftDown
= event
.m_shiftDown
;
2387 child_event
.m_altDown
= event
.m_altDown
;
2388 child_event
.m_metaDown
= event
.m_metaDown
;
2390 m_captureMouse
->ProcessCanvasObjectEvent( child_event
);
2395 wxCanvasObject
*obj
= m_root
->IsHitWorld(xw
,yw
,margin
);
2397 if (obj
&& !obj
->IsControl())
2399 wxMouseEvent
child_event( wxEVT_MOTION
);
2400 child_event
.SetEventObject( obj
);
2401 child_event
.m_x
= x
;
2402 child_event
.m_y
= y
;
2403 child_event
.m_leftDown
= event
.m_leftDown
;
2404 child_event
.m_rightDown
= event
.m_rightDown
;
2405 child_event
.m_middleDown
= event
.m_middleDown
;
2406 child_event
.m_controlDown
= event
.m_controlDown
;
2407 child_event
.m_shiftDown
= event
.m_shiftDown
;
2408 child_event
.m_altDown
= event
.m_altDown
;
2409 child_event
.m_metaDown
= event
.m_metaDown
;
2411 if ((obj
!= m_lastMouse
) && (m_lastMouse
!= NULL
))
2413 child_event
.SetEventType( wxEVT_LEAVE_WINDOW
);
2414 child_event
.SetEventObject( m_lastMouse
);
2415 child_event
.m_x
= x
;
2416 child_event
.m_y
= y
;
2417 m_lastMouse
->ProcessCanvasObjectEvent( child_event
);
2420 child_event
.SetEventType( wxEVT_ENTER_WINDOW
);
2421 child_event
.SetEventObject( m_lastMouse
);
2422 child_event
.m_x
= x
;
2423 child_event
.m_y
= y
;
2424 m_lastMouse
->ProcessCanvasObjectEvent( child_event
);
2426 child_event
.SetEventType( wxEVT_MOTION
);
2427 child_event
.SetEventObject( obj
);
2430 obj
->ProcessCanvasObjectEvent( child_event
);
2436 wxMouseEvent
child_event( wxEVT_LEAVE_WINDOW
);
2437 child_event
.SetEventObject( m_lastMouse
);
2438 child_event
.m_x
= x
;
2439 child_event
.m_y
= y
;
2440 child_event
.m_leftDown
= event
.m_leftDown
;
2441 child_event
.m_rightDown
= event
.m_rightDown
;
2442 child_event
.m_middleDown
= event
.m_middleDown
;
2443 child_event
.m_controlDown
= event
.m_controlDown
;
2444 child_event
.m_shiftDown
= event
.m_shiftDown
;
2445 child_event
.m_altDown
= event
.m_altDown
;
2446 child_event
.m_metaDown
= event
.m_metaDown
;
2447 m_lastMouse
->ProcessCanvasObjectEvent( child_event
);
2449 m_lastMouse
= (wxCanvasObject
*) NULL
;
2455 if (m_captureMouse
) //no matter what go to this one
2457 wxMouseEvent
child_event( event
.GetEventType() );
2458 child_event
.SetEventObject(m_captureMouse
);
2459 child_event
.m_x
= x
;
2460 child_event
.m_y
= y
;
2461 child_event
.m_leftDown
= event
.m_leftDown
;
2462 child_event
.m_rightDown
= event
.m_rightDown
;
2463 child_event
.m_middleDown
= event
.m_middleDown
;
2464 child_event
.m_controlDown
= event
.m_controlDown
;
2465 child_event
.m_shiftDown
= event
.m_shiftDown
;
2466 child_event
.m_altDown
= event
.m_altDown
;
2467 child_event
.m_metaDown
= event
.m_metaDown
;
2468 m_captureMouse
->ProcessCanvasObjectEvent( child_event
);
2472 wxCanvasObject
*obj
= m_root
->IsHitWorld(xw
,yw
,margin
);
2474 if (obj
&& !obj
->IsControl())
2476 wxMouseEvent
child_event( event
.GetEventType() );
2477 child_event
.SetEventObject( obj
);
2478 child_event
.m_x
= x
;
2479 child_event
.m_y
= y
;
2480 child_event
.m_leftDown
= event
.m_leftDown
;
2481 child_event
.m_rightDown
= event
.m_rightDown
;
2482 child_event
.m_middleDown
= event
.m_middleDown
;
2483 child_event
.m_controlDown
= event
.m_controlDown
;
2484 child_event
.m_shiftDown
= event
.m_shiftDown
;
2485 child_event
.m_altDown
= event
.m_altDown
;
2486 child_event
.m_metaDown
= event
.m_metaDown
;
2488 obj
->ProcessCanvasObjectEvent( child_event
);
2497 void wxCanvas::OnSize(wxSizeEvent
&event
)
2501 GetClientSize( &w
, &h
);
2504 m_buffer
= wxImage( w
, h
);
2507 m_buffer
= wxBitmap( w
, h
);
2508 dc
.SelectObject( m_buffer
);
2509 dc
.SetPen( *wxTRANSPARENT_PEN
);
2510 wxBrush
brush( m_background
, wxSOLID
);
2511 dc
.SetBrush( brush
);
2512 dc
.DrawRectangle( 0, 0, m_buffer
.GetWidth(), m_buffer
.GetHeight() );
2513 dc
.SelectObject( wxNullBitmap
);
2517 wxNode
*node
= m_updateRects
.First();
2520 wxRect
*rect
= (wxRect
*) node
->Data();
2522 m_updateRects
.DeleteNode( node
);
2523 node
= m_updateRects
.First();
2528 m_admin
->SetActive(this);
2529 SetMappingScroll(m_virt_minX
,m_virt_minY
,m_virt_maxX
,m_virt_maxY
,FALSE
);
2531 Update( 0,0, m_buffer
.GetWidth(), m_buffer
.GetHeight(), FALSE
);
2536 void wxCanvas::OnIdle(wxIdleEvent
&event
)
2538 m_admin
->SetActive(this);
2543 void wxCanvas::OnSetFocus(wxFocusEvent
&event
)
2545 m_admin
->SetActive(this);
2548 void wxCanvas::OnKillFocus(wxFocusEvent
&event
)
2553 void wxCanvas::OnEraseBackground(wxEraseEvent
&event
)
2559 // maps the virtual window (Real drawing to the window coordinates
2560 // also used for zooming
2561 void wxCanvas::SetMappingScroll( double vx1
, double vy1
, double vx2
, double vy2
, bool border
)
2564 GetClientSize(&dwxi
,&dwyi
);
2566 if (vx2
==vx1
) vx2
=vx1
+100000;
2567 if (vy2
==vy1
) vy2
=vy1
+100000;
2578 double dvx
= m_virt_maxX
- m_virt_minX
;
2579 double dvy
= m_virt_maxY
- m_virt_minY
;
2581 // calculate the scaling factor for the virtual window
2584 if ((dvy
/ dvx
) < (dwy
/ dwx
))
2586 dvy
= dvx
* (dwy
/ dwx
);
2587 // calculate the change in the coordinates
2588 temp_y
= (dvy
- (m_virt_maxY
- m_virt_minY
) )/ 2.0;
2592 dvx
= dvy
* (dwx
/ dwy
);
2593 // calculate the change in the coordinates
2594 temp_x
= (dvx
- (m_virt_maxX
- m_virt_minX
) )/ 2.0;
2597 // add or substract the change from the original coordinates
2598 m_virt_minX
=m_virt_minX
-temp_x
;
2599 m_virt_minY
=m_virt_minY
-temp_y
;
2601 m_virt_maxX
=m_virt_maxX
+temp_x
;
2602 m_virt_maxY
=m_virt_maxY
+temp_y
;
2604 // initialize the mapping_matrix used for mapping the
2605 // virtual windows to the drawing window
2607 // make mappingmatrix
2608 m_mapping_matrix
.Identity();
2610 // translate the drawing to 0,0
2612 m_mapping_matrix
.Translate(-m_virt_minX
,-m_virt_maxY
);
2614 m_mapping_matrix
.Translate(-m_virt_minX
,-m_virt_minY
);
2617 // make a small white border around the drawing
2618 m_virt_minX
=m_virt_minX
- 0.05 * dvx
;
2619 m_virt_minY
=m_virt_minY
- 0.05 * dvy
;
2621 m_virt_maxX
=m_virt_maxX
+ 0.05 * dvx
;
2622 m_virt_maxY
=m_virt_maxY
+ 0.05 * dvy
;
2624 // translate the drawing to 0,0
2626 m_mapping_matrix
.Translate(-m_virt_minX
,-m_virt_maxY
);
2628 m_mapping_matrix
.Translate(-m_virt_minX
,-m_virt_minY
);
2631 double scalefactor_x
= dwx
;
2632 scalefactor_x
/= (m_virt_maxX
- m_virt_minX
);
2634 double scalefactor_y
= dwy
;
2635 scalefactor_y
/= (m_virt_maxY
- m_virt_minY
);
2637 // scale the drawing so it fit's in the window
2638 m_mapping_matrix
.Scale(scalefactor_x
, scalefactor_y
, 0, 0);
2640 // because of coordinate change mirror over X
2641 // 0,0 in graphic computerscreens: upperleft corner
2642 // 0,0 in cartesian: lowerleft corner
2645 m_mapping_matrix
.Mirror();
2647 // make inverse of mapping matrix
2648 // this is to set coordinates in the statusbar
2649 // and the calculate screencoordinates to world coordinates used
2651 m_inverse_mapping
=m_mapping_matrix
;
2652 m_inverse_mapping
.Invert();
2655 SetScroll(m_virtm_minX
,m_virtm_minY
,m_virtm_maxX
,m_virtm_maxY
);
2658 GetClientSize(&dx2
,&dy2
);
2659 if ( dwxi
!= dx2
|| dwyi
!= dy2
) //scrollbar is/became empty
2660 SetScroll(m_virtm_minX
,m_virtm_minY
,m_virtm_maxX
,m_virtm_maxY
);
2664 void wxCanvas::SetScroll(double vx1
,double vy1
,double vx2
,double vy2
)
2671 double dvx
= m_virt_maxX
- m_virt_minX
;
2672 double dvy
= m_virt_maxY
- m_virt_minY
;
2673 double dmvx
= m_virtm_maxX
- m_virtm_minX
;
2674 double dmvy
= m_virtm_maxY
- m_virtm_minY
;
2676 SetScrollbar(wxHORIZONTAL
,(m_virt_minX
-m_virtm_minX
)/dmvx
*1000,dvx
/dmvx
*1000,1000,true);
2679 SetScrollbar(wxVERTICAL
,(m_virtm_maxY
-m_virt_maxY
)/dmvy
*1000,dvy
/dmvy
*1000,1000,true);
2683 SetScrollbar(wxVERTICAL
,(m_virt_minY
-m_virtm_minY
)/dmvy
*1000,dvy
/dmvy
*1000,1000,true);
2689 // coordinates conversions
2690 // -----------------------
2691 double wxCanvas::DeviceToLogicalX(int x
) const
2693 return m_inverse_mapping
.GetValue(0,0) * x
+ m_inverse_mapping
.GetValue(2,0);
2696 double wxCanvas::DeviceToLogicalY(int y
) const
2698 return m_inverse_mapping
.GetValue(1,1) * y
+ m_inverse_mapping
.GetValue(2,1);
2701 double wxCanvas::DeviceToLogicalXRel(int x
) const
2703 return x
*m_inverse_mapping
.GetValue(0,0);
2706 double wxCanvas::DeviceToLogicalYRel(int y
) const
2708 return y
*m_inverse_mapping
.GetValue(1,1);
2711 int wxCanvas::LogicalToDeviceX(double x
) const
2713 return (int) (m_mapping_matrix
.GetValue(0,0) * x
+ m_mapping_matrix
.GetValue(2,0) + 0.5);
2716 int wxCanvas::LogicalToDeviceY(double y
) const
2718 return (int) (m_mapping_matrix
.GetValue(1,1) * y
+ m_mapping_matrix
.GetValue(2,1) + 0.5);
2721 int wxCanvas::LogicalToDeviceXRel(double x
) const
2723 return (int) (x
*m_mapping_matrix
.GetValue(0,0) + 0.5);
2726 int wxCanvas::LogicalToDeviceYRel(double y
) const
2728 return (int) (y
*m_mapping_matrix
.GetValue(1,1) + 0.5);
2733 // return the inverse mapping matrix for zooming or coordinates
2734 wxTransformMatrix
wxCanvas::GetInverseMappingMatrix()
2736 return m_inverse_mapping
;
2739 wxTransformMatrix
wxCanvas::GetMappingMatrix()
2741 return m_mapping_matrix
;
2745 // ----------------------------------------------------------------------------
2746 // scrolling behaviour
2747 // ----------------------------------------------------------------------------
2749 void wxCanvas::OnScroll(wxScrollWinEvent
& event
)
2751 if (event
.GetEventType()==wxEVT_SCROLLWIN_THUMBRELEASE
)
2753 if (event
.GetOrientation()==wxHORIZONTAL
)
2755 double x
=m_virtm_minX
+event
.GetPosition()/1000.0*(m_virtm_maxX
-m_virtm_minX
);
2756 x
=LogicalToDeviceXRel(x
-m_virt_minX
);
2757 ScrollWindow(-x
, 0, (const wxRect
*) NULL
);
2761 double y
=m_virtm_minY
+event
.GetPosition()/1000.0*(m_virtm_maxY
-m_virtm_minY
);
2762 y
=LogicalToDeviceYRel(y
-m_virt_minY
);
2763 ScrollWindow(0, -y
, (const wxRect
*) NULL
);
2766 else if (event
.GetEventType()==wxEVT_SCROLLWIN_PAGEUP
)
2768 if (event
.GetOrientation()==wxHORIZONTAL
)
2770 double x
=GetBufferWidth();
2771 ScrollWindow(x
, 0, (const wxRect
*) NULL
);
2775 double y
=GetBufferHeight();
2776 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2779 else if (event
.GetEventType()==wxEVT_SCROLLWIN_PAGEDOWN
)
2781 if (event
.GetOrientation()==wxHORIZONTAL
)
2783 double x
=-GetBufferWidth();
2784 ScrollWindow(x
, 0, (const wxRect
*) NULL
);
2788 double y
=-GetBufferHeight();
2789 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2792 else if (event
.GetEventType()==wxEVT_SCROLLWIN_LINEUP
)
2794 if (event
.GetOrientation()==wxHORIZONTAL
)
2796 int x
=GetBufferWidth()/10;
2797 ScrollWindow(x
, 0, (const wxRect
*) NULL
);
2801 int y
=GetBufferHeight()/10;
2802 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2805 else if (event
.GetEventType()==wxEVT_SCROLLWIN_LINEDOWN
)
2807 if (event
.GetOrientation()==wxHORIZONTAL
)
2809 int x
=-GetBufferWidth()/10;
2810 ScrollWindow(x
, 0, (const wxRect
*) NULL
);
2814 int y
=-GetBufferHeight()/10;
2815 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2821 void wxCanvas::OnChar(wxKeyEvent
& event
)
2823 switch ( event
.KeyCode() )
2828 double y
=GetBufferHeight();
2829 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2835 double y
=-GetBufferHeight();
2836 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2841 double y
=m_virtm_minY
;
2842 y
=LogicalToDeviceYRel(y
-m_virt_minY
);
2843 ScrollWindow(0, -y
, (const wxRect
*) NULL
);
2848 double y
=m_virtm_minY
+(m_virtm_maxY
-m_virtm_minY
);
2849 y
=LogicalToDeviceYRel(y
-m_virt_minY
);
2850 ScrollWindow(0, -y
, (const wxRect
*) NULL
);
2856 if (!event
.ControlDown())
2857 y
=GetBufferHeight()/10;
2859 y
=GetBufferHeight();
2860 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2867 if (!event
.ControlDown())
2868 y
=-GetBufferHeight()/10;
2870 y
=-GetBufferHeight();
2871 ScrollWindow(0, y
, (const wxRect
*) NULL
);
2878 if (!event
.ControlDown())
2879 x
=GetBufferWidth()/10;
2882 ScrollWindow(x
, 0, (const wxRect
*) NULL
);
2888 if (!event
.ControlDown())
2889 x
=-GetBufferWidth()/10;
2891 x
=-GetBufferWidth();
2892 ScrollWindow(x
, 0, (const wxRect
*) NULL
);
2904 //--------------------------------------------------------------------
2906 //--------------------------------------------------------------------
2908 class wxCanvasModule
: public wxModule
2911 virtual bool OnInit();
2912 virtual void OnExit();
2915 DECLARE_DYNAMIC_CLASS(wxCanvasModule
)
2918 IMPLEMENT_DYNAMIC_CLASS(wxCanvasModule
, wxModule
)
2920 bool wxCanvasModule::OnInit()
2923 int error
= FT_Init_FreeType( &g_freetypeLibrary
);
2924 if (error
) return FALSE
;
2930 void wxCanvasModule::OnExit()
2933 FT_Done_FreeType( g_freetypeLibrary
);
2938 wxCanvasAdmin::wxCanvasAdmin()
2943 wxCanvasAdmin::~wxCanvasAdmin()
2950 void wxCanvasAdmin::Append( wxCanvas
* canvas
)
2952 m_canvaslist
.Append( canvas
);
2955 void wxCanvasAdmin::Remove( wxCanvas
* canvas
)
2957 m_canvaslist
.DeleteObject( canvas
);
2960 void wxCanvasAdmin::Update(wxCanvasObject
* obj
, double x
, double y
, double width
, double height
)
2962 wxNode
*node
= m_canvaslist
.First();
2966 wxCanvas
*canvas
= (wxCanvas
*) node
->Data();
2968 if (m_active
== canvas
)
2970 int xi
= canvas
->LogicalToDeviceX( x
);
2971 int yi
= canvas
->LogicalToDeviceY( y
);
2972 int wi
= canvas
->LogicalToDeviceXRel( width
);
2973 int hi
= canvas
->LogicalToDeviceYRel( height
);
2974 //update a little more then is strictly needed,
2975 //to get rid of the 1 bit bugs
2976 if (canvas
->GetYaxis())
2977 canvas
->Update( xi
-2, yi
+hi
-2, wi
+4, -hi
+4);
2979 canvas
->Update( xi
-2, yi
-2, wi
+4, hi
+4);
2982 { wxCanvasObject
* topobj
=canvas
->GetRoot()->Contains(obj
);
2985 wxCanvas
* tcanvas
= m_active
;
2989 //KKK TODO somehow the next does not work for update i do not know why
2990 canvas->GetRoot()->CalcBoundingBox();
2991 int xi = topobj->GetX();
2992 int yi = topobj->GetY();
2993 int wi = topobj->GetWidth();
2994 int hi = topobj->GetHeight();
2996 canvas
->Update( 0,0, canvas
->GetBufferWidth(),canvas
->GetBufferHeight());
3001 node
= node
->Next();
3005 void wxCanvasAdmin::UpdateNow()
3007 wxNode
*node
= m_canvaslist
.First();
3010 wxCanvas
*canvas
= (wxCanvas
*) node
->Data();
3012 canvas
->UpdateNow();
3013 node
= node
->Next();
3017 // coordinates conversions
3018 // -----------------------
3019 double wxCanvasAdmin::DeviceToLogicalX(int x
) const
3021 return m_active
->DeviceToLogicalX(x
);
3024 double wxCanvasAdmin::DeviceToLogicalY(int y
) const
3026 return m_active
->DeviceToLogicalY(y
);
3029 double wxCanvasAdmin::DeviceToLogicalXRel(int x
) const
3031 return m_active
->DeviceToLogicalXRel(x
);
3034 double wxCanvasAdmin::DeviceToLogicalYRel(int y
) const
3036 return m_active
->DeviceToLogicalYRel(y
);
3039 int wxCanvasAdmin::LogicalToDeviceX(double x
) const
3041 return m_active
->LogicalToDeviceX(x
);
3044 int wxCanvasAdmin::LogicalToDeviceY(double y
) const
3046 return m_active
->LogicalToDeviceY(y
);
3049 int wxCanvasAdmin::LogicalToDeviceXRel(double x
) const
3051 return m_active
->LogicalToDeviceXRel(x
);
3054 int wxCanvasAdmin::LogicalToDeviceYRel(double y
) const
3056 return m_active
->LogicalToDeviceYRel(y
);
3059 void wxCanvasAdmin::SetActive(wxCanvas
* activate
)
3061 wxNode
*node
= m_canvaslist
.First();
3064 wxCanvas
*canvas
= (wxCanvas
*) node
->Data();
3066 if (activate
== canvas
)
3071 node
= node
->Next();