git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@9207 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
968 lines
29 KiB
C++
968 lines
29 KiB
C++
/////////////////////////////////////////////////////////////////////////////
|
|
// Name: bitmap.cpp
|
|
// Purpose: wxBitmap
|
|
// Author: David Webster
|
|
// Modified by:
|
|
// Created: 08/08/99
|
|
// RCS-ID: $Id$
|
|
// Copyright: (c) David Webster
|
|
// Licence: wxWindows licence
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
#ifdef __GNUG__
|
|
#pragma implementation "bitmap.h"
|
|
#endif
|
|
|
|
// For compilers that support precompilation, includes "wx.h".
|
|
#include "wx/wxprec.h"
|
|
|
|
#ifndef WX_PRECOMP
|
|
#include <stdio.h>
|
|
|
|
#include "wx/list.h"
|
|
#include "wx/utils.h"
|
|
#include "wx/app.h"
|
|
#include "wx/palette.h"
|
|
#include "wx/dcmemory.h"
|
|
#include "wx/bitmap.h"
|
|
#include "wx/icon.h"
|
|
#endif
|
|
|
|
#include "wx/os2/private.h"
|
|
#include "wx/log.h"
|
|
|
|
//#include "wx/msw/dib.h"
|
|
#include "wx/image.h"
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// macros
|
|
// ----------------------------------------------------------------------------
|
|
|
|
IMPLEMENT_DYNAMIC_CLASS(wxBitmap, wxGDIObject)
|
|
IMPLEMENT_DYNAMIC_CLASS(wxMask, wxObject)
|
|
|
|
IMPLEMENT_DYNAMIC_CLASS(wxBitmapHandler, wxObject)
|
|
|
|
// ============================================================================
|
|
// implementation
|
|
// ============================================================================
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxBitmapRefData
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxBitmapRefData::wxBitmapRefData()
|
|
{
|
|
m_nQuality = 0;
|
|
m_pSelectedInto = NULL;
|
|
m_nNumColors = 0;
|
|
m_pBitmapMask = NULL;
|
|
m_hBitmap = (WXHBITMAP) NULL;
|
|
} // end of wxBitmapRefData::wxBitmapRefData
|
|
|
|
void wxBitmapRefData::Free()
|
|
{
|
|
wxASSERT_MSG( !m_pSelectedInto,
|
|
wxT("deleting bitmap still selected into wxMemoryDC") );
|
|
|
|
if (m_hBitmap)
|
|
{
|
|
if ( !::GpiDeleteBitmap((HBITMAP)m_hBitmap) )
|
|
{
|
|
wxLogLastError("GpiDeleteBitmap(hbitmap)");
|
|
}
|
|
}
|
|
|
|
delete m_pBitmapMask;
|
|
m_pBitmapMask = NULL;
|
|
} // end of wxBitmapRefData::Free
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxBitmap creation
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// this function should be called from all wxBitmap ctors
|
|
void wxBitmap::Init()
|
|
{
|
|
// m_refData = NULL; done in the base class ctor
|
|
|
|
if (wxTheBitmapList)
|
|
wxTheBitmapList->AddBitmap(this);
|
|
} // end of wxBitmap::Init
|
|
|
|
bool wxBitmap::CopyFromIconOrCursor(
|
|
const wxGDIImage& rIcon
|
|
)
|
|
{
|
|
HPOINTER hIcon = (HPOINTER)rIcon.GetHandle();
|
|
POINTERINFO SIconInfo;
|
|
|
|
if (!::WinQueryPointerInfo(hIcon, &SIconInfo))
|
|
{
|
|
wxLogLastError(wxT("WinQueryPointerInfo"));
|
|
return FALSE;
|
|
}
|
|
wxBitmapRefData* pRefData = new wxBitmapRefData;
|
|
|
|
m_refData = pRefData;
|
|
|
|
int nWidth = rIcon.GetWidth();
|
|
int nHeight = rIcon.GetHeight();
|
|
|
|
pRefData->m_nWidth = nWidth;
|
|
pRefData->m_nHeight = nHeight;
|
|
pRefData->m_nDepth = wxDisplayDepth();
|
|
|
|
pRefData->m_hBitmap = (WXHBITMAP)SIconInfo.hbmColor;
|
|
|
|
//
|
|
// No mask in the Info struct in OS/2
|
|
//
|
|
return(TRUE);
|
|
} // end of wxBitmap::CopyFromIconOrCursor
|
|
|
|
bool wxBitmap::CopyFromCursor(
|
|
const wxCursor& rCursor
|
|
)
|
|
{
|
|
UnRef();
|
|
|
|
if (!rCursor.Ok())
|
|
return(FALSE);
|
|
return(CopyFromIconOrCursor(rCursor));
|
|
} // end of wxBitmap::CopyFromCursor
|
|
|
|
bool wxBitmap::CopyFromIcon(
|
|
const wxIcon& rIcon
|
|
)
|
|
{
|
|
UnRef();
|
|
|
|
if (!rIcon.Ok())
|
|
return(FALSE);
|
|
|
|
return CopyFromIconOrCursor(rIcon);
|
|
} // end of wxBitmap::CopyFromIcon
|
|
|
|
wxBitmap::~wxBitmap()
|
|
{
|
|
if (wxTheBitmapList)
|
|
wxTheBitmapList->DeleteObject(this);
|
|
} // end of wxBitmap::~wxBitmap
|
|
|
|
wxBitmap::wxBitmap(
|
|
const char zBits[]
|
|
, int nTheWidth
|
|
, int nTheHeight
|
|
, int nNoBits
|
|
)
|
|
{
|
|
Init();
|
|
|
|
wxBitmapRefData* pRefData = new wxBitmapRefData;
|
|
BITMAPINFOHEADER2 vHeader;
|
|
BITMAPINFO2 vInfo;
|
|
HDC hDc;
|
|
HPS hPs;
|
|
DEVOPENSTRUC vDop = { NULL, "DISPLAY", NULL, NULL, NULL, NULL, NULL, NULL, NULL };
|
|
SIZEL vSize = {0, 0};
|
|
|
|
wxASSERT(vHabmain != NULL);
|
|
|
|
hDc = ::DevOpenDC(vHabmain, OD_MEMORY, (PSZ)"*", 1L, (PDEVOPENDATA)&vDop, 0L);
|
|
|
|
vHeader.cbFix = sizeof(vHeader);
|
|
vHeader.cx = (USHORT)nTheWidth;
|
|
vHeader.cy = (USHORT)nTheHeight;
|
|
vHeader.cPlanes = 1L;
|
|
vHeader.cBitCount = nNoBits;
|
|
vHeader.ulCompression = BCA_UNCOMP;
|
|
vHeader.cxResolution = 0;
|
|
vHeader.cyResolution = 0;
|
|
vHeader.cclrUsed = 0;
|
|
vHeader.cclrImportant = 0;
|
|
vHeader.usUnits = BRU_METRIC;
|
|
vHeader.usRecording = BRA_BOTTOMUP;
|
|
vHeader.usRendering = BRH_NOTHALFTONED;
|
|
vHeader.cSize1 = 0;
|
|
vHeader.cSize2 = 0;
|
|
vHeader.ulColorEncoding = 0;
|
|
vHeader.ulIdentifier = 0;
|
|
|
|
hPs = ::GpiCreatePS(vHabmain, hDc, &vSize, GPIA_ASSOC | PU_PELS);
|
|
if (hPs == 0)
|
|
{
|
|
wxLogLastError("GpiCreatePS Failure");
|
|
}
|
|
|
|
m_refData = pRefData;
|
|
|
|
pRefData->m_nWidth = nTheWidth;
|
|
pRefData->m_nHeight = nTheHeight;
|
|
pRefData->m_nDepth = nNoBits;
|
|
pRefData->m_nNumColors = 0;
|
|
pRefData->m_pSelectedInto = NULL;
|
|
|
|
HBITMAP hBmp = ::GpiCreateBitmap(hPs, &vHeader, 0L, NULL, &vInfo);
|
|
if (!hBmp)
|
|
{
|
|
wxLogLastError("CreateBitmap");
|
|
}
|
|
SetHBITMAP((WXHBITMAP)hBmp);
|
|
} // end of wxBitmap::wxBitmap
|
|
|
|
//
|
|
// Create from XPM data
|
|
//
|
|
wxBitmap::wxBitmap(
|
|
char** ppData
|
|
)
|
|
{
|
|
Init();
|
|
|
|
(void)Create( (void *)ppData
|
|
,wxBITMAP_TYPE_XPM_DATA
|
|
,0
|
|
,0
|
|
,0
|
|
);
|
|
} // end of wxBitmap::wxBitmap
|
|
|
|
wxBitmap::wxBitmap(
|
|
const char** ppData
|
|
)
|
|
{
|
|
Init();
|
|
|
|
(void)Create( (void *)ppData
|
|
,wxBITMAP_TYPE_XPM_DATA
|
|
,0
|
|
,0
|
|
,0
|
|
);
|
|
} // end of wxBitmap::wxBitmap
|
|
|
|
wxBitmap::wxBitmap(
|
|
int nW
|
|
, int nH
|
|
, int nD
|
|
)
|
|
{
|
|
Init();
|
|
|
|
(void)Create( nW
|
|
,nH
|
|
,nD
|
|
);
|
|
} // end of wxBitmap::wxBitmap
|
|
|
|
wxBitmap::wxBitmap(
|
|
void* pData
|
|
, long lType
|
|
, int nWidth
|
|
, int nHeight
|
|
, int nDepth
|
|
)
|
|
{
|
|
Init();
|
|
|
|
(void)Create( pData
|
|
,lType
|
|
,nWidth
|
|
,nHeight
|
|
,nDepth
|
|
);
|
|
} // end of wxBitmap::wxBitmap
|
|
|
|
wxBitmap::wxBitmap(
|
|
const wxString& rFilename
|
|
, long lType
|
|
)
|
|
{
|
|
Init();
|
|
|
|
LoadFile( rFilename
|
|
,(int)lType
|
|
);
|
|
} // end of wxBitmap::wxBitmap
|
|
|
|
bool wxBitmap::Create(
|
|
int nW
|
|
, int nH
|
|
, int nD
|
|
)
|
|
{
|
|
HBITMAP hBmp;
|
|
BITMAPINFOHEADER2 vHeader;
|
|
|
|
wxASSERT(vHabmain != NULL);
|
|
UnRef();
|
|
m_refData = new wxBitmapRefData;
|
|
GetBitmapData()->m_nWidth = nW;
|
|
GetBitmapData()->m_nHeight = nH;
|
|
GetBitmapData()->m_nDepth = nD;
|
|
|
|
if (nD > 0)
|
|
{
|
|
DEVOPENSTRUC vDop = {0L, "DISPLAY", NULL, 0L, 0L, 0L, 0L, 0L, 0L};
|
|
SIZEL vSize = {0, 0};
|
|
HDC hDC = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
|
|
HPS hPS = ::GpiCreatePS(vHabmain, hDC, &vSize, PU_PELS | GPIA_ASSOC);
|
|
|
|
memset(&vHeader, '\0', sizeof(BITMAPINFOHEADER2));
|
|
vHeader.cbFix = sizeof(BITMAPINFOHEADER2);
|
|
vHeader.cx = nW;
|
|
vHeader.cy = nH;
|
|
vHeader.cPlanes = 1;
|
|
vHeader.cBitCount = nD;
|
|
|
|
hBmp = ::GpiCreateBitmap( hPS
|
|
,&vHeader
|
|
,0L
|
|
,NULL
|
|
,NULL
|
|
);
|
|
::GpiDestroyPS(hPS);
|
|
::DevCloseDC(hDC);
|
|
}
|
|
else
|
|
{
|
|
HPS hPSScreen;
|
|
HDC hDCScreen;
|
|
LONG lBitCount;
|
|
|
|
hPSScreen = ::WinGetScreenPS(HWND_DESKTOP);
|
|
hDCScreen = ::GpiQueryDevice(hPSScreen);
|
|
::DevQueryCaps(hDCScreen, CAPS_COLOR_BITCOUNT, 1L, &lBitCount);
|
|
|
|
memset(&vHeader, '\0', sizeof(BITMAPINFOHEADER2));
|
|
vHeader.cbFix = sizeof(BITMAPINFOHEADER2);
|
|
vHeader.cx = nW;
|
|
vHeader.cy = nH;
|
|
vHeader.cPlanes = 1;
|
|
vHeader.cBitCount = lBitCount;
|
|
|
|
hBmp = ::GpiCreateBitmap( hPSScreen
|
|
,&vHeader
|
|
,0L
|
|
,NULL
|
|
,NULL
|
|
);
|
|
|
|
GetBitmapData()->m_nDepth = wxDisplayDepth();
|
|
::WinReleasePS(hPSScreen);
|
|
}
|
|
SetHBITMAP((WXHBITMAP)hBmp);
|
|
|
|
#if WXWIN_COMPATIBILITY_2
|
|
GetBitmapData()->m_bOk = hBmp != 0;
|
|
#endif // WXWIN_COMPATIBILITY_2
|
|
|
|
return Ok();
|
|
} // end of wxBitmap::Create
|
|
|
|
bool wxBitmap::LoadFile(
|
|
const wxString& rFilename
|
|
, long lType
|
|
)
|
|
{
|
|
HPS hPs = NULLHANDLE;
|
|
|
|
UnRef();
|
|
|
|
wxBitmapHandler* pHandler = wxDynamicCast( FindHandler(lType)
|
|
,wxBitmapHandler
|
|
);
|
|
|
|
if (pHandler)
|
|
{
|
|
m_refData = new wxBitmapRefData;
|
|
|
|
return(pHandler->LoadFile( this
|
|
,rFilename
|
|
,hPs
|
|
,lType
|
|
, -1
|
|
, -1
|
|
));
|
|
}
|
|
else
|
|
{
|
|
wxImage vImage;
|
|
|
|
if (!vImage.LoadFile(rFilename, lType) || !vImage.Ok() )
|
|
return(FALSE);
|
|
|
|
*this = vImage.ConvertToBitmap();
|
|
|
|
return(TRUE);
|
|
}
|
|
} // end of wxBitmap::LoadFile
|
|
|
|
bool wxBitmap::Create(
|
|
void* pData
|
|
, long lType
|
|
, int nWidth
|
|
, int nHeight
|
|
, int nDepth
|
|
)
|
|
{
|
|
UnRef();
|
|
|
|
wxBitmapHandler* pHandler = wxDynamicCast( FindHandler(lType)
|
|
,wxBitmapHandler
|
|
);
|
|
|
|
if (!pHandler)
|
|
{
|
|
wxLogDebug(wxT("Failed to create bitmap: no bitmap handler for "
|
|
"type %d defined."), lType);
|
|
|
|
return(FALSE);
|
|
}
|
|
|
|
m_refData = new wxBitmapRefData;
|
|
|
|
return(pHandler->Create( this
|
|
,pData
|
|
,lType
|
|
,nWidth
|
|
,nHeight
|
|
,nDepth
|
|
));
|
|
} // end of wxBitmap::Create
|
|
|
|
bool wxBitmap::SaveFile(
|
|
const wxString& rFilename
|
|
, int lType
|
|
, const wxPalette* pPalette
|
|
)
|
|
{
|
|
wxBitmapHandler* pHandler = wxDynamicCast( FindHandler(lType)
|
|
,wxBitmapHandler
|
|
);
|
|
|
|
if (pHandler)
|
|
{
|
|
return pHandler->SaveFile( this
|
|
,rFilename
|
|
,lType
|
|
,pPalette
|
|
);
|
|
}
|
|
else
|
|
{
|
|
// FIXME what about palette? shouldn't we use it?
|
|
wxImage vImage(*this);
|
|
|
|
if (!vImage.Ok())
|
|
return(FALSE);
|
|
|
|
return(vImage.SaveFile( rFilename
|
|
,lType
|
|
));
|
|
}
|
|
} // end of wxBitmap::SaveFile
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// sub bitmap extraction
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxBitmap wxBitmap::GetSubBitmap(
|
|
const wxRect& rRect
|
|
) const
|
|
{
|
|
wxCHECK_MSG( Ok() &&
|
|
(rRect.x >= 0) && (rRect.y >= 0) &&
|
|
(rRect.x + rRect.width <= GetWidth()) &&
|
|
(rRect.y + rRect.height <= GetHeight()),
|
|
wxNullBitmap, wxT("Invalid bitmap or bitmap region") );
|
|
|
|
wxBitmap vRet( rRect.width
|
|
,rRect.height
|
|
,GetDepth()
|
|
);
|
|
wxASSERT_MSG( vRet.Ok(), wxT("GetSubBitmap error") );
|
|
|
|
|
|
//
|
|
// Copy bitmap data
|
|
//
|
|
SIZEL vSize = {0, 0};
|
|
DEVOPENSTRUC vDop = {0L, "DISPLAY", NULL, 0L, 0L, 0L, 0L, 0L, 0L};
|
|
HDC hDCSrc = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
|
|
HDC hDCDst = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
|
|
HPS hPSSrc = ::GpiCreatePS(vHabmain, hDCSrc, &vSize, PU_PELS | GPIA_ASSOC);
|
|
HPS hPSDst = ::GpiCreatePS(vHabmain, hDCDst, &vSize, PU_PELS | GPIA_ASSOC);
|
|
POINTL vPoint[4] = { rRect.x, rRect.y,
|
|
rRect.x + rRect.width, rRect.y + rRect.height,
|
|
0, 0, GetWidth(), GetHeight()
|
|
};
|
|
|
|
::GpiSetBitmap(hPSSrc, (HBITMAP) GetHBITMAP());
|
|
::GpiSetBitmap(hPSDst, (HBITMAP) vRet.GetHBITMAP());
|
|
::GpiBitBlt( hPSDst
|
|
,hPSSrc
|
|
,4L
|
|
,vPoint
|
|
,ROP_SRCCOPY
|
|
,BBO_IGNORE
|
|
);
|
|
|
|
//
|
|
// Copy mask if there is one
|
|
//
|
|
if (GetMask())
|
|
{
|
|
BITMAPINFOHEADER2 vBmih;
|
|
|
|
memset(&vBmih, '\0', sizeof(BITMAPINFOHEADER2));
|
|
vBmih.cbFix = sizeof(BITMAPINFOHEADER2);
|
|
vBmih.cx = rRect.width;
|
|
vBmih.cy = rRect.height;
|
|
vBmih.cPlanes = 1;
|
|
vBmih.cBitCount = 1;
|
|
|
|
HBITMAP hBmpMask = ::GpiCreateBitmap( hPSDst
|
|
,&vBmih
|
|
,0L
|
|
,NULL
|
|
,NULL
|
|
);
|
|
|
|
::GpiSetBitmap(hPSSrc, (HBITMAP) GetHBITMAP());
|
|
::GpiSetBitmap(hPSDst, (HBITMAP) vRet.GetHBITMAP());
|
|
|
|
::GpiSetBitmap(hPSSrc, (HBITMAP) GetMask()->GetMaskBitmap());
|
|
::GpiSetBitmap(hPSDst, (HBITMAP) hBmpMask);
|
|
::GpiBitBlt( hPSDst
|
|
,hPSSrc
|
|
,4L
|
|
,vPoint
|
|
,ROP_SRCCOPY
|
|
,BBO_IGNORE
|
|
);
|
|
|
|
wxMask* pMask = new wxMask((WXHBITMAP)hBmpMask);
|
|
vRet.SetMask(pMask);
|
|
}
|
|
|
|
::GpiSetBitmap(hPSSrc, NULL);
|
|
::GpiSetBitmap(hPSDst, NULL);
|
|
::GpiDestroyPS(hPSSrc);
|
|
::GpiDestroyPS(hPSDst);
|
|
::DevCloseDC(hDCSrc);
|
|
::DevCloseDC(hDCDst);
|
|
return vRet;
|
|
} // end of wxBitmap::GetSubBitmap
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxBitmap accessors
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void wxBitmap::SetQuality(
|
|
int nQ
|
|
)
|
|
{
|
|
EnsureHasData();
|
|
|
|
GetBitmapData()->m_nQuality = nQ;
|
|
} // end of wxBitmap::SetQuality
|
|
|
|
#if WXWIN_COMPATIBILITY_2
|
|
void wxBitmap::SetOk(
|
|
bool bOk
|
|
)
|
|
{
|
|
EnsureHasData();
|
|
|
|
GetBitmapData()->m_bOk = bOk;
|
|
} // end of wxBitmap::SetOk
|
|
#endif // WXWIN_COMPATIBILITY_2
|
|
|
|
void wxBitmap::SetPalette(
|
|
const wxPalette& rPalette
|
|
)
|
|
{
|
|
EnsureHasData();
|
|
|
|
GetBitmapData()->m_vBitmapPalette = rPalette;
|
|
} // end of wxBitmap::SetPalette
|
|
|
|
void wxBitmap::SetMask(
|
|
wxMask* pMask
|
|
)
|
|
{
|
|
EnsureHasData();
|
|
|
|
GetBitmapData()->m_pBitmapMask = pMask;
|
|
} // end of wxBitmap::SetMask
|
|
|
|
//
|
|
// Will try something for OS/2 but not really sure how close
|
|
// to the msw intent this is.
|
|
//
|
|
wxBitmap wxBitmap::GetBitmapForDC(
|
|
wxDC& rDc
|
|
) const
|
|
{
|
|
wxMemoryDC vMemDC;
|
|
wxBitmap vTmpBitmap( this->GetWidth()
|
|
,this->GetHeight()
|
|
,rDc.GetDepth()
|
|
);
|
|
WXHBITMAP vOldBitmap;
|
|
HPS hMemoryPS;
|
|
HPS hPs;
|
|
POINTL vPoint[4];
|
|
SIZEL vSize = {0,0};
|
|
|
|
hMemoryPS = ::GpiCreatePS(vHabmain, (HDC)vMemDC.GetHDC(), &vSize, PU_PELS | GPIT_MICRO | GPIA_ASSOC);
|
|
hPs = ::GpiCreatePS(vHabmain, (HDC)rDc.GetHDC(), &vSize, PU_PELS | GPIT_MICRO | GPIA_ASSOC);
|
|
|
|
// TODO: Set the points
|
|
|
|
vOldBitmap = (WXHBITMAP)::GpiSetBitmap(hPs, (HBITMAP)vTmpBitmap.GetHBITMAP());
|
|
::GpiBitBlt(hPs, hMemoryPS, 4L, vPoint, ROP_SRCCOPY, BBO_IGNORE);
|
|
|
|
return(vTmpBitmap);
|
|
} // end of wxBitmap::GetBitmapForDC
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxMask
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxMask::wxMask()
|
|
{
|
|
m_hMaskBitmap = 0;
|
|
} // end of wxMask::wxMask
|
|
|
|
// Construct a mask from a bitmap and a colour indicating
|
|
// the transparent area
|
|
wxMask::wxMask(
|
|
const wxBitmap& rBitmap
|
|
, const wxColour& rColour
|
|
)
|
|
{
|
|
m_hMaskBitmap = 0;
|
|
Create( rBitmap
|
|
,rColour
|
|
);
|
|
} // end of wxMask::wxMask
|
|
|
|
// Construct a mask from a bitmap and a palette index indicating
|
|
// the transparent area
|
|
wxMask::wxMask(
|
|
const wxBitmap& rBitmap
|
|
, int nPaletteIndex
|
|
)
|
|
{
|
|
m_hMaskBitmap = 0;
|
|
Create( rBitmap
|
|
,nPaletteIndex
|
|
);
|
|
} // end of wxMask::wxMask
|
|
|
|
// Construct a mask from a mono bitmap (copies the bitmap).
|
|
wxMask::wxMask(
|
|
const wxBitmap& rBitmap
|
|
)
|
|
{
|
|
m_hMaskBitmap = 0;
|
|
Create(rBitmap);
|
|
} // end of wxMask::wxMask
|
|
|
|
wxMask::~wxMask()
|
|
{
|
|
if (m_hMaskBitmap)
|
|
::GpiDeleteBitmap((HBITMAP)m_hMaskBitmap);
|
|
} // end of wxMask::~wxMask
|
|
|
|
// Create a mask from a mono bitmap (copies the bitmap).
|
|
bool wxMask::Create(
|
|
const wxBitmap& rBitmap
|
|
)
|
|
{
|
|
BITMAPINFOHEADER2 vBmih;
|
|
SIZEL vSize = {0, 0};
|
|
DEVOPENSTRUC vDop = {0L, "DISPLAY", NULL, 0L, 0L, 0L, 0L, 0L, 0L};
|
|
HDC hDCSrc = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
|
|
HDC hDCDst = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
|
|
HPS hPSSrc = ::GpiCreatePS(vHabmain, hDCSrc, &vSize, PU_PELS | GPIA_ASSOC);
|
|
HPS hPSDst = ::GpiCreatePS(vHabmain, hDCDst, &vSize, PU_PELS | GPIA_ASSOC);
|
|
POINTL vPoint[4] = { 0 ,0, rBitmap.GetWidth(), rBitmap.GetHeight(),
|
|
0, 0, rBitmap.GetWidth(), rBitmap.GetHeight()
|
|
};
|
|
|
|
if (m_hMaskBitmap)
|
|
{
|
|
::GpiDeleteBitmap((HBITMAP) m_hMaskBitmap);
|
|
m_hMaskBitmap = 0;
|
|
}
|
|
if (!rBitmap.Ok() || rBitmap.GetDepth() != 1)
|
|
{
|
|
return(FALSE);
|
|
}
|
|
|
|
memset(&vBmih, '\0', sizeof(BITMAPINFOHEADER2));
|
|
vBmih.cbFix = sizeof(BITMAPINFOHEADER2);
|
|
vBmih.cx = rBitmap.GetWidth();
|
|
vBmih.cy = rBitmap.GetHeight();
|
|
vBmih.cPlanes = 1;
|
|
vBmih.cBitCount = 1;
|
|
|
|
m_hMaskBitmap = ::GpiCreateBitmap( hPSDst
|
|
,&vBmih
|
|
,0L
|
|
,NULL
|
|
,NULL
|
|
);
|
|
|
|
::GpiSetBitmap(hPSSrc, (HBITMAP) rBitmap.GetHBITMAP());
|
|
::GpiSetBitmap(hPSDst, (HBITMAP) m_hMaskBitmap);
|
|
::GpiBitBlt( hPSDst
|
|
,hPSSrc
|
|
,4L
|
|
,vPoint
|
|
,ROP_SRCCOPY
|
|
,BBO_IGNORE
|
|
);
|
|
|
|
::GpiDestroyPS(hPSSrc);
|
|
::GpiDestroyPS(hPSDst);
|
|
::DevCloseDC(hDCSrc);
|
|
::DevCloseDC(hDCDst);
|
|
return(TRUE);
|
|
} // end of wxMask::Create
|
|
|
|
// Create a mask from a bitmap and a palette index indicating
|
|
// the transparent area
|
|
bool wxMask::Create(
|
|
const wxBitmap& rBitmap
|
|
, int nPaletteIndex
|
|
)
|
|
{
|
|
if (m_hMaskBitmap)
|
|
{
|
|
::GpiDeleteBitmap((HBITMAP) m_hMaskBitmap);
|
|
m_hMaskBitmap = 0;
|
|
}
|
|
if (rBitmap.Ok() && rBitmap.GetPalette()->Ok())
|
|
{
|
|
unsigned char cRed;
|
|
unsigned char cGreen;
|
|
unsigned char cBlue;
|
|
|
|
if (rBitmap.GetPalette()->GetRGB( nPaletteIndex
|
|
,&cRed
|
|
,&cGreen
|
|
,&cBlue
|
|
))
|
|
{
|
|
wxColour vTransparentColour( cRed
|
|
,cGreen
|
|
,cBlue
|
|
);
|
|
|
|
return (Create( rBitmap
|
|
,vTransparentColour
|
|
));
|
|
}
|
|
}
|
|
return(FALSE);
|
|
} // end of wxMask::Create
|
|
|
|
// Create a mask from a bitmap and a colour indicating
|
|
// the transparent area
|
|
bool wxMask::Create(
|
|
const wxBitmap& rBitmap
|
|
, const wxColour& rColour
|
|
)
|
|
{
|
|
bool bOk = TRUE;
|
|
COLORREF vMaskColour = OS2RGB( rColour.Red()
|
|
,rColour.Green()
|
|
,rColour.Blue()
|
|
);
|
|
BITMAPINFOHEADER2 vBmih;
|
|
SIZEL vSize = {0, 0};
|
|
DEVOPENSTRUC vDop = { NULL, "DISPLAY", NULL, NULL, NULL, NULL, NULL, NULL, NULL };
|
|
HDC hDCSrc = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
|
|
HDC hDCDst = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
|
|
HPS hPSSrc = ::GpiCreatePS(vHabmain, hDCSrc, &vSize, PU_PELS | GPIA_ASSOC);
|
|
HPS hPSDst = ::GpiCreatePS(vHabmain, hDCDst, &vSize, PU_PELS | GPIA_ASSOC);
|
|
POINTL vPoint[4] = { 0 ,0, rBitmap.GetWidth(), rBitmap.GetHeight(),
|
|
0, 0, rBitmap.GetWidth(), rBitmap.GetHeight()
|
|
};
|
|
|
|
if (m_hMaskBitmap)
|
|
{
|
|
::GpiDeleteBitmap((HBITMAP) m_hMaskBitmap);
|
|
m_hMaskBitmap = 0;
|
|
}
|
|
if (!rBitmap.Ok())
|
|
{
|
|
return(FALSE);
|
|
}
|
|
|
|
//
|
|
// Scan the bitmap for the transparent colour and set
|
|
// the corresponding pixels in the mask to BLACK and
|
|
// the rest to WHITE
|
|
//
|
|
|
|
memset(&vBmih, '\0', sizeof(BITMAPINFOHEADER2));
|
|
vBmih.cbFix = sizeof(BITMAPINFOHEADER2);
|
|
vBmih.cx = rBitmap.GetWidth();
|
|
vBmih.cy = rBitmap.GetHeight();
|
|
vBmih.cPlanes = 1;
|
|
vBmih.cBitCount = 1;
|
|
|
|
m_hMaskBitmap = ::GpiCreateBitmap( hPSDst
|
|
,&vBmih
|
|
,0L
|
|
,NULL
|
|
,NULL
|
|
);
|
|
|
|
::GpiSetBitmap(hPSSrc, (HBITMAP) rBitmap.GetHBITMAP());
|
|
::GpiSetBitmap(hPSDst, (HBITMAP) m_hMaskBitmap);
|
|
|
|
//
|
|
// This is not very efficient, but I can't think
|
|
// of a better way of doing it
|
|
//
|
|
for (int w = 0; w < rBitmap.GetWidth(); w++)
|
|
{
|
|
for (int h = 0; h < rBitmap.GetHeight(); h++)
|
|
{
|
|
POINTL vPt = {w, h};
|
|
COLORREF vCol = (COLORREF)::GpiQueryPel(hPSSrc, &vPt);
|
|
if (vCol == (COLORREF)CLR_NOINDEX)
|
|
{
|
|
//
|
|
// Doesn't make sense to continue
|
|
//
|
|
bOk = FALSE;
|
|
break;
|
|
}
|
|
|
|
if (vCol == vMaskColour)
|
|
{
|
|
::GpiSetColor(hPSDst, OS2RGB(0, 0, 0));
|
|
::GpiSetPel(hPSDst, &vPt);
|
|
}
|
|
else
|
|
{
|
|
::GpiSetColor(hPSDst, OS2RGB(255, 255, 255));
|
|
::GpiSetPel(hPSDst, &vPt);
|
|
}
|
|
}
|
|
}
|
|
::GpiSetBitmap(hPSSrc, NULL);
|
|
::GpiSetBitmap(hPSDst, NULL);
|
|
::GpiDestroyPS(hPSSrc);
|
|
::GpiDestroyPS(hPSDst);
|
|
::DevCloseDC(hDCSrc);
|
|
::DevCloseDC(hDCDst);
|
|
return(TRUE);
|
|
} // end of wxMask::Create
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxBitmapHandler
|
|
// ----------------------------------------------------------------------------
|
|
|
|
bool wxBitmapHandler::Create(
|
|
wxGDIImage* pImage
|
|
, void* pData
|
|
, long lFlags
|
|
, int nWidth
|
|
, int nHeight
|
|
, int nDepth
|
|
)
|
|
{
|
|
wxBitmap* pBitmap = wxDynamicCast( pImage
|
|
,wxBitmap
|
|
);
|
|
|
|
return(pBitmap ? Create( pBitmap
|
|
,pData
|
|
,nWidth
|
|
,nHeight
|
|
,nDepth
|
|
) : FALSE);
|
|
}
|
|
|
|
bool wxBitmapHandler::Load(
|
|
wxGDIImage* pImage
|
|
, const wxString& rName
|
|
, HPS hPs
|
|
, long lFlags
|
|
, int nWidth
|
|
, int nHeight
|
|
)
|
|
{
|
|
wxBitmap* pBitmap = wxDynamicCast( pImage
|
|
,wxBitmap
|
|
);
|
|
|
|
return(pBitmap ? LoadFile( pBitmap
|
|
,rName
|
|
,hPs
|
|
,lFlags
|
|
,nWidth
|
|
,nHeight
|
|
) : FALSE);
|
|
}
|
|
|
|
bool wxBitmapHandler::Save(
|
|
wxGDIImage* pImage
|
|
, const wxString& rName
|
|
, int lType
|
|
)
|
|
{
|
|
wxBitmap* pBitmap = wxDynamicCast( pImage
|
|
,wxBitmap
|
|
);
|
|
|
|
return(pBitmap ? SaveFile( pBitmap
|
|
,rName
|
|
,lType
|
|
) : FALSE);
|
|
}
|
|
|
|
bool wxBitmapHandler::Create(
|
|
wxBitmap* WXUNUSED(pBitmap)
|
|
, void* WXUNUSED(pData)
|
|
, long WXUNUSED(lType)
|
|
, int WXUNUSED(nWidth)
|
|
, int WXUNUSED(nHeight)
|
|
, int WXUNUSED(nDepth)
|
|
)
|
|
{
|
|
return(FALSE);
|
|
}
|
|
|
|
bool wxBitmapHandler::LoadFile(
|
|
wxBitmap* WXUNUSED(pBitmap)
|
|
, const wxString& WXUNUSED(rName)
|
|
, HPS WXUNUSED(hPs)
|
|
, long WXUNUSED(lType)
|
|
, int WXUNUSED(nDesiredWidth)
|
|
, int WXUNUSED(nDesiredHeight)
|
|
)
|
|
{
|
|
return(FALSE);
|
|
}
|
|
|
|
bool wxBitmapHandler::SaveFile(
|
|
wxBitmap* WXUNUSED(pBitmap)
|
|
, const wxString& WXUNUSED(rName)
|
|
, int WXUNUSED(nType)
|
|
, const wxPalette* WXUNUSED(pPalette)
|
|
)
|
|
{
|
|
return(FALSE);
|
|
}
|
|
|