X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/a2327a9f80fa3240ab652da8f95a66b5c25576de..cdbd62d6ff290fd58acd1bc5574dfc79db3a6f70:/src/msw/enhmeta.cpp diff --git a/src/msw/enhmeta.cpp b/src/msw/enhmeta.cpp index 189d2980b8..f7c0f7abb0 100644 --- a/src/msw/enhmeta.cpp +++ b/src/msw/enhmeta.cpp @@ -1,12 +1,11 @@ /////////////////////////////////////////////////////////////////////////////// -// Name: msw/enhmeta.cpp +// Name: src/msw/enhmeta.cpp // Purpose: implementation of wxEnhMetaFileXXX classes // Author: Vadim Zeitlin // Modified by: // Created: 13.01.00 -// RCS-ID: $Id$ // Copyright: (c) 2000 Vadim Zeitlin -// Licence: wxWindows license +// Licence: wxWindows licence /////////////////////////////////////////////////////////////////////////////// // ============================================================================ @@ -17,10 +16,6 @@ // headers // ---------------------------------------------------------------------------- -#ifdef __GNUG__ - #pragma implementation "enhmeta.h" -#endif - // For compilers that support precompilation, includes "wx.h". #include "wx/wxprec.h" @@ -33,9 +28,14 @@ #ifndef WX_PRECOMP #include "wx/string.h" #include "wx/log.h" + #include "wx/intl.h" #endif //WX_PRECOMP +#include "wx/dc.h" +#include "wx/msw/dc.h" + #include "wx/metafile.h" +#include "wx/clipbrd.h" #include "wx/msw/private.h" @@ -44,7 +44,6 @@ // ---------------------------------------------------------------------------- IMPLEMENT_DYNAMIC_CLASS(wxEnhMetaFile, wxObject) -IMPLEMENT_ABSTRACT_CLASS(wxEnhMetaFileDC, wxDC) // ---------------------------------------------------------------------------- // macros @@ -59,7 +58,7 @@ IMPLEMENT_ABSTRACT_CLASS(wxEnhMetaFileDC, wxDC) // we must pass NULL if the string is empty to metafile functions static inline const wxChar *GetMetaFileName(const wxString& fn) - { return !fn ? (wxChar *)NULL : fn.c_str(); } + { return !fn ? NULL : wxMSW_CONV_LPCTSTR(fn); } // ============================================================================ // implementation @@ -69,6 +68,38 @@ static inline const wxChar *GetMetaFileName(const wxString& fn) // wxEnhMetaFile // ---------------------------------------------------------------------------- +wxGDIRefData *wxEnhMetaFile::CreateGDIRefData() const +{ + wxFAIL_MSG( wxT("must be implemented if used") ); + + return NULL; +} + +wxGDIRefData * +wxEnhMetaFile::CloneGDIRefData(const wxGDIRefData *WXUNUSED(data)) const +{ + wxFAIL_MSG( wxT("must be implemented if used") ); + + return NULL; +} + +void wxEnhMetaFile::Init() +{ + if ( m_filename.empty() ) + { + m_hMF = 0; + } + else // have valid file name, load metafile from it + { + m_hMF = (WXHANDLE)::GetEnhMetaFile(m_filename.t_str()); + if ( !m_hMF ) + { + wxLogSysError(_("Failed to load metafile from file \"%s\"."), + m_filename.c_str()); + } + } +} + void wxEnhMetaFile::Assign(const wxEnhMetaFile& mf) { if ( &mf == this ) @@ -80,7 +111,7 @@ void wxEnhMetaFile::Assign(const wxEnhMetaFile& mf) GetMetaFileName(m_filename)); if ( !m_hMF ) { - wxLogLastError(_T("CopyEnhMetaFile")); + wxLogLastError(wxT("CopyEnhMetaFile")); } } else @@ -95,15 +126,15 @@ void wxEnhMetaFile::Free() { if ( !::DeleteEnhMetaFile(GetEMF()) ) { - wxLogLastError(_T("DeleteEnhMetaFile")); + wxLogLastError(wxT("DeleteEnhMetaFile")); } } } bool wxEnhMetaFile::Play(wxDC *dc, wxRect *rectBound) { - wxCHECK_MSG( Ok(), FALSE, _T("can't play invalid enhanced metafile") ); - wxCHECK_MSG( dc, FALSE, _T("invalid wxDC in wxEnhMetaFile::Play") ); + wxCHECK_MSG( IsOk(), false, wxT("can't play invalid enhanced metafile") ); + wxCHECK_MSG( dc, false, wxT("invalid wxDC in wxEnhMetaFile::Play") ); RECT rect; if ( rectBound ) @@ -123,26 +154,31 @@ bool wxEnhMetaFile::Play(wxDC *dc, wxRect *rectBound) rect.bottom = size.y; } - if ( !::PlayEnhMetaFile(GetHdcOf(*dc), GetEMF(), &rect) ) + wxDCImpl *impl = dc->GetImpl(); + wxMSWDCImpl *msw_impl = wxDynamicCast( impl, wxMSWDCImpl ); + if (!msw_impl) + return false; + + if ( !::PlayEnhMetaFile(GetHdcOf(*msw_impl), GetEMF(), &rect) ) { - wxLogLastError(_T("PlayEnhMetaFile")); + wxLogLastError(wxT("PlayEnhMetaFile")); - return FALSE; + return false; } - return TRUE; + return true; } wxSize wxEnhMetaFile::GetSize() const { wxSize size = wxDefaultSize; - if ( Ok() ) + if ( IsOk() ) { ENHMETAHEADER hdr; if ( !::GetEnhMetaFileHeader(GetEMF(), sizeof(hdr), &hdr) ) { - wxLogLastError(_T("GetEnhMetaFileHeader")); + wxLogLastError(wxT("GetEnhMetaFileHeader")); } else { @@ -161,17 +197,80 @@ wxSize wxEnhMetaFile::GetSize() const return size; } +bool wxEnhMetaFile::SetClipboard(int WXUNUSED(width), int WXUNUSED(height)) +{ +#if wxUSE_DRAG_AND_DROP && wxUSE_CLIPBOARD + wxCHECK_MSG( m_hMF, false, wxT("can't copy invalid metafile to clipboard") ); + + return wxTheClipboard->AddData(new wxEnhMetaFileDataObject(*this)); +#else // !wxUSE_DRAG_AND_DROP + wxFAIL_MSG(wxT("not implemented")); + return false; +#endif // wxUSE_DRAG_AND_DROP/!wxUSE_DRAG_AND_DROP +} + // ---------------------------------------------------------------------------- -// wxEnhMetaFileDC +// wxEnhMetaFileDCImpl // ---------------------------------------------------------------------------- -wxEnhMetaFileDC::wxEnhMetaFileDC(const wxString& filename, +class wxEnhMetaFileDCImpl : public wxMSWDCImpl +{ +public: + wxEnhMetaFileDCImpl( wxEnhMetaFileDC *owner, + const wxString& filename, int width, int height, + const wxString& description ); + wxEnhMetaFileDCImpl( wxEnhMetaFileDC *owner, + const wxDC& referenceDC, + const wxString& filename, int width, int height, + const wxString& description ); + virtual ~wxEnhMetaFileDCImpl(); + + // obtain a pointer to the new metafile (caller should delete it) + wxEnhMetaFile *Close(); + +protected: + virtual void DoGetSize(int *width, int *height) const; + +private: + void Create(HDC hdcRef, + const wxString& filename, int width, int height, + const wxString& description); + + // size passed to ctor and returned by DoGetSize() + int m_width, + m_height; +}; + + +wxEnhMetaFileDCImpl::wxEnhMetaFileDCImpl( wxEnhMetaFileDC* owner, + const wxString& filename, + int width, int height, + const wxString& description ) + : wxMSWDCImpl( owner ) +{ + Create(ScreenHDC(), filename, width, height, description); +} + +wxEnhMetaFileDCImpl::wxEnhMetaFileDCImpl( wxEnhMetaFileDC* owner, + const wxDC& referenceDC, + const wxString& filename, + int width, int height, + const wxString& description ) + : wxMSWDCImpl( owner ) +{ + Create(GetHdcOf(referenceDC), filename, width, height, description); +} + +void wxEnhMetaFileDCImpl::Create(HDC hdcRef, + const wxString& filename, int width, int height, const wxString& description) { - ScreenHDC hdcRef; + m_width = width; + m_height = height; - RECT rect, *pRect; + RECT rect; + RECT *pRect; if ( width && height ) { rect.top = @@ -180,7 +279,7 @@ wxEnhMetaFileDC::wxEnhMetaFileDC(const wxString& filename, rect.bottom = height; // CreateEnhMetaFile() wants them in HIMETRIC - PixelToHIMETRIC(&rect.right, &rect.bottom); + PixelToHIMETRIC(&rect.right, &rect.bottom, hdcRef); pRect = ▭ } @@ -191,21 +290,29 @@ wxEnhMetaFileDC::wxEnhMetaFileDC(const wxString& filename, } m_hDC = (WXHDC)::CreateEnhMetaFile(hdcRef, GetMetaFileName(filename), - pRect, description); + pRect, description.t_str()); if ( !m_hDC ) { - wxLogLastError(_T("CreateEnhMetaFile")); + wxLogLastError(wxT("CreateEnhMetaFile")); } } -wxEnhMetaFile *wxEnhMetaFileDC::Close() +void wxEnhMetaFileDCImpl::DoGetSize(int *width, int *height) const { - wxCHECK_MSG( Ok(), NULL, _T("invalid wxEnhMetaFileDC") ); + if ( width ) + *width = m_width; + if ( height ) + *height = m_height; +} + +wxEnhMetaFile *wxEnhMetaFileDCImpl::Close() +{ + wxCHECK_MSG( IsOk(), NULL, wxT("invalid wxEnhMetaFileDC") ); HENHMETAFILE hMF = ::CloseEnhMetaFile(GetHdc()); if ( !hMF ) { - wxLogLastError(_T("CloseEnhMetaFile")); + wxLogLastError(wxT("CloseEnhMetaFile")); return NULL; } @@ -215,18 +322,55 @@ wxEnhMetaFile *wxEnhMetaFileDC::Close() return mf; } -wxEnhMetaFileDC::~wxEnhMetaFileDC() +wxEnhMetaFileDCImpl::~wxEnhMetaFileDCImpl() { // avoid freeing it in the base class dtor m_hDC = 0; } // ---------------------------------------------------------------------------- -// wxEnhMetaFileDataObject +// wxEnhMetaFileDC // ---------------------------------------------------------------------------- +IMPLEMENT_ABSTRACT_CLASS(wxEnhMetaFileDC, wxDC) + +wxEnhMetaFileDC::wxEnhMetaFileDC(const wxString& filename, + int width, int height, + const wxString& description) + : wxDC(new wxEnhMetaFileDCImpl(this, + filename, + width, height, + description)) +{ +} + +wxEnhMetaFileDC::wxEnhMetaFileDC(const wxDC& referenceDC, + const wxString& filename, + int width, int height, + const wxString& description) + : wxDC(new wxEnhMetaFileDCImpl(this, + referenceDC, + filename, + width, height, + description)) +{ +} + +wxEnhMetaFile *wxEnhMetaFileDC::Close() +{ + wxEnhMetaFileDCImpl * const + impl = static_cast(GetImpl()); + wxCHECK_MSG( impl, NULL, wxT("no wxEnhMetaFileDC implementation") ); + + return impl->Close(); +} + #if wxUSE_DRAG_AND_DROP +// ---------------------------------------------------------------------------- +// wxEnhMetaFileDataObject +// ---------------------------------------------------------------------------- + wxDataFormat wxEnhMetaFileDataObject::GetPreferredFormat(Direction WXUNUSED(dir)) const { @@ -255,7 +399,7 @@ size_t wxEnhMetaFileDataObject::GetDataSize(const wxDataFormat& format) const } else { - wxASSERT_MSG( format == wxDF_METAFILE, _T("unsupported format") ); + wxASSERT_MSG( format == wxDF_METAFILE, wxT("unsupported format") ); return sizeof(METAFILEPICT); } @@ -263,7 +407,7 @@ size_t wxEnhMetaFileDataObject::GetDataSize(const wxDataFormat& format) const bool wxEnhMetaFileDataObject::GetDataHere(const wxDataFormat& format, void *buf) const { - wxCHECK_MSG( m_metafile.Ok(), FALSE, _T("copying invalid enh metafile") ); + wxCHECK_MSG( m_metafile.IsOk(), false, wxT("copying invalid enh metafile") ); HENHMETAFILE hEMF = (HENHMETAFILE)m_metafile.GetHENHMETAFILE(); @@ -272,16 +416,16 @@ bool wxEnhMetaFileDataObject::GetDataHere(const wxDataFormat& format, void *buf) HENHMETAFILE hEMFCopy = ::CopyEnhMetaFile(hEMF, NULL); if ( !hEMFCopy ) { - wxLogLastError(_T("CopyEnhMetaFile")); + wxLogLastError(wxT("CopyEnhMetaFile")); - return FALSE; + return false; } *(HENHMETAFILE *)buf = hEMFCopy; } else { - wxASSERT_MSG( format == wxDF_METAFILE, _T("unsupported format") ); + wxASSERT_MSG( format == wxDF_METAFILE, wxT("unsupported format") ); // convert to WMF @@ -289,18 +433,18 @@ bool wxEnhMetaFileDataObject::GetDataHere(const wxDataFormat& format, void *buf) // first get the buffer size and alloc memory size_t size = ::GetWinMetaFileBits(hEMF, 0, NULL, MM_ANISOTROPIC, hdc); - wxCHECK_MSG( size, FALSE, _T("GetWinMetaFileBits() failed") ); + wxCHECK_MSG( size, false, wxT("GetWinMetaFileBits() failed") ); BYTE *bits = (BYTE *)malloc(size); // then get the enh metafile bits if ( !::GetWinMetaFileBits(hEMF, size, bits, MM_ANISOTROPIC, hdc) ) { - wxLogLastError(_T("GetWinMetaFileBits")); + wxLogLastError(wxT("GetWinMetaFileBits")); free(bits); - return FALSE; + return false; } // and finally convert them to the WMF @@ -308,9 +452,9 @@ bool wxEnhMetaFileDataObject::GetDataHere(const wxDataFormat& format, void *buf) free(bits); if ( !hMF ) { - wxLogLastError(_T("SetMetaFileBitsEx")); + wxLogLastError(wxT("SetMetaFileBitsEx")); - return FALSE; + return false; } METAFILEPICT *mfpict = (METAFILEPICT *)buf; @@ -324,7 +468,7 @@ bool wxEnhMetaFileDataObject::GetDataHere(const wxDataFormat& format, void *buf) PixelToHIMETRIC(&mfpict->xExt, &mfpict->yExt); } - return TRUE; + return true; } bool wxEnhMetaFileDataObject::SetData(const wxDataFormat& format, @@ -337,28 +481,28 @@ bool wxEnhMetaFileDataObject::SetData(const wxDataFormat& format, { hEMF = *(HENHMETAFILE *)buf; - wxCHECK_MSG( hEMF, FALSE, _T("pasting invalid enh metafile") ); + wxCHECK_MSG( hEMF, false, wxT("pasting invalid enh metafile") ); } else { - wxASSERT_MSG( format == wxDF_METAFILE, _T("unsupported format") ); + wxASSERT_MSG( format == wxDF_METAFILE, wxT("unsupported format") ); // convert from WMF const METAFILEPICT *mfpict = (const METAFILEPICT *)buf; // first get the buffer size size_t size = ::GetMetaFileBitsEx(mfpict->hMF, 0, NULL); - wxCHECK_MSG( size, FALSE, _T("GetMetaFileBitsEx() failed") ); + wxCHECK_MSG( size, false, wxT("GetMetaFileBitsEx() failed") ); // then get metafile bits BYTE *bits = (BYTE *)malloc(size); if ( !::GetMetaFileBitsEx(mfpict->hMF, size, bits) ) { - wxLogLastError(_T("GetMetaFileBitsEx")); + wxLogLastError(wxT("GetMetaFileBitsEx")); free(bits); - return FALSE; + return false; } ScreenHDC hdcRef; @@ -368,16 +512,57 @@ bool wxEnhMetaFileDataObject::SetData(const wxDataFormat& format, free(bits); if ( !hEMF ) { - wxLogLastError(_T("SetWinMetaFileBits")); + wxLogLastError(wxT("SetWinMetaFileBits")); - return FALSE; + return false; } } m_metafile.SetHENHMETAFILE((WXHANDLE)hEMF); - return TRUE; + return true; } + +// ---------------------------------------------------------------------------- +// wxEnhMetaFileSimpleDataObject +// ---------------------------------------------------------------------------- + +size_t wxEnhMetaFileSimpleDataObject::GetDataSize() const +{ + // we pass data by handle and not HGLOBAL + return 0u; +} + +bool wxEnhMetaFileSimpleDataObject::GetDataHere(void *buf) const +{ + wxCHECK_MSG( m_metafile.IsOk(), false, wxT("copying invalid enh metafile") ); + + HENHMETAFILE hEMF = (HENHMETAFILE)m_metafile.GetHENHMETAFILE(); + + HENHMETAFILE hEMFCopy = ::CopyEnhMetaFile(hEMF, NULL); + if ( !hEMFCopy ) + { + wxLogLastError(wxT("CopyEnhMetaFile")); + + return false; + } + + *(HENHMETAFILE *)buf = hEMFCopy; + return true; +} + +bool wxEnhMetaFileSimpleDataObject::SetData(size_t WXUNUSED(len), + const void *buf) +{ + HENHMETAFILE hEMF = *(HENHMETAFILE *)buf; + + wxCHECK_MSG( hEMF, false, wxT("pasting invalid enh metafile") ); + m_metafile.SetHENHMETAFILE((WXHANDLE)hEMF); + + return true; +} + + #endif // wxUSE_DRAG_AND_DROP #endif // wxUSE_ENH_METAFILE