]> git.saurik.com Git - wxWidgets.git/blob - contrib/src/ogl/mfutils.cpp
1fc7848fcb93d0c5e35228686f9adbbc76afcddb
[wxWidgets.git] / contrib / src / ogl / mfutils.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: mfutils.cpp
3 // Purpose: Metafile utillities
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 "mfutils.h"
14 #endif
15
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
18
19 #ifdef __BORLANDC__
20 #pragma hdrstop
21 #endif
22
23 #ifndef WX_PRECOMP
24 #include <wx/wx.h>
25 #endif
26
27 #include <wx/metafile.h>
28 #include <wx/utils.h>
29
30 #include "wx/ogl/ogl.h"
31
32 #include <stdio.h>
33
34 static char _buf[1024]; // a temp buffer to use inplace of wxBuffer, which is deprecated.
35
36 // 16-bit unsigned integer
37 static unsigned int getshort(FILE *fp)
38 {
39 int c, c1;
40 c = getc(fp); c1 = getc(fp);
41 unsigned int res = ((unsigned int) c) + (((unsigned int) c1) << 8);
42 return res;
43 }
44
45 // 16-bit signed integer
46 static int getsignedshort(FILE *fp)
47 {
48 int c, c1;
49 c = getc(fp); c1 = getc(fp);
50 #if 0
51 // this is not used value, no need to execute it
52 int testRes = ((unsigned int) c) + (((unsigned int) c1) << 8);
53 #endif
54 unsigned long res1 = ((unsigned int) c) + (((unsigned int) c1) << 8);
55 int res;
56 if (res1 > 32767)
57 res = (int)(res1 - 65536);
58 else
59 res = (int)(res1);
60 return res;
61 }
62
63 // 32-bit integer
64 static long getint(FILE *fp)
65 {
66 int c, c1, c2, c3;
67 c = getc(fp); c1 = getc(fp); c2 = getc(fp); c3 = getc(fp);
68 long res = (long)((long) c) +
69 (((long) c1) << 8) +
70 (((long) c2) << 16) +
71 (((long) c3) << 24);
72 return res;
73 }
74
75
76 /* Placeable metafile header
77 struct mfPLACEABLEHEADER {
78 DWORD key; // 32-bit
79 HANDLE hmf; // 16-bit
80 RECT bbox; // 4x16 bit
81 WORD inch; // 16-bit
82 DWORD reserved; // 32-bit
83 WORD checksum; // 16-bit
84 };
85 */
86
87 wxMetaRecord::~wxMetaRecord(void)
88 {
89 if (points) delete[] points;
90 if (stringParam) delete[] stringParam;
91 }
92
93 wxXMetaFile::wxXMetaFile(const wxChar *file)
94 {
95 ok = FALSE;
96 top = 0.0;
97 bottom = 0.0;
98 left = 0.0;
99 right = 0.0;
100
101 if (file)
102 ok = ReadFile(file);
103 }
104
105 /*
106 Handle table gdiObjects
107 ------------ ----------
108 [0] wxPen
109 [1]----param2--- wxBrush
110 [2] | wxFont
111 [3] | -> wxPen
112
113 The handle table works as follows.
114 When a GDI object is created whilst reading in the
115 metafile, the (e.g.) createpen record is added to the
116 first free entry in the handle table. The createpen
117 record's param1 is a pointer to the actual wxPen, and
118 its param2 is the index into the gdiObjects list, which only
119 grows and never shrinks (unlike the handle table.)
120
121 When SelectObject(index) is found, the index in the file
122 refers to the position in the handle table. BUT we then
123 set param2 to be the position of the wxPen in gdiObjects,
124 i.e. to param2 of the CreatePen record, itself found in
125 the handle table.
126
127 When an object is deleted, the entry in the handletable is
128 NULLed but the gdiObjects entry is not removed (no point, and
129 allows us to create all GDI objects in advance of playing the
130 metafile).
131 */
132
133
134 static wxMetaRecord *HandleTable[100];
135 static int HandleTableSize = 0;
136
137 void DeleteMetaRecordHandle(int index)
138 {
139 HandleTable[index] = NULL;
140 }
141
142 int AddMetaRecordHandle(wxMetaRecord *record)
143 {
144 for (int i = 0; i < HandleTableSize; i++)
145 if (!HandleTable[i])
146 {
147 HandleTable[i] = record;
148 return i;
149 }
150 // No free spaces in table, so append.
151
152 HandleTable[HandleTableSize] = record;
153 HandleTableSize ++;
154 return (HandleTableSize - 1);
155 }
156
157 bool wxXMetaFile::ReadFile(const wxChar *file)
158 {
159 HandleTableSize = 0;
160
161 FILE *handle = wxFopen(file, wxT("rb"));
162 if (!handle) return FALSE;
163
164 // Read placeable metafile header, if any
165 long key = getint(handle);
166
167 if (key == (long) 0x9AC6CDD7)
168 {
169 /* long hmf = */ getshort(handle);
170 int iLeft, iTop, iRight, iBottom;
171 iLeft = getsignedshort(handle);
172 iTop = getsignedshort(handle);
173 iRight = getsignedshort(handle);
174 iBottom = getsignedshort(handle);
175
176 left = (double)iLeft;
177 top = (double)iTop;
178 right = (double)iRight;
179 bottom = (double)iBottom;
180
181 /* int inch = */ getshort(handle);
182 /* long reserved = */ getint(handle);
183 /* int checksum = */ getshort(handle);
184 /*
185 double widthInUnits = (double)right - left;
186 double heightInUnits = (double)bottom - top;
187 *width = (int)((widthInUnits*1440.0)/inch);
188 *height = (int)((heightInUnits*1440.0)/inch);
189 */
190 }
191 else rewind(handle);
192
193 // Read METAHEADER
194 int mtType = getshort(handle);
195
196 if (mtType != 1 && mtType != 2)
197 {
198 fclose(handle);
199 return FALSE;
200 }
201
202 /* int mtHeaderSize = */ getshort(handle);
203 int mtVersion = getshort(handle);
204
205 if (mtVersion != 0x0300 && mtVersion != 0x0100)
206 {
207 fclose(handle);
208 return FALSE;
209 }
210
211 /* long mtSize = */ getint(handle);
212 /* int mtNoObjects = */ getshort(handle);
213 /* long mtMaxRecord = */ getint(handle);
214 /* int mtNoParameters = */ getshort(handle);
215
216 while (!feof(handle))
217 {
218 long rdSize = getint(handle); // 4 bytes
219 int rdFunction = getshort(handle); // 2 bytes
220 if (feof(handle))
221 break;
222
223 switch (rdFunction)
224 {
225 case META_SETBKCOLOR:
226 {
227 wxMetaRecord *rec = new wxMetaRecord(META_SETBKCOLOR);
228 long colorref = getint(handle); // COLORREF
229 rec->param1 = GetRValue(colorref);
230 rec->param2 = GetGValue(colorref);
231 rec->param3 = GetBValue(colorref);
232 metaRecords.Append(rec);
233 break;
234 }
235 case META_SETBKMODE:
236 {
237 wxMetaRecord *rec = new wxMetaRecord(META_SETBKMODE);
238 rec->param1 = getshort(handle); // Background mode
239 if (rec->param1 == OPAQUE) rec->param1 = wxSOLID;
240 else rec->param1 = wxTRANSPARENT;
241 metaRecords.Append(rec);
242 break;
243 }
244 case META_SETMAPMODE:
245 {
246 wxMetaRecord *rec = new wxMetaRecord(META_SETMAPMODE);
247 rec->param1 = getshort(handle);
248 metaRecords.Append(rec);
249 break;
250 }
251 // case META_SETROP2:
252 // case META_SETRELABS:
253 // case META_SETPOLYFILLMODE:
254 // case META_SETSTRETCHBLTMODE:
255 // case META_SETTEXTCHAREXTRA:
256 case META_SETTEXTCOLOR:
257 {
258 wxMetaRecord *rec = new wxMetaRecord(META_SETTEXTCOLOR);
259 long colorref = getint(handle); // COLORREF
260 rec->param1 = GetRValue(colorref);
261 rec->param2 = GetGValue(colorref);
262 rec->param3 = GetBValue(colorref);
263 metaRecords.Append(rec);
264 break;
265 }
266 // case META_SETTEXTJUSTIFICATION:
267 case META_SETWINDOWORG:
268 {
269 wxMetaRecord *rec = new wxMetaRecord(META_SETWINDOWORG);
270 rec->param2 = getshort(handle);
271 rec->param1 = getshort(handle);
272 metaRecords.Append(rec);
273 break;
274 }
275 case META_SETWINDOWEXT:
276 {
277 wxMetaRecord *rec = new wxMetaRecord(META_SETWINDOWEXT);
278 rec->param2 = getshort(handle);
279 rec->param1 = getshort(handle);
280 metaRecords.Append(rec);
281 break;
282 }
283 // case META_SETVIEWPORTORG:
284 // case META_SETVIEWPORTEXT:
285 // case META_OFFSETWINDOWORG:
286 // case META_SCALEWINDOWEXT:
287 // case META_OFFSETVIEWPORTORG:
288 // case META_SCALEVIEWPORTEXT:
289 case META_LINETO:
290 {
291 wxMetaRecord *rec = new wxMetaRecord(META_LINETO);
292 rec->param1 = getshort(handle); // x1
293 rec->param2 = getshort(handle); // y1
294 metaRecords.Append(rec);
295 break;
296 }
297 case META_MOVETO:
298 {
299 wxMetaRecord *rec = new wxMetaRecord(META_MOVETO);
300 rec->param1 = getshort(handle); // x1
301 rec->param2 = getshort(handle); // y1
302 metaRecords.Append(rec);
303 break;
304 }
305 case META_EXCLUDECLIPRECT:
306 {
307 wxMetaRecord *rec = new wxMetaRecord(META_EXCLUDECLIPRECT);
308 rec->param4 = getshort(handle); // y2
309 rec->param3 = getshort(handle); // x2
310 rec->param2 = getshort(handle); // y1
311 rec->param1 = getshort(handle); // x1
312 metaRecords.Append(rec);
313 break;
314 }
315 case META_INTERSECTCLIPRECT:
316 {
317 wxMetaRecord *rec = new wxMetaRecord(META_INTERSECTCLIPRECT);
318 rec->param4 = getshort(handle); // y2
319 rec->param3 = getshort(handle); // x2
320 rec->param2 = getshort(handle); // y1
321 rec->param1 = getshort(handle); // x1
322 metaRecords.Append(rec);
323 break;
324 }
325 // case META_ARC: // DO!!!
326 case META_ELLIPSE:
327 {
328 wxMetaRecord *rec = new wxMetaRecord(META_ELLIPSE);
329 rec->param4 = getshort(handle); // y2
330 rec->param3 = getshort(handle); // x2
331 rec->param2 = getshort(handle); // y1
332 rec->param1 = getshort(handle); // x1
333 metaRecords.Append(rec);
334 break;
335 }
336 // case META_FLOODFILL:
337 // case META_PIE: // DO!!!
338 case META_RECTANGLE:
339 {
340 wxMetaRecord *rec = new wxMetaRecord(META_RECTANGLE);
341 rec->param4 = getshort(handle); // y2
342 rec->param3 = getshort(handle); // x2
343 rec->param2 = getshort(handle); // y1
344 rec->param1 = getshort(handle); // x1
345 metaRecords.Append(rec);
346 break;
347 }
348 case META_ROUNDRECT:
349 {
350 wxMetaRecord *rec = new wxMetaRecord(META_ROUNDRECT);
351 rec->param6 = getshort(handle); // width
352 rec->param5 = getshort(handle); // height
353 rec->param4 = getshort(handle); // y2
354 rec->param3 = getshort(handle); // x2
355 rec->param2 = getshort(handle); // y1
356 rec->param1 = getshort(handle); // x1
357 metaRecords.Append(rec);
358 break;
359 }
360 // case META_PATBLT:
361 // case META_SAVEDC:
362 case META_SETPIXEL:
363 {
364 wxMetaRecord *rec = new wxMetaRecord(META_SETPIXEL);
365 rec->param1 = getshort(handle); // x1
366 rec->param2 = getshort(handle); // y1
367 rec->param3 = getint(handle); // COLORREF
368 metaRecords.Append(rec);
369 break;
370 }
371 // case META_OFFSETCLIPRGN:
372 case META_TEXTOUT:
373 {
374 wxMetaRecord *rec = new wxMetaRecord(META_TEXTOUT);
375 int count = getshort(handle);
376 rec->stringParam = new wxChar[count+1];
377 fread((void *)rec->stringParam, sizeof(wxChar), count, handle);
378 rec->stringParam[count] = 0;
379 rec->param2 = getshort(handle); // Y
380 rec->param1 = getshort(handle); // X
381 metaRecords.Append(rec);
382 break;
383 }
384 /*
385 case META_EXTTEXTOUT:
386 {
387 wxMetaRecord *rec = new wxMetaRecord(META_EXTTEXTOUT);
388 int cellSpacing = getshort(handle);
389 int count = getshort(handle);
390 rec->stringParam = new char[count+1];
391 fread((void *)rec->stringParam, sizeof(char), count, handle);
392 rec->stringParam[count] = 0;
393 // Rectangle
394 int rectY2 = getshort(handle);
395 int rectX2 = getshort(handle);
396 int rectY1 = getshort(handle);
397 int rectX1 = getshort(handle);
398 int rectType = getshort(handle);
399 rec->param2 = getshort(handle); // Y
400 rec->param1 = getshort(handle); // X
401 metaRecords.Append(rec);
402 break;
403 }
404 */
405 // case META_BITBLT:
406 // case META_STRETCHBLT:
407 case META_POLYGON:
408 {
409 wxMetaRecord *rec = new wxMetaRecord(META_POLYGON);
410 rec->param1 = getshort(handle);
411 rec->points = new wxRealPoint[(int)rec->param1];
412 for (int i = 0; i < rec->param1; i++)
413 {
414 rec->points[i].x = getshort(handle);
415 rec->points[i].y = getshort(handle);
416 }
417
418 metaRecords.Append(rec);
419 break;
420 }
421 case META_POLYLINE:
422 {
423 wxMetaRecord *rec = new wxMetaRecord(META_POLYLINE);
424 rec->param1 = (long)getshort(handle);
425 rec->points = new wxRealPoint[(int)rec->param1];
426 for (int i = 0; i < rec->param1; i++)
427 {
428 rec->points[i].x = getshort(handle);
429 rec->points[i].y = getshort(handle);
430 }
431
432 metaRecords.Append(rec);
433 break;
434 }
435 // case META_ESCAPE:
436 // case META_RESTOREDC:
437 // case META_FILLREGION:
438 // case META_FRAMEREGION:
439 // case META_INVERTREGION:
440 // case META_PAINTREGION:
441 // case META_SELECTCLIPREGION: // DO THIS!
442 case META_SELECTOBJECT:
443 {
444 wxMetaRecord *rec = new wxMetaRecord(META_SELECTOBJECT);
445 rec->param1 = (long)getshort(handle); // Position of object in gdiObjects list
446 metaRecords.Append(rec);
447 // param2 gives the index into gdiObjects, which is different from
448 // the index into the handle table.
449 rec->param2 = HandleTable[(int)rec->param1]->param2;
450 break;
451 }
452 // case META_SETTEXTALIGN:
453 // case META_DRAWTEXT:
454 // case META_CHORD:
455 // case META_SETMAPPERFLAGS:
456 // case META_EXTTEXTOUT:
457 // case META_SETDIBTODEV:
458 // case META_SELECTPALETTE:
459 // case META_REALIZEPALETTE:
460 // case META_ANIMATEPALETTE:
461 // case META_SETPALENTRIES:
462 // case META_POLYPOLYGON:
463 // case META_RESIZEPALETTE:
464 // case META_DIBBITBLT:
465 // case META_DIBSTRETCHBLT:
466 case META_DIBCREATEPATTERNBRUSH:
467 {
468 wxMetaRecord *rec = new wxMetaRecord(META_DIBCREATEPATTERNBRUSH);
469 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
470
471 metaRecords.Append(rec);
472 gdiObjects.Append(rec);
473 AddMetaRecordHandle(rec);
474 rec->param2 = (long)(gdiObjects.GetCount() - 1);
475 break;
476 }
477 // case META_STRETCHDIB:
478 // case META_EXTFLOODFILL:
479 // case META_RESETDC:
480 // case META_STARTDOC:
481 // case META_STARTPAGE:
482 // case META_ENDPAGE:
483 // case META_ABORTDOC:
484 // case META_ENDDOC:
485 case META_DELETEOBJECT:
486 {
487 int index = getshort(handle);
488 DeleteMetaRecordHandle(index);
489 break;
490 }
491 case META_CREATEPALETTE:
492 {
493 wxMetaRecord *rec = new wxMetaRecord(META_CREATEPALETTE);
494 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
495
496 metaRecords.Append(rec);
497 gdiObjects.Append(rec);
498 AddMetaRecordHandle(rec);
499 rec->param2 = (long)(gdiObjects.GetCount() - 1);
500 break;
501 }
502 case META_CREATEBRUSH:
503 {
504 wxMetaRecord *rec = new wxMetaRecord(META_CREATEBRUSH);
505 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
506 metaRecords.Append(rec);
507 gdiObjects.Append(rec);
508 AddMetaRecordHandle(rec);
509 rec->param2 = (long)(gdiObjects.GetCount() - 1);
510 break;
511 }
512 case META_CREATEPATTERNBRUSH:
513 {
514 wxMetaRecord *rec = new wxMetaRecord(META_CREATEPATTERNBRUSH);
515 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
516 metaRecords.Append(rec);
517 gdiObjects.Append(rec);
518 AddMetaRecordHandle(rec);
519 rec->param2 = (long)(gdiObjects.GetCount() - 1);
520 break;
521 }
522 case META_CREATEPENINDIRECT:
523 {
524 wxMetaRecord *rec = new wxMetaRecord(META_CREATEPENINDIRECT);
525 int msStyle = getshort(handle); // Style: 2 bytes
526 int x = getshort(handle); // X: 2 bytes
527 /* int y = */ getshort(handle); // Y: 2 bytes
528 long colorref = getint(handle); // COLORREF 4 bytes
529
530 int style;
531 if (msStyle == PS_DOT)
532 style = wxDOT;
533 else if (msStyle == PS_DASH)
534 style = wxSHORT_DASH;
535 else if (msStyle == PS_NULL)
536 style = wxTRANSPARENT;
537 else style = wxSOLID;
538
539 wxColour colour(GetRValue(colorref), GetGValue(colorref), GetBValue(colorref));
540 rec->param1 = (long)wxThePenList->FindOrCreatePen(colour, x, style);
541 metaRecords.Append(rec);
542 gdiObjects.Append(rec);
543
544 AddMetaRecordHandle(rec);
545 rec->param2 = (long)(gdiObjects.GetCount() - 1);
546
547 // For some reason, the size of this record is sometimes 9 words!!!
548 // instead of the usual 8. So read 2 characters extra.
549 if (rdSize == 9)
550 {
551 (void) getshort(handle);
552 }
553 break;
554 }
555 case META_CREATEFONTINDIRECT:
556 {
557 wxMetaRecord *rec = new wxMetaRecord(META_CREATEFONTINDIRECT);
558 int lfHeight = getshort(handle); // 2 bytes
559 /* int lfWidth = */ getshort(handle); // 2 bytes
560 /* int lfEsc = */ getshort(handle); // 2 bytes
561 /* int lfOrient = */ getshort(handle); // 2 bytes
562 int lfWeight = getshort(handle); // 2 bytes
563 char lfItalic = getc(handle); // 1 byte
564 char lfUnderline = getc(handle); // 1 byte
565 /* char lfStrikeout = */ getc(handle); // 1 byte
566 /* char lfCharSet = */ getc(handle); // 1 byte
567 /* char lfOutPrecision = */ getc(handle); // 1 byte
568 /* char lfClipPrecision = */ getc(handle); // 1 byte
569 /* char lfQuality = */ getc(handle); // 1 byte
570 char lfPitchAndFamily = getc(handle); // 1 byte (18th)
571 char lfFacename[32];
572 // Read the rest of the record, which is total record size
573 // minus the number of bytes already read (18 record, 6 metarecord
574 // header)
575 fread((void *)lfFacename, sizeof(char), (int)((2*rdSize) - 18 - 6), handle);
576
577 int family;
578 if (lfPitchAndFamily & FF_MODERN)
579 family = wxMODERN;
580 else if (lfPitchAndFamily & FF_MODERN)
581 family = wxMODERN;
582 else if (lfPitchAndFamily & FF_ROMAN)
583 family = wxROMAN;
584 else if (lfPitchAndFamily & FF_SWISS)
585 family = wxSWISS;
586 else if (lfPitchAndFamily & FF_DECORATIVE)
587 family = wxDECORATIVE;
588 else
589 family = wxDEFAULT;
590
591 int weight;
592 if (lfWeight == 300)
593 weight = wxLIGHT;
594 else if (lfWeight == 400)
595 weight = wxNORMAL;
596 else if (lfWeight == 900)
597 weight = wxBOLD;
598 else weight = wxNORMAL;
599
600 int style;
601 if (lfItalic != 0)
602 style = wxITALIC;
603 else
604 style = wxNORMAL;
605
606 // About how many pixels per inch???
607 int logPixelsY = 100;
608 int pointSize = (int)(lfHeight*72.0/logPixelsY);
609
610 wxFont *theFont =
611 wxTheFontList->FindOrCreateFont(pointSize, family, style, weight, (lfUnderline != 0));
612
613 rec->param1 = (long) theFont;
614 metaRecords.Append(rec);
615 gdiObjects.Append(rec);
616 AddMetaRecordHandle(rec);
617 rec->param2 = (long)(gdiObjects.GetCount() - 1);
618 break;
619 }
620 case META_CREATEBRUSHINDIRECT:
621 {
622 wxMetaRecord *rec = new wxMetaRecord(META_CREATEBRUSHINDIRECT);
623 int msStyle = getshort(handle); // Style: 2 bytes
624 long colorref = getint(handle); // COLORREF: 4 bytes
625 int hatchStyle = getshort(handle); // Hatch style 2 bytes
626
627 int style;
628 switch (msStyle)
629 {
630 case BS_HATCHED:
631 {
632 switch (hatchStyle)
633 {
634 case HS_BDIAGONAL:
635 style = wxBDIAGONAL_HATCH;
636 break;
637 case HS_DIAGCROSS:
638 style = wxCROSSDIAG_HATCH;
639 break;
640 case HS_FDIAGONAL:
641 style = wxFDIAGONAL_HATCH;
642 break;
643 case HS_HORIZONTAL:
644 style = wxHORIZONTAL_HATCH;
645 break;
646 case HS_VERTICAL:
647 style = wxVERTICAL_HATCH;
648 break;
649 default:
650 case HS_CROSS:
651 style = wxCROSS_HATCH;
652 break;
653 }
654 break;
655 }
656 case BS_SOLID:
657 default:
658 style = wxSOLID;
659 break;
660 }
661 if (msStyle == PS_DOT)
662 style = wxDOT;
663 else if (msStyle == PS_DASH)
664 style = wxSHORT_DASH;
665 else if (msStyle == PS_NULL)
666 style = wxTRANSPARENT;
667 else style = wxSOLID;
668
669 wxColour colour(GetRValue(colorref), GetGValue(colorref), GetBValue(colorref));
670 rec->param1 = (long)wxTheBrushList->FindOrCreateBrush(colour, style);
671 metaRecords.Append(rec);
672 gdiObjects.Append(rec);
673 AddMetaRecordHandle(rec);
674 rec->param2 = (long)(gdiObjects.GetCount() - 1);
675 break;
676 }
677 case META_CREATEBITMAPINDIRECT:
678 {
679 wxMetaRecord *rec = new wxMetaRecord(META_CREATEBITMAPINDIRECT);
680 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
681
682 metaRecords.Append(rec);
683 gdiObjects.Append(rec);
684 AddMetaRecordHandle(rec);
685 rec->param2 = (long)(gdiObjects.GetCount() - 1);
686 break;
687 }
688 case META_CREATEBITMAP:
689 {
690 wxMetaRecord *rec = new wxMetaRecord(META_CREATEBITMAP);
691 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
692
693 metaRecords.Append(rec);
694 gdiObjects.Append(rec);
695 AddMetaRecordHandle(rec);
696 rec->param2 = (long)(gdiObjects.GetCount() - 1);
697 break;
698 }
699 case META_CREATEREGION:
700 {
701 wxMetaRecord *rec = new wxMetaRecord(META_CREATEREGION);
702 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
703
704 metaRecords.Append(rec);
705 gdiObjects.Append(rec);
706 AddMetaRecordHandle(rec);
707 rec->param2 = (long)(gdiObjects.GetCount() - 1);
708 break;
709 }
710 default:
711 {
712 fread((void *)_buf, sizeof(char), (int)((2*rdSize) - 6), handle);
713 break;
714 }
715 }
716 }
717 fclose(handle);
718 return TRUE;
719 }
720
721 wxXMetaFile::~wxXMetaFile(void)
722 {
723 wxNode *node = metaRecords.GetFirst();
724 while (node)
725 {
726 wxMetaRecord *rec = (wxMetaRecord *)node->GetData();
727 delete rec;
728 wxNode *next = node->GetNext();
729 delete node;
730 node = next;
731 }
732 }
733
734 bool wxXMetaFile::SetClipboard(int WXUNUSED(width), int WXUNUSED(height))
735 {
736 return FALSE;
737 }
738
739 bool wxXMetaFile::Play(wxDC *dc)
740 {
741 wxNode *node = metaRecords.GetFirst();
742 while (node)
743 {
744 wxMetaRecord *rec = (wxMetaRecord *)node->GetData();
745 int rdFunction = rec->metaFunction;
746
747 switch (rdFunction)
748 {
749 case META_SETBKCOLOR:
750 {
751 break;
752 }
753 case META_SETBKMODE:
754 {
755 break;
756 }
757 case META_SETMAPMODE:
758 {
759 break;
760 }
761 // case META_SETROP2:
762 // case META_SETRELABS:
763 // case META_SETPOLYFILLMODE:
764 // case META_SETSTRETCHBLTMODE:
765 // case META_SETTEXTCHAREXTRA:
766 case META_SETTEXTCOLOR:
767 {
768 break;
769 }
770 // case META_SETTEXTJUSTIFICATION:
771 case META_SETWINDOWORG:
772 {
773 break;
774 }
775 case META_SETWINDOWEXT:
776 {
777 break;
778 }
779 // case META_SETVIEWPORTORG:
780 // case META_SETVIEWPORTEXT:
781 // case META_OFFSETWINDOWORG:
782 // case META_SCALEWINDOWEXT:
783 // case META_OFFSETVIEWPORTORG:
784 // case META_SCALEVIEWPORTEXT:
785 case META_LINETO:
786 {
787 long x1 = rec->param1;
788 long y1 = rec->param2;
789 dc->DrawLine((long) lastX, (long) lastY, x1, y1);
790 break;
791 }
792 case META_MOVETO:
793 {
794 lastX = (double)rec->param1;
795 lastY = (double)rec->param2;
796 break;
797 }
798 case META_EXCLUDECLIPRECT:
799 {
800 break;
801 }
802 case META_INTERSECTCLIPRECT:
803 {
804 break;
805 }
806 // case META_ARC: // DO!!!
807 case META_ELLIPSE:
808 {
809 break;
810 }
811 // case META_FLOODFILL:
812 // case META_PIE: // DO!!!
813 case META_RECTANGLE:
814 {
815 dc->DrawRectangle((long)rec->param1, (long)rec->param2,
816 (long)rec->param3 - rec->param1,
817 (long)rec->param4 - rec->param2);
818 break;
819 }
820 case META_ROUNDRECT:
821 {
822 dc->DrawRoundedRectangle((long)rec->param1, (long)rec->param2,
823 (long)rec->param3 - rec->param1,
824 (long)rec->param4 - rec->param2,
825 (long)rec->param5);
826 break;
827 }
828 // case META_PATBLT:
829 // case META_SAVEDC:
830 case META_SETPIXEL:
831 {
832 // rec->param1 = getshort(handle); // x1
833 // rec->param2 = getshort(handle); // y1
834 // rec->param3 = getint(handle); // COLORREF
835 break;
836 }
837 // case META_OFFSETCLIPRGN:
838 case META_TEXTOUT:
839 {
840 /*
841 int count = getshort(handle);
842 rec->stringParam = new char[count+1];
843 fread((void *)rec->stringParam, sizeof(char), count, handle);
844 rec->stringParam[count] = 0;
845 rec->param2 = getshort(handle); // Y
846 rec->param1 = getshort(handle); // X
847 */
848 break;
849 }
850 // case META_BITBLT:
851 // case META_STRETCHBLT:
852 case META_POLYGON:
853 {
854 /*
855 rec->param1 = getshort(handle);
856 rec->points = new wxRealPoint[(int)rec->param1];
857 for (int i = 0; i < rec->param1; i++)
858 {
859 rec->points[i].x = getshort(handle);
860 rec->points[i].y = getshort(handle);
861 }
862 */
863 break;
864 }
865 case META_POLYLINE:
866 {
867 /*
868 wxMetaRecord *rec = new wxMetaRecord(META_POLYLINE);
869 rec->param1 = (long)getshort(handle);
870 rec->points = new wxRealPoint[(int)rec->param1];
871 for (int i = 0; i < rec->param1; i++)
872 {
873 rec->points[i].x = getshort(handle);
874 rec->points[i].y = getshort(handle);
875 }
876 */
877 break;
878 }
879 // case META_ESCAPE:
880 // case META_RESTOREDC:
881 // case META_FILLREGION:
882 // case META_FRAMEREGION:
883 // case META_INVERTREGION:
884 // case META_PAINTREGION:
885 // case META_SELECTCLIPREGION: // DO THIS!
886 case META_SELECTOBJECT:
887 {
888 /*
889 wxMetaRecord *rec = new wxMetaRecord(META_SELECTOBJECT);
890 rec->param1 = (long)getshort(handle); // Position of object in gdiObjects list
891 */
892 break;
893 }
894 // case META_SETTEXTALIGN:
895 // case META_DRAWTEXT:
896 // case META_CHORD:
897 // case META_SETMAPPERFLAGS:
898 // case META_EXTTEXTOUT:
899 // case META_SETDIBTODEV:
900 // case META_SELECTPALETTE:
901 // case META_REALIZEPALETTE:
902 // case META_ANIMATEPALETTE:
903 // case META_SETPALENTRIES:
904 // case META_POLYPOLYGON:
905 // case META_RESIZEPALETTE:
906 // case META_DIBBITBLT:
907 // case META_DIBSTRETCHBLT:
908 case META_DIBCREATEPATTERNBRUSH:
909 {
910 /*
911 fread((void *)wxBuffer, sizeof(char), (int)(rdSize - 3), handle);
912 */
913 break;
914 }
915 // case META_STRETCHDIB:
916 // case META_EXTFLOODFILL:
917 // case META_RESETDC:
918 // case META_STARTDOC:
919 // case META_STARTPAGE:
920 // case META_ENDPAGE:
921 // case META_ABORTDOC:
922 // case META_ENDDOC:
923 // case META_DELETEOBJECT: // DO!!
924 case META_CREATEPALETTE:
925 {
926 /*
927 wxMetaRecord *rec = new wxMetaRecord(META_CREATEPALETTE);
928 fread((void *)wxBuffer, sizeof(char), (int)(rdSize - 3), handle);
929 */
930 break;
931 }
932 case META_CREATEBRUSH:
933 {
934 /*
935 fread((void *)wxBuffer, sizeof(char), (int)(rdSize - 3), handle);
936 */
937 break;
938 }
939 case META_CREATEPATTERNBRUSH:
940 {
941 /*
942 fread((void *)wxBuffer, sizeof(char), (int)(rdSize - 3), handle);
943 */
944 break;
945 }
946 case META_CREATEPENINDIRECT:
947 {
948 /*
949 int msStyle = getshort(handle); // Style: 2 bytes
950 int x = getshort(handle); // X: 2 bytes
951 int y = getshort(handle); // Y: 2 bytes
952 int colorref = getint(handle); // COLORREF 4 bytes
953
954 int style;
955 if (msStyle == PS_DOT)
956 style = wxDOT;
957 else if (msStyle == PS_DASH)
958 style = wxSHORT_DASH;
959 else if (msStyle == PS_NULL)
960 style = wxTRANSPARENT;
961 else style = wxSOLID;
962
963 wxColour colour(GetRValue(colorref), GetGValue(colorref), GetBValue(colorref));
964 rec->param1 = (long)wxThePenList->FindOrCreatePen(&colour, x, style);
965 */
966 break;
967 }
968 case META_CREATEFONTINDIRECT:
969 {
970 /*
971 int lfHeight = getshort(handle);
972 int lfWidth = getshort(handle);
973 int lfEsc = getshort(handle);
974 int lfOrient = getshort(handle);
975 int lfWeight = getshort(handle);
976 char lfItalic = getc(handle);
977 char lfUnderline = getc(handle);
978 char lfStrikeout = getc(handle);
979 char lfCharSet = getc(handle);
980 char lfOutPrecision = getc(handle);
981 char lfClipPrecision = getc(handle);
982 char lfQuality = getc(handle);
983 char lfPitchAndFamily = getc(handle);
984 char lfFacename[32];
985 fread((void *)lfFacename, sizeof(char), 32, handle);
986
987 int family;
988 if (lfPitchAndFamily & FF_MODERN)
989 family = wxMODERN;
990 else if (lfPitchAndFamily & FF_MODERN)
991 family = wxMODERN;
992 else if (lfPitchAndFamily & FF_ROMAN)
993 family = wxROMAN;
994 else if (lfPitchAndFamily & FF_SWISS)
995 family = wxSWISS;
996 else if (lfPitchAndFamily & FF_DECORATIVE)
997 family = wxDECORATIVE;
998 else
999 family = wxDEFAULT;
1000
1001 int weight;
1002 if (lfWeight == 300)
1003 weight = wxLIGHT;
1004 else if (lfWeight == 400)
1005 weight = wxNORMAL;
1006 else if (lfWeight == 900)
1007 weight = wxBOLD;
1008 else weight = wxNORMAL;
1009
1010 int style;
1011 if ((bool)lfItalic)
1012 style = wxITALIC;
1013 else
1014 style = wxNORMAL;
1015
1016 // About how many pixels per inch???
1017 int logPixelsY = 100;
1018 int pointSize = (int)(lfHeight*72.0/logPixelsY);
1019
1020 wxFont *theFont =
1021 wxTheFontList->FindOrCreateFont(pointSize, family, style, weight, (bool)lfUnderline);
1022
1023 rec->param1 = (long)theFont;
1024 */
1025 break;
1026 }
1027 case META_CREATEBRUSHINDIRECT:
1028 {
1029 /*
1030 int msStyle = getshort(handle); // Style: 2 bytes
1031 int colorref = getint(handle); // COLORREF: 4 bytes
1032 int hatchStyle = getshort(handle); // Hatch style 2 bytes
1033
1034 int style;
1035 if (msStyle == PS_DOT)
1036 style = wxDOT;
1037 else if (msStyle == PS_DASH)
1038 style = wxSHORT_DASH;
1039 else if (msStyle == PS_NULL)
1040 style = wxTRANSPARENT;
1041 else style = wxSOLID;
1042
1043 wxColour colour(GetRValue(colorref), GetGValue(colorref), GetBValue(colorref));
1044 rec->param1 = (long)wxTheBrushList->FindOrCreateBrush(&colour, wxSOLID);
1045 */
1046 break;
1047 }
1048 case META_CREATEBITMAPINDIRECT:
1049 {
1050 /*
1051 fread((void *)wxBuffer, sizeof(char), (int)(rdSize - 3), handle);
1052 */
1053 break;
1054 }
1055 case META_CREATEBITMAP:
1056 {
1057 /*
1058 fread((void *)wxBuffer, sizeof(char), (int)(rdSize - 3), handle);
1059 */
1060 break;
1061 }
1062 case META_CREATEREGION:
1063 {
1064 dc->DestroyClippingRegion();
1065 /*
1066 rec->param1 = getshort(handle); // Style: 2 bytes
1067 */
1068 break;
1069 }
1070 default:
1071 {
1072 break;
1073 }
1074 }
1075 node = node->GetNext();
1076 }
1077 return TRUE;
1078 }
1079