]> git.saurik.com Git - wxWidgets.git/blob - src/os2/metafile.cpp
updates for DoGetBestSize
[wxWidgets.git] / src / os2 / metafile.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: metafile.cpp
3 // Purpose: wxMetaFile, wxMetaFileDC etc. These classes are optional.
4 // Author: David Webster
5 // Modified by:
6 // Created: 10/10/99
7 // RCS-ID: $Id$
8 // Copyright: (c) David Webster
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // For compilers that support precompilation, includes "wx.h".
13 #include "wx/wxprec.h"
14
15 #ifndef WX_PRECOMP
16 #include "wx/setup.h"
17 #endif
18
19 #if wxUSE_METAFILE
20
21 #ifndef WX_PRECOMP
22 #include "wx/utils.h"
23 #include "wx/app.h"
24 #endif
25
26 #include "wx/metafile.h"
27 #include "wx/clipbrd.h"
28 #include "wx/os2/private.h"
29
30 #include <stdio.h>
31 #include <string.h>
32
33 extern bool wxClipboardIsOpen;
34
35 #if !USE_SHARED_LIBRARY
36 IMPLEMENT_DYNAMIC_CLASS(wxMetafile, wxObject)
37 IMPLEMENT_ABSTRACT_CLASS(wxMetafileDC, wxDC)
38 #endif
39
40 /*
41 * Metafiles
42 * Currently, the only purpose for making a metafile is to put
43 * it on the clipboard.
44 */
45
46 wxMetafileRefData::wxMetafileRefData(void)
47 {
48 m_metafile = 0;
49 m_windowsMappingMode = wxMM_ANISOTROPIC;
50 }
51
52 wxMetafileRefData::~wxMetafileRefData(void)
53 {
54 if (m_metafile)
55 {
56 // TODO: DeleteMetaFile((HMETAFILE) m_metafile);
57 m_metafile = 0;
58 }
59 }
60
61 wxMetafile::wxMetafile(const wxString& file)
62 {
63 m_refData = new wxMetafileRefData;
64
65 M_METAFILEDATA->m_windowsMappingMode = wxMM_ANISOTROPIC;
66 M_METAFILEDATA->m_metafile = 0;
67 if (!file.IsNull() && (file.Cmp(wxT("")) == 0))
68 M_METAFILEDATA->m_metafile = (WXHANDLE)0; // TODO: GetMetaFile(file);
69 }
70
71 wxMetafile::~wxMetafile(void)
72 {
73 }
74
75 bool wxMetafile::SetClipboard(int width, int height)
76 {
77 if (!m_refData)
78 return FALSE;
79
80 bool alreadyOpen=wxClipboardOpen();
81 if (!alreadyOpen)
82 {
83 wxOpenClipboard();
84 if (!wxEmptyClipboard()) return FALSE;
85 }
86 bool success = wxSetClipboardData(wxDF_METAFILE, this, width,height);
87 if (!alreadyOpen) wxCloseClipboard();
88 return (bool) success;
89 }
90
91 bool wxMetafile::Play(wxDC *dc)
92 {
93 if (!m_refData)
94 return FALSE;
95
96 dc->BeginDrawing();
97
98 // if (dc->GetHDC() && M_METAFILEDATA->m_metafile)
99 // PlayMetaFile((HDC) dc->GetHDC(), (HMETAFILE) M_METAFILEDATA->m_metafile);
100
101 dc->EndDrawing();
102
103 return TRUE;
104 }
105
106 void wxMetafile::SetHMETAFILE(WXHANDLE mf)
107 {
108 if (m_refData)
109 m_refData = new wxMetafileRefData;
110
111 M_METAFILEDATA->m_metafile = mf;
112 }
113
114 void wxMetafile::SetWindowsMappingMode(int mm)
115 {
116 if (m_refData)
117 m_refData = new wxMetafileRefData;
118
119 M_METAFILEDATA->m_windowsMappingMode = mm;
120 }
121
122 /*
123 * Metafile device context
124 *
125 */
126
127 // Original constructor that does not takes origin and extent. If you use this,
128 // *DO* give origin/extent arguments to wxMakeMetafilePlaceable.
129 wxMetafileDC::wxMetafileDC(const wxString& file)
130 {
131 m_metaFile = NULL;
132 m_minX = 10000;
133 m_minY = 10000;
134 m_maxX = -10000;
135 m_maxY = -10000;
136 // m_title = NULL;
137
138 if (!file.IsNull() && wxFileExists(file))
139 wxRemoveFile(file);
140
141 // TODO
142 /*
143 if (!file.IsNull() && (file != wxT("")))
144 m_hDC = (WXHDC) CreateMetaFile(file);
145 else
146 m_hDC = (WXHDC) CreateMetaFile(NULL);
147 */
148
149 m_ok = (m_hDC != (WXHDC) 0) ;
150
151 // Actual Windows mapping mode, for future reference.
152 m_windowsMappingMode = wxMM_TEXT;
153
154 SetMapMode(wxMM_TEXT); // NOTE: does not set HDC mapmode (this is correct)
155 }
156
157 // New constructor that takes origin and extent. If you use this, don't
158 // give origin/extent arguments to wxMakeMetafilePlaceable.
159 wxMetafileDC::wxMetafileDC(const wxString& file, int xext, int yext, int xorg, int yorg)
160 {
161 m_minX = 10000;
162 m_minY = 10000;
163 m_maxX = -10000;
164 m_maxY = -10000;
165 if (file != wxT("") && wxFileExists(file)) wxRemoveFile(file);
166 // m_hDC = (WXHDC) CreateMetaFile(file);
167
168 m_ok = TRUE;
169
170 // ::SetWindowOrgEx((HDC) m_hDC,xorg,yorg, NULL);
171 // ::SetWindowExtEx((HDC) m_hDC,xext,yext, NULL);
172
173 // Actual Windows mapping mode, for future reference.
174 m_windowsMappingMode = wxMM_ANISOTROPIC;
175
176 SetMapMode(wxMM_TEXT); // NOTE: does not set HDC mapmode (this is correct)
177 }
178
179 wxMetafileDC::~wxMetafileDC(void)
180 {
181 m_hDC = 0;
182 }
183
184 void wxMetafileDC::GetTextExtent(const wxString& string, long *x, long *y,
185 long *descent, long *externalLeading, wxFont *theFont, bool use16bit) const
186 {
187 wxFont *fontToUse = theFont;
188 if (!fontToUse)
189 fontToUse = (wxFont*) &m_font;
190
191 // TODO:
192 /*
193 HDC dc = GetDC(NULL);
194
195 SIZE sizeRect;
196 TEXTMETRIC tm;
197 GetTextExtentPoint(dc, WXSTRINGCAST string, wxStrlen(WXSTRINGCAST string), &sizeRect);
198 GetTextMetrics(dc, &tm);
199
200 ReleaseDC(NULL, dc);
201
202 if ( x )
203 *x = sizeRect.cx;
204 if ( y )
205 *y = sizeRect.cy;
206 if ( descent )
207 *descent = tm.tmDescent;
208 if ( externalLeading )
209 *externalLeading = tm.tmExternalLeading;
210 */
211 }
212
213 wxMetafile *wxMetafileDC::Close(void)
214 {
215 SelectOldObjects(m_hDC);
216 HANDLE mf = 0; // TODO: CloseMetaFile((HDC) m_hDC);
217 m_hDC = 0;
218 if (mf)
219 {
220 wxMetafile *wx_mf = new wxMetafile;
221 wx_mf->SetHMETAFILE((WXHANDLE) mf);
222 wx_mf->SetWindowsMappingMode(m_windowsMappingMode);
223 return wx_mf;
224 }
225 return NULL;
226 }
227
228 void wxMetafileDC::SetMapMode(int mode)
229 {
230 m_mappingMode = mode;
231
232 // int pixel_width = 0;
233 // int pixel_height = 0;
234 // int mm_width = 0;
235 // int mm_height = 0;
236
237 float mm2pixelsX = 10.0;
238 float mm2pixelsY = 10.0;
239
240 switch (mode)
241 {
242 case wxMM_TWIPS:
243 {
244 m_logicalScaleX = (float)(twips2mm * mm2pixelsX);
245 m_logicalScaleY = (float)(twips2mm * mm2pixelsY);
246 break;
247 }
248 case wxMM_POINTS:
249 {
250 m_logicalScaleX = (float)(pt2mm * mm2pixelsX);
251 m_logicalScaleY = (float)(pt2mm * mm2pixelsY);
252 break;
253 }
254 case wxMM_METRIC:
255 {
256 m_logicalScaleX = mm2pixelsX;
257 m_logicalScaleY = mm2pixelsY;
258 break;
259 }
260 case wxMM_LOMETRIC:
261 {
262 m_logicalScaleX = (float)(mm2pixelsX/10.0);
263 m_logicalScaleY = (float)(mm2pixelsY/10.0);
264 break;
265 }
266 default:
267 case wxMM_TEXT:
268 {
269 m_logicalScaleX = 1.0;
270 m_logicalScaleY = 1.0;
271 break;
272 }
273 }
274 m_windowExtX = 100;
275 m_windowExtY = 100;
276 }
277
278 #ifdef __WIN32__
279 struct RECT32
280 {
281 short left;
282 short top;
283 short right;
284 short bottom;
285 };
286
287 struct mfPLACEABLEHEADER {
288 DWORD key;
289 short hmf;
290 RECT32 bbox;
291 WORD inch;
292 DWORD reserved;
293 WORD checksum;
294 };
295 #else
296 struct mfPLACEABLEHEADER {
297 DWORD key;
298 HANDLE hmf;
299 RECT bbox;
300 WORD inch;
301 DWORD reserved;
302 WORD checksum;
303 };
304 #endif
305
306 /*
307 * Pass filename of existing non-placeable metafile, and bounding box.
308 * Adds a placeable metafile header, sets the mapping mode to anisotropic,
309 * and sets the window origin and extent to mimic the wxMM_TEXT mapping mode.
310 *
311 */
312
313 bool wxMakeMetafilePlaceable(const wxString& filename, float scale)
314 {
315 return wxMakeMetafilePlaceable(filename, 0, 0, 0, 0, scale, FALSE);
316 }
317
318 bool wxMakeMetafilePlaceable(const wxString& filename, int x1, int y1, int x2, int y2, float scale, bool useOriginAndExtent)
319 {
320 // TODO: the OS/2 PM/MM way to do this
321 /*
322 // I'm not sure if this is the correct way of suggesting a scale
323 // to the client application, but it's the only way I can find.
324 int unitsPerInch = (int)(576/scale);
325
326 mfPLACEABLEHEADER header;
327 header.key = 0x9AC6CDD7L;
328 header.hmf = 0;
329 header.bbox.xLeft = (int)(x1);
330 header.bbox.yTop = (int)(y1);
331 header.bbox.xRight = (int)(x2);
332 header.bbox.yBottom = (int)(y2);
333 header.inch = unitsPerInch;
334 header.reserved = 0;
335
336 // Calculate checksum
337 WORD *p;
338 mfPLACEABLEHEADER *pMFHead = &header;
339 for (p =(WORD *)pMFHead,pMFHead -> checksum = 0;
340 p < (WORD *)&pMFHead ->checksum; ++p)
341 pMFHead ->checksum ^= *p;
342
343 FILE *fd = fopen(filename.fn_str(), "rb");
344 if (!fd) return FALSE;
345
346 wxChar tempFileBuf[256];
347 wxGetTempFileName(wxT("mf"), tempFileBuf);
348 FILE *fHandle = fopen(wxConvFile.cWX2MB(tempFileBuf), "wb");
349 if (!fHandle)
350 return FALSE;
351 fwrite((void *)&header, sizeof(unsigned char), sizeof(mfPLACEABLEHEADER), fHandle);
352
353 // Calculate origin and extent
354 int originX = x1;
355 int originY = y1;
356 int extentX = x2 - x1;
357 int extentY = (y2 - y1);
358
359 // Read metafile header and write
360 METAHEADER metaHeader;
361 fread((void *)&metaHeader, sizeof(unsigned char), sizeof(metaHeader), fd);
362
363 if (useOriginAndExtent)
364 metaHeader.mtSize += 15;
365 else
366 metaHeader.mtSize += 5;
367
368 fwrite((void *)&metaHeader, sizeof(unsigned char), sizeof(metaHeader), fHandle);
369
370 // Write SetMapMode, SetWindowOrigin and SetWindowExt records
371 char modeBuffer[8];
372 char originBuffer[10];
373 char extentBuffer[10];
374 METARECORD *modeRecord = (METARECORD *)&modeBuffer;
375
376 METARECORD *originRecord = (METARECORD *)&originBuffer;
377 METARECORD *extentRecord = (METARECORD *)&extentBuffer;
378
379 modeRecord->rdSize = 4;
380 modeRecord->rdFunction = META_SETMAPMODE;
381 modeRecord->rdParm[0] = MM_ANISOTROPIC;
382
383 originRecord->rdSize = 5;
384 originRecord->rdFunction = META_SETWINDOWORG;
385 originRecord->rdParm[0] = originY;
386 originRecord->rdParm[1] = originX;
387
388 extentRecord->rdSize = 5;
389 extentRecord->rdFunction = META_SETWINDOWEXT;
390 extentRecord->rdParm[0] = extentY;
391 extentRecord->rdParm[1] = extentX;
392
393 fwrite((void *)modeBuffer, sizeof(char), 8, fHandle);
394
395 if (useOriginAndExtent)
396 {
397 fwrite((void *)originBuffer, sizeof(char), 10, fHandle);
398 fwrite((void *)extentBuffer, sizeof(char), 10, fHandle);
399 }
400
401 int ch = -2;
402 while (ch != EOF)
403 {
404 ch = getc(fd);
405 if (ch != EOF)
406 {
407 putc(ch, fHandle);
408 }
409 }
410 fclose(fHandle);
411 fclose(fd);
412 wxRemoveFile(filename);
413 wxCopyFile(tempFileBuf, filename);
414 wxRemoveFile(tempFileBuf);
415 */
416 return TRUE;
417 }
418
419 #endif // wxUSE_METAFILE
420