]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/enhmeta.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: msw/enhmeta.cpp
3 // Purpose: implementation of wxEnhMetaFileXXX classes
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 2000 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows license
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "enhmeta.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
31 #if wxUSE_ENH_METAFILE
34 #include "wx/string.h"
38 #include "wx/metafile.h"
39 #include "wx/clipbrd.h"
41 #include "wx/msw/private.h"
43 // ----------------------------------------------------------------------------
45 // ----------------------------------------------------------------------------
47 IMPLEMENT_DYNAMIC_CLASS(wxEnhMetaFile
, wxObject
)
48 IMPLEMENT_ABSTRACT_CLASS(wxEnhMetaFileDC
, wxDC
)
50 // ----------------------------------------------------------------------------
52 // ----------------------------------------------------------------------------
54 #define GetEMF() ((HENHMETAFILE)m_hMF)
55 #define GetEMFOf(mf) ((HENHMETAFILE)((mf).m_hMF))
57 // ----------------------------------------------------------------------------
59 // ----------------------------------------------------------------------------
61 // we must pass NULL if the string is empty to metafile functions
62 static inline const wxChar
*GetMetaFileName(const wxString
& fn
)
63 { return !fn
? (wxChar
*)NULL
: fn
.c_str(); }
65 // ============================================================================
67 // ============================================================================
69 // ----------------------------------------------------------------------------
71 // ----------------------------------------------------------------------------
73 void wxEnhMetaFile::Assign(const wxEnhMetaFile
& mf
)
80 m_hMF
= (WXHANDLE
)::CopyEnhMetaFile(GetEMFOf(mf
),
81 GetMetaFileName(m_filename
));
84 wxLogLastError(_T("CopyEnhMetaFile"));
93 void wxEnhMetaFile::Free()
97 if ( !::DeleteEnhMetaFile(GetEMF()) )
99 wxLogLastError(_T("DeleteEnhMetaFile"));
104 bool wxEnhMetaFile::Play(wxDC
*dc
, wxRect
*rectBound
)
106 wxCHECK_MSG( Ok(), FALSE
, _T("can't play invalid enhanced metafile") );
107 wxCHECK_MSG( dc
, FALSE
, _T("invalid wxDC in wxEnhMetaFile::Play") );
112 rect
.top
= rectBound
->y
;
113 rect
.left
= rectBound
->x
;
114 rect
.right
= rectBound
->x
+ rectBound
->width
;
115 rect
.bottom
= rectBound
->y
+ rectBound
->height
;
119 wxSize size
= GetSize();
124 rect
.bottom
= size
.y
;
127 if ( !::PlayEnhMetaFile(GetHdcOf(*dc
), GetEMF(), &rect
) )
129 wxLogLastError(_T("PlayEnhMetaFile"));
137 wxSize
wxEnhMetaFile::GetSize() const
139 wxSize size
= wxDefaultSize
;
144 if ( !::GetEnhMetaFileHeader(GetEMF(), sizeof(hdr
), &hdr
) )
146 wxLogLastError(_T("GetEnhMetaFileHeader"));
150 // the width and height are in HIMETRIC (0.01mm) units, transform
152 LONG w
= hdr
.rclFrame
.right
,
153 h
= hdr
.rclFrame
.bottom
;
155 HIMETRICToPixel(&w
, &h
);
165 bool wxEnhMetaFile::SetClipboard(int WXUNUSED(width
), int WXUNUSED(height
))
167 wxCHECK_MSG( m_hMF
, FALSE
, _T("can't copy invalid metafile to clipboard") );
169 return wxTheClipboard
->AddData(new wxEnhMetaFileDataObject(*this));
172 // ----------------------------------------------------------------------------
174 // ----------------------------------------------------------------------------
176 wxEnhMetaFileDC::wxEnhMetaFileDC(const wxString
& filename
,
177 int width
, int height
,
178 const wxString
& description
)
183 if ( width
&& height
)
188 rect
.bottom
= height
;
190 // CreateEnhMetaFile() wants them in HIMETRIC
191 PixelToHIMETRIC(&rect
.right
, &rect
.bottom
);
197 // GDI will try to find out the size for us (not recommended)
198 pRect
= (LPRECT
)NULL
;
201 m_hDC
= (WXHDC
)::CreateEnhMetaFile(hdcRef
, GetMetaFileName(filename
),
205 wxLogLastError(_T("CreateEnhMetaFile"));
209 wxEnhMetaFile
*wxEnhMetaFileDC::Close()
211 wxCHECK_MSG( Ok(), NULL
, _T("invalid wxEnhMetaFileDC") );
213 HENHMETAFILE hMF
= ::CloseEnhMetaFile(GetHdc());
216 wxLogLastError(_T("CloseEnhMetaFile"));
221 wxEnhMetaFile
*mf
= new wxEnhMetaFile
;
222 mf
->SetHENHMETAFILE((WXHANDLE
)hMF
);
226 wxEnhMetaFileDC::~wxEnhMetaFileDC()
228 // avoid freeing it in the base class dtor
232 // ----------------------------------------------------------------------------
233 // wxEnhMetaFileDataObject
234 // ----------------------------------------------------------------------------
236 #if wxUSE_DRAG_AND_DROP
239 wxEnhMetaFileDataObject::GetPreferredFormat(Direction
WXUNUSED(dir
)) const
241 return wxDF_ENHMETAFILE
;
244 size_t wxEnhMetaFileDataObject::GetFormatCount(Direction
WXUNUSED(dir
)) const
246 // wxDF_ENHMETAFILE and wxDF_METAFILE
250 void wxEnhMetaFileDataObject::GetAllFormats(wxDataFormat
*formats
,
251 Direction
WXUNUSED(dir
)) const
253 formats
[0] = wxDF_ENHMETAFILE
;
254 formats
[1] = wxDF_METAFILE
;
257 size_t wxEnhMetaFileDataObject::GetDataSize(const wxDataFormat
& format
) const
259 if ( format
== wxDF_ENHMETAFILE
)
261 // we pass data by handle and not HGLOBAL
266 wxASSERT_MSG( format
== wxDF_METAFILE
, _T("unsupported format") );
268 return sizeof(METAFILEPICT
);
272 bool wxEnhMetaFileDataObject::GetDataHere(const wxDataFormat
& format
, void *buf
) const
274 wxCHECK_MSG( m_metafile
.Ok(), FALSE
, _T("copying invalid enh metafile") );
276 HENHMETAFILE hEMF
= (HENHMETAFILE
)m_metafile
.GetHENHMETAFILE();
278 if ( format
== wxDF_ENHMETAFILE
)
280 HENHMETAFILE hEMFCopy
= ::CopyEnhMetaFile(hEMF
, NULL
);
283 wxLogLastError(_T("CopyEnhMetaFile"));
288 *(HENHMETAFILE
*)buf
= hEMFCopy
;
292 wxASSERT_MSG( format
== wxDF_METAFILE
, _T("unsupported format") );
298 // first get the buffer size and alloc memory
299 size_t size
= ::GetWinMetaFileBits(hEMF
, 0, NULL
, MM_ANISOTROPIC
, hdc
);
300 wxCHECK_MSG( size
, FALSE
, _T("GetWinMetaFileBits() failed") );
302 BYTE
*bits
= (BYTE
*)malloc(size
);
304 // then get the enh metafile bits
305 if ( !::GetWinMetaFileBits(hEMF
, size
, bits
, MM_ANISOTROPIC
, hdc
) )
307 wxLogLastError(_T("GetWinMetaFileBits"));
314 // and finally convert them to the WMF
315 HMETAFILE hMF
= ::SetMetaFileBitsEx(size
, bits
);
319 wxLogLastError(_T("SetMetaFileBitsEx"));
324 METAFILEPICT
*mfpict
= (METAFILEPICT
*)buf
;
326 wxSize sizeMF
= m_metafile
.GetSize();
328 mfpict
->mm
= MM_ANISOTROPIC
;
329 mfpict
->xExt
= sizeMF
.x
;
330 mfpict
->yExt
= sizeMF
.y
;
332 PixelToHIMETRIC(&mfpict
->xExt
, &mfpict
->yExt
);
338 bool wxEnhMetaFileDataObject::SetData(const wxDataFormat
& format
,
339 size_t WXUNUSED(len
),
344 if ( format
== wxDF_ENHMETAFILE
)
346 hEMF
= *(HENHMETAFILE
*)buf
;
348 wxCHECK_MSG( hEMF
, FALSE
, _T("pasting invalid enh metafile") );
352 wxASSERT_MSG( format
== wxDF_METAFILE
, _T("unsupported format") );
355 const METAFILEPICT
*mfpict
= (const METAFILEPICT
*)buf
;
357 // first get the buffer size
358 size_t size
= ::GetMetaFileBitsEx(mfpict
->hMF
, 0, NULL
);
359 wxCHECK_MSG( size
, FALSE
, _T("GetMetaFileBitsEx() failed") );
361 // then get metafile bits
362 BYTE
*bits
= (BYTE
*)malloc(size
);
363 if ( !::GetMetaFileBitsEx(mfpict
->hMF
, size
, bits
) )
365 wxLogLastError(_T("GetMetaFileBitsEx"));
374 // and finally create an enhanced metafile from them
375 hEMF
= ::SetWinMetaFileBits(size
, bits
, hdcRef
, mfpict
);
379 wxLogLastError(_T("SetWinMetaFileBits"));
385 m_metafile
.SetHENHMETAFILE((WXHANDLE
)hEMF
);
389 #endif // wxUSE_DRAG_AND_DROP
391 #endif // wxUSE_ENH_METAFILE