/************************************************************************* * * $RCSfile: bitmapex.cxx,v $ * * $Revision: 1.12 $ * * last change: $Author: rt $ $Date: 2004-09-08 15:05:36 $ * * The Contents of this file are made available subject to the terms of * either of the following licenses * * - GNU Lesser General Public License Version 2.1 * - Sun Industry Standards Source License Version 1.1 * * Sun Microsystems Inc., October, 2000 * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2000 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * * * Sun Industry Standards Source License Version 1.1 * ================================================= * The contents of this file are subject to the Sun Industry Standards * Source License Version 1.1 (the "License"); You may not use this file * except in compliance with the License. You may obtain a copy of the * License at http://www.openoffice.org/license.html. * * Software provided under this License is provided on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, * WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS, * MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING. * See the License for the specific provisions governing your rights and * obligations concerning the Software. * * The Initial Developer of the Original Code is: Sun Microsystems, Inc. * * Copyright: 2000 by Sun Microsystems, Inc. * * All Rights Reserved. * * Contributor(s): _______________________________________ * * ************************************************************************/ #include #ifndef _RTL_CRC_H_ #include #endif #ifndef _SV_SALBTYPE_HXX #include #endif #ifndef _STREAM_HXX #include #endif #ifndef _DEBUG_HXX #include #endif #ifndef _SV_OUTDEV_HXX #include #endif #ifndef _SV_ALPHA_HXX #include #endif #ifndef _SV_IMAGE_H #include #endif #ifndef _SV_BITMAPEX_HXX #include #endif #ifndef _SV_PNGREAD_HXX #include #endif #ifndef _SV_IMPIMAGETREE_H #include #endif #ifndef _SV_RC_H #include #endif // ------------ // - BitmapEx - // ------------ BitmapEx::BitmapEx() : eTransparent( TRANSPARENT_NONE ), bAlpha ( FALSE ) { } // ------------------------------------------------------------------ BitmapEx::BitmapEx( const BitmapEx& rBitmapEx ) : aBitmap ( rBitmapEx.aBitmap ), aMask ( rBitmapEx.aMask ), aBitmapSize ( rBitmapEx.aBitmapSize ), aTransparentColor ( rBitmapEx.aTransparentColor ), eTransparent ( rBitmapEx.eTransparent ), bAlpha ( rBitmapEx.bAlpha ) { } // ------------------------------------------------------------------ BitmapEx::BitmapEx( const ResId& rResId ) : eTransparent( TRANSPARENT_NONE ), bAlpha ( FALSE ) { static ImplImageTreeSingletonRef aImageTree; ResMgr* pResMgr = NULL; ResMgr::GetResourceSkipHeader( rResId.SetRT( RSC_BITMAP ), &pResMgr ); pResMgr->ReadShort(); pResMgr->ReadShort(); const String aFileName( pResMgr->ReadString() ); if( !aImageTree->loadImage( aFileName, *this ) ) { #ifdef DBG_UTIL ByteString aErrorStr( "BitmapEx::BitmapEx( const ResId& rResId ): could not load image <" ); DBG_ERROR( ( ( aErrorStr += ByteString( aFileName, RTL_TEXTENCODING_ASCII_US ) ) += '>' ).GetBuffer() ); #endif } } // ------------------------------------------------------------------ BitmapEx::BitmapEx( const Bitmap& rBmp ) : aBitmap ( rBmp ), aBitmapSize ( aBitmap.GetSizePixel() ), eTransparent( TRANSPARENT_NONE ), bAlpha ( FALSE ) { } // ------------------------------------------------------------------ BitmapEx::BitmapEx( const Bitmap& rBmp, const Bitmap& rMask ) : aBitmap ( rBmp ), aMask ( rMask ), aBitmapSize ( aBitmap.GetSizePixel() ), eTransparent ( !rMask ? TRANSPARENT_NONE : TRANSPARENT_BITMAP ), bAlpha ( FALSE ) { // #105489# Ensure a mask is exactly one bit deep if( !!aMask && aMask.GetBitCount() != 1 ) { OSL_TRACE("BitmapEx: forced mask to monochrome"); aMask.ImplMakeMono( 255 ); } } // ------------------------------------------------------------------ BitmapEx::BitmapEx( const Bitmap& rBmp, const AlphaMask& rAlphaMask ) : aBitmap ( rBmp ), aMask ( rAlphaMask.ImplGetBitmap() ), aBitmapSize ( aBitmap.GetSizePixel() ), eTransparent ( !rAlphaMask ? TRANSPARENT_NONE : TRANSPARENT_BITMAP ), bAlpha ( !rAlphaMask ? FALSE : TRUE ) { } // ------------------------------------------------------------------ BitmapEx::BitmapEx( const Bitmap& rBmp, const Color& rTransparentColor ) : aBitmap ( rBmp ), aBitmapSize ( aBitmap.GetSizePixel() ), aTransparentColor ( rTransparentColor ), eTransparent ( TRANSPARENT_BITMAP ), bAlpha ( FALSE ) { aMask = aBitmap.CreateMask( aTransparentColor ); } // ------------------------------------------------------------------ BitmapEx::~BitmapEx() { } // ------------------------------------------------------------------ // ------------------------------------------------------------------ BitmapEx& BitmapEx::operator=( const BitmapEx& rBitmapEx ) { if( &rBitmapEx != this ) { aBitmap = rBitmapEx.aBitmap; aMask = rBitmapEx.aMask; aBitmapSize = rBitmapEx.aBitmapSize; aTransparentColor = rBitmapEx.aTransparentColor; eTransparent = rBitmapEx.eTransparent; bAlpha = rBitmapEx.bAlpha; } return *this; } // ------------------------------------------------------------------ BOOL BitmapEx::operator==( const BitmapEx& rBitmapEx ) const { if( eTransparent != rBitmapEx.eTransparent ) return FALSE; if( aBitmap != rBitmapEx.aBitmap ) return FALSE; if( aBitmapSize != rBitmapEx.aBitmapSize ) return FALSE; if( eTransparent == TRANSPARENT_NONE ) return TRUE; if( eTransparent == TRANSPARENT_COLOR ) return aTransparentColor == rBitmapEx.aTransparentColor; return( ( aMask == rBitmapEx.aMask ) && ( bAlpha == rBitmapEx.bAlpha ) ); } // ------------------------------------------------------------------ BOOL BitmapEx::IsEqual( const BitmapEx& rBmpEx ) const { return( rBmpEx.eTransparent == eTransparent && rBmpEx.bAlpha == bAlpha && rBmpEx.aBitmap.IsEqual( aBitmap ) && rBmpEx.aMask.IsEqual( aMask ) ); } // ------------------------------------------------------------------ BOOL BitmapEx::IsEmpty() const { return( aBitmap.IsEmpty() && aMask.IsEmpty() ); } // ------------------------------------------------------------------ void BitmapEx::SetEmpty() { aBitmap.SetEmpty(); aMask.SetEmpty(); eTransparent = TRANSPARENT_NONE; bAlpha = FALSE; } // ------------------------------------------------------------------ void BitmapEx::Clear() { SetEmpty(); } // ------------------------------------------------------------------ BOOL BitmapEx::IsTransparent() const { return( eTransparent != TRANSPARENT_NONE ); } // ------------------------------------------------------------------ BOOL BitmapEx::IsAlpha() const { return( IsTransparent() && bAlpha ); } // ------------------------------------------------------------------ Bitmap BitmapEx::GetBitmap( const Color* pTransReplaceColor ) const { Bitmap aRetBmp( aBitmap ); if( pTransReplaceColor && ( eTransparent != TRANSPARENT_NONE ) ) { Bitmap aTempMask; if( eTransparent == TRANSPARENT_COLOR ) aTempMask = aBitmap.CreateMask( aTransparentColor ); else aTempMask = aMask; if( !IsAlpha() ) aRetBmp.Replace( aTempMask, *pTransReplaceColor ); else aRetBmp.Replace( GetAlpha(), *pTransReplaceColor ); } return aRetBmp; } // ------------------------------------------------------------------ BitmapEx BitmapEx::GetColorTransformedBitmapEx( BmpColorMode eColorMode ) const { BitmapEx aRet; if( BMP_COLOR_HIGHCONTRAST == eColorMode ) { aRet = *this; aRet.aBitmap = aBitmap.GetColorTransformedBitmap( eColorMode ); } else if( BMP_COLOR_MONOCHROME_BLACK == eColorMode || BMP_COLOR_MONOCHROME_WHITE == eColorMode ) { aRet = *this; aRet.aBitmap = aRet.aBitmap.GetColorTransformedBitmap( eColorMode ); if( !aRet.aMask.IsEmpty() ) { aRet.aMask.CombineSimple( aRet.aBitmap, BMP_COMBINE_OR ); aRet.aBitmap.Erase( ( BMP_COLOR_MONOCHROME_BLACK == eColorMode ) ? COL_BLACK : COL_WHITE ); } } return aRet; } // ------------------------------------------------------------------ Bitmap BitmapEx::GetMask() const { Bitmap aRet( aMask ); if( IsAlpha() ) aRet.ImplMakeMono( 255 ); return aRet; } // ------------------------------------------------------------------ AlphaMask BitmapEx::GetAlpha() const { AlphaMask aAlpha; if( IsAlpha() ) aAlpha.ImplSetBitmap( aMask ); else aAlpha = aMask; return aAlpha; } // ------------------------------------------------------------------ ULONG BitmapEx::GetSizeBytes() const { ULONG nSizeBytes = aBitmap.GetSizeBytes(); if( eTransparent == TRANSPARENT_BITMAP ) nSizeBytes += aMask.GetSizeBytes(); return nSizeBytes; } // ------------------------------------------------------------------ ULONG BitmapEx::GetChecksum() const { sal_uInt32 nCrc = aBitmap.GetChecksum(); SVBT32 aBT32; LongToSVBT32( (long) eTransparent, aBT32 ); nCrc = rtl_crc32( nCrc, aBT32, 4 ); LongToSVBT32( (long) bAlpha, aBT32 ); nCrc = rtl_crc32( nCrc, aBT32, 4 ); if( ( TRANSPARENT_BITMAP == eTransparent ) && !aMask.IsEmpty() ) { LongToSVBT32( aMask.GetChecksum(), aBT32 ); nCrc = rtl_crc32( nCrc, aBT32, 4 ); } return nCrc; } // ------------------------------------------------------------------ void BitmapEx::SetSizePixel( const Size& rNewSize ) { Scale( rNewSize ); } // ------------------------------------------------------------------ BOOL BitmapEx::Invert() { BOOL bRet = FALSE; if( !!aBitmap ) { bRet = aBitmap.Invert(); if( bRet && ( eTransparent == TRANSPARENT_COLOR ) ) aTransparentColor = BitmapColor( aTransparentColor ).Invert(); } return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::Mirror( ULONG nMirrorFlags ) { BOOL bRet = FALSE; if( !!aBitmap ) { bRet = aBitmap.Mirror( nMirrorFlags ); if( bRet && ( eTransparent == TRANSPARENT_BITMAP ) && !!aMask ) aMask.Mirror( nMirrorFlags ); } return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::Scale( const double& rScaleX, const double& rScaleY, ULONG nScaleFlag ) { BOOL bRet = FALSE; if( !!aBitmap ) { bRet = aBitmap.Scale( rScaleX, rScaleY, nScaleFlag ); if( bRet && ( eTransparent == TRANSPARENT_BITMAP ) && !!aMask ) aMask.Scale( rScaleX, rScaleY, BMP_SCALE_FAST ); aBitmapSize = aBitmap.GetSizePixel(); } return bRet; } // ------------------------------------------------------------------------ BOOL BitmapEx::Scale( const Size& rNewSize, ULONG nScaleFlag ) { BOOL bRet; if( aBitmapSize.Width() && aBitmapSize.Height() ) { bRet = Scale( (double) rNewSize.Width() / aBitmapSize.Width(), (double) rNewSize.Height() / aBitmapSize.Height(), nScaleFlag ); } else bRet = TRUE; return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::Rotate( long nAngle10, const Color& rFillColor ) { BOOL bRet = FALSE; if( !!aBitmap ) { const BOOL bTransRotate = ( Color( COL_TRANSPARENT ) == rFillColor ); if( bTransRotate ) { if( eTransparent == TRANSPARENT_COLOR ) bRet = aBitmap.Rotate( nAngle10, aTransparentColor ); else { bRet = aBitmap.Rotate( nAngle10, COL_BLACK ); if( eTransparent == TRANSPARENT_NONE ) { aMask = Bitmap( aBitmapSize, 1 ); aMask.Erase( COL_BLACK ); eTransparent = TRANSPARENT_BITMAP; } if( bRet && !!aMask ) aMask.Rotate( nAngle10, COL_WHITE ); } } else { bRet = aBitmap.Rotate( nAngle10, rFillColor ); if( bRet && ( eTransparent == TRANSPARENT_BITMAP ) && !!aMask ) aMask.Rotate( nAngle10, COL_WHITE ); } aBitmapSize = aBitmap.GetSizePixel(); } return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::Crop( const Rectangle& rRectPixel ) { BOOL bRet = FALSE; if( !!aBitmap ) { bRet = aBitmap.Crop( rRectPixel ); if( bRet && ( eTransparent == TRANSPARENT_BITMAP ) && !!aMask ) aMask.Crop( rRectPixel ); aBitmapSize = aBitmap.GetSizePixel(); } return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::Convert( BmpConversion eConversion ) { return( !!aBitmap ? aBitmap.Convert( eConversion ) : FALSE ); } // ------------------------------------------------------------------ BOOL BitmapEx::ReduceColors( USHORT nNewColorCount, BmpReduce eReduce ) { return( !!aBitmap ? aBitmap.ReduceColors( nNewColorCount, eReduce ) : FALSE ); } // ------------------------------------------------------------------ BOOL BitmapEx::Expand( ULONG nDX, ULONG nDY, const Color* pInitColor, BOOL bExpandTransparent ) { BOOL bRet = FALSE; if( !!aBitmap ) { bRet = aBitmap.Expand( nDX, nDY, pInitColor ); if( bRet && ( eTransparent == TRANSPARENT_BITMAP ) && !!aMask ) { Color aColor( bExpandTransparent ? COL_WHITE : COL_BLACK ); aMask.Expand( nDX, nDY, &aColor ); } aBitmapSize = aBitmap.GetSizePixel(); } return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::CopyPixel( const Rectangle& rRectDst, const Rectangle& rRectSrc, const BitmapEx* pBmpExSrc ) { BOOL bRet = FALSE; if( !pBmpExSrc || pBmpExSrc->IsEmpty() ) { if( !aBitmap.IsEmpty() ) { bRet = aBitmap.CopyPixel( rRectDst, rRectSrc ); if( bRet && ( eTransparent == TRANSPARENT_BITMAP ) && !!aMask ) aMask.CopyPixel( rRectDst, rRectSrc ); } } else { if( !aBitmap.IsEmpty() ) { bRet = aBitmap.CopyPixel( rRectDst, rRectSrc, &pBmpExSrc->aBitmap ); if( bRet ) { if( pBmpExSrc->IsAlpha() ) { if( IsAlpha() ) aMask.CopyPixel( rRectDst, rRectSrc, &pBmpExSrc->aMask ); else if( IsTransparent() ) { AlphaMask* pAlpha = new AlphaMask( aMask ); aMask = pAlpha->ImplGetBitmap(); delete pAlpha; bAlpha = TRUE; aMask.CopyPixel( rRectDst, rRectSrc, &pBmpExSrc->aMask ); } else { sal_uInt8 cBlack = 0; AlphaMask* pAlpha = new AlphaMask( GetSizePixel(), &cBlack ); aMask = pAlpha->ImplGetBitmap(); delete pAlpha; eTransparent = TRANSPARENT_BITMAP; bAlpha = TRUE; aMask.CopyPixel( rRectDst, rRectSrc, &pBmpExSrc->aMask ); } } else if( pBmpExSrc->IsTransparent() ) { if( IsAlpha() ) { AlphaMask aAlpha( pBmpExSrc->aMask ); aMask.CopyPixel( rRectDst, rRectSrc, &aAlpha.ImplGetBitmap() ); } else if( IsTransparent() ) aMask.CopyPixel( rRectDst, rRectSrc, &pBmpExSrc->aMask ); else { aMask = Bitmap( GetSizePixel(), 1 ); aMask.Erase( Color( COL_BLACK ) ); eTransparent = TRANSPARENT_BITMAP; aMask.CopyPixel( rRectDst, rRectSrc, &pBmpExSrc->aMask ); } } else if( IsAlpha() ) { sal_uInt8 cBlack = 0; const AlphaMask aAlphaSrc( pBmpExSrc->GetSizePixel(), &cBlack ); aMask.CopyPixel( rRectDst, rRectSrc, &aAlphaSrc.ImplGetBitmap() ); } else if( IsTransparent() ) { Bitmap aMaskSrc( pBmpExSrc->GetSizePixel(), 1 ); aMaskSrc.Erase( Color( COL_BLACK ) ); aMask.CopyPixel( rRectDst, rRectSrc, &aMaskSrc ); } } } } return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::Erase( const Color& rFillColor ) { BOOL bRet = FALSE; if( !!aBitmap ) { bRet = aBitmap.Erase( rFillColor ); if( bRet && ( eTransparent == TRANSPARENT_BITMAP ) && !!aMask ) { // #104416# Respect transparency on fill color if( rFillColor.GetTransparency() ) { const Color aFill( rFillColor.GetTransparency(), rFillColor.GetTransparency(), rFillColor.GetTransparency() ); aMask.Erase( aFill ); } else { const Color aBlack( COL_BLACK ); aMask.Erase( aBlack ); } } } return bRet; } // ------------------------------------------------------------------ BOOL BitmapEx::Dither( ULONG nDitherFlags, const BitmapPalette* pDitherPal ) { return( !!aBitmap ? aBitmap.Dither( nDitherFlags, pDitherPal ) : FALSE ); } // ------------------------------------------------------------------ BOOL BitmapEx::Replace( const Color& rSearchColor, const Color& rReplaceColor, ULONG nTol ) { return( !!aBitmap ? aBitmap.Replace( rSearchColor, rReplaceColor, nTol ) : FALSE ); } // ------------------------------------------------------------------ BOOL BitmapEx::Replace( const Color* pSearchColors, const Color* pReplaceColors, ULONG nColorCount, const ULONG* pTols ) { return( !!aBitmap ? aBitmap.Replace( pSearchColors, pReplaceColors, nColorCount, (ULONG*) pTols ) : FALSE ); } // ------------------------------------------------------------------ BOOL BitmapEx::Adjust( short nLuminancePercent, short nContrastPercent, short nChannelRPercent, short nChannelGPercent, short nChannelBPercent, double fGamma, BOOL bInvert ) { return( !!aBitmap ? aBitmap.Adjust( nLuminancePercent, nContrastPercent, nChannelRPercent, nChannelGPercent, nChannelBPercent, fGamma, bInvert ) : FALSE ); } // ------------------------------------------------------------------ BOOL BitmapEx::Filter( BmpFilter eFilter, const BmpFilterParam* pFilterParam, const Link* pProgress ) { return( !!aBitmap ? aBitmap.Filter( eFilter, pFilterParam, pProgress ) : FALSE ); } // ------------------------------------------------------------------ void BitmapEx::Draw( OutputDevice* pOutDev, const Point& rDestPt ) const { pOutDev->DrawBitmapEx( rDestPt, *this ); } // ------------------------------------------------------------------ void BitmapEx::Draw( OutputDevice* pOutDev, const Point& rDestPt, const Size& rDestSize ) const { pOutDev->DrawBitmapEx( rDestPt, rDestSize, *this ); } // ------------------------------------------------------------------ void BitmapEx::Draw( OutputDevice* pOutDev, const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel ) const { pOutDev->DrawBitmapEx( rDestPt, rDestSize, rSrcPtPixel, rSrcSizePixel, *this ); } // ------------------------------------------------------------------ SvStream& operator<<( SvStream& rOStm, const BitmapEx& rBitmapEx ) { rBitmapEx.aBitmap.Write( rOStm ); rOStm << (UINT32) 0x25091962; rOStm << (UINT32) 0xACB20201; rOStm << (BYTE) rBitmapEx.eTransparent; if( rBitmapEx.eTransparent == TRANSPARENT_BITMAP ) rBitmapEx.aMask.Write( rOStm ); else if( rBitmapEx.eTransparent == TRANSPARENT_COLOR ) rOStm << rBitmapEx.aTransparentColor; return rOStm; } // ------------------------------------------------------------------ SvStream& operator>>( SvStream& rIStm, BitmapEx& rBitmapEx ) { Bitmap aBmp; rIStm >> aBmp; if( !rIStm.GetError() ) { const ULONG nStmPos = rIStm.Tell(); UINT32 nMagic1; UINT32 nMagic2; rIStm >> nMagic1 >> nMagic2; if( ( nMagic1 != 0x25091962 ) || ( nMagic2 != 0xACB20201 ) || rIStm.GetError() ) { rIStm.Seek( nStmPos ); rBitmapEx = aBmp; } else { BYTE bTransparent; rIStm >> bTransparent; if( bTransparent == (BYTE) TRANSPARENT_BITMAP ) { Bitmap aMask; rIStm >> aMask; if( !!aMask) { // do we have an alpha mask? if( ( 8 == aMask.GetBitCount() ) && aMask.HasGreyPalette() ) { AlphaMask aAlpha; // create alpha mask quickly (without greyscale conversion) aAlpha.ImplSetBitmap( aMask ); rBitmapEx = BitmapEx( aBmp, aAlpha ); } else rBitmapEx = BitmapEx( aBmp, aMask ); } else rBitmapEx = aBmp; } else if( bTransparent == (BYTE) TRANSPARENT_COLOR ) { Color aTransparentColor; rIStm >> aTransparentColor; rBitmapEx = BitmapEx( aBmp, aTransparentColor ); } else rBitmapEx = aBmp; } } return rIStm; }