#include "wx/dcprint.h"
#include "wx/module.h"
#include "wx/dynload.h"
+
+#ifdef wxHAVE_RAW_BITMAP
#include "wx/rawbmp.h"
+#endif
#include <string.h>
#include <math.h>
// convert degrees to radians
static inline double DegToRad(double deg) { return (deg * M_PI) / 180.0; }
+// call AlphaBlend() to blit contents of hdcSrc to hdcDst using alpha
+//
+// NB: bmpSrc is the bitmap selected in hdcSrc, it is not really needed
+// to pass it to this function but as we already have it at the point
+// of call anyhow we do
+//
+// return true if we could draw the bitmap in one way or the other, false
+// otherwise
+static bool AlphaBlt(HDC hdcDst,
+ int x, int y, int w, int h,
+ HDC hdcSrc,
+ const wxBitmap& bmpSrc);
+
+#ifdef wxHAVE_RAW_BITMAP
// our (limited) AlphaBlend() replacement
static void
-wxAlphaBlend(wxDC& dc, int x, int y, int w, int h, const wxBitmap& bmp);
+wxAlphaBlend(HDC hdcDst, int x, int y, int w, int h, const wxBitmap& bmp);
+#endif
// ----------------------------------------------------------------------------
// private classes
wxCoord x2 = x1 + width,
y2 = y1 + height;
-#if defined(__WIN32__) && !defined(__SC__) && !defined(__WXMICROWIN__)
+#if defined(__WIN32__) && !defined(__SYMANTEC__) && !defined(__WXMICROWIN__)
RECT rect;
rect.left = x1;
rect.top = y1;
HPALETTE oldPal = 0;
#endif // wxUSE_PALETTE
- // do we have AlphaBlend() and company in the headers?
-#ifdef AC_SRC_OVER
if ( bmp.HasAlpha() )
{
- // yes, now try to see if we have it during run-time
-
- typedef BOOL (WINAPI *AlphaBlend_t)(HDC,int,int,int,int,
- HDC,int,int,int,int,
- BLENDFUNCTION);
-
- // bitmaps can be drawn only from GUI thread so there is no need to
- // protect this static variable from multiple threads
- static bool s_triedToLoad = FALSE;
- static AlphaBlend_t pfnAlphaBlend = NULL;
- if ( !s_triedToLoad )
- {
- s_triedToLoad = TRUE;
-
- // don't give errors about the DLL being unavailable, we're
- // prepared to handle this
- wxLogNull nolog;
-
- wxDynamicLibrary dll(_T("msimg32.dll"));
- if ( dll.IsLoaded() )
- {
- pfnAlphaBlend = (AlphaBlend_t)dll.GetSymbol(_T("AlphaBlend"));
- if ( pfnAlphaBlend )
- {
- // we must keep the DLL loaded if we want to be able to
- // call AlphaBlend() so just never unload it at all, not a
- // big deal
- dll.Detach();
- }
- }
- }
-
- if ( pfnAlphaBlend )
- {
- MemoryHDC hdcMem;
- SelectInHDC select(hdcMem, GetHbitmapOf(bmp));
-
- BLENDFUNCTION bf;
- bf.BlendOp = AC_SRC_OVER;
- bf.BlendFlags = 0;
- bf.SourceConstantAlpha = 0xff;
- bf.AlphaFormat = AC_SRC_ALPHA;
-
- if ( !pfnAlphaBlend(GetHdc(), x, y, width, height,
- hdcMem, 0, 0, width, height,
- bf) )
- {
- wxLogLastError(_T("AlphaBlend"));
- }
- }
- else // use our own (probably much slower) implementation
- {
- wxAlphaBlend(*this, x, y, width, height, bmp);
- }
+ MemoryHDC hdcMem;
+ SelectInHDC select(hdcMem, GetHbitmapOf(bmp));
- return;
+ if ( AlphaBlt(GetHdc(), x, y, width, height, hdcMem, bmp) )
+ return;
}
-#endif // defined(AC_SRC_OVER)
if ( useMask )
{
m_backgroundBrush = brush;
- if (!m_backgroundBrush.Ok())
- return;
-
- if (m_canvas)
- {
- bool customColours = TRUE;
- // If we haven't specified wxUSER_COLOURS, don't allow the panel/dialog box to
- // change background colours from the control-panel specified colours.
- if (m_canvas->IsKindOf(CLASSINFO(wxWindow)) && ((m_canvas->GetWindowStyleFlag() & wxUSER_COLOURS) != wxUSER_COLOURS))
- customColours = FALSE;
-
- if (customColours)
- {
- if (m_backgroundBrush.GetStyle()==wxTRANSPARENT)
- {
- m_canvas->SetTransparent(TRUE);
- }
- else
- {
- // New behaviour, 10/2/99: setting the background brush of a DC
- // doesn't affect the window background colour. However,
- // I'm leaving in the transparency setting because it's needed by
- // various controls (e.g. wxStaticText) to determine whether to draw
- // transparently or not. TODO: maybe this should be a new function
- // wxWindow::SetTransparency(). Should that apply to the child itself, or the
- // parent?
- // m_canvas->SetBackgroundColour(m_backgroundBrush.GetColour());
- m_canvas->SetTransparent(FALSE);
- }
- }
- }
- COLORREF new_color = m_backgroundBrush.GetColour().GetPixel();
+ if ( m_backgroundBrush.Ok() )
{
- (void)SetBkColor(GetHdc(), new_color);
+ (void)SetBkColor(GetHdc(), m_backgroundBrush.GetColour().GetPixel());
}
}
return;
}
- double mm2pixelsX = pixel_width / mm_width,
- mm2pixelsY = pixel_height / mm_height;
+ double mm2pixelsX = (double)pixel_width / mm_width,
+ mm2pixelsY = (double)pixel_height / mm_height;
switch (mode)
{
int rop, bool useMask,
wxCoord xsrcMask, wxCoord ysrcMask)
{
+ wxCHECK_MSG( source, FALSE, _T("wxDC::Blit(): NULL wxDC pointer") );
+
#ifdef __WXMICROWIN__
if (!GetHDC()) return FALSE;
#endif
const wxBitmap& bmpSrc = source->m_selectedBitmap;
+ if ( bmpSrc.Ok() && bmpSrc.HasAlpha() )
+ {
+ if ( AlphaBlt(GetHdc(), xdest, ydest, width, height,
+ GetHdcOf(*source), bmpSrc) )
+ return TRUE;
+ }
wxMask *mask = NULL;
if ( useMask )
#endif // wxUSE_DC_CACHEING
// ----------------------------------------------------------------------------
-// wxAlphaBlend: our fallback if ::AlphaBlend() is unavailable
+// alpha channel support
// ----------------------------------------------------------------------------
+static bool AlphaBlt(HDC hdcDst,
+ int x, int y, int width, int height,
+ HDC hdcSrc,
+ const wxBitmap& bmp)
+{
+ wxASSERT_MSG( bmp.Ok() && bmp.HasAlpha(), _T("AlphaBlt(): invalid bitmap") );
+ wxASSERT_MSG( hdcDst && hdcSrc, _T("AlphaBlt(): invalid HDC") );
+
+ // do we have AlphaBlend() and company in the headers?
+#if defined(AC_SRC_OVER) && wxUSE_DYNLIB_CLASS
+ // yes, now try to see if we have it during run-time
+ typedef BOOL (WINAPI *AlphaBlend_t)(HDC,int,int,int,int,
+ HDC,int,int,int,int,
+ BLENDFUNCTION);
+
+ // bitmaps can be drawn only from GUI thread so there is no need to
+ // protect this static variable from multiple threads
+ static bool s_triedToLoad = FALSE;
+ static AlphaBlend_t pfnAlphaBlend = NULL;
+ if ( !s_triedToLoad )
+ {
+ s_triedToLoad = TRUE;
+
+ // don't give errors about the DLL being unavailable, we're
+ // prepared to handle this
+ wxLogNull nolog;
+
+ wxDynamicLibrary dll(_T("msimg32.dll"));
+ if ( dll.IsLoaded() )
+ {
+ pfnAlphaBlend = (AlphaBlend_t)dll.GetSymbol(_T("AlphaBlend"));
+ if ( pfnAlphaBlend )
+ {
+ // we must keep the DLL loaded if we want to be able to
+ // call AlphaBlend() so just never unload it at all, not a
+ // big deal
+ dll.Detach();
+ }
+ }
+ }
+
+ if ( pfnAlphaBlend )
+ {
+ BLENDFUNCTION bf;
+ bf.BlendOp = AC_SRC_OVER;
+ bf.BlendFlags = 0;
+ bf.SourceConstantAlpha = 0xff;
+ bf.AlphaFormat = AC_SRC_ALPHA;
+
+ if ( pfnAlphaBlend(hdcDst, x, y, width, height,
+ hdcSrc, 0, 0, width, height,
+ bf) )
+ {
+ // skip wxAlphaBlend() call below
+ return TRUE;
+ }
+
+ wxLogLastError(_T("AlphaBlend"));
+ }
+#endif // defined(AC_SRC_OVER)
+
+ // AlphaBlend() unavailable of failed: use our own (probably much slower)
+ // implementation
+#ifdef wxHAVE_RAW_BITMAP
+ wxAlphaBlend(hdcDst, x, y, width, height, bmp);
+
+ return TRUE;
+#else // !wxHAVE_RAW_BITMAP
+ // no wxAlphaBlend() neither, fall back to using simple BitBlt() (we lose
+ // alpha but at least something will be shown like this)
+ return FALSE;
+#endif // wxHAVE_RAW_BITMAP
+}
+
+
+// wxAlphaBlend: our fallback if ::AlphaBlend() is unavailable
+#ifdef wxHAVE_RAW_BITMAP
+
static void
-wxAlphaBlend(wxDC& dc, int xDst, int yDst, int w, int h, const wxBitmap& bmpSrc)
+wxAlphaBlend(HDC hdcDst, int xDst, int yDst, int w, int h, const wxBitmap& bmpSrc)
{
// get the destination DC pixels
wxBitmap bmpDst(w, h, 32 /* force creating RGBA DIB */);
MemoryHDC hdcMem;
SelectInHDC select(hdcMem, GetHbitmapOf(bmpDst));
- if ( !::BitBlt(hdcMem, 0, 0, w, h, GetHdcOf(dc), 0, 0, SRCCOPY) )
+ if ( !::BitBlt(hdcMem, 0, 0, w, h, hdcDst, 0, 0, SRCCOPY) )
{
wxLogLastError(_T("BitBlt"));
}
// combine them with the source bitmap using alpha
- wxRawBitmapData dataDst(bmpDst),
- dataSrc(bmpSrc);
+ wxAlphaPixelData dataDst(bmpDst),
+ dataSrc((wxBitmap &)bmpSrc);
wxCHECK_RET( dataDst && dataSrc,
_T("failed to get raw data in wxAlphaBlend") );
- wxRawBitmapIterator pDst(dataDst),
- pSrc(dataSrc);
+ wxAlphaPixelData::Iterator pDst(dataDst),
+ pSrc(dataSrc);
for ( int y = 0; y < h; y++ )
{
- wxRawBitmapIterator pDstRowStart = pDst,
- pSrcRowStart = pSrc;
+ wxAlphaPixelData::Iterator pDstRowStart = pDst,
+ pSrcRowStart = pSrc;
for ( int x = 0; x < w; x++ )
{
pDst = pDstRowStart;
pSrc = pSrcRowStart;
- pDst.OffsetY(1);
- pSrc.OffsetY(1);
+ pDst.OffsetY(dataDst, 1);
+ pSrc.OffsetY(dataSrc, 1);
}
// and finally blit them back to the destination DC
- if ( !::BitBlt(GetHdcOf(dc), xDst, yDst, w, h, hdcMem, 0, 0, SRCCOPY) )
+ if ( !::BitBlt(hdcDst, xDst, yDst, w, h, hdcMem, 0, 0, SRCCOPY) )
{
wxLogLastError(_T("BitBlt"));
}
}
+#endif // #ifdef wxHAVE_RAW_BITMAP