]> git.saurik.com Git - wxWidgets.git/blob - contrib/src/ogl/drawn.cpp
35129107ddb472918b5d48523f1f67134f0fbd97
[wxWidgets.git] / contrib / src / ogl / drawn.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: drawn.cpp
3 // Purpose: wxDrawnShape
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 12/07/98
7 // RCS-ID: $Id$
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 #ifdef __GNUG__
13 #pragma implementation "drawn.h"
14 #pragma implementation "drawnp.h"
15 #endif
16
17 // For compilers that support precompilation, includes "wx.h".
18 #include "wx/wxprec.h"
19
20 #ifdef __BORLANDC__
21 #pragma hdrstop
22 #endif
23
24 #ifndef WX_PRECOMP
25 #include <wx/wx.h>
26 #endif
27
28 #if wxUSE_PROLOGIO
29 #include <wx/deprecated/wxexpr.h>
30 #endif
31
32 #include "wx/ogl/ogl.h"
33
34 static void IntToHex(unsigned int dec, wxChar *buf);
35 static unsigned long HexToInt(wxChar *buf);
36 extern wxChar *oglBuffer;
37
38 #define gyTYPE_PEN 40
39 #define gyTYPE_BRUSH 41
40 #define gyTYPE_FONT 42
41
42 /*
43 * Drawn object
44 *
45 */
46
47 IMPLEMENT_DYNAMIC_CLASS(wxDrawnShape, wxRectangleShape)
48
49 wxDrawnShape::wxDrawnShape():wxRectangleShape(100.0, 50.0)
50 {
51 m_saveToFile = true;
52 m_currentAngle = oglDRAWN_ANGLE_0;
53 }
54
55 wxDrawnShape::~wxDrawnShape()
56 {
57 }
58
59 void wxDrawnShape::OnDraw(wxDC& dc)
60 {
61 // Pass pen and brush in case we have force outline
62 // and fill colours
63 if (m_shadowMode != SHADOW_NONE)
64 {
65 if (m_shadowBrush)
66 m_metafiles[m_currentAngle].m_fillBrush = m_shadowBrush;
67 m_metafiles[m_currentAngle].m_outlinePen = g_oglTransparentPen;
68 m_metafiles[m_currentAngle].Draw(dc, m_xpos + m_shadowOffsetX, m_ypos + m_shadowOffsetY);
69 }
70
71 m_metafiles[m_currentAngle].m_outlinePen = m_pen;
72 m_metafiles[m_currentAngle].m_fillBrush = m_brush;
73 m_metafiles[m_currentAngle].Draw(dc, m_xpos, m_ypos);
74 }
75
76 void wxDrawnShape::SetSize(double w, double h, bool WXUNUSED(recursive))
77 {
78 SetAttachmentSize(w, h);
79
80 double scaleX;
81 double scaleY;
82 if (GetWidth() == 0.0)
83 scaleX = 1.0;
84 else scaleX = w/GetWidth();
85 if (GetHeight() == 0.0)
86 scaleY = 1.0;
87 else scaleY = h/GetHeight();
88
89 for (int i = 0; i < 4; i++)
90 {
91 if (m_metafiles[i].IsValid())
92 m_metafiles[i].Scale(scaleX, scaleY);
93 }
94 m_width = w;
95 m_height = h;
96 SetDefaultRegionSize();
97 }
98
99 void wxDrawnShape::Scale(double sx, double sy)
100 {
101 int i;
102 for (i = 0; i < 4; i++)
103 {
104 if (m_metafiles[i].IsValid())
105 {
106 m_metafiles[i].Scale(sx, sy);
107 m_metafiles[i].CalculateSize(this);
108 }
109 }
110 }
111
112 void wxDrawnShape::Translate(double x, double y)
113 {
114 int i;
115 for (i = 0; i < 4; i++)
116 {
117 if (m_metafiles[i].IsValid())
118 {
119 m_metafiles[i].Translate(x, y);
120 m_metafiles[i].CalculateSize(this);
121 }
122 }
123 }
124
125 // theta is absolute rotation from the zero position
126 void wxDrawnShape::Rotate(double x, double y, double theta)
127 {
128 m_currentAngle = DetermineMetaFile(theta);
129
130 if (m_currentAngle == 0)
131 {
132 // Rotate metafile
133 if (!m_metafiles[0].GetRotateable())
134 return;
135
136 m_metafiles[0].Rotate(x, y, theta);
137 }
138
139 double actualTheta = theta-m_rotation;
140
141 // Rotate attachment points
142 double sinTheta = (double)sin(actualTheta);
143 double cosTheta = (double)cos(actualTheta);
144 wxNode *node = m_attachmentPoints.GetFirst();
145 while (node)
146 {
147 wxAttachmentPoint *point = (wxAttachmentPoint *)node->GetData();
148 double x1 = point->m_x;
149 double y1 = point->m_y;
150 point->m_x = x1*cosTheta - y1*sinTheta + x*(1.0 - cosTheta) + y*sinTheta;
151 point->m_y = x1*sinTheta + y1*cosTheta + y*(1.0 - cosTheta) + x*sinTheta;
152 node = node->GetNext();
153 }
154 m_rotation = theta;
155
156 m_metafiles[m_currentAngle].CalculateSize(this);
157 }
158
159 // Which metafile do we use now? Based on current rotation and validity
160 // of metafiles.
161
162 int wxDrawnShape::DetermineMetaFile(double rotation)
163 {
164 double tolerance = 0.0001;
165 const double pi = M_PI ;
166 double angle1 = 0.0;
167 double angle2 = pi/2.0;
168 double angle3 = pi;
169 double angle4 = 3.0*pi/2.0;
170
171 int whichMetafile = 0;
172
173 if (oglRoughlyEqual(rotation, angle1, tolerance))
174 {
175 whichMetafile = 0;
176 }
177 else if (oglRoughlyEqual(rotation, angle2, tolerance))
178 {
179 whichMetafile = 1;
180 }
181 else if (oglRoughlyEqual(rotation, angle3, tolerance))
182 {
183 whichMetafile = 2;
184 }
185 else if (oglRoughlyEqual(rotation, angle4, tolerance))
186 {
187 whichMetafile = 3;
188 }
189
190 if ((whichMetafile > 0) && !m_metafiles[whichMetafile].IsValid())
191 whichMetafile = 0;
192
193 return whichMetafile;
194 }
195
196 void wxDrawnShape::OnDrawOutline(wxDC& dc, double x, double y, double w, double h)
197 {
198 if (m_metafiles[m_currentAngle].GetOutlineOp() != -1)
199 {
200 wxNode* node = m_metafiles[m_currentAngle].GetOps().Item(m_metafiles[m_currentAngle].GetOutlineOp());
201 wxASSERT (node != NULL);
202 wxDrawOp* op = (wxDrawOp*) node->GetData();
203
204 if (op->OnDrawOutline(dc, x, y, w, h, m_width, m_height))
205 return;
206 }
207
208 // Default... just use a rectangle
209 wxRectangleShape::OnDrawOutline(dc, x, y, w, h);
210 }
211
212 // Get the perimeter point using the special outline op, if there is one,
213 // otherwise use default wxRectangleShape scheme
214 bool wxDrawnShape::GetPerimeterPoint(double x1, double y1,
215 double x2, double y2,
216 double *x3, double *y3)
217 {
218 if (m_metafiles[m_currentAngle].GetOutlineOp() != -1)
219 {
220 wxNode* node = m_metafiles[m_currentAngle].GetOps().Item(m_metafiles[m_currentAngle].GetOutlineOp());
221 wxASSERT (node != NULL);
222 wxDrawOp* op = (wxDrawOp*) node->GetData();
223
224 if (op->GetPerimeterPoint(x1, y1, x2, y2, x3, y3, GetX(), GetY(), GetAttachmentMode()))
225 return true;
226 }
227
228 // Default... just use a rectangle
229 return wxRectangleShape::GetPerimeterPoint(x1, y1, x2, y2, x3, y3);
230 }
231
232 #if wxUSE_PROLOGIO
233 void wxDrawnShape::WriteAttributes(wxExpr *clause)
234 {
235 wxRectangleShape::WriteAttributes(clause);
236
237 clause->AddAttributeValue(_T("current_angle"), (long)m_currentAngle);
238 clause->AddAttributeValue(_T("save_metafile"), (long)m_saveToFile);
239 if (m_saveToFile)
240 {
241 for (int i = 0; i < 4; i++)
242 {
243 if (m_metafiles[i].IsValid())
244 m_metafiles[i].WriteAttributes(clause, i);
245 }
246 }
247 }
248
249 void wxDrawnShape::ReadAttributes(wxExpr *clause)
250 {
251 wxRectangleShape::ReadAttributes(clause);
252
253 int iVal = (int) m_saveToFile;
254 clause->GetAttributeValue(_T("save_metafile"), iVal);
255 clause->GetAttributeValue(_T("current_angle"), m_currentAngle);
256 m_saveToFile = (iVal != 0);
257
258 if (m_saveToFile)
259 {
260 for (int i = 0; i < 4; i++)
261 {
262 m_metafiles[i].ReadAttributes(clause, i);
263 }
264 }
265 }
266 #endif
267
268 // Does the copying for this object
269 void wxDrawnShape::Copy(wxShape& copy)
270 {
271 wxRectangleShape::Copy(copy);
272
273 wxASSERT( copy.IsKindOf(CLASSINFO(wxDrawnShape)) ) ;
274
275 wxDrawnShape& drawnCopy = (wxDrawnShape&) copy;
276
277 for (int i = 0; i < 4; i++)
278 {
279 m_metafiles[i].Copy(drawnCopy.m_metafiles[i]);
280 }
281 drawnCopy.m_saveToFile = m_saveToFile;
282 drawnCopy.m_currentAngle = m_currentAngle;
283 }
284
285 bool wxDrawnShape::LoadFromMetaFile(const wxString& filename)
286 {
287 return m_metafiles[0].LoadFromMetaFile(filename, &m_width, &m_height);
288 }
289
290 // Set of functions for drawing into a pseudo metafile.
291 // They use integers, but doubles are used internally for accuracy
292 // when scaling.
293
294 void wxDrawnShape::DrawLine(const wxPoint& pt1, const wxPoint& pt2)
295 {
296 m_metafiles[m_currentAngle].DrawLine(pt1, pt2);
297 }
298
299 void wxDrawnShape::DrawRectangle(const wxRect& rect)
300 {
301 m_metafiles[m_currentAngle].DrawRectangle(rect);
302 }
303
304 void wxDrawnShape::DrawRoundedRectangle(const wxRect& rect, double radius)
305 {
306 m_metafiles[m_currentAngle].DrawRoundedRectangle(rect, radius);
307 }
308
309 void wxDrawnShape::DrawEllipse(const wxRect& rect)
310 {
311 m_metafiles[m_currentAngle].DrawEllipse(rect);
312 }
313
314 void wxDrawnShape::DrawArc(const wxPoint& centrePt, const wxPoint& startPt, const wxPoint& endPt)
315 {
316 m_metafiles[m_currentAngle].DrawArc(centrePt, startPt, endPt);
317 }
318
319 void wxDrawnShape::DrawEllipticArc(const wxRect& rect, double startAngle, double endAngle)
320 {
321 m_metafiles[m_currentAngle].DrawEllipticArc(rect, startAngle, endAngle);
322 }
323
324 void wxDrawnShape::DrawPoint(const wxPoint& pt)
325 {
326 m_metafiles[m_currentAngle].DrawPoint(pt);
327 }
328
329 void wxDrawnShape::DrawText(const wxString& text, const wxPoint& pt)
330 {
331 m_metafiles[m_currentAngle].DrawText(text, pt);
332 }
333
334 void wxDrawnShape::DrawLines(int n, wxPoint pts[])
335 {
336 m_metafiles[m_currentAngle].DrawLines(n, pts);
337 }
338
339 void wxDrawnShape::DrawPolygon(int n, wxPoint pts[], int flags)
340 {
341 if (flags & oglMETAFLAGS_ATTACHMENTS)
342 {
343 ClearAttachments();
344 int i;
345 for (i = 0; i < n; i++)
346 m_attachmentPoints.Append(new wxAttachmentPoint(i, pts[i].x, pts[i].y));
347 }
348 m_metafiles[m_currentAngle].DrawPolygon(n, pts, flags);
349 }
350
351 void wxDrawnShape::DrawSpline(int n, wxPoint pts[])
352 {
353 m_metafiles[m_currentAngle].DrawSpline(n, pts);
354 }
355
356 void wxDrawnShape::SetClippingRect(const wxRect& rect)
357 {
358 m_metafiles[m_currentAngle].SetClippingRect(rect);
359 }
360
361 void wxDrawnShape::DestroyClippingRect()
362 {
363 m_metafiles[m_currentAngle].DestroyClippingRect();
364 }
365
366 void wxDrawnShape::SetDrawnPen(wxPen* pen, bool isOutline)
367 {
368 m_metafiles[m_currentAngle].SetPen(pen, isOutline);
369 }
370
371 void wxDrawnShape::SetDrawnBrush(wxBrush* brush, bool isFill)
372 {
373 m_metafiles[m_currentAngle].SetBrush(brush, isFill);
374 }
375
376 void wxDrawnShape::SetDrawnFont(wxFont* font)
377 {
378 m_metafiles[m_currentAngle].SetFont(font);
379 }
380
381 void wxDrawnShape::SetDrawnTextColour(const wxColour& colour)
382 {
383 m_metafiles[m_currentAngle].SetTextColour(colour);
384 }
385
386 void wxDrawnShape::SetDrawnBackgroundColour(const wxColour& colour)
387 {
388 m_metafiles[m_currentAngle].SetBackgroundColour(colour);
389 }
390
391 void wxDrawnShape::SetDrawnBackgroundMode(int mode)
392 {
393 m_metafiles[m_currentAngle].SetBackgroundMode(mode);
394 }
395
396
397 /*
398 * Individual operations
399 *
400 */
401
402 /*
403 * Set font, brush, text colour
404 *
405 */
406
407 wxOpSetGDI::wxOpSetGDI(int theOp, wxPseudoMetaFile *theImage, int theGdiIndex, int theMode):
408 wxDrawOp(theOp)
409 {
410 m_gdiIndex = theGdiIndex;
411 m_image = theImage;
412 m_mode = theMode;
413 }
414
415 void wxOpSetGDI::Do(wxDC& dc, double WXUNUSED(xoffset), double WXUNUSED(yoffset))
416 {
417 switch (m_op)
418 {
419 case DRAWOP_SET_PEN:
420 {
421 // Check for overriding this operation for outline
422 // colour
423 if (m_image->m_outlineColours.Member((wxObject *)m_gdiIndex))
424 {
425 if (m_image->m_outlinePen)
426 dc.SetPen(* m_image->m_outlinePen);
427 }
428 else
429 {
430 wxNode *node = m_image->m_gdiObjects.Item(m_gdiIndex);
431 if (node)
432 {
433 wxPen *pen = (wxPen *)node->GetData();
434 if (pen)
435 dc.SetPen(* pen);
436 }
437 }
438 break;
439 }
440 case DRAWOP_SET_BRUSH:
441 {
442 // Check for overriding this operation for outline or fill
443 // colour
444 if (m_image->m_outlineColours.Member((wxObject *)m_gdiIndex))
445 {
446 // Need to construct a brush to match the outline pen's colour
447 if (m_image->m_outlinePen)
448 {
449 wxBrush *br = wxTheBrushList->FindOrCreateBrush(m_image->m_outlinePen->GetColour(), wxSOLID);
450 if (br)
451 dc.SetBrush(* br);
452 }
453 }
454 else if (m_image->m_fillColours.Member((wxObject *)m_gdiIndex))
455 {
456 if (m_image->m_fillBrush)
457 {
458 dc.SetBrush(* m_image->m_fillBrush);
459 }
460 }
461 else
462 {
463 wxNode *node = m_image->m_gdiObjects.Item(m_gdiIndex);
464 if (node)
465 {
466 wxBrush *brush = (wxBrush *)node->GetData();
467 if (brush)
468 dc.SetBrush(* brush);
469 }
470 }
471 break;
472 }
473 case DRAWOP_SET_FONT:
474 {
475 wxNode *node = m_image->m_gdiObjects.Item(m_gdiIndex);
476 if (node)
477 {
478 wxFont *font = (wxFont *)node->GetData();
479 if (font)
480 dc.SetFont(* font);
481 }
482 break;
483 }
484 case DRAWOP_SET_TEXT_COLOUR:
485 {
486 wxColour col(m_r,m_g,m_b);
487 dc.SetTextForeground(col);
488 break;
489 }
490 case DRAWOP_SET_BK_COLOUR:
491 {
492 wxColour col(m_r,m_g,m_b);
493 dc.SetTextBackground(col);
494 break;
495 }
496 case DRAWOP_SET_BK_MODE:
497 {
498 dc.SetBackgroundMode(m_mode);
499 break;
500 }
501 default:
502 break;
503 }
504 }
505
506 wxDrawOp *wxOpSetGDI::Copy(wxPseudoMetaFile *newImage)
507 {
508 wxOpSetGDI *newOp = new wxOpSetGDI(m_op, newImage, m_gdiIndex, m_mode);
509 newOp->m_r = m_r;
510 newOp->m_g = m_g;
511 newOp->m_b = m_b;
512 return newOp;
513 }
514
515 #if wxUSE_PROLOGIO
516 wxExpr *wxOpSetGDI::WriteExpr(wxPseudoMetaFile *WXUNUSED(image))
517 {
518 wxExpr *expr = new wxExpr(wxExprList);
519 expr->Append(new wxExpr((long)m_op));
520 switch (m_op)
521 {
522 case DRAWOP_SET_PEN:
523 case DRAWOP_SET_BRUSH:
524 case DRAWOP_SET_FONT:
525 {
526 expr->Append(new wxExpr((long)m_gdiIndex));
527 break;
528 }
529 case DRAWOP_SET_TEXT_COLOUR:
530 case DRAWOP_SET_BK_COLOUR:
531 {
532 expr->Append(new wxExpr((long)m_r));
533 expr->Append(new wxExpr((long)m_g));
534 expr->Append(new wxExpr((long)m_b));
535 break;
536 }
537 case DRAWOP_SET_BK_MODE:
538 {
539 expr->Append(new wxExpr((long)m_mode));
540 break;
541 }
542 default:
543 break;
544 }
545 return expr;
546 }
547
548 void wxOpSetGDI::ReadExpr(wxPseudoMetaFile *WXUNUSED(image), wxExpr *expr)
549 {
550 switch (m_op)
551 {
552 case DRAWOP_SET_PEN:
553 case DRAWOP_SET_BRUSH:
554 case DRAWOP_SET_FONT:
555 {
556 m_gdiIndex = (int)expr->Nth(1)->IntegerValue();
557 break;
558 }
559 case DRAWOP_SET_TEXT_COLOUR:
560 case DRAWOP_SET_BK_COLOUR:
561 {
562 m_r = (unsigned char)expr->Nth(1)->IntegerValue();
563 m_g = (unsigned char)expr->Nth(2)->IntegerValue();
564 m_b = (unsigned char)expr->Nth(3)->IntegerValue();
565 break;
566 }
567 case DRAWOP_SET_BK_MODE:
568 {
569 m_mode = (int)expr->Nth(1)->IntegerValue();
570 break;
571 }
572 default:
573 break;
574 }
575 }
576 #endif
577
578 /*
579 * Set/destroy clipping
580 *
581 */
582
583 wxOpSetClipping::wxOpSetClipping(int theOp, double theX1, double theY1,
584 double theX2, double theY2):wxDrawOp(theOp)
585 {
586 m_x1 = theX1;
587 m_y1 = theY1;
588 m_x2 = theX2;
589 m_y2 = theY2;
590 }
591
592 wxDrawOp *wxOpSetClipping::Copy(wxPseudoMetaFile *WXUNUSED(newImage))
593 {
594 wxOpSetClipping *newOp = new wxOpSetClipping(m_op, m_x1, m_y1, m_x2, m_y2);
595 return newOp;
596 }
597
598 void wxOpSetClipping::Do(wxDC& dc, double xoffset, double yoffset)
599 {
600 switch (m_op)
601 {
602 case DRAWOP_SET_CLIPPING_RECT:
603 {
604 dc.SetClippingRegion((long)(m_x1 + xoffset), (long)(m_y1 + yoffset), (long)(m_x2 + xoffset), (long)(m_y2 + yoffset));
605 break;
606 }
607 case DRAWOP_DESTROY_CLIPPING_RECT:
608 {
609 dc.DestroyClippingRegion();
610 break;
611 }
612 default:
613 break;
614 }
615 }
616
617 void wxOpSetClipping::Scale(double xScale, double yScale)
618 {
619 m_x1 *= xScale;
620 m_y1 *= yScale;
621 m_x2 *= xScale;
622 m_y2 *= yScale;
623 }
624
625 void wxOpSetClipping::Translate(double x, double y)
626 {
627 m_x1 += x;
628 m_y1 += y;
629 }
630
631 #if wxUSE_PROLOGIO
632 wxExpr *wxOpSetClipping::WriteExpr(wxPseudoMetaFile *WXUNUSED(image))
633 {
634 wxExpr *expr = new wxExpr(wxExprList);
635 expr->Append(new wxExpr((long)m_op));
636 switch (m_op)
637 {
638 case DRAWOP_SET_CLIPPING_RECT:
639 {
640 expr->Append(new wxExpr(m_x1));
641 expr->Append(new wxExpr(m_y1));
642 expr->Append(new wxExpr(m_x2));
643 expr->Append(new wxExpr(m_y2));
644 break;
645 }
646 default:
647 break;
648 }
649 return expr;
650 }
651
652 void wxOpSetClipping::ReadExpr(wxPseudoMetaFile *WXUNUSED(image), wxExpr *expr)
653 {
654 switch (m_op)
655 {
656 case DRAWOP_SET_CLIPPING_RECT:
657 {
658 m_x1 = expr->Nth(1)->RealValue();
659 m_y1 = expr->Nth(2)->RealValue();
660 m_x2 = expr->Nth(3)->RealValue();
661 m_y2 = expr->Nth(4)->RealValue();
662 break;
663 }
664 default:
665 break;
666 }
667 }
668 #endif
669
670 /*
671 * Draw line, rectangle, rounded rectangle, ellipse, point, arc, text
672 *
673 */
674
675 wxOpDraw::wxOpDraw(int theOp, double theX1, double theY1, double theX2, double theY2,
676 double theRadius, const wxString& s) : wxDrawOp(theOp)
677 {
678 m_x1 = theX1;
679 m_y1 = theY1;
680 m_x2 = theX2;
681 m_y2 = theY2;
682 m_x3 = 0.0;
683 m_y3 = 0.0;
684 m_radius = theRadius;
685 m_textString = s;
686 }
687
688 wxOpDraw::~wxOpDraw()
689 {
690 }
691
692 wxDrawOp *wxOpDraw::Copy(wxPseudoMetaFile *WXUNUSED(newImage))
693 {
694 wxOpDraw *newOp = new wxOpDraw(m_op, m_x1, m_y1, m_x2, m_y2, m_radius, m_textString);
695 newOp->m_x3 = m_x3;
696 newOp->m_y3 = m_y3;
697 return newOp;
698 }
699
700 void wxOpDraw::Do(wxDC& dc, double xoffset, double yoffset)
701 {
702 switch (m_op)
703 {
704 case DRAWOP_DRAW_LINE:
705 {
706 dc.DrawLine(WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset), WXROUND(m_x2+xoffset), WXROUND(m_y2+yoffset));
707 break;
708 }
709 case DRAWOP_DRAW_RECT:
710 {
711 dc.DrawRectangle(WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset), WXROUND(m_x2), WXROUND(m_y2));
712 break;
713 }
714 case DRAWOP_DRAW_ROUNDED_RECT:
715 {
716 dc.DrawRoundedRectangle(WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset), WXROUND(m_x2), WXROUND(m_y2), m_radius);
717 break;
718 }
719 case DRAWOP_DRAW_ELLIPSE:
720 {
721 dc.DrawEllipse(WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset), WXROUND(m_x2), WXROUND(m_y2));
722 break;
723 }
724 case DRAWOP_DRAW_ARC:
725 {
726 dc.DrawArc(WXROUND(m_x2+xoffset), WXROUND(m_y2+yoffset),
727 WXROUND(m_x3+xoffset), WXROUND(m_y3+yoffset),
728 WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset));
729 break;
730 }
731 case DRAWOP_DRAW_ELLIPTIC_ARC:
732 {
733 const double pi = M_PI ;
734
735 // Convert back to degrees
736 dc.DrawEllipticArc(
737 WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset),
738 WXROUND(m_x2), WXROUND(m_y2),
739 WXROUND(m_x3*(360.0/(2.0*pi))), WXROUND(m_y3*(360.0/(2.0*pi))));
740 break;
741 }
742 case DRAWOP_DRAW_POINT:
743 {
744 dc.DrawPoint(WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset));
745 break;
746 }
747 case DRAWOP_DRAW_TEXT:
748 {
749 dc.DrawText(m_textString, WXROUND(m_x1+xoffset), WXROUND(m_y1+yoffset));
750 break;
751 }
752 default:
753 break;
754 }
755 }
756
757 void wxOpDraw::Scale(double scaleX, double scaleY)
758 {
759 m_x1 *= scaleX;
760 m_y1 *= scaleY;
761 m_x2 *= scaleX;
762 m_y2 *= scaleY;
763
764 if (m_op != DRAWOP_DRAW_ELLIPTIC_ARC)
765 {
766 m_x3 *= scaleX;
767 m_y3 *= scaleY;
768 }
769
770 m_radius *= scaleX;
771 }
772
773 void wxOpDraw::Translate(double x, double y)
774 {
775 m_x1 += x;
776 m_y1 += y;
777
778 switch (m_op)
779 {
780 case DRAWOP_DRAW_LINE:
781 {
782 m_x2 += x;
783 m_y2 += y;
784 break;
785 }
786 case DRAWOP_DRAW_ARC:
787 {
788 m_x2 += x;
789 m_y2 += y;
790 m_x3 += x;
791 m_y3 += y;
792 break;
793 }
794 case DRAWOP_DRAW_ELLIPTIC_ARC:
795 {
796 break;
797 }
798 default:
799 break;
800 }
801 }
802
803 void wxOpDraw::Rotate(double x, double y, double theta, double sinTheta, double cosTheta)
804 {
805 double newX1 = m_x1*cosTheta - m_y1*sinTheta + x*(1.0 - cosTheta) + y*sinTheta;
806 double newY1 = m_x1*sinTheta + m_y1*cosTheta + y*(1.0 - cosTheta) + x*sinTheta;
807
808 switch (m_op)
809 {
810 case DRAWOP_DRAW_LINE:
811 {
812 double newX2 = m_x2*cosTheta - m_y2*sinTheta + x*(1.0 - cosTheta) + y*sinTheta;
813 double newY2 = m_x2*sinTheta + m_y2*cosTheta + y*(1.0 - cosTheta) + x*sinTheta;
814
815 m_x1 = newX1;
816 m_y1 = newY1;
817 m_x2 = newX2;
818 m_y2 = newY2;
819 break;
820 }
821 case DRAWOP_DRAW_RECT:
822 case DRAWOP_DRAW_ROUNDED_RECT:
823 case DRAWOP_DRAW_ELLIPTIC_ARC:
824 {
825 // Assume only 0, 90, 180, 270 degree rotations.
826 // oldX1, oldY1 represents the top left corner. Find the
827 // bottom right, and rotate that. Then the width/height is the difference
828 // between x/y values.
829 double oldBottomRightX = m_x1 + m_x2;
830 double oldBottomRightY = m_y1 + m_y2;
831 double newBottomRightX = oldBottomRightX*cosTheta - oldBottomRightY*sinTheta + x*(1.0 - cosTheta) + y*sinTheta;
832 double newBottomRightY = oldBottomRightX*sinTheta + oldBottomRightY*cosTheta + y*(1.0 - cosTheta) + x*sinTheta;
833
834 // Now find the new top-left, bottom-right coordinates.
835 double minX = wxMin(newX1, newBottomRightX);
836 double minY = wxMin(newY1, newBottomRightY);
837 double maxX = wxMax(newX1, newBottomRightX);
838 double maxY = wxMax(newY1, newBottomRightY);
839
840 m_x1 = minX;
841 m_y1 = minY;
842 m_x2 = maxX - minX; // width
843 m_y2 = maxY - minY; // height
844
845 if (m_op == DRAWOP_DRAW_ELLIPTIC_ARC)
846 {
847 // Add rotation to angles
848 m_x3 += theta;
849 m_y3 += theta;
850 }
851
852 break;
853 }
854 case DRAWOP_DRAW_ARC:
855 {
856 double newX2 = m_x2*cosTheta - m_y2*sinTheta + x*(1.0 - cosTheta) + y*sinTheta;
857 double newY2 = m_x2*sinTheta + m_y2*cosTheta + y*(1.0 - cosTheta) + x*sinTheta;
858 double newX3 = m_x3*cosTheta - m_y3*sinTheta + x*(1.0 - cosTheta) + y*sinTheta;
859 double newY3 = m_x3*sinTheta + m_y3*cosTheta + y*(1.0 - cosTheta) + x*sinTheta;
860
861 m_x1 = newX1;
862 m_y1 = newY1;
863 m_x2 = newX2;
864 m_y2 = newY2;
865 m_x3 = newX3;
866 m_y3 = newY3;
867
868 break;
869 }
870 default:
871 break;
872 }
873 }
874
875 #if wxUSE_PROLOGIO
876 wxExpr *wxOpDraw::WriteExpr(wxPseudoMetaFile *WXUNUSED(image))
877 {
878 wxExpr *expr = new wxExpr(wxExprList);
879 expr->Append(new wxExpr((long)m_op));
880 switch (m_op)
881 {
882 case DRAWOP_DRAW_LINE:
883 case DRAWOP_DRAW_RECT:
884 case DRAWOP_DRAW_ELLIPSE:
885 {
886 expr->Append(new wxExpr(m_x1));
887 expr->Append(new wxExpr(m_y1));
888 expr->Append(new wxExpr(m_x2));
889 expr->Append(new wxExpr(m_y2));
890 break;
891 }
892 case DRAWOP_DRAW_ROUNDED_RECT:
893 {
894 expr->Append(new wxExpr(m_x1));
895 expr->Append(new wxExpr(m_y1));
896 expr->Append(new wxExpr(m_x2));
897 expr->Append(new wxExpr(m_y2));
898 expr->Append(new wxExpr(m_radius));
899 break;
900 }
901 case DRAWOP_DRAW_POINT:
902 {
903 expr->Append(new wxExpr(m_x1));
904 expr->Append(new wxExpr(m_y1));
905 break;
906 }
907 case DRAWOP_DRAW_TEXT:
908 {
909 expr->Append(new wxExpr(m_x1));
910 expr->Append(new wxExpr(m_y1));
911 expr->Append(new wxExpr(wxExprString, m_textString));
912 break;
913 }
914 case DRAWOP_DRAW_ARC:
915 case DRAWOP_DRAW_ELLIPTIC_ARC:
916 {
917 expr->Append(new wxExpr(m_x1));
918 expr->Append(new wxExpr(m_y1));
919 expr->Append(new wxExpr(m_x2));
920 expr->Append(new wxExpr(m_y2));
921 expr->Append(new wxExpr(m_x3));
922 expr->Append(new wxExpr(m_y3));
923 break;
924 }
925 default:
926 {
927 break;
928 }
929 }
930 return expr;
931 }
932
933 void wxOpDraw::ReadExpr(wxPseudoMetaFile *WXUNUSED(image), wxExpr *expr)
934 {
935 switch (m_op)
936 {
937 case DRAWOP_DRAW_LINE:
938 case DRAWOP_DRAW_RECT:
939 case DRAWOP_DRAW_ELLIPSE:
940 {
941 m_x1 = expr->Nth(1)->RealValue();
942 m_y1 = expr->Nth(2)->RealValue();
943 m_x2 = expr->Nth(3)->RealValue();
944 m_y2 = expr->Nth(4)->RealValue();
945 break;
946 }
947 case DRAWOP_DRAW_ROUNDED_RECT:
948 {
949 m_x1 = expr->Nth(1)->RealValue();
950 m_y1 = expr->Nth(2)->RealValue();
951 m_x2 = expr->Nth(3)->RealValue();
952 m_y2 = expr->Nth(4)->RealValue();
953 m_radius = expr->Nth(5)->RealValue();
954 break;
955 }
956 case DRAWOP_DRAW_POINT:
957 {
958 m_x1 = expr->Nth(1)->RealValue();
959 m_y1 = expr->Nth(2)->RealValue();
960 break;
961 }
962 case DRAWOP_DRAW_TEXT:
963 {
964 m_x1 = expr->Nth(1)->RealValue();
965 m_y1 = expr->Nth(2)->RealValue();
966 m_textString = wxString(expr->Nth(3)->StringValue());
967 break;
968 }
969 case DRAWOP_DRAW_ARC:
970 case DRAWOP_DRAW_ELLIPTIC_ARC:
971 {
972 m_x1 = expr->Nth(1)->RealValue();
973 m_y1 = expr->Nth(2)->RealValue();
974 m_x2 = expr->Nth(3)->RealValue();
975 m_y2 = expr->Nth(4)->RealValue();
976 m_x3 = expr->Nth(5)->RealValue();
977 m_y3 = expr->Nth(6)->RealValue();
978 break;
979 }
980 default:
981 {
982 break;
983 }
984 }
985 }
986 #endif
987
988 /*
989 * Draw polygon, polyline, spline
990 *
991 */
992
993 wxOpPolyDraw::wxOpPolyDraw(int theOp, int n, wxRealPoint *thePoints):wxDrawOp(theOp)
994 {
995 m_noPoints = n;
996 m_points = thePoints;
997 }
998
999 wxOpPolyDraw::~wxOpPolyDraw()
1000 {
1001 delete[] m_points;
1002 }
1003
1004 wxDrawOp *wxOpPolyDraw::Copy(wxPseudoMetaFile *WXUNUSED(newImage))
1005 {
1006 wxRealPoint *newPoints = new wxRealPoint[m_noPoints];
1007 for (int i = 0; i < m_noPoints; i++)
1008 {
1009 newPoints[i].x = m_points[i].x;
1010 newPoints[i].y = m_points[i].y;
1011 }
1012 wxOpPolyDraw *newOp = new wxOpPolyDraw(m_op, m_noPoints, newPoints);
1013 return newOp;
1014 }
1015
1016 void wxOpPolyDraw::Do(wxDC& dc, double xoffset, double yoffset)
1017 {
1018 switch (m_op)
1019 {
1020 case DRAWOP_DRAW_POLYLINE:
1021 {
1022 wxPoint *actualPoints = new wxPoint[m_noPoints];
1023 int i;
1024 for (i = 0; i < m_noPoints; i++)
1025 {
1026 actualPoints[i].x = WXROUND(m_points[i].x);
1027 actualPoints[i].y = WXROUND(m_points[i].y);
1028 }
1029
1030 dc.DrawLines(m_noPoints, actualPoints, WXROUND(xoffset), WXROUND(yoffset));
1031
1032 delete[] actualPoints;
1033 break;
1034 }
1035 case DRAWOP_DRAW_POLYGON:
1036 {
1037 wxPoint *actualPoints = new wxPoint[m_noPoints];
1038 int i;
1039 for (i = 0; i < m_noPoints; i++)
1040 {
1041 actualPoints[i].x = WXROUND(m_points[i].x);
1042 actualPoints[i].y = WXROUND(m_points[i].y);
1043 }
1044
1045 dc.DrawPolygon(m_noPoints, actualPoints, WXROUND(xoffset), WXROUND(yoffset));
1046
1047 delete[] actualPoints;
1048 break;
1049 }
1050 case DRAWOP_DRAW_SPLINE:
1051 {
1052 wxPoint *actualPoints = new wxPoint[m_noPoints];
1053 int i;
1054 for (i = 0; i < m_noPoints; i++)
1055 {
1056 actualPoints[i].x = WXROUND(m_points[i].x);
1057 actualPoints[i].y = WXROUND(m_points[i].y);
1058 }
1059
1060 dc.DrawSpline(m_noPoints, actualPoints); // no offsets in DrawSpline // , xoffset, yoffset);
1061
1062 delete[] actualPoints;
1063 break;
1064 }
1065 default:
1066 break;
1067 }
1068 }
1069
1070 void wxOpPolyDraw::Scale(double scaleX, double scaleY)
1071 {
1072 for (int i = 0; i < m_noPoints; i++)
1073 {
1074 m_points[i].x *= scaleX;
1075 m_points[i].y *= scaleY;
1076 }
1077 }
1078
1079 void wxOpPolyDraw::Translate(double x, double y)
1080 {
1081 for (int i = 0; i < m_noPoints; i++)
1082 {
1083 m_points[i].x += x;
1084 m_points[i].y += y;
1085 }
1086 }
1087
1088 void wxOpPolyDraw::Rotate(double x, double y, double WXUNUSED(theta), double sinTheta, double cosTheta)
1089 {
1090 for (int i = 0; i < m_noPoints; i++)
1091 {
1092 double x1 = m_points[i].x;
1093 double y1 = m_points[i].y;
1094 m_points[i].x = x1*cosTheta - y1*sinTheta + x*(1.0 - cosTheta) + y*sinTheta;
1095 m_points[i].y = x1*sinTheta + y1*cosTheta + y*(1.0 - cosTheta) + x*sinTheta;
1096 }
1097 }
1098
1099 #if wxUSE_PROLOGIO
1100 wxExpr *wxOpPolyDraw::WriteExpr(wxPseudoMetaFile *WXUNUSED(image))
1101 {
1102 wxExpr *expr = new wxExpr(wxExprList);
1103 expr->Append(new wxExpr((long)m_op));
1104 expr->Append(new wxExpr((long)m_noPoints));
1105
1106 // char buf1[9];
1107 wxChar buf2[5];
1108 wxChar buf3[5];
1109
1110 oglBuffer[0] = 0;
1111
1112 /*
1113 * Store each coordinate pair in a hex string to save space.
1114 * E.g. "1B9080CD". 4 hex digits per coordinate pair.
1115 *
1116 */
1117
1118 for (int i = 0; i < m_noPoints; i++)
1119 {
1120 long signedX = (long)(m_points[i].x*100.0);
1121 long signedY = (long)(m_points[i].y*100.0);
1122
1123 // Scale to 0 -> 64K
1124 long unSignedX = (long)(signedX + 32767.0);
1125 long unSignedY = (long)(signedY + 32767.0);
1126
1127 // IntToHex((unsigned int)signedX, buf2);
1128 // IntToHex((unsigned int)signedY, buf3);
1129 IntToHex((int)unSignedX, buf2);
1130 IntToHex((int)unSignedY, buf3);
1131
1132 // Don't overrun the buffer
1133 if ((i*8) < 3000)
1134 {
1135 wxStrcat(oglBuffer, buf2);
1136 wxStrcat(oglBuffer, buf3);
1137 }
1138 }
1139 expr->Append(new wxExpr(wxExprString, oglBuffer));
1140 return expr;
1141 }
1142
1143 void wxOpPolyDraw::ReadExpr(wxPseudoMetaFile *WXUNUSED(image), wxExpr *expr)
1144 {
1145 m_noPoints = (int)expr->Nth(1)->IntegerValue();
1146
1147 wxChar buf1[5];
1148 wxChar buf2[5];
1149
1150 m_points = new wxRealPoint[m_noPoints];
1151 int i = 0;
1152 int bufPtr = 0;
1153 wxString hexString = expr->Nth(2)->StringValue();
1154 while (i < m_noPoints)
1155 {
1156 buf1[0] = hexString[(size_t)bufPtr];
1157 buf1[1] = hexString[(size_t)(bufPtr + 1)];
1158 buf1[2] = hexString[(size_t)(bufPtr + 2)];
1159 buf1[3] = hexString[(size_t)(bufPtr + 3)];
1160 buf1[4] = 0;
1161
1162 buf2[0] = hexString[(size_t)(bufPtr + 4)];
1163 buf2[1] = hexString[(size_t)(bufPtr + 5)];
1164 buf2[2] = hexString[(size_t)(bufPtr + 6)];
1165 buf2[3] = hexString[(size_t)(bufPtr + 7)];
1166 buf2[4] = 0;
1167
1168 bufPtr += 8;
1169
1170 // int signedX = (signed int)HexToInt(buf1);
1171 // int signedY = (signed int)HexToInt(buf2);
1172 long unSignedX = HexToInt(buf1);
1173 long unSignedY = HexToInt(buf2);
1174 // Scale -32K -> +32K
1175 long signedX = unSignedX - 32767;
1176 long signedY = unSignedY - 32767;
1177 #if defined(__WXMSW__) && 0
1178 int testX = (signed int)unSignedX;
1179 int testY = (signed int)unSignedY;
1180 #endif
1181
1182 m_points[i].x = (double)(signedX / 100.0);
1183 m_points[i].y = (double)(signedY / 100.0);
1184
1185 i ++;
1186 }
1187 }
1188 #endif
1189
1190 // Draw an outline using the current operation.
1191 bool wxOpPolyDraw::OnDrawOutline(wxDC& dc, double x, double y, double w, double h, double oldW, double oldH)
1192 {
1193 dc.SetBrush(* wxTRANSPARENT_BRUSH);
1194
1195 // Multiply all points by proportion of new size to old size
1196 double x_proportion = (double)(fabs(w/oldW));
1197 double y_proportion = (double)(fabs(h/oldH));
1198
1199 int n = m_noPoints;
1200 wxPoint *intPoints = new wxPoint[n];
1201 int i;
1202 for (i = 0; i < n; i++)
1203 {
1204 intPoints[i].x = WXROUND (x_proportion * m_points[i].x);
1205 intPoints[i].y = WXROUND (y_proportion * m_points[i].y);
1206 }
1207 dc.DrawPolygon(n, intPoints, (long) x, (long) y);
1208 delete[] intPoints;
1209 return true;
1210 }
1211
1212 // Assume (x1, y1) is centre of box (most generally, line end at box)
1213 bool wxOpPolyDraw::GetPerimeterPoint(double x1, double y1,
1214 double x2, double y2,
1215 double *x3, double *y3,
1216 double xOffset, double yOffset,
1217 int attachmentMode)
1218 {
1219 int n = m_noPoints;
1220
1221 // First check for situation where the line is vertical,
1222 // and we would want to connect to a point on that vertical --
1223 // oglFindEndForPolyline can't cope with this (the arrow
1224 // gets drawn to the wrong place).
1225 if ((attachmentMode == ATTACHMENT_MODE_NONE) && (x1 == x2))
1226 {
1227 // Look for the point we'd be connecting to. This is
1228 // a heuristic...
1229 int i;
1230 for (i = 0; i < n; i++)
1231 {
1232 wxRealPoint *point = & (m_points[i]);
1233 if (point->x == 0.0)
1234 {
1235 if ((y2 > y1) && (point->y > 0.0))
1236 {
1237 *x3 = point->x + xOffset;
1238 *y3 = point->y + yOffset;
1239 return true;
1240 }
1241 else if ((y2 < y1) && (point->y < 0.0))
1242 {
1243 *x3 = point->x + xOffset;
1244 *y3 = point->y + yOffset;
1245 return true;
1246 }
1247 }
1248 }
1249 }
1250
1251 double *xpoints = new double[n];
1252 double *ypoints = new double[n];
1253
1254 for (int i = 0; i < n; i++)
1255 {
1256 wxRealPoint *point = & (m_points[i]);
1257 xpoints[i] = point->x + xOffset;
1258 ypoints[i] = point->y + yOffset;
1259 }
1260
1261 oglFindEndForPolyline(n, xpoints, ypoints,
1262 x1, y1, x2, y2, x3, y3);
1263
1264 delete[] xpoints;
1265 delete[] ypoints;
1266
1267 return true;
1268 }
1269
1270
1271 /*
1272 * Utilities
1273 *
1274 */
1275
1276 static char hexArray[] = {
1277 _T('0'), _T('1'), _T('2'), _T('3'), _T('4'), _T('5'), _T('6'), _T('7'),
1278 _T('8'), _T('9'), _T('A'), _T('B'), _T('C'), _T('D'), _T('E'), _T('F') };
1279
1280 // Convert unsigned 16-bit integer to 4-character hex string
1281 static void IntToHex(unsigned int dec, wxChar *buf)
1282 {
1283 int digit1 = (int)(dec/4096);
1284 int digit2 = (int)((dec - (digit1*4096))/256);
1285 int digit3 = (int)((dec - (digit1*4096) - (digit2*256))/16);
1286 int digit4 = dec - (digit1*4096 + digit2*256 + digit3*16);
1287
1288 buf[0] = hexArray[digit1];
1289 buf[1] = hexArray[digit2];
1290 buf[2] = hexArray[digit3];
1291 buf[3] = hexArray[digit4];
1292 buf[4] = 0;
1293 }
1294
1295 // One hex digit to decimal number
1296 static int HexToInt1(wxChar hex)
1297 {
1298 switch (hex)
1299 {
1300 case _T('0'):
1301 return 0;
1302 case _T('1'):
1303 return 1;
1304 case _T('2'):
1305 return 2;
1306 case _T('3'):
1307 return 3;
1308 case _T('4'):
1309 return 4;
1310 case _T('5'):
1311 return 5;
1312 case _T('6'):
1313 return 6;
1314 case _T('7'):
1315 return 7;
1316 case _T('8'):
1317 return 8;
1318 case _T('9'):
1319 return 9;
1320 case _T('A'):
1321 return 10;
1322 case _T('B'):
1323 return 11;
1324 case _T('C'):
1325 return 12;
1326 case _T('D'):
1327 return 13;
1328 case _T('E'):
1329 return 14;
1330 case _T('F'):
1331 return 15;
1332 }
1333
1334 return 0;
1335 }
1336
1337 // 4-digit hex string to unsigned integer
1338 static unsigned long HexToInt(wxChar *buf)
1339 {
1340 long d1 = (long)(HexToInt1(buf[0])*4096.0) ;
1341 long d2 = (long)(HexToInt1(buf[1])*256.0) ;
1342 long d3 = (long)(HexToInt1(buf[2])*16.0) ;
1343 long d4 = (long)(HexToInt1(buf[3])) ;
1344 unsigned long n = (long)(d1 + d2 + d3 + d4) ;
1345 return n;
1346 }
1347
1348 /*
1349 * wxPseudo meta-file
1350 *
1351 */
1352
1353 IMPLEMENT_DYNAMIC_CLASS(wxPseudoMetaFile, wxObject)
1354
1355 wxPseudoMetaFile::wxPseudoMetaFile()
1356 {
1357 m_currentRotation = 0;
1358 m_rotateable = true;
1359 m_width = 0.0;
1360 m_height = 0.0;
1361 m_outlinePen = NULL;
1362 m_fillBrush = NULL;
1363 m_outlineOp = -1;
1364 }
1365
1366 wxPseudoMetaFile::wxPseudoMetaFile(wxPseudoMetaFile& mf):wxObject()
1367 {
1368 mf.Copy(*this);
1369 }
1370
1371 wxPseudoMetaFile::~wxPseudoMetaFile()
1372 {
1373 Clear();
1374 }
1375
1376 void wxPseudoMetaFile::Clear()
1377 {
1378 wxNode *node = m_ops.GetFirst();
1379 while (node)
1380 {
1381 wxDrawOp *op = (wxDrawOp *)node->GetData();
1382 delete op;
1383 node = node->GetNext();
1384 }
1385 m_ops.Clear();
1386 m_gdiObjects.Clear();
1387 m_outlineColours.Clear();
1388 m_fillColours.Clear();
1389 m_outlineOp = -1;
1390 }
1391
1392 void wxPseudoMetaFile::Draw(wxDC& dc, double xoffset, double yoffset)
1393 {
1394 wxNode *node = m_ops.GetFirst();
1395 while (node)
1396 {
1397 wxDrawOp *op = (wxDrawOp *)node->GetData();
1398 op->Do(dc, xoffset, yoffset);
1399 node = node->GetNext();
1400 }
1401 }
1402
1403 void wxPseudoMetaFile::Scale(double sx, double sy)
1404 {
1405 wxNode *node = m_ops.GetFirst();
1406 while (node)
1407 {
1408 wxDrawOp *op = (wxDrawOp *)node->GetData();
1409 op->Scale(sx, sy);
1410 node = node->GetNext();
1411 }
1412 m_width *= sx;
1413 m_height *= sy;
1414 }
1415
1416 void wxPseudoMetaFile::Translate(double x, double y)
1417 {
1418 wxNode *node = m_ops.GetFirst();
1419 while (node)
1420 {
1421 wxDrawOp *op = (wxDrawOp *)node->GetData();
1422 op->Translate(x, y);
1423 node = node->GetNext();
1424 }
1425 }
1426
1427 void wxPseudoMetaFile::Rotate(double x, double y, double theta)
1428 {
1429 double theta1 = theta-m_currentRotation;
1430 if (theta1 == 0.0) return;
1431 double cosTheta = (double)cos(theta1);
1432 double sinTheta = (double)sin(theta1);
1433
1434 wxNode *node = m_ops.GetFirst();
1435 while (node)
1436 {
1437 wxDrawOp *op = (wxDrawOp *)node->GetData();
1438 op->Rotate(x, y, theta, sinTheta, cosTheta);
1439 node = node->GetNext();
1440 }
1441 m_currentRotation = theta;
1442 }
1443
1444 #if wxUSE_PROLOGIO
1445 void wxPseudoMetaFile::WriteAttributes(wxExpr *clause, int whichAngle)
1446 {
1447 wxString widthStr;
1448 widthStr.Printf(wxT("meta_width%d"), whichAngle);
1449
1450 wxString heightStr;
1451 heightStr.Printf(wxT("meta_height%d"), whichAngle);
1452
1453 wxString outlineStr;
1454 outlineStr.Printf(wxT("outline_op%d"), whichAngle);
1455
1456 wxString rotateableStr;
1457 rotateableStr.Printf(wxT("meta_rotateable%d"), whichAngle);
1458
1459 // Write width and height
1460 clause->AddAttributeValue(widthStr, m_width);
1461 clause->AddAttributeValue(heightStr, m_height);
1462 clause->AddAttributeValue(rotateableStr, (long)m_rotateable);
1463 clause->AddAttributeValue(outlineStr, (long)m_outlineOp);
1464
1465 // Write GDI objects
1466 wxChar buf[50];
1467 int i = 1;
1468 wxNode *node = m_gdiObjects.GetFirst();
1469 while (node)
1470 {
1471 wxSprintf(buf, _T("gdi%d_%d"), whichAngle, i);
1472 wxObject *obj = (wxObject *)node->GetData();
1473 wxExpr *expr = NULL;
1474 if (obj)
1475 {
1476 if (obj->IsKindOf(CLASSINFO(wxPen)))
1477 {
1478 wxPen *thePen = (wxPen *)obj;
1479 expr = new wxExpr(wxExprList);
1480 expr->Append(new wxExpr((long)gyTYPE_PEN));
1481 expr->Append(new wxExpr((long)thePen->GetWidth()));
1482 expr->Append(new wxExpr((long)thePen->GetStyle()));
1483 expr->Append(new wxExpr((long)thePen->GetColour().Red()));
1484 expr->Append(new wxExpr((long)thePen->GetColour().Green()));
1485 expr->Append(new wxExpr((long)thePen->GetColour().Blue()));
1486 }
1487 else if (obj->IsKindOf(CLASSINFO(wxBrush)))
1488 {
1489 wxBrush *theBrush = (wxBrush *)obj;
1490 expr = new wxExpr(wxExprList);
1491 expr->Append(new wxExpr((long)gyTYPE_BRUSH));
1492 expr->Append(new wxExpr((long)theBrush->GetStyle()));
1493 expr->Append(new wxExpr((long)theBrush->GetColour().Red()));
1494 expr->Append(new wxExpr((long)theBrush->GetColour().Green()));
1495 expr->Append(new wxExpr((long)theBrush->GetColour().Blue()));
1496 }
1497 else if (obj->IsKindOf(CLASSINFO(wxFont)))
1498 {
1499 wxFont *theFont = (wxFont *)obj;
1500 expr = new wxExpr(wxExprList);
1501 expr->Append(new wxExpr((long)gyTYPE_FONT));
1502 expr->Append(new wxExpr((long)theFont->GetPointSize()));
1503 expr->Append(new wxExpr((long)theFont->GetFamily()));
1504 expr->Append(new wxExpr((long)theFont->GetStyle()));
1505 expr->Append(new wxExpr((long)theFont->GetWeight()));
1506 expr->Append(new wxExpr((long)theFont->GetUnderlined()));
1507 }
1508 }
1509 else
1510 {
1511 // If no recognised GDI object, append a place holder anyway.
1512 expr = new wxExpr(wxExprList);
1513 expr->Append(new wxExpr((long)0));
1514 }
1515
1516 if (expr)
1517 {
1518 clause->AddAttributeValue(buf, expr);
1519 i ++;
1520 }
1521 node = node->GetNext();
1522 }
1523
1524 // Write drawing operations
1525 i = 1;
1526 node = m_ops.GetFirst();
1527 while (node)
1528 {
1529 wxSprintf(buf, _T("op%d_%d"), whichAngle, i);
1530 wxDrawOp *op = (wxDrawOp *)node->GetData();
1531 wxExpr *expr = op->WriteExpr(this);
1532 if (expr)
1533 {
1534 clause->AddAttributeValue(buf, expr);
1535 i ++;
1536 }
1537 node = node->GetNext();
1538 }
1539
1540 // Write outline and fill GDI op lists (if any)
1541 if (m_outlineColours.GetCount() > 0)
1542 {
1543 wxExpr *outlineExpr = new wxExpr(wxExprList);
1544 node = m_outlineColours.GetFirst();
1545 while (node)
1546 {
1547 outlineExpr->Append(new wxExpr((long)node->GetData()));
1548 node = node->GetNext();
1549 }
1550 wxString outlineObjectsStr;
1551 outlineObjectsStr.Printf(wxT("outline_objects%d"), whichAngle);
1552
1553 clause->AddAttributeValue(outlineObjectsStr, outlineExpr);
1554 }
1555 if (m_fillColours.GetCount() > 0)
1556 {
1557 wxExpr *fillExpr = new wxExpr(wxExprList);
1558 node = m_fillColours.GetFirst();
1559 while (node)
1560 {
1561 fillExpr->Append(new wxExpr((long)node->GetData()));
1562 node = node->GetNext();
1563 }
1564 wxString fillObjectsStr;
1565 fillObjectsStr.Printf(wxT("fill_objects%d"), whichAngle);
1566
1567 clause->AddAttributeValue(fillObjectsStr, fillExpr);
1568 }
1569
1570 }
1571
1572 void wxPseudoMetaFile::ReadAttributes(wxExpr *clause, int whichAngle)
1573 {
1574 wxString widthStr;
1575 widthStr.Printf(wxT("meta_width%d"), whichAngle);
1576
1577 wxString heightStr;
1578 heightStr.Printf(wxT("meta_height%d"), whichAngle);
1579
1580 wxString outlineStr;
1581 outlineStr.Printf(wxT("outline_op%d"), whichAngle);
1582
1583 wxString rotateableStr;
1584 rotateableStr.Printf(wxT("meta_rotateable%d"), whichAngle);
1585
1586 clause->GetAttributeValue(widthStr, m_width);
1587 clause->GetAttributeValue(heightStr, m_height);
1588 clause->GetAttributeValue(outlineStr, m_outlineOp);
1589
1590 int iVal = (int) m_rotateable;
1591 clause->GetAttributeValue(rotateableStr, iVal);
1592 m_rotateable = (iVal != 0);
1593
1594 // Read GDI objects
1595 wxChar buf[50];
1596 int i = 1;
1597 bool keepGoing = true;
1598 while (keepGoing)
1599 {
1600 wxSprintf(buf, _T("gdi%d_%d"), whichAngle, i);
1601 wxExpr *expr = NULL;
1602 clause->GetAttributeValue(buf, &expr);
1603 if (!expr)
1604 {
1605 keepGoing = false;
1606 }
1607 else
1608 {
1609 wxExpr *idExpr = expr->Nth(0);
1610 switch (idExpr->IntegerValue())
1611 {
1612 case gyTYPE_PEN:
1613 {
1614 int penWidth = (int)expr->Nth(1)->IntegerValue();
1615 int penStyle = (int)expr->Nth(2)->IntegerValue();
1616 unsigned char penRed = (unsigned char)expr->Nth(3)->IntegerValue();
1617 unsigned char penGreen = (unsigned char)expr->Nth(4)->IntegerValue();
1618 unsigned char penBlue = (unsigned char)expr->Nth(5)->IntegerValue();
1619 wxColour col(penRed, penGreen, penBlue);
1620 wxPen *p = wxThePenList->FindOrCreatePen(col, penWidth, penStyle);
1621 if (!p)
1622 p = wxBLACK_PEN;
1623 m_gdiObjects.Append(p);
1624 break;
1625 }
1626 case gyTYPE_BRUSH:
1627 {
1628 int brushStyle = (int)expr->Nth(1)->IntegerValue();
1629 unsigned char brushRed = (unsigned char)expr->Nth(2)->IntegerValue();
1630 unsigned char brushGreen = (unsigned char)expr->Nth(3)->IntegerValue();
1631 unsigned char brushBlue = (unsigned char)expr->Nth(4)->IntegerValue();
1632 wxColour col(brushRed, brushGreen, brushBlue);
1633 wxBrush *b = wxTheBrushList->FindOrCreateBrush(col, brushStyle);
1634 if (!b)
1635 b = wxWHITE_BRUSH;
1636 m_gdiObjects.Append(b);
1637 break;
1638 }
1639 case gyTYPE_FONT:
1640 {
1641 int fontPointSize = (int)expr->Nth(1)->IntegerValue();
1642 int fontFamily = (int)expr->Nth(2)->IntegerValue();
1643 int fontStyle = (int)expr->Nth(3)->IntegerValue();
1644 int fontWeight = (int)expr->Nth(4)->IntegerValue();
1645 int fontUnderlined = (int)expr->Nth(5)->IntegerValue();
1646 m_gdiObjects.Append(wxTheFontList->FindOrCreateFont(fontPointSize,
1647 fontFamily, fontStyle, fontWeight, (fontUnderlined != 0)));
1648 break;
1649 }
1650 default:
1651 {
1652 // Place holder
1653 m_gdiObjects.Append(NULL);
1654 break;
1655 }
1656 }
1657 i ++;
1658 }
1659 }
1660
1661 // Now read in the operations
1662 keepGoing = true;
1663 i = 1;
1664 while (keepGoing)
1665 {
1666 wxSprintf(buf, _T("op%d_%d"), whichAngle, i);
1667 wxExpr *expr = NULL;
1668 clause->GetAttributeValue(buf, &expr);
1669 if (!expr)
1670 {
1671 keepGoing = false;
1672 }
1673 else
1674 {
1675 wxExpr *idExpr = expr->Nth(0);
1676 int opId = (int)idExpr->IntegerValue();
1677 switch (opId)
1678 {
1679 case DRAWOP_SET_PEN:
1680 case DRAWOP_SET_BRUSH:
1681 case DRAWOP_SET_FONT:
1682 case DRAWOP_SET_TEXT_COLOUR:
1683 case DRAWOP_SET_BK_COLOUR:
1684 case DRAWOP_SET_BK_MODE:
1685 {
1686 wxOpSetGDI *theOp = new wxOpSetGDI(opId, this, 0);
1687 theOp->ReadExpr(this, expr);
1688 m_ops.Append(theOp);
1689 break;
1690 }
1691
1692 case DRAWOP_SET_CLIPPING_RECT:
1693 case DRAWOP_DESTROY_CLIPPING_RECT:
1694 {
1695 wxOpSetClipping *theOp = new wxOpSetClipping(opId, 0.0, 0.0, 0.0, 0.0);
1696 theOp->ReadExpr(this, expr);
1697 m_ops.Append(theOp);
1698 break;
1699 }
1700
1701 case DRAWOP_DRAW_LINE:
1702 case DRAWOP_DRAW_RECT:
1703 case DRAWOP_DRAW_ROUNDED_RECT:
1704 case DRAWOP_DRAW_ELLIPSE:
1705 case DRAWOP_DRAW_POINT:
1706 case DRAWOP_DRAW_ARC:
1707 case DRAWOP_DRAW_TEXT:
1708 {
1709 wxOpDraw *theOp = new wxOpDraw(opId, 0.0, 0.0, 0.0, 0.0);
1710 theOp->ReadExpr(this, expr);
1711 m_ops.Append(theOp);
1712 break;
1713 }
1714 case DRAWOP_DRAW_SPLINE:
1715 case DRAWOP_DRAW_POLYLINE:
1716 case DRAWOP_DRAW_POLYGON:
1717 {
1718 wxOpPolyDraw *theOp = new wxOpPolyDraw(opId, 0, NULL);
1719 theOp->ReadExpr(this, expr);
1720 m_ops.Append(theOp);
1721 break;
1722 }
1723 default:
1724 break;
1725 }
1726 }
1727 i ++;
1728 }
1729
1730 wxString outlineObjectsStr;
1731 outlineObjectsStr.Printf(wxT("outline_objects%d"), whichAngle);
1732
1733 // Now read in the list of outline and fill operations, if any
1734 wxExpr *expr1 = clause->AttributeValue(outlineObjectsStr);
1735 if (expr1)
1736 {
1737 wxExpr *eachExpr = expr1->GetFirst();
1738 while (eachExpr)
1739 {
1740 m_outlineColours.Append((wxObject *)eachExpr->IntegerValue());
1741 eachExpr = eachExpr->GetNext();
1742 }
1743 }
1744
1745 wxString fillObjectsStr;
1746 fillObjectsStr.Printf(wxT("fill_objects%d"), whichAngle);
1747
1748 expr1 = clause->AttributeValue(fillObjectsStr);
1749 if (expr1)
1750 {
1751 wxExpr *eachExpr = expr1->GetFirst();
1752 while (eachExpr)
1753 {
1754 m_fillColours.Append((wxObject *)eachExpr->IntegerValue());
1755 eachExpr = eachExpr->GetNext();
1756 }
1757 }
1758 }
1759 #endif
1760
1761 // Does the copying for this object
1762 void wxPseudoMetaFile::Copy(wxPseudoMetaFile& copy)
1763 {
1764 copy.Clear();
1765
1766 copy.m_currentRotation = m_currentRotation;
1767 copy.m_width = m_width;
1768 copy.m_height = m_height;
1769 copy.m_rotateable = m_rotateable;
1770 copy.m_fillBrush = m_fillBrush;
1771 copy.m_outlinePen = m_outlinePen;
1772 copy.m_outlineOp = m_outlineOp;
1773
1774 // Copy the GDI objects
1775 wxNode *node = m_gdiObjects.GetFirst();
1776 while (node)
1777 {
1778 wxObject *obj = (wxObject *)node->GetData();
1779 copy.m_gdiObjects.Append(obj);
1780 node = node->GetNext();
1781 }
1782
1783 // Copy the operations
1784 node = m_ops.GetFirst();
1785 while (node)
1786 {
1787 wxDrawOp *op = (wxDrawOp *)node->GetData();
1788 copy.m_ops.Append(op->Copy(&copy));
1789 node = node->GetNext();
1790 }
1791
1792 // Copy the outline/fill operations
1793 node = m_outlineColours.GetFirst();
1794 while (node)
1795 {
1796 copy.m_outlineColours.Append((wxObject *)node->GetData());
1797 node = node->GetNext();
1798 }
1799 node = m_fillColours.GetFirst();
1800 while (node)
1801 {
1802 copy.m_fillColours.Append((wxObject *)node->GetData());
1803 node = node->GetNext();
1804 }
1805 }
1806
1807 /*
1808 * Pass size of existing image; scale height to
1809 * fit width and return new width and height.
1810 *
1811 */
1812
1813 bool wxPseudoMetaFile::LoadFromMetaFile(const wxString& filename, double *rwidth, double *rheight)
1814 {
1815 if (!wxFileExists(filename))
1816 return false;
1817
1818 wxXMetaFile *metaFile = new wxXMetaFile;
1819
1820 if (!metaFile->ReadFile(filename))
1821 {
1822 delete metaFile;
1823 return false;
1824 }
1825
1826 double lastX = 0.0;
1827 double lastY = 0.0;
1828
1829 // Convert from metafile records to wxDrawnShape records
1830 wxNode *node = metaFile->metaRecords.GetFirst();
1831 while (node)
1832 {
1833 wxMetaRecord *record = (wxMetaRecord *)node->GetData();
1834 switch (record->metaFunction)
1835 {
1836 case META_SETBKCOLOR:
1837 {
1838 wxOpSetGDI *op = new wxOpSetGDI(DRAWOP_SET_BK_COLOUR, this, 0);
1839 op->m_r = (unsigned char)record->param1;
1840 op->m_g = (unsigned char)record->param2;
1841 op->m_b = (unsigned char)record->param3;
1842 m_ops.Append(op);
1843 break;
1844 }
1845 case META_SETBKMODE:
1846 {
1847 wxOpSetGDI *op = new wxOpSetGDI(DRAWOP_SET_BK_MODE, this, 0, (int)record->param1);
1848 m_ops.Append(op);
1849 break;
1850 }
1851 case META_SETMAPMODE:
1852 {
1853 break;
1854 }
1855 // case META_SETROP2:
1856 // case META_SETRELABS:
1857 // case META_SETPOLYFILLMODE:
1858 // case META_SETSTRETCHBLTMODE:
1859 // case META_SETTEXTCHAREXTRA:
1860 case META_SETTEXTCOLOR:
1861 {
1862 wxOpSetGDI *op = new wxOpSetGDI(DRAWOP_SET_TEXT_COLOUR, this, 0);
1863 op->m_r = (unsigned char)record->param1;
1864 op->m_g = (unsigned char)record->param2;
1865 op->m_b = (unsigned char)record->param3;
1866 m_ops.Append(op);
1867 break;
1868 }
1869 // case META_SETTEXTJUSTIFICATION:
1870 // case META_SETWINDOWORG:
1871 // case META_SETWINDOWEXT:
1872 // case META_SETVIEWPORTORG:
1873 // case META_SETVIEWPORTEXT:
1874 // case META_OFFSETWINDOWORG:
1875 // case META_SCALEWINDOWEXT:
1876 // case META_OFFSETVIEWPORTORG:
1877 // case META_SCALEVIEWPORTEXT:
1878 case META_LINETO:
1879 {
1880 wxOpDraw *op = new wxOpDraw(DRAWOP_DRAW_LINE, (double)lastX, (double)lastY,
1881 (double)record->param1, (double)record->param2);
1882 m_ops.Append(op);
1883 break;
1884 }
1885 case META_MOVETO:
1886 {
1887 lastX = (double)record->param1;
1888 lastY = (double)record->param2;
1889 break;
1890 }
1891 case META_EXCLUDECLIPRECT:
1892 {
1893 /*
1894 wxMetaRecord *rec = new wxMetaRecord(META_EXCLUDECLIPRECT);
1895 rec->param4 = getshort(handle); // m_y2
1896 rec->param3 = getshort(handle); // x2
1897 rec->param2 = getshort(handle); // y1
1898 rec->param1 = getshort(handle); // x1
1899 */
1900 break;
1901 }
1902 case META_INTERSECTCLIPRECT:
1903 {
1904 /*
1905 rec->param4 = getshort(handle); // m_y2
1906 rec->param3 = getshort(handle); // x2
1907 rec->param2 = getshort(handle); // y1
1908 rec->param1 = getshort(handle); // x1
1909 */
1910 break;
1911 }
1912 // case META_ARC: // DO!!!
1913 case META_ELLIPSE:
1914 {
1915 wxOpDraw *op = new wxOpDraw(DRAWOP_DRAW_ELLIPSE,
1916 (double)record->param1, (double)record->param2,
1917 (double)(record->param3 - record->param1),
1918 (double)(record->param4 - record->param2));
1919 m_ops.Append(op);
1920 break;
1921 }
1922 // case META_FLOODFILL:
1923 // case META_PIE: // DO!!!
1924 case META_RECTANGLE:
1925 {
1926 wxOpDraw *op = new wxOpDraw(DRAWOP_DRAW_RECT,
1927 (double)record->param1, (double)record->param2,
1928 (double)(record->param3 - record->param1),
1929 (double)(record->param4 - record->param2));
1930 m_ops.Append(op);
1931 break;
1932 }
1933 case META_ROUNDRECT:
1934 {
1935 wxOpDraw *op = new wxOpDraw(DRAWOP_DRAW_ROUNDED_RECT,
1936 (double)record->param1, (double)record->param2,
1937 (double)(record->param3 - record->param1),
1938 (double)(record->param4 - record->param2), (double)record->param5);
1939 m_ops.Append(op);
1940 break;
1941 }
1942 // case META_PATBLT:
1943 // case META_SAVEDC:
1944 case META_SETPIXEL:
1945 {
1946 wxOpDraw *op = new wxOpDraw(DRAWOP_DRAW_POINT,
1947 (double)record->param1, (double)record->param2,
1948 0.0, 0.0);
1949
1950 // SHOULD SET THE COLOUR - SET PEN?
1951 // rec->param3 = getint(handle); // COLORREF
1952 m_ops.Append(op);
1953 break;
1954 }
1955 // case META_OFFSETCLIPRGN:
1956 case META_TEXTOUT:
1957 {
1958 wxOpDraw *op = new wxOpDraw(DRAWOP_DRAW_TEXT,
1959 (double)record->param1, (double)record->param2,
1960 0.0, 0.0, 0.0, record->stringParam);
1961 m_ops.Append(op);
1962 break;
1963 }
1964 // case META_BITBLT:
1965 // case META_STRETCHBLT:
1966 case META_POLYGON:
1967 {
1968 int n = (int)record->param1;
1969 wxRealPoint *newPoints = new wxRealPoint[n];
1970 for (int i = 0; i < n; i++)
1971 {
1972 newPoints[i].x = record->points[i].x;
1973 newPoints[i].y = record->points[i].y;
1974 }
1975
1976 wxOpPolyDraw *op = new wxOpPolyDraw(DRAWOP_DRAW_POLYGON, n, newPoints);
1977 m_ops.Append(op);
1978 break;
1979 }
1980 case META_POLYLINE:
1981 {
1982 int n = (int)record->param1;
1983 wxRealPoint *newPoints = new wxRealPoint[n];
1984 for (int i = 0; i < n; i++)
1985 {
1986 newPoints[i].x = record->points[i].x;
1987 newPoints[i].y = record->points[i].y;
1988 }
1989
1990 wxOpPolyDraw *op = new wxOpPolyDraw(DRAWOP_DRAW_POLYLINE, n, newPoints);
1991 m_ops.Append(op);
1992 break;
1993 }
1994 // case META_ESCAPE:
1995 // case META_RESTOREDC:
1996 // case META_FILLREGION:
1997 // case META_FRAMEREGION:
1998 // case META_INVERTREGION:
1999 // case META_PAINTREGION:
2000 // case META_SELECTCLIPREGION: // DO THIS!
2001 case META_SELECTOBJECT:
2002 {
2003 // The pen, brush etc. has already been created when the metafile
2004 // was read in, so we don't create it - we set it.
2005 wxNode *recNode = metaFile->gdiObjects.Item((int)record->param2);
2006 if (recNode)
2007 {
2008 wxMetaRecord *gdiRec = (wxMetaRecord *)recNode->GetData();
2009 if (gdiRec && (gdiRec->param1 != 0))
2010 {
2011 wxObject *obj = (wxObject *)gdiRec->param1;
2012 if (obj->IsKindOf(CLASSINFO(wxPen)))
2013 {
2014 wxOpSetGDI *op = new wxOpSetGDI(DRAWOP_SET_PEN, this, (int)record->param2);
2015 m_ops.Append(op);
2016 }
2017 else if (obj->IsKindOf(CLASSINFO(wxBrush)))
2018 {
2019 wxOpSetGDI *op = new wxOpSetGDI(DRAWOP_SET_BRUSH, this, (int)record->param2);
2020 m_ops.Append(op);
2021 }
2022 else if (obj->IsKindOf(CLASSINFO(wxFont)))
2023 {
2024 wxOpSetGDI *op = new wxOpSetGDI(DRAWOP_SET_FONT, this, (int)record->param2);
2025 m_ops.Append(op);
2026 }
2027 }
2028 }
2029 break;
2030 }
2031 // case META_SETTEXTALIGN:
2032 // case META_DRAWTEXT:
2033 // case META_CHORD:
2034 // case META_SETMAPPERFLAGS:
2035 // case META_EXTTEXTOUT:
2036 // case META_SETDIBTODEV:
2037 // case META_SELECTPALETTE:
2038 // case META_REALIZEPALETTE:
2039 // case META_ANIMATEPALETTE:
2040 // case META_SETPALENTRIES:
2041 // case META_POLYPOLYGON:
2042 // case META_RESIZEPALETTE:
2043 // case META_DIBBITBLT:
2044 // case META_DIBSTRETCHBLT:
2045 case META_DIBCREATEPATTERNBRUSH:
2046 {
2047 // Place holder
2048 m_gdiObjects.Append(NULL);
2049 break;
2050 }
2051 // case META_STRETCHDIB:
2052 // case META_EXTFLOODFILL:
2053 // case META_RESETDC:
2054 // case META_STARTDOC:
2055 // case META_STARTPAGE:
2056 // case META_ENDPAGE:
2057 // case META_ABORTDOC:
2058 // case META_ENDDOC:
2059 // case META_DELETEOBJECT: // DO!!
2060 case META_CREATEPALETTE:
2061 {
2062 // Place holder
2063 m_gdiObjects.Append(NULL);
2064 break;
2065 }
2066 case META_CREATEBRUSH:
2067 {
2068 // Place holder
2069 m_gdiObjects.Append(NULL);
2070 break;
2071 }
2072 case META_CREATEPATTERNBRUSH:
2073 {
2074 // Place holder
2075 m_gdiObjects.Append(NULL);
2076 break;
2077 }
2078 case META_CREATEPENINDIRECT:
2079 {
2080 // The pen is created when the metafile is read in.
2081 // We keep track of all the GDI objects needed for this
2082 // image so when reading the wxDrawnShape from file,
2083 // we can read in all the GDI objects, then refer
2084 // to them by an index starting from zero thereafter.
2085 m_gdiObjects.Append((wxObject *)record->param1);
2086 break;
2087 }
2088 case META_CREATEFONTINDIRECT:
2089 {
2090 m_gdiObjects.Append((wxObject *)record->param1);
2091 break;
2092 }
2093 case META_CREATEBRUSHINDIRECT:
2094 {
2095 // Don't have to do anything here: the pen is created
2096 // when the metafile is read in.
2097 m_gdiObjects.Append((wxObject *)record->param1);
2098 break;
2099 }
2100 case META_CREATEBITMAPINDIRECT:
2101 {
2102 // Place holder
2103 m_gdiObjects.Append(NULL);
2104 break;
2105 }
2106 case META_CREATEBITMAP:
2107 {
2108 // Place holder
2109 m_gdiObjects.Append(NULL);
2110 break;
2111 }
2112 case META_CREATEREGION:
2113 {
2114 // Place holder
2115 m_gdiObjects.Append(NULL);
2116 break;
2117 }
2118 default:
2119 {
2120 break;
2121 }
2122 }
2123 node = node->GetNext();
2124 }
2125 double actualWidth = (double)fabs(metaFile->right - metaFile->left);
2126 double actualHeight = (double)fabs(metaFile->bottom - metaFile->top);
2127
2128 double initialScaleX = 1.0;
2129 double initialScaleY = 1.0;
2130
2131 double xoffset, yoffset;
2132
2133 // Translate so origin is at centre of rectangle
2134 if (metaFile->bottom > metaFile->top)
2135 yoffset = - (double)((metaFile->bottom - metaFile->top)/2.0);
2136 else
2137 yoffset = - (double)((metaFile->top - metaFile->bottom)/2.0);
2138
2139 if (metaFile->right > metaFile->left)
2140 xoffset = - (double)((metaFile->right - metaFile->left)/2.0);
2141 else
2142 xoffset = - (double)((metaFile->left - metaFile->right)/2.0);
2143
2144 Translate(xoffset, yoffset);
2145
2146 // Scale to a reasonable size (take the width of this wxDrawnShape
2147 // as a guide)
2148 if (actualWidth != 0.0)
2149 {
2150 initialScaleX = (double)((*rwidth) / actualWidth);
2151 initialScaleY = initialScaleX;
2152 (*rheight) = initialScaleY*actualHeight;
2153 }
2154 Scale(initialScaleX, initialScaleY);
2155
2156 m_width = (actualWidth*initialScaleX);
2157 m_height = *rheight;
2158
2159 delete metaFile;
2160 return true;
2161 }
2162
2163 // Scale to fit size
2164 void wxPseudoMetaFile::ScaleTo(double w, double h)
2165 {
2166 double scaleX = (double)(w/m_width);
2167 double scaleY = (double)(h/m_height);
2168
2169 // Do the scaling
2170 Scale(scaleX, scaleY);
2171 }
2172
2173 void wxPseudoMetaFile::GetBounds(double *boundMinX, double *boundMinY, double *boundMaxX, double *boundMaxY)
2174 {
2175 double maxX = (double) -99999.9;
2176 double maxY = (double) -99999.9;
2177 double minX = (double) 99999.9;
2178 double minY = (double) 99999.9;
2179
2180 wxNode *node = m_ops.GetFirst();
2181 while (node)
2182 {
2183 wxDrawOp *op = (wxDrawOp *)node->GetData();
2184 switch (op->GetOp())
2185 {
2186 case DRAWOP_DRAW_LINE:
2187 case DRAWOP_DRAW_RECT:
2188 case DRAWOP_DRAW_ROUNDED_RECT:
2189 case DRAWOP_DRAW_ELLIPSE:
2190 case DRAWOP_DRAW_POINT:
2191 case DRAWOP_DRAW_TEXT:
2192 {
2193 wxOpDraw *opDraw = (wxOpDraw *)op;
2194 if (opDraw->m_x1 < minX) minX = opDraw->m_x1;
2195 if (opDraw->m_x1 > maxX) maxX = opDraw->m_x1;
2196 if (opDraw->m_y1 < minY) minY = opDraw->m_y1;
2197 if (opDraw->m_y1 > maxY) maxY = opDraw->m_y1;
2198 if (op->GetOp() == DRAWOP_DRAW_LINE)
2199 {
2200 if (opDraw->m_x2 < minX) minX = opDraw->m_x2;
2201 if (opDraw->m_x2 > maxX) maxX = opDraw->m_x2;
2202 if (opDraw->m_y2 < minY) minY = opDraw->m_y2;
2203 if (opDraw->m_y2 > maxY) maxY = opDraw->m_y2;
2204 }
2205 else if (op->GetOp() == DRAWOP_DRAW_RECT ||
2206 op->GetOp() == DRAWOP_DRAW_ROUNDED_RECT ||
2207 op->GetOp() == DRAWOP_DRAW_ELLIPSE)
2208 {
2209 if ((opDraw->m_x1 + opDraw->m_x2) < minX) minX = (opDraw->m_x1 + opDraw->m_x2);
2210 if ((opDraw->m_x1 + opDraw->m_x2) > maxX) maxX = (opDraw->m_x1 + opDraw->m_x2);
2211 if ((opDraw->m_y1 + opDraw->m_y2) < minY) minY = (opDraw->m_y1 + opDraw->m_y2);
2212 if ((opDraw->m_y1 + opDraw->m_y2) > maxY) maxY = (opDraw->m_y1 + opDraw->m_y2);
2213 }
2214 break;
2215 }
2216 case DRAWOP_DRAW_ARC:
2217 {
2218 // TODO: don't yet know how to calculate the bounding box
2219 // for an arc. So pretend it's a line; to get a correct
2220 // bounding box, draw a blank rectangle first, of the correct
2221 // size.
2222 wxOpDraw *opDraw = (wxOpDraw *)op;
2223 if (opDraw->m_x1 < minX) minX = opDraw->m_x1;
2224 if (opDraw->m_x1 > maxX) maxX = opDraw->m_x1;
2225 if (opDraw->m_y1 < minY) minY = opDraw->m_y1;
2226 if (opDraw->m_y1 > maxY) maxY = opDraw->m_y1;
2227 if (opDraw->m_x2 < minX) minX = opDraw->m_x2;
2228 if (opDraw->m_x2 > maxX) maxX = opDraw->m_x2;
2229 if (opDraw->m_y2 < minY) minY = opDraw->m_y2;
2230 if (opDraw->m_y2 > maxY) maxY = opDraw->m_y2;
2231 break;
2232 }
2233 case DRAWOP_DRAW_POLYLINE:
2234 case DRAWOP_DRAW_POLYGON:
2235 case DRAWOP_DRAW_SPLINE:
2236 {
2237 wxOpPolyDraw *poly = (wxOpPolyDraw *)op;
2238 for (int i = 0; i < poly->m_noPoints; i++)
2239 {
2240 if (poly->m_points[i].x < minX) minX = poly->m_points[i].x;
2241 if (poly->m_points[i].x > maxX) maxX = poly->m_points[i].x;
2242 if (poly->m_points[i].y < minY) minY = poly->m_points[i].y;
2243 if (poly->m_points[i].y > maxY) maxY = poly->m_points[i].y;
2244 }
2245 break;
2246 }
2247 default:
2248 break;
2249 }
2250 node = node->GetNext();
2251 }
2252
2253 *boundMinX = minX;
2254 *boundMinY = minY;
2255 *boundMaxX = maxX;
2256 *boundMaxY = maxY;
2257 /*
2258 *w = (double)fabs(maxX - minX);
2259 *h = (double)fabs(maxY - minY);
2260 */
2261 }
2262
2263 // Calculate size from current operations
2264 void wxPseudoMetaFile::CalculateSize(wxDrawnShape* shape)
2265 {
2266 double boundMinX, boundMinY, boundMaxX, boundMaxY;
2267
2268 GetBounds(& boundMinX, & boundMinY, & boundMaxX, & boundMaxY);
2269
2270 SetSize(boundMaxX - boundMinX, boundMaxY - boundMinY);
2271
2272 if (shape)
2273 {
2274 shape->SetWidth(m_width);
2275 shape->SetHeight(m_height);
2276 }
2277 }
2278
2279 // Set of functions for drawing into a pseudo metafile.
2280 // They use integers, but doubles are used internally for accuracy
2281 // when scaling.
2282
2283 void wxPseudoMetaFile::DrawLine(const wxPoint& pt1, const wxPoint& pt2)
2284 {
2285 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_LINE,
2286 (double) pt1.x, (double) pt1.y, (double) pt2.x, (double) pt2.y);
2287
2288 m_ops.Append(theOp);
2289 }
2290
2291 void wxPseudoMetaFile::DrawRectangle(const wxRect& rect)
2292 {
2293 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_RECT,
2294 (double) rect.x, (double) rect.y, (double) rect.width, (double) rect.height);
2295
2296 m_ops.Append(theOp);
2297 }
2298
2299 void wxPseudoMetaFile::DrawRoundedRectangle(const wxRect& rect, double radius)
2300 {
2301 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_ROUNDED_RECT,
2302 (double) rect.x, (double) rect.y, (double) rect.width, (double) rect.height);
2303
2304 theOp->m_radius = radius;
2305
2306 m_ops.Append(theOp);
2307 }
2308
2309 void wxPseudoMetaFile::DrawEllipse(const wxRect& rect)
2310 {
2311 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_ELLIPSE,
2312 (double) rect.x, (double) rect.y, (double) rect.width, (double) rect.height);
2313
2314 m_ops.Append(theOp);
2315 }
2316
2317 void wxPseudoMetaFile::DrawArc(const wxPoint& centrePt, const wxPoint& startPt, const wxPoint& endPt)
2318 {
2319 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_ARC,
2320 (double) centrePt.x, (double) centrePt.y, (double) startPt.x, (double) startPt.y);
2321
2322 theOp->m_x3 = (double) endPt.x;
2323 theOp->m_y3 = (double) endPt.y;
2324
2325 m_ops.Append(theOp);
2326 }
2327
2328 void wxPseudoMetaFile::DrawEllipticArc(const wxRect& rect, double startAngle, double endAngle)
2329 {
2330 const double pi = M_PI ;
2331
2332 double startAngleRadians = startAngle* (pi*2.0/360.0);
2333 double endAngleRadians = endAngle* (pi*2.0/360.0);
2334
2335 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_ELLIPTIC_ARC,
2336 (double) rect.x, (double) rect.y, (double) rect.width, (double) rect.height);
2337
2338 theOp->m_x3 = startAngleRadians;
2339 theOp->m_y3 = endAngleRadians;
2340
2341 m_ops.Append(theOp);
2342 }
2343
2344 void wxPseudoMetaFile::DrawPoint(const wxPoint& pt)
2345 {
2346 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_POINT,
2347 (double) pt.x, (double) pt.y, 0.0, 0.0);
2348
2349 m_ops.Append(theOp);
2350 }
2351
2352 void wxPseudoMetaFile::DrawText(const wxString& text, const wxPoint& pt)
2353 {
2354 wxOpDraw *theOp = new wxOpDraw(DRAWOP_DRAW_TEXT,
2355 (double) pt.x, (double) pt.y, 0.0, 0.0);
2356
2357 theOp->m_textString = text;
2358
2359 m_ops.Append(theOp);
2360 }
2361
2362 void wxPseudoMetaFile::DrawLines(int n, wxPoint pts[])
2363 {
2364 wxRealPoint* realPoints = new wxRealPoint[n];
2365 int i;
2366 for (i = 0; i < n; i++)
2367 {
2368 realPoints[i].x = pts[i].x;
2369 realPoints[i].y = pts[i].y;
2370 }
2371 wxOpPolyDraw* theOp = new wxOpPolyDraw(DRAWOP_DRAW_POLYLINE, n, realPoints);
2372 m_ops.Append(theOp);
2373 }
2374
2375 void wxPseudoMetaFile::DrawPolygon(int n, wxPoint pts[], int flags)
2376 {
2377 wxRealPoint* realPoints = new wxRealPoint[n];
2378 int i;
2379 for (i = 0; i < n; i++)
2380 {
2381 realPoints[i].x = pts[i].x;
2382 realPoints[i].y = pts[i].y;
2383 }
2384 wxOpPolyDraw* theOp = new wxOpPolyDraw(DRAWOP_DRAW_POLYGON, n, realPoints);
2385 m_ops.Append(theOp);
2386
2387 if (flags & oglMETAFLAGS_OUTLINE)
2388 m_outlineOp = (m_ops.GetCount() - 1);
2389 }
2390
2391 void wxPseudoMetaFile::DrawSpline(int n, wxPoint pts[])
2392 {
2393 wxRealPoint* realPoints = new wxRealPoint[n];
2394 int i;
2395 for (i = 0; i < n; i++)
2396 {
2397 realPoints[i].x = pts[i].x;
2398 realPoints[i].y = pts[i].y;
2399 }
2400 wxOpPolyDraw* theOp = new wxOpPolyDraw(DRAWOP_DRAW_SPLINE, n, realPoints);
2401 m_ops.Append(theOp);
2402 }
2403
2404 void wxPseudoMetaFile::SetClippingRect(const wxRect& rect)
2405 {
2406 /* wxOpSetClipping* theOp = */ new wxOpSetClipping(DRAWOP_SET_CLIPPING_RECT,
2407 (double) rect.x, (double) rect.y, (double) rect.width, (double) rect.height);
2408 }
2409
2410 void wxPseudoMetaFile::DestroyClippingRect()
2411 {
2412 wxOpSetClipping* theOp = new wxOpSetClipping(DRAWOP_DESTROY_CLIPPING_RECT,
2413 0.0, 0.0, 0.0, 0.0);
2414
2415 m_ops.Append(theOp);
2416 }
2417
2418 void wxPseudoMetaFile::SetPen(wxPen* pen, bool isOutline)
2419 {
2420 m_gdiObjects.Append(pen);
2421 int n = m_gdiObjects.GetCount();
2422
2423 wxOpSetGDI* theOp = new wxOpSetGDI(DRAWOP_SET_PEN, this, n - 1);
2424
2425 m_ops.Append(theOp);
2426
2427 if (isOutline)
2428 {
2429 m_outlineColours.Append((wxObject*) (n - 1));
2430 }
2431 }
2432
2433 void wxPseudoMetaFile::SetBrush(wxBrush* brush, bool isFill)
2434 {
2435 m_gdiObjects.Append(brush);
2436 int n = m_gdiObjects.GetCount();
2437
2438 wxOpSetGDI* theOp = new wxOpSetGDI(DRAWOP_SET_BRUSH, this, n - 1);
2439
2440 m_ops.Append(theOp);
2441
2442 if (isFill)
2443 {
2444 m_fillColours.Append((wxObject*) (n - 1));
2445 }
2446 }
2447
2448 void wxPseudoMetaFile::SetFont(wxFont* font)
2449 {
2450 m_gdiObjects.Append(font);
2451 int n = m_gdiObjects.GetCount();
2452
2453 wxOpSetGDI* theOp = new wxOpSetGDI(DRAWOP_SET_FONT, this, n - 1);
2454
2455 m_ops.Append(theOp);
2456 }
2457
2458 void wxPseudoMetaFile::SetTextColour(const wxColour& colour)
2459 {
2460 wxOpSetGDI* theOp = new wxOpSetGDI(DRAWOP_SET_TEXT_COLOUR, this, 0);
2461 theOp->m_r = colour.Red();
2462 theOp->m_g = colour.Green();
2463 theOp->m_b = colour.Blue();
2464
2465 m_ops.Append(theOp);
2466 }
2467
2468 void wxPseudoMetaFile::SetBackgroundColour(const wxColour& colour)
2469 {
2470 wxOpSetGDI* theOp = new wxOpSetGDI(DRAWOP_SET_BK_COLOUR, this, 0);
2471 theOp->m_r = colour.Red();
2472 theOp->m_g = colour.Green();
2473 theOp->m_b = colour.Blue();
2474
2475 m_ops.Append(theOp);
2476 }
2477
2478 void wxPseudoMetaFile::SetBackgroundMode(int mode)
2479 {
2480 wxOpSetGDI* theOp = new wxOpSetGDI(DRAWOP_SET_BK_MODE, this, 0, mode);
2481
2482 m_ops.Append(theOp);
2483 }
2484