X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/33ac7e6f01acbac1cff0ad400d8ea7f0bfd0a62f..397cfa3ae9b44ad6a4cc5328e2637fa886c43e12:/src/msw/dcprint.cpp?ds=sidebyside diff --git a/src/msw/dcprint.cpp b/src/msw/dcprint.cpp index c77f88f080..c0f9a2c097 100644 --- a/src/msw/dcprint.cpp +++ b/src/msw/dcprint.cpp @@ -5,7 +5,7 @@ // Modified by: // Created: 01/02/97 // RCS-ID: $Id$ -// Copyright: (c) Julian Smart and Markus Holzem +// Copyright: (c) Julian Smart // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// @@ -38,10 +38,15 @@ #if wxUSE_PRINTING_ARCHITECTURE #include "wx/msw/private.h" + +#if wxUSE_WXDIB +#include "wx/msw/dib.h" +#endif + #include "wx/dcprint.h" #include "math.h" -#if wxUSE_COMMON_DIALOGS || defined(__WXWINE__) +#if wxUSE_COMMON_DIALOGS #include #endif @@ -50,7 +55,7 @@ #endif // mingw32 defines GDI_ERROR incorrectly -#ifdef __GNUWIN32__ +#if defined(__GNUWIN32__) || !defined(GDI_ERROR) #undef GDI_ERROR #define GDI_ERROR ((int)-1) #endif @@ -194,7 +199,7 @@ bool wxPrinterDC::StartDoc(const wxString& message) if (ret <= 0) { DWORD lastError = GetLastError(); - wxLogDebug(wxT("wxDC::StartDoc failed with error: %d\n"), lastError); + wxLogDebug(wxT("wxDC::StartDoc failed with error: %ld\n"), lastError); } #endif @@ -224,16 +229,18 @@ static bool wxGetDefaultDeviceName(wxString& deviceName, wxString& portName) deviceName.clear(); LPDEVNAMES lpDevNames; - LPSTR lpszDriverName; - LPSTR lpszDeviceName; - LPSTR lpszPortName; + LPTSTR lpszDriverName; + LPTSTR lpszDeviceName; + LPTSTR lpszPortName; PRINTDLG pd; // Cygwin has trouble believing PRINTDLG is 66 bytes - thinks it is 68 #ifdef __GNUWIN32__ + memset(&pd, 0, 66); pd.lStructSize = 66; // sizeof(PRINTDLG); #else + memset(&pd, 0, sizeof(PRINTDLG)); pd.lStructSize = sizeof(PRINTDLG); #endif @@ -256,9 +263,9 @@ static bool wxGetDefaultDeviceName(wxString& deviceName, wxString& portName) if (pd.hDevNames) { lpDevNames = (LPDEVNAMES)GlobalLock(pd.hDevNames); - lpszDriverName = (LPSTR)lpDevNames + lpDevNames->wDriverOffset; - lpszDeviceName = (LPSTR)lpDevNames + lpDevNames->wDeviceOffset; - lpszPortName = (LPSTR)lpDevNames + lpDevNames->wOutputOffset; + lpszDriverName = (LPTSTR)lpDevNames + lpDevNames->wDriverOffset; + lpszDeviceName = (LPTSTR)lpDevNames + lpDevNames->wDeviceOffset; + lpszPortName = (LPTSTR)lpDevNames + lpDevNames->wOutputOffset; deviceName = lpszDeviceName; portName = lpszPortName; @@ -273,85 +280,8 @@ static bool wxGetDefaultDeviceName(wxString& deviceName, wxString& portName) GlobalFree(pd.hDevMode); pd.hDevMode=NULL; } - return ( deviceName != wxT("") ); -} - -#if 0 -// This uses defaults, except for orientation, so we should eliminate this function -// and use the 2nd form (passing wxPrintData) instead. -WXHDC wxGetPrinterDC(int orientation) -{ - HDC hDC; - LPDEVMODE lpDevMode = NULL; - LPDEVNAMES lpDevNames; - LPSTR lpszDriverName; - LPSTR lpszDeviceName; - LPSTR lpszPortName; - - PRINTDLG pd; - // __GNUWIN32__ has trouble believing PRINTDLG is 66 bytes - thinks it is 68 -#ifdef __GNUWIN32__ - pd.lStructSize = 66; // sizeof(PRINTDLG); -#else - pd.lStructSize = sizeof(PRINTDLG); -#endif - pd.hwndOwner = (HWND)NULL; - pd.hDevMode = NULL; // Will be created by PrintDlg - pd.hDevNames = NULL; // Ditto - pd.Flags = PD_RETURNDEFAULT; - pd.nCopies = 1; - - if (!PrintDlg((LPPRINTDLG)&pd)) - { - if ( pd.hDevMode ) - GlobalFree(pd.hDevMode); - if (pd.hDevNames) - GlobalFree(pd.hDevNames); - - return(0); - } - - if (!pd.hDevNames) - { - if ( pd.hDevMode ) - GlobalFree(pd.hDevMode); - } - - lpDevNames = (LPDEVNAMES)GlobalLock(pd.hDevNames); - lpszDriverName = (LPSTR)lpDevNames + lpDevNames->wDriverOffset; - lpszDeviceName = (LPSTR)lpDevNames + lpDevNames->wDeviceOffset; - lpszPortName = (LPSTR)lpDevNames + lpDevNames->wOutputOffset; - GlobalUnlock(pd.hDevNames); - - if ( pd.hDevMode ) - { - lpDevMode = (DEVMODE*) GlobalLock(pd.hDevMode); - lpDevMode->dmOrientation = orientation; - lpDevMode->dmFields |= DM_ORIENTATION; - } - -#ifdef __WIN32__ - hDC = CreateDC(lpszDriverName, lpszDeviceName, lpszPortName, (DEVMODE *)lpDevMode); -#else - hDC = CreateDC(lpszDriverName, lpszDeviceName, lpszPortName, (LPSTR)lpDevMode); -#endif - - if (pd.hDevMode && lpDevMode) - GlobalUnlock(pd.hDevMode); - - if (pd.hDevNames) - { - GlobalFree(pd.hDevNames); - pd.hDevNames=NULL; - } - if (pd.hDevMode) - { - GlobalFree(pd.hDevMode); - pd.hDevMode=NULL; - } - return (WXHDC) hDC; + return ( deviceName != wxEmptyString ); } -#endif // 0 // Gets an HDC for the specified printer configuration WXHDC WXDLLEXPORT wxGetPrinterDC(const wxPrintData& printDataConst) @@ -409,12 +339,51 @@ WXHDC WXDLLEXPORT wxGetPrinterDC(const wxPrintData& printDataConst) // wxPrinterDC bit blitting/bitmap drawing // ---------------------------------------------------------------------------- -// Win16 doesn't define GDI_ERROR. -#ifndef GDI_ERROR -#define GDI_ERROR -1 +// helper of DoDrawBitmap() and DoBlit() +static +bool DrawBitmapUsingStretchDIBits(HDC hdc, + const wxBitmap& bmp, + wxCoord x, wxCoord y) +{ +#if wxUSE_WXDIB + wxDIB dib(bmp); + if ( !dib.IsOk() ) + return FALSE; + + DIBSECTION ds; + if ( !::GetObject(dib.GetHandle(), sizeof(ds), &ds) ) + { + wxLogLastError(_T("GetObject(DIBSECTION)")); + + return FALSE; + } + + // ok, we've got all data we need, do blit it + if ( ::StretchDIBits + ( + hdc, + x, y, + ds.dsBmih.biWidth, ds.dsBmih.biHeight, + 0, 0, + ds.dsBmih.biWidth, ds.dsBmih.biHeight, + ds.dsBm.bmBits, + (LPBITMAPINFO)&ds.dsBmih, + DIB_RGB_COLORS, + SRCCOPY + ) == GDI_ERROR ) + { + wxLogLastError(wxT("StretchDIBits")); + + return FALSE; + } + + return TRUE; +#else + return FALSE; #endif +} -void wxPrinterDC::DoDrawBitmap(const wxBitmap &bmp, +void wxPrinterDC::DoDrawBitmap(const wxBitmap& bmp, wxCoord x, wxCoord y, bool useMask) { @@ -423,42 +392,10 @@ void wxPrinterDC::DoDrawBitmap(const wxBitmap &bmp, int width = bmp.GetWidth(), height = bmp.GetHeight(); - if ( ::GetDeviceCaps(GetHdc(), RASTERCAPS) & RC_STRETCHDIB ) - { - BITMAPINFO *info = (BITMAPINFO *) malloc( sizeof( BITMAPINFOHEADER ) + 256 * sizeof(RGBQUAD ) ); - memset( info, 0, sizeof( BITMAPINFOHEADER ) ); - - int iBitsSize = ((width + 3 ) & ~3 ) * height; - - void* bits = malloc( iBitsSize ); - - info->bmiHeader.biSize = sizeof( BITMAPINFOHEADER ); - info->bmiHeader.biWidth = width; - info->bmiHeader.biHeight = height; - info->bmiHeader.biPlanes = 1; - info->bmiHeader.biBitCount = 8; - info->bmiHeader.biCompression = BI_RGB; - - ScreenHDC display; - if ( GetDIBits(display, GetHbitmapOf(bmp), 0, - bmp.GetHeight(), bits, info, - DIB_RGB_COLORS) ) - { - if ( ::StretchDIBits(GetHdc(), x, y, - width, height, - 0 , 0, width, height, - bits, info, - DIB_RGB_COLORS, SRCCOPY) == GDI_ERROR ) - { - wxLogLastError(wxT("StretchDIBits")); - } - } - - free(bits); - free(info); - } - else // no support for StretchDIBits() + if ( !(::GetDeviceCaps(GetHdc(), RASTERCAPS) & RC_STRETCHDIB) || + !DrawBitmapUsingStretchDIBits(GetHdc(), bmp, x, y) ) { + // no support for StretchDIBits() or an error occured if we got here wxMemoryDC memDC; memDC.SelectObject(bmp); @@ -471,29 +408,29 @@ void wxPrinterDC::DoDrawBitmap(const wxBitmap &bmp, bool wxPrinterDC::DoBlit(wxCoord xdest, wxCoord ydest, wxCoord width, wxCoord height, wxDC *source, - wxCoord xsrc, wxCoord ysrc, - int WXUNUSED(rop), bool useMask) + wxCoord WXUNUSED(xsrc), wxCoord WXUNUSED(ysrc), + int WXUNUSED(rop), bool useMask, + wxCoord WXUNUSED(xsrcMask), wxCoord WXUNUSED(ysrcMask)) { - bool success = TRUE; - - if ( useMask ) + wxBitmap& bmp = source->GetSelectedBitmap(); + wxMask *mask = useMask ? bmp.GetMask() : NULL; + if ( mask ) { - // If we are printing source colours are screen colours - // not printer colours and so we need copy the bitmap - // pixel by pixel. + // If we are printing source colours are screen colours not printer + // colours and so we need copy the bitmap pixel by pixel. RECT rect; - HDC dc_src = GetHdcOf(*source); - HDC dc_mask = ::CreateCompatibleDC(dc_src); + HDC dcSrc = GetHdcOf(*source); + MemoryHDC dcMask(dcSrc); + SelectInHDC selectMask(dcMask, (HBITMAP)mask->GetMaskBitmap()); - ::SelectObject(dc_mask, (HBITMAP) source->GetSelectedBitmap().GetMask()->GetMaskBitmap()); for (int x = 0; x < width; x++) { for (int y = 0; y < height; y++) { - COLORREF cref = ::GetPixel(dc_mask, x, y); + COLORREF cref = ::GetPixel(dcMask, x, y); if (cref) { - HBRUSH brush = ::CreateSolidBrush(::GetPixel(dc_src, x, y)); + HBRUSH brush = ::CreateSolidBrush(::GetPixel(dcSrc, x, y)); rect.left = xdest + x; rect.right = rect.left + 1; rect.top = ydest + y; @@ -503,76 +440,30 @@ bool wxPrinterDC::DoBlit(wxCoord xdest, wxCoord ydest, } } } - ::SelectObject(dc_mask, 0); - ::DeleteDC(dc_mask); } else // no mask { - if ( ::GetDeviceCaps(GetHdc(), RASTERCAPS) & RC_STRETCHDIB ) + if ( !(::GetDeviceCaps(GetHdc(), RASTERCAPS) & RC_STRETCHDIB) || + !DrawBitmapUsingStretchDIBits(GetHdc(), bmp, xdest, ydest) ) { - wxBitmap& bmp = source->GetSelectedBitmap(); - int width = bmp.GetWidth(), - height = bmp.GetHeight(); - - BITMAPINFO *info = (BITMAPINFO *) malloc( sizeof( BITMAPINFOHEADER ) + 256 * sizeof(RGBQUAD ) ); - int iBitsSize = ((width + 3 ) & ~3 ) * height; - - void* bits = malloc( iBitsSize ); - - memset( info , 0 , sizeof( BITMAPINFOHEADER ) ); + // no support for StretchDIBits - info->bmiHeader.biSize = sizeof( BITMAPINFOHEADER ); - info->bmiHeader.biWidth = width; - info->bmiHeader.biHeight = height; - info->bmiHeader.biPlanes = 1; - info->bmiHeader.biBitCount = 8; - info->bmiHeader.biCompression = BI_RGB; - - ScreenHDC display; - if ( !::GetDIBits(display, GetHbitmapOf(bmp), 0, - height, bits, info, DIB_RGB_COLORS) ) - { - wxLogLastError(wxT("GetDIBits")); - - success = FALSE; - } - - if ( success ) - { - success = ::StretchDIBits(GetHdc(), xdest, ydest, - width, height, - xsrc, ysrc, - width, height, - bits, info , - DIB_RGB_COLORS, - SRCCOPY) != GDI_ERROR; - if ( !success ) - { - wxLogLastError(wxT("StretchDIBits")); - } - } - - free(bits); - free(info); - } - else // no support for StretchDIBits - { - // as we are printing, source colours are screen colours not printer - // colours and so we need copy the bitmap pixel by pixel. - HDC dc_src = GetHdcOf(*source); + // as we are printing, source colours are screen colours not + // printer colours and so we need copy the bitmap pixel by pixel. + HDC dcSrc = GetHdcOf(*source); RECT rect; for (int y = 0; y < height; y++) { - // This is Stefan Csomor's optimisation, where identical adjacent - // pixels are drawn together. + // optimization: draw identical adjacent pixels together. for (int x = 0; x < width; x++) { - COLORREF col = ::GetPixel(dc_src, x, y); + COLORREF col = ::GetPixel(dcSrc, x, y); HBRUSH brush = ::CreateSolidBrush( col ); rect.left = xdest + x; rect.top = ydest + y; - while( (x + 1 < width) && (::GetPixel(dc_src, x + 1, y) == col ) ) + while( (x + 1 < width) && + (::GetPixel(dcSrc, x + 1, y) == col ) ) { ++x; } @@ -585,7 +476,7 @@ bool wxPrinterDC::DoBlit(wxCoord xdest, wxCoord ydest, } } - return success; + return TRUE; } #endif