home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
OS/2 Shareware BBS: 10 Tools
/
10-Tools.zip
/
wxos2233.zip
/
wxOS2-2_3_3.zip
/
wxWindows-2.3.3
/
src
/
os2
/
bitmap.cpp
< prev
next >
Wrap
C/C++ Source or Header
|
2002-07-22
|
52KB
|
1,615 lines
/////////////////////////////////////////////////////////////////////////////
// Name: bitmap.cpp
// Purpose: wxBitmap
// Author: David Webster
// Modified by:
// Created: 08/08/99
// RCS-ID: $Id: BITMAP.CPP,v 1.40 2002/07/22 03:29:06 DW Exp $
// 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"
#include "wx/xpmdecod.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()
{
if ( m_pSelectedInto )
{
wxLogLastError("GpiDeleteBitmap(hbitmap)");
}
if (m_hBitmap)
{
if (!::GpiDeleteBitmap((HBITMAP)m_hBitmap))
{
wxLogLastError("GpiDeleteBitmap(hbitmap)");
}
}
if (m_pBitmapMask)
{
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_bIsMono = FALSE;
//
// True for all bitmaps created from bits, wxImages, Xpms
//
} // 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;
wxMask* pMask = new wxMask(SIconInfo.hbmPointer);
pMask->SetMaskBitmap(GetHBITMAP());
SetMask(pMask);
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()
{
} // end of wxBitmap::~wxBitmap
wxBitmap::wxBitmap(
const char zBits[]
, int nWidth
, int nHeight
, int nDepth
)
{
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};
char* pzData;
wxASSERT(vHabmain != NULL);
m_refData = pRefData;
pRefData->m_nWidth = nWidth;
pRefData->m_nHeight = nHeight;
pRefData->m_nDepth = nDepth;
pRefData->m_nNumColors = 0;
pRefData->m_pSelectedInto = NULL;
hDc = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
hPs = ::GpiCreatePS(vHabmain, hDc, &vSize, GPIA_ASSOC | PU_PELS);
if (hPs == 0)
{
wxLogLastError("GpiCreatePS Failure");
}
if (nDepth == 1)
{
//
// We assume that it is in XBM format which is not quite the same as
// the format CreateBitmap() wants because the order of bytes in the
// line is inversed!
//
const size_t nBytesPerLine = (nWidth + 7) / 8;
const size_t nPadding = nBytesPerLine % 2;
const size_t nLen = nHeight * (nPadding + nBytesPerLine);
const char* pzSrc = zBits;
int nRows;
size_t nCols;
pzData = (char *)malloc(nLen);
char* pzDst = pzData;
for (nRows = 0; nRows < nHeight; nRows++)
{
for (nCols = 0; nCols < nBytesPerLine; nCols++)
{
unsigned char ucVal = *pzSrc++;
unsigned char ucReversed = 0;
int nBits;
for (nBits = 0; nBits < 8; nBits++)
{
ucReversed <<= 1;
ucReversed |= (ucVal & 0x01);
ucVal >>= 1;
}
*pzDst++ = ucReversed;
}
if (nPadding)
*pzDst++ = 0;
}
}
else
{
//
// Bits should already be in Windows standard format
//
pzData = (char *)zBits; // const_cast is harmless
}
if (nDepth > 24)
nDepth = 24; // MAX supported in PM
memset(&vHeader, '\0', 16);
vHeader.cbFix = 16;
vHeader.cx = (USHORT)nWidth;
vHeader.cy = (USHORT)nHeight;
vHeader.cPlanes = 1L;
vHeader.cBitCount = nDepth;
vHeader.usReserved = 0;
memset(&vInfo, '\0', 16);
vInfo.cbFix = 16;
vInfo.cx = (USHORT)nWidth;
vInfo.cy = (USHORT)nHeight;
vInfo.cPlanes = 1L;
vInfo.cBitCount = nDepth;
HBITMAP hBmp = ::GpiCreateBitmap(hPs, &vHeader, CBM_INIT, (PBYTE)pzData, &vInfo);
if (!hBmp)
{
wxLogLastError("CreateBitmap");
}
::GpiDestroyPS(hPs);
::DevCloseDC(hDc);
SetHBITMAP((WXHBITMAP)hBmp);
} // 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(
int nId
, long lType
)
{
Init();
LoadFile( nId
,(int)lType
);
SetId(nId);
} // 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;
//
// Xpms and bitmaps from other images can also be mono's, but only
// mono's need help changing their colors with MemDC changes
//
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);
if (nD == 1)
m_bIsMono = TRUE;
memset(&vHeader, '\0', 16);
vHeader.cbFix = 16;
vHeader.cx = nW;
vHeader.cy = nH;
vHeader.cPlanes = 1;
vHeader.cBitCount = 24; //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);
if (lBitCount > 24)
lBitCount = 24;
memset(&vHeader, '\0', 16);
vHeader.cbFix = 16;
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::CreateFromXpm(
const char** ppData
)
{
#if wxUSE_IMAGE && wxUSE_XPM
Init();
wxCHECK_MSG(ppData != NULL, FALSE, wxT("invalid bitmap data"))
wxXPMDecoder vDecoder;
wxImage vImg = vDecoder.ReadData(ppData);
wxCHECK_MSG(vImg.Ok(), FALSE, wxT("invalid bitmap data"))
*this = wxBitmap(vImg);
return TRUE;
#else
return FALSE;
#endif
} // end of wxBitmap::CreateFromXpm
bool wxBitmap::LoadFile(
int nId
, long lType
)
{
HPS hPs = NULLHANDLE;
UnRef();
wxBitmapHandler* pHandler = wxDynamicCast( FindHandler(lType)
,wxBitmapHandler
);
if (pHandler)
{
m_refData = new wxBitmapRefData;
return(pHandler->LoadFile( this
,nId
,lType
, -1
, -1
));
}
else
{
return(FALSE);
}
} // 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 = ConvertToImage();
if (!vImage.Ok())
return(FALSE);
return(vImage.SaveFile( rFilename
,lType
));
}
} // end of wxBitmap::SaveFile
// ----------------------------------------------------------------------------
// wxImage-wxBitmap convertion
// ----------------------------------------------------------------------------
bool wxBitmap::CreateFromImage (
const wxImage& rImage
, int nDepth
)
{
wxCHECK_MSG(rImage.Ok(), FALSE, wxT("invalid image"));
m_refData = new wxBitmapRefData();
int nSizeLimit = 1024 * 768 * 3;
int nWidth = rImage.GetWidth();
int nBmpHeight = rImage.GetHeight();
int nBytePerLine = nWidth * 3;
int nSizeDWORD = sizeof(DWORD);
int nLineBoundary = nBytePerLine % nSizeDWORD;
int nPadding = 0;
if (nLineBoundary > 0)
{
nPadding = nSizeDWORD - nLineBoundary;
nBytePerLine += nPadding;
}
//
// Calc the number of DIBs and heights of DIBs
//
int nNumDIB = 1;
int nHRemain = 0;
int nHeight = nSizeLimit / nBytePerLine;
if (nHeight >= nBmpHeight)
nHeight = nBmpHeight;
else
{
nNumDIB = nBmpHeight / nHeight;
nHRemain = nBmpHeight % nHeight;
if (nHRemain > 0)
nNumDIB++;
}
//
// Set bitmap parameters
//
wxCHECK_MSG(rImage.Ok(), FALSE, wxT("invalid image"));
SetWidth(nWidth);
SetHeight(nBmpHeight);
if (nDepth == 1)
m_bIsMono = TRUE;
else
m_bIsMono = FALSE;
if (nDepth == -1)
nDepth = wxDisplayDepth();
SetDepth(nDepth);
#if wxUSE_PALETTE
//
// Copy the palette from the source image
//
SetPalette(rImage.GetPalette());
#endif // wxUSE_PALETTE
//
// Create a DIB header
//
BITMAPINFOHEADER2 vHeader;
BITMAPINFO2 vInfo;
//
// Fill in the DIB header
//
memset(&vHeader, '\0', 16);
vHeader.cbFix = 16;
vHeader.cx = (ULONG)nWidth;
vHeader.cy = (ULONG)nHeight;
vHeader.cPlanes = 1L;
vHeader.cBitCount = 24;
//
// Memory for DIB data
//
unsigned char* pucBits;
pucBits = (unsigned char *)malloc(nBytePerLine * nHeight);
if(!pucBits)
{
wxFAIL_MSG(wxT("could not allocate memory for DIB"));
return FALSE;
}
memset(pucBits, '\0', (nBytePerLine * nHeight));
//
// Create and set the device-dependent bitmap
//
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);
LONG lScans;
HDC hDCScreen = ::DevOpenDC(vHabmain, OD_MEMORY, "*", 5L, (PDEVOPENDATA)&vDop, NULLHANDLE);
HPS hPSScreen;
HBITMAP hBmp;
HBITMAP hBmpOld;
memset(&vInfo, '\0', 16);
vInfo.cbFix = 16;
vInfo.cx = (ULONG)nWidth;
vInfo.cy = (ULONG)nHeight;
vInfo.cPlanes = 1;
vInfo.cBitCount = 24; // Set to desired count going in
hBmp = ::GpiCreateBitmap( hPS
,&vHeader
,0L
,NULL
,NULL
);
#if wxUSE_PALETTE
HPAL hOldPalette = NULLHANDLE;
if (rImage.GetPalette().Ok())
{
hOldPalette = ::GpiSelectPalette(hPS, (HPAL)rImage.GetPalette().GetHPALETTE());
}
#endif // wxUSE_PALETTE
//
// Copy image data into DIB data and then into DDB (in a loop)
//
unsigned char* pData = rImage.GetData();
int i;
int j;
int n;
int nOrigin = 0;
unsigned char* ptdata = pData;
unsigned char* ptbits;
if ((hBmpOld = ::GpiSetBitmap(hPS, hBmp)) == HBM_ERROR)
{
ERRORID vError;
wxString sError;
vError = ::WinGetLastError(vHabmain);
sError = wxPMErrorToStr(vError);
}
for (n = 0; n < nNumDIB; n++)
{
if (nNumDIB > 1 && n == nNumDIB - 1 && nHRemain > 0)
{
//
// Redefine height and size of the (possibly) last smaller DIB
// memory is not reallocated
//
nHeight = nHRemain;
vHeader.cy = (DWORD)(nHeight);
vHeader.cbImage = nBytePerLine * nHeight;
}
ptbits = pucBits;
for (j = 0; j < nHeight; j++)
{
for (i = 0; i < nWidth; i++)
{
*(ptbits++) = *(ptdata + 2);
*(ptbits++) = *(ptdata + 1);
*(ptbits++) = *(ptdata);
ptdata += 3;
}
for (i = 0; i < nPadding; i++)
*(ptbits++) = 0;
}
//
// Have to do something similar to WIN32's StretchDIBits, use GpiBitBlt
// in combination with setting the bits into the selected bitmap
//
if ((lScans = ::GpiSetBitmapBits( hPS
,0 // Start at the bottom
,(LONG)nHeight // One line per scan
,(PBYTE)pucBits
,&vInfo
)) == GPI_ALTERROR)
{
ERRORID vError;
wxString sError;
vError = ::WinGetLastError(vHabmain);
sError = wxPMErrorToStr(vError);
}
hPSScreen = ::GpiCreatePS( vHabmain
,hDCScreen
,&vSize
,PU_PELS | GPIA_ASSOC
);
POINTL vPoint[4] = { 0, nOrigin,
nWidth, nHeight,
0, 0, nWidth, nHeight
};
::GpiBitBlt( hPSScreen
,hPS
,4
,vPoint
,ROP_SRCCOPY
,BBO_IGNORE
);
::GpiDestroyPS(hPSScreen);
nOrigin += nHeight;
}
SetHBITMAP((WXHBITMAP)hBmp);
#if wxUSE_PALETTE
if (hOldPalette)
::GpiSelectPalette(hPS, hOldPalette);
#endif // wxUSE_PALETTE
//
// Similarly, created an mono-bitmap for the possible mask
//
if (rImage.HasMask())
{
vHeader.cbFix = 16;
vHeader.cx = nWidth;
vHeader.cy = nHeight;
vHeader.cPlanes = 1;
vHeader.cBitCount = 24;
hBmp = ::GpiCreateBitmap( hPS
,&vHeader
,0L
,NULL
,NULL
);
hBmpOld = ::GpiSetBitmap(hPS, hBmp);
if (nNumDIB == 1)
nHeight = nBmpHeight;
else
nHeight = nSizeLimit / nBytePerLine;
vHeader.cy = (DWORD)(nHeight);
nOrigin = 0;
unsigned char cRed = rImage.GetMaskRed();
unsigned char cGreen = rImage.GetMaskGreen();
unsigned char cBlue = rImage.GetMaskBlue();
unsigned char cZero = 0;
unsigned char cOne = 255;
ptdata = pData;
for (n = 0; n < nNumDIB; n++)
{
if (nNumDIB > 1 && n == nNumDIB - 1 && nHRemain > 0)
{
//
// Redefine height and size of the (possibly) last smaller DIB
// memory is not reallocated
//
nHeight = nHRemain;
vHeader.cy = (DWORD)(nHeight);
vHeader.cbImage = nBytePerLine * nHeight;
}
ptbits = pucBits;
for (int j = 0; j < nHeight; j++)
{
for (i = 0; i < nWidth; i++)
{
unsigned char cRedImage = (*(ptdata++)) ;
unsigned char cGreenImage = (*(ptdata++)) ;
unsigned char cBlueImage = (*(ptdata++)) ;
if ((cRedImage != cRed) || (cGreenImage != cGreen) || (cBlueImage != cBlue))
{
*(ptbits++) = cOne;
*(ptbits++) = cOne;
*(ptbits++) = cOne;
}
else
{
*(ptbits++) = cZero;
*(ptbits++) = cZero;
*(ptbits++) = cZero;
}
}
for (i = 0; i < nPadding; i++)
*(ptbits++) = cZero;
}
lScans = ::GpiSetBitmapBits( hPS
,0 // Start at the bottom
,(LONG)nHeight // One line per scan
,(PBYTE)pucBits
,&vInfo
);
hPSScreen = ::GpiCreatePS( vHabmain
,hDCScreen
,&vSize
,PU_PELS | GPIA_ASSOC
);
POINTL vPoint2[4] = { 0, nOrigin,
nWidth, nHeight,
0, 0, nWidth, nHeight
};
::GpiBitBlt( hPSScreen
,hPS
,4
,vPoint2
,ROP_SRCCOPY
,BBO_IGNORE
);
::GpiDestroyPS(hPSScreen);
nOrigin += nHeight;
}
//
// Create a wxMask object
//
wxMask* pMask = new wxMask();
pMask->SetMaskBitmap((WXHBITMAP)hBmp);
SetMask(pMask);
hBmpOld = ::GpiSetBitmap(hPS, hBmpOld);
}
//
// Free allocated resources
//
::GpiSetBitmap(hPS, NULLHANDLE);
::GpiDestroyPS(hPS);
::DevCloseDC(hDCScreen);
::DevCloseDC(hDC);
free(pucBits);
return TRUE;
} // end of wxBitmap::CreateFromImage
wxImage wxBitmap::ConvertToImage() const
{
wxImage vImage;
wxDC* pDC;
wxCHECK_MSG( Ok(), wxNullImage, wxT("invalid bitmap") );
//
// Create an wxImage object
//
int nWidth = GetWidth();
int nHeight = GetHeight();
int nDevWidth;
int nDevHeight;
int nBytePerLine = nWidth * 3;
int nSizeDWORD = sizeof(DWORD);
int nLineBoundary = nBytePerLine % nSizeDWORD;
int nPadding = 0;
unsigned char* pData;
unsigned char* lpBits;
long lScans;
BITMAPINFOHEADER2 vDIBh;
BITMAPINFO2 vDIBInfo;
HPS hPSMem;
HPS hPS;
HBITMAP hBitmap;
HBITMAP hOldBitmap;
DEVOPENSTRUC vDop = {0L, "DISPLAY", NULL, 0L, 0L, 0L, 0L, 0L, 0L};
SIZEL vSizlPage = {0,0};
HDC hDCMem;
vImage.Create( nWidth
,nHeight
);
pData = vImage.GetData();
if(!pData)
{
wxFAIL_MSG( wxT("could not allocate data for image") );
return wxNullImage;
}
if(nLineBoundary > 0)
{
nPadding = nSizeDWORD - nLineBoundary;
nBytePerLine += nPadding;
}
wxDisplaySize( &nDevWidth
,&nDevHeight
);
//
// Create and fill a DIB header
//
memset(&vDIBh, '\0', 16);
vDIBh.cbFix = 16;
vDIBh.cx = nWidth;
vDIBh.cy = nHeight;
vDIBh.cPlanes = 1;
vDIBh.cBitCount = 24;
memset(&vDIBInfo, '\0', 16);
vDIBInfo.cbFix = 16;
vDIBInfo.cx = nWidth;
vDIBInfo.cy = nHeight;
vDIBInfo.cPlanes = 1;
vDIBInfo.cBitCount = 24;
lpBits = (unsigned char *)malloc(nBytePerLine * nHeight);
if (!lpBits)
{
wxFAIL_MSG(wxT("could not allocate data for DIB"));
free(pData);
return wxNullImage;
}
memset(lpBits, '\0', (nBytePerLine * nHeight));
hBitmap = (HBITMAP)GetHBITMAP();
//
// May already be selected into a PS
//
if ((pDC = GetSelectedInto()) != NULL)
{
hPSMem = pDC->GetHPS();
}
else
{
hDCMem = ::DevOpenDC( vHabmain
,OD_MEMORY
,"*"
,5L
,(PDEVOPENDATA)&vDop
,NULLHANDLE
);
hPSMem = ::GpiCreatePS( vHabmain
,hDCMem
,&vSizlPage
,PU_PELS | GPIA_ASSOC
);
}
if ((hOldBitmap = ::GpiSetBitmap(hPSMem, hBitmap)) == HBM_ERROR)
{
ERRORID vError;
wxString sError;
vError = ::WinGetLastError(vHabmain);
sError = wxPMErrorToStr(vError);
}
//
// Copy data from the device-dependent bitmap to the DIB
//
if ((lScans = ::GpiQueryBitmapBits( hPSMem
,0L
,(LONG)nHeight
,(PBYTE)lpBits
,&vDIBInfo
)) == GPI_ALTERROR)
{
ERRORID vError;
wxString sError;
vError = ::WinGetLastError(vHabmain);
sError = wxPMErrorToStr(vError);
}
//
// Copy DIB data into the wxImage object
//
int i;
int j;
unsigned char* ptdata = pData;
unsigned char* ptbits = lpBits;
for (i = 0; i < nHeight; i++)
{
for (j = 0; j < nWidth; j++)
{
*(ptdata++) = *(ptbits+2);
*(ptdata++) = *(ptbits+1);
*(ptdata++) = *(ptbits );
ptbits += 3;
}
ptbits += nPadding;
}
if ((pDC = GetSelectedInto()) == NULL)
{
::GpiSetBitmap(hPSMem, NULLHANDLE);
::GpiDestroyPS(hPSMem);
::DevCloseDC(hDCMem);
}
//
// Similarly, set data according to the possible mask bitmap
//
if (GetMask() && GetMask()->GetMaskBitmap())
{
hBitmap = (HBITMAP)GetMask()->GetMaskBitmap();
//
// Memory DC/PS created, color set, data copied, and memory DC/PS deleted
//
HDC hMemDC = ::DevOpenDC( vHabmain
,OD_MEMORY
,"*"
,5L
,(PDEVOPENDATA)&vDop
,NULLHANDLE
);
HPS hMemPS = ::GpiCreatePS( vHabmain
,hMemDC
,&vSizlPage
,PU_PELS | GPIA_ASSOC
);
::GpiSetColor(hMemPS, OS2RGB(0, 0, 0));
::GpiSetBackColor(hMemPS, OS2RGB(255, 255, 255) );
::GpiSetBitmap(hMemPS, hBitmap);
::GpiQueryBitmapBits( hPSMem
,0L
,(LONG)nHeight
,(PBYTE)lpBits
,&vDIBInfo
);
::GpiSetBitmap(hMemPS, NULLHANDLE);
::GpiDestroyPS(hMemPS);
::DevCloseDC(hMemDC);
//
// Background color set to RGB(16,16,16) in consistent with wxGTK
//
unsigned char ucRed = 16;
unsigned char ucGreen = 16;
unsigned char ucBlue = 16;
ptdata = pData;
ptbits = lpBits;
for (i = 0; i < nHeight; i++)
{
for (j = 0; j < nWidth; j++)
{
if (*ptbits != 0)
ptdata += 3;
else
{
*(ptdata++) = ucRed;
*(ptdata++) = ucGreen;
*(ptdata++) = ucBlue;
}
ptbits += 3;
}
ptbits += nPadding;
}
vImage.SetMaskColour( ucRed
,ucGreen
,ucBlue
);
vImage.SetMask(TRUE);
}
else
{
vImage.SetMask(FALSE);
}
//
// Free allocated resources
//
free(lpBits);
return vImage;
} // end of wxBitmap::ConvertToImage
// ----------------------------------------------------------------------------
// 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] = { 0, 0, rRect.width, rRect.height,
rRect.x, rRect.y,
rRect.x + rRect.width, rRect.y + rRect.height
};
::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 = 24;
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
wxBitmap wxBitmap::GetBitmapForDC(
wxDC& rDc
) const
{
return(*this);
} // 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 = 24;
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
, int nId
, long lFlags
, int nWidth
, int nHeight
)
{
wxBitmap* pBitmap = wxDynamicCast( pImage
,wxBitmap
);
return(pBitmap ? LoadFile( pBitmap
,nId
,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)
, int WXUNUSED(nId)
, 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);
}
// ----------------------------------------------------------------------------
// Utility functions
// ----------------------------------------------------------------------------
HBITMAP wxInvertMask(
HBITMAP hBmpMask
, int nWidth
, int nHeight
)
{
HBITMAP hBmpInvMask = 0;
wxCHECK_MSG( hBmpMask, 0, _T("invalid bitmap in wxInvertMask") );
//
// Get width/height from the bitmap if not given
//
if (!nWidth || !nHeight)
{
BITMAPINFOHEADER2 vBmhdr;
::GpiQueryBitmapInfoHeader( hBmpMask
,&vBmhdr
);
nWidth = (int)vBmhdr.cx;
nHeight = (int)vBmhdr.cy;
}
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, nWidth, nHeight,
0, 0, nWidth, nHeight
};
memset(&vBmih, '\0', 16);
vBmih.cbFix = 16;
vBmih.cx = nWidth;
vBmih.cy = nHeight;
vBmih.cPlanes = 1;
vBmih.cBitCount = 24;
hBmpInvMask = ::GpiCreateBitmap( hPSDst
,&vBmih
,0L
,NULL
,NULL
);
::GpiSetBitmap(hPSSrc, (HBITMAP) hBmpMask);
::GpiSetBitmap(hPSDst, (HBITMAP) hBmpInvMask);
::GpiBitBlt( hPSDst
,hPSSrc
,4L
,vPoint
,ROP_SRCINVERT
,BBO_IGNORE
);
::GpiDestroyPS(hPSSrc);
::GpiDestroyPS(hPSDst);
::DevCloseDC(hDCSrc);
::DevCloseDC(hDCDst);
return hBmpInvMask;
} // end of WxWinGdi_InvertMask