/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: salbmp.cxx,v $ * * $Revision: 1.21 $ * * last change: $Author: hr $ $Date: 2006-06-19 19:53:52 $ * * The Contents of this file are made available subject to * the terms of GNU Lesser General Public License Version 2.1. * * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2005 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 * ************************************************************************/ #include #include #include #ifdef FREEBSD #include #endif #include #include #include #ifndef _OSL_ENDIAN_H_ #include #endif #ifndef _RTL_MEMORY_H_ #include #endif #ifndef _SV_SALBTYPE_HXX #include #endif #ifndef _SV_SALDATA_HXX #include #endif #ifndef _SV_SALDISP_HXX #include #endif #ifndef _SV_SALGDI_H #include #endif #ifndef _SV_SALBMP_H #include #endif #ifndef _SV_SALINST_H #include #endif #ifndef _SV_BITMAP_HXX #include #endif // ----------- // - Defines - // ----------- #define SAL_DRAWPIXMAP_MAX_EXT 4096 // ------------- // - SalBitmap - // ------------- SalBitmap* X11SalInstance::CreateSalBitmap() { return new X11SalBitmap(); } ImplSalBitmapCache* X11SalBitmap::mpCache = NULL; ULONG X11SalBitmap::mnCacheInstCount = 0; // ----------------------------------------------------------------------------- X11SalBitmap::X11SalBitmap() : mpDIB( NULL ), mpDDB( NULL ) { } // ----------------------------------------------------------------------------- X11SalBitmap::~X11SalBitmap() { Destroy(); } // ----------------------------------------------------------------------------- void X11SalBitmap::ImplCreateCache() { if( !mnCacheInstCount++ ) mpCache = new ImplSalBitmapCache; } // ----------------------------------------------------------------------------- void X11SalBitmap::ImplDestroyCache() { DBG_ASSERT( mnCacheInstCount, "X11SalBitmap::ImplDestroyCache(): underflow" ); if( mnCacheInstCount && !--mnCacheInstCount ) delete mpCache, mpCache = NULL; } // ----------------------------------------------------------------------------- void X11SalBitmap::ImplRemovedFromCache() { if( mpDDB ) delete mpDDB, mpDDB = NULL; } // ----------------------------------------------------------------------------- BitmapBuffer* X11SalBitmap::ImplCreateDIB( const Size& rSize, USHORT nBitCount, const BitmapPalette& rPal ) { DBG_ASSERT( nBitCount == 1 || nBitCount == 4 || nBitCount == 8 || nBitCount == 24, "Unsupported BitCount!" ); BitmapBuffer* pDIB; if( rSize.Width() && rSize.Height() ) { pDIB = new BitmapBuffer; if( pDIB ) { const USHORT nColors = ( nBitCount <= 8 ) ? ( 1 << nBitCount ) : 0; pDIB->mnFormat = BMP_FORMAT_BOTTOM_UP; switch( nBitCount ) { case( 1 ): pDIB->mnFormat |= BMP_FORMAT_1BIT_MSB_PAL; break; case( 4 ): pDIB->mnFormat |= BMP_FORMAT_4BIT_MSN_PAL; break; case( 8 ): pDIB->mnFormat |= BMP_FORMAT_8BIT_PAL; break; default: nBitCount = 24; //fall through case 24: pDIB->mnFormat |= BMP_FORMAT_24BIT_TC_BGR; break; } pDIB->mnWidth = rSize.Width(); pDIB->mnHeight = rSize.Height(); pDIB->mnScanlineSize = AlignedWidth4Bytes( pDIB->mnWidth * nBitCount ); pDIB->mnBitCount = nBitCount; if( nColors ) { pDIB->maPalette = rPal; pDIB->maPalette.SetEntryCount( nColors ); } pDIB->mpBits = new BYTE[ pDIB->mnScanlineSize * pDIB->mnHeight ]; } } else pDIB = NULL; return pDIB; } // ----------------------------------------------------------------------------- BitmapBuffer* X11SalBitmap::ImplCreateDIB( Drawable aDrawable, long nDrawableDepth, long nX, long nY, long nWidth, long nHeight ) { BitmapBuffer* pDIB = NULL; if( aDrawable && nWidth && nHeight && nDrawableDepth ) { SalDisplay* pSalDisp = GetSalData()->GetDisplay(); SalXLib* pXLib = pSalDisp->GetXLib(); Display* pXDisp = pSalDisp->GetDisplay(); // do not die on XError here // alternatively one could check the coordinates for being offscreen // but this call can actually work on servers with backing store // defaults even if the rectangle is offscreen // so better catch the XError BOOL bXError = pXLib->GetIgnoreXErrors(); pXLib->SetIgnoreXErrors( TRUE ); XImage* pImage = XGetImage( pXDisp, aDrawable, nX, nY, nWidth, nHeight, AllPlanes, ZPixmap ); BOOL bWasError = pXLib->WasXError(); pXLib->SetIgnoreXErrors( bXError ); if( ! bWasError && pImage && pImage->data ) { const SalTwoRect aTwoRect = { 0, 0, nWidth, nHeight, 0, 0, nWidth, nHeight }; BitmapBuffer aSrcBuf; ULONG nDstFormat = BMP_FORMAT_BOTTOM_UP; const BitmapPalette* pDstPal = NULL; aSrcBuf.mnFormat = BMP_FORMAT_TOP_DOWN; aSrcBuf.mnWidth = nWidth; aSrcBuf.mnHeight = nHeight; aSrcBuf.mnBitCount = pImage->bits_per_pixel; aSrcBuf.mnScanlineSize = pImage->bytes_per_line; aSrcBuf.mpBits = (BYTE*) pImage->data; pImage->red_mask = pSalDisp->GetVisual()->red_mask; pImage->green_mask = pSalDisp->GetVisual()->green_mask; pImage->blue_mask = pSalDisp->GetVisual()->blue_mask; switch( aSrcBuf.mnBitCount ) { case( 1 ): { aSrcBuf.mnFormat |= ( LSBFirst == pImage->bitmap_bit_order ? BMP_FORMAT_1BIT_LSB_PAL : BMP_FORMAT_1BIT_MSB_PAL ); nDstFormat |= BMP_FORMAT_1BIT_MSB_PAL; } break; case( 4 ): { aSrcBuf.mnFormat |= ( LSBFirst == pImage->bitmap_bit_order ? BMP_FORMAT_4BIT_LSN_PAL : BMP_FORMAT_4BIT_MSN_PAL ); nDstFormat |= BMP_FORMAT_4BIT_MSN_PAL; } break; case( 8 ): { aSrcBuf.mnFormat |= BMP_FORMAT_8BIT_PAL; nDstFormat |= BMP_FORMAT_8BIT_PAL; } break; case( 16 ): { nDstFormat |= BMP_FORMAT_24BIT_TC_BGR; aSrcBuf.maColorMask = ColorMask( pImage->red_mask, pImage->green_mask, pImage->blue_mask ); if( LSBFirst == pImage->byte_order ) { aSrcBuf.mnFormat |= BMP_FORMAT_16BIT_TC_LSB_MASK; } else { aSrcBuf.mnFormat |= BMP_FORMAT_16BIT_TC_MSB_MASK; // aSrcBuf.maColorMask = ColorMask( pImage->red_mask ), SWAPSHORT( pImage->green_mask ), SWAPSHORT( pImage->blue_mask ) ); } } break; case( 24 ): { if( ( LSBFirst == pImage->byte_order ) && ( pImage->red_mask == 0xFF ) ) aSrcBuf.mnFormat |= BMP_FORMAT_24BIT_TC_RGB; else aSrcBuf.mnFormat |= BMP_FORMAT_24BIT_TC_BGR; nDstFormat |= BMP_FORMAT_24BIT_TC_BGR; } break; case( 32 ): { if( LSBFirst == pImage->byte_order ) aSrcBuf.mnFormat |= ( pSalDisp->GetVisual()->red_mask == 0xFF ? BMP_FORMAT_32BIT_TC_RGBA : BMP_FORMAT_32BIT_TC_BGRA ); else aSrcBuf.mnFormat |= ( pSalDisp->GetVisual()->red_mask == 0xFF ? BMP_FORMAT_32BIT_TC_ABGR : BMP_FORMAT_32BIT_TC_ARGB ); nDstFormat |= BMP_FORMAT_24BIT_TC_BGR; } break; } BitmapPalette& rPal = aSrcBuf.maPalette; if( aSrcBuf.mnBitCount == 1 ) { rPal.SetEntryCount( 2 ); pDstPal = &rPal; rPal[ 0 ] = Color( COL_BLACK ); rPal[ 1 ] = Color( COL_WHITE ); } else if( aSrcBuf.mnBitCount <= 8 ) { SalColormap& rColMap = pSalDisp->GetColormap(); const USHORT nCols = Min( (ULONG)rColMap.GetUsed(), (ULONG)(1 << nDrawableDepth) ); rPal.SetEntryCount( nCols ); pDstPal = &rPal; for( USHORT i = 0; i < nCols; i++ ) { const SalColor nColor( rColMap.GetColor( i ) ); BitmapColor& rBmpCol = rPal[ i ]; rBmpCol.SetRed( SALCOLOR_RED( nColor ) ); rBmpCol.SetGreen( SALCOLOR_GREEN( nColor ) ); rBmpCol.SetBlue( SALCOLOR_BLUE( nColor ) ); } } nDstFormat = aSrcBuf.mnFormat; pDIB = StretchAndConvert( aSrcBuf, aTwoRect, nDstFormat, const_cast(pDstPal), &aSrcBuf.maColorMask ); XDestroyImage( pImage ); } } return pDIB; } // ----------------------------------------------------------------------------- XImage* X11SalBitmap::ImplCreateXImage( SalDisplay *pSalDisp, long nDepth, const SalTwoRect& rTwoRect ) const { XImage* pImage = NULL; if( !mpDIB && mpDDB ) { const_cast(this)->mpDIB = ImplCreateDIB( mpDDB->ImplGetPixmap(), mpDDB->ImplGetDepth(), 0, 0, mpDDB->ImplGetWidth(), mpDDB->ImplGetHeight() ); } if( mpDIB && mpDIB->mnWidth && mpDIB->mnHeight ) { Display* pXDisp = pSalDisp->GetDisplay(); long nWidth = rTwoRect.mnDestWidth; long nHeight = rTwoRect.mnDestHeight; if( 1 == GetBitCount() ) nDepth = 1; pImage = XCreateImage( pXDisp, pSalDisp->GetVisual()->GetVisual(), nDepth, ( 1 == nDepth ) ? XYBitmap :ZPixmap, 0, NULL, nWidth, nHeight, 32, 0 ); if( pImage ) { BitmapBuffer* pDstBuf; ULONG nDstFormat = BMP_FORMAT_TOP_DOWN; BitmapPalette* pPal = NULL; ColorMask* pMask = NULL; switch( pImage->bits_per_pixel ) { case( 1 ): nDstFormat |= ( LSBFirst == pImage->bitmap_bit_order ? BMP_FORMAT_1BIT_LSB_PAL : BMP_FORMAT_1BIT_MSB_PAL ); break; case( 4 ): nDstFormat |= ( LSBFirst == pImage->bitmap_bit_order ? BMP_FORMAT_4BIT_LSN_PAL : BMP_FORMAT_4BIT_MSN_PAL ); break; case( 8 ): nDstFormat |= BMP_FORMAT_8BIT_PAL; break; case( 16 ): { #ifdef OSL_BIGENDIAN if( MSBFirst == pImage->byte_order ) nDstFormat |= BMP_FORMAT_16BIT_TC_MSB_MASK; else nDstFormat |= BMP_FORMAT_16BIT_TC_LSB_MASK; #else /* OSL_LITENDIAN */ nDstFormat |= BMP_FORMAT_16BIT_TC_LSB_MASK; if( MSBFirst == pImage->byte_order ) pImage->byte_order = LSBFirst; #endif pMask = new ColorMask( pImage->red_mask, pImage->green_mask, pImage->blue_mask ); } break; case( 24 ): { if( ( LSBFirst == pImage->byte_order ) && ( pImage->red_mask == 0xFF ) ) nDstFormat |= BMP_FORMAT_24BIT_TC_RGB; else nDstFormat |= BMP_FORMAT_24BIT_TC_BGR; } break; case( 32 ): { if( LSBFirst == pImage->byte_order ) nDstFormat |= ( pImage->red_mask == 0xFF ? BMP_FORMAT_32BIT_TC_RGBA : BMP_FORMAT_32BIT_TC_BGRA ); else nDstFormat |= ( pImage->red_mask == 0xFF ? BMP_FORMAT_32BIT_TC_ABGR : BMP_FORMAT_32BIT_TC_ARGB ); } break; } if( pImage->depth == 1 ) { pPal = new BitmapPalette( 2 ); (*pPal)[ 0 ] = Color( COL_BLACK ); (*pPal)[ 1 ] = Color( COL_WHITE ); } else if( pImage->depth <= 8 ) { SalColormap& rColMap = pSalDisp->GetColormap(); const USHORT nCols = Min( (ULONG)rColMap.GetUsed(), (ULONG)(1 << pImage->depth) ); pPal = new BitmapPalette( nCols ); for( USHORT i = 0; i < nCols; i++ ) { const SalColor nColor( rColMap.GetColor( i ) ); BitmapColor& rBmpCol = (*pPal)[ i ]; rBmpCol.SetRed( SALCOLOR_RED( nColor ) ); rBmpCol.SetGreen( SALCOLOR_GREEN( nColor ) ); rBmpCol.SetBlue( SALCOLOR_BLUE( nColor ) ); } } pDstBuf = StretchAndConvert( *mpDIB, rTwoRect, nDstFormat, pPal, pMask ); delete pPal; delete pMask; if( pDstBuf && pDstBuf->mpBits ) { // set data in buffer as data member in pImage pImage->data = (char*) pDstBuf->mpBits; // destroy buffer; don't destroy allocated data in buffer delete pDstBuf; } else { XDestroyImage( pImage ); pImage = NULL; } } } return pImage; } // ----------------------------------------------------------------------------- #ifdef _USE_PRINT_EXTENSION_ bool X11SalBitmap::ImplCreateFromDrawable( SalDisplay* pDisplay, Drawable aDrawable, long nDrawableDepth, long nX, long nY, long nWidth, long nHeight ) { Destroy(); if( aDrawable && nWidth && nHeight && nDrawableDepth ) mpDDB = new ImplSalDDB( pDisplay, aDrawable, nDrawableDepth, nX, nY, nWidth, nHeight ); return( mpDDB != NULL ); } #else bool X11SalBitmap::ImplCreateFromDrawable( Drawable aDrawable, long nDrawableDepth, long nX, long nY, long nWidth, long nHeight ) { Destroy(); if( aDrawable && nWidth && nHeight && nDrawableDepth ) mpDDB = new ImplSalDDB( aDrawable, nDrawableDepth, nX, nY, nWidth, nHeight ); return( mpDDB != NULL ); } // ----------------------------------------------------------------------------- #endif bool X11SalBitmap::SnapShot (Display* pDisplay, XLIB_Window hWindow) { if (hWindow != None) { XWindowAttributes aAttribute; XGetWindowAttributes (pDisplay, hWindow, &aAttribute); if (aAttribute.map_state == IsViewable) { // get coordinates relative to root window XLIB_Window hPetitFleur; int nRootX, nRootY; if (XTranslateCoordinates (pDisplay, hWindow, aAttribute.root, 0, 0, &nRootX, &nRootY, &hPetitFleur)) { XWindowAttributes aRootAttribute; XGetWindowAttributes (pDisplay, aAttribute.root, &aRootAttribute); int width = aAttribute.width; int height = aAttribute.height; int x = nRootX; int y = nRootY; // horizontal range check if (x < 0) { width = width + x; x = 0; } else if (x > aRootAttribute.width) { width = 0; x = aRootAttribute.width; } else if (x + width > aRootAttribute.width) { width = aRootAttribute.width - x; } // vertical range check if (y < 0) { height = height + y; y = 0; } else if (y > aRootAttribute.height) { height = 0; y = aRootAttribute.height; } else if (y + height > aRootAttribute.height) { height = aRootAttribute.height - y; } if ((width > 0) && (height > 0)) { XImage* pImage = XGetImage (pDisplay, aAttribute.root, x, y, width, height, AllPlanes, ZPixmap); bool bSnapShot = ImplCreateFromXImage (pDisplay, aAttribute.root, pImage); XDestroyImage (pImage); return bSnapShot; } } } } return False; } bool X11SalBitmap::ImplCreateFromXImage (Display* pDisplay, XLIB_Window hWindow, XImage* pImage) { Destroy(); if (pImage != NULL && pImage->width != 0 && pImage->height != 0 && pImage->depth != 0) { mpDDB = new ImplSalDDB (pDisplay, hWindow, pImage); return True; } return False; } #ifdef _USE_PRINT_EXTENSION_ void X11SalBitmap::ImplDraw( SalDisplay *pDisplay, Drawable aDrawable, long nDrawableDepth, const SalTwoRect& rTwoRect, const GC& rGC ) const { if( !mpDDB || !mpDDB->ImplMatches( nDrawableDepth, rTwoRect ) ) { if( mpDDB ) { // do we already have a DIB? if not, create aDIB from current DDB first if( !mpDIB ) { ( (SalBitmap*) this )->mpDIB = ImplCreateDIB( mpDDB->ImplGetPixmap(), mpDDB->ImplGetDepth(), 0, 0, mpDDB->ImplGetWidth(), mpDDB->ImplGetHeight() ); } delete mpDDB, ( (SalBitmap*) this )->mpDDB = NULL; } if( mpCache ) mpCache->ImplRemove( const_cast(this) ); SalTwoRect aTwoRect( rTwoRect ); // create new DDB from DIB if( aTwoRect.mnSrcWidth == aTwoRect.mnDestWidth && aTwoRect.mnSrcHeight == aTwoRect.mnDestHeight ) { const Size aSize( GetSize() ); aTwoRect.mnSrcX = aTwoRect.mnSrcY = aTwoRect.mnDestX = aTwoRect.mnDestY = 0; aTwoRect.mnSrcWidth = aTwoRect.mnDestWidth = aSize.Width(); aTwoRect.mnSrcHeight = aTwoRect.mnDestHeight = aSize.Height(); } XImage* pImage = ImplCreateXImage( pDisplay, nDrawableDepth, aTwoRect ); if( pImage ) { ( (SalBitmap*) this )->mpDDB = new ImplSalDDB( pDisplay, pImage, aDrawable, aTwoRect ); delete[] pImage->data, pImage->data = NULL; XDestroyImage( pImage ); if( mpCache ) mpCache->ImplAdd( const_cast(this), mpDDB->ImplGetMemSize() ); } } if( mpDDB ) mpDDB->ImplDraw( pDisplay, aDrawable, nDrawableDepth, rTwoRect, rGC ); } #else void X11SalBitmap::ImplDraw( Drawable aDrawable, long nDrawableDepth, const SalTwoRect& rTwoRect, const GC& rGC ) const { if( !mpDDB || !mpDDB->ImplMatches( nDrawableDepth, rTwoRect ) ) { if( mpDDB ) { // do we already have a DIB? if not, create aDIB from current DDB first if( !mpDIB ) { const_cast(this)->mpDIB = ImplCreateDIB( mpDDB->ImplGetPixmap(), mpDDB->ImplGetDepth(), 0, 0, mpDDB->ImplGetWidth(), mpDDB->ImplGetHeight() ); } delete mpDDB, const_cast(this)->mpDDB = NULL; } if( mpCache ) mpCache->ImplRemove( const_cast(this) ); SalTwoRect aTwoRect( rTwoRect ); if( aTwoRect.mnSrcX < 0 ) { aTwoRect.mnSrcWidth += aTwoRect.mnSrcX; aTwoRect.mnSrcX = 0; } if( aTwoRect.mnSrcY < 0 ) { aTwoRect.mnSrcHeight += aTwoRect.mnSrcY; aTwoRect.mnSrcY = 0; } // create new DDB from DIB const Size aSize( GetSize() ); if( aTwoRect.mnSrcWidth == aTwoRect.mnDestWidth && aTwoRect.mnSrcHeight == aTwoRect.mnDestHeight ) { aTwoRect.mnSrcX = aTwoRect.mnSrcY = aTwoRect.mnDestX = aTwoRect.mnDestY = 0; aTwoRect.mnSrcWidth = aTwoRect.mnDestWidth = aSize.Width(); aTwoRect.mnSrcHeight = aTwoRect.mnDestHeight = aSize.Height(); } else if( aTwoRect.mnSrcWidth+aTwoRect.mnSrcX > aSize.Width() || aTwoRect.mnSrcHeight+aTwoRect.mnSrcY > aSize.Height() ) { // #i47823# this should not happen at all, but does nonetheless // because BitmapEx allows for mask bitmaps of different size // than image bitmap (broken) if( aTwoRect.mnSrcX >= aSize.Width() || aTwoRect.mnSrcY >= aSize.Height() ) return; // this would be a really mad case if( aTwoRect.mnSrcWidth+aTwoRect.mnSrcX > aSize.Width() ) { aTwoRect.mnSrcWidth = aSize.Width()-aTwoRect.mnSrcX; if( aTwoRect.mnSrcWidth < 1 ) { aTwoRect.mnSrcX = 0; aTwoRect.mnSrcWidth = aSize.Width(); } } if( aTwoRect.mnSrcHeight+aTwoRect.mnSrcY > aSize.Height() ) { aTwoRect.mnSrcHeight = aSize.Height() - aTwoRect.mnSrcY; if( aTwoRect.mnSrcHeight < 1 ) { aTwoRect.mnSrcY = 0; aTwoRect.mnSrcHeight = aSize.Height(); } } } XImage* pImage = ImplCreateXImage( GetSalData()->GetDisplay(), nDrawableDepth, aTwoRect ); if( pImage ) { const_cast(this)->mpDDB = new ImplSalDDB( pImage, aDrawable, aTwoRect ); delete[] pImage->data, pImage->data = NULL; XDestroyImage( pImage ); if( mpCache ) mpCache->ImplAdd( const_cast(this), mpDDB->ImplGetMemSize() ); } } if( mpDDB ) mpDDB->ImplDraw( aDrawable, nDrawableDepth, rTwoRect, rGC ); } #endif // ----------------------------------------------------------------------------- bool X11SalBitmap::Create( const Size& rSize, USHORT nBitCount, const BitmapPalette& rPal ) { Destroy(); mpDIB = ImplCreateDIB( rSize, nBitCount, rPal ); return( mpDIB != NULL ); } // ----------------------------------------------------------------------------- bool X11SalBitmap::Create( const SalBitmap& rSSalBmp ) { Destroy(); const X11SalBitmap& rSalBmp = static_cast( rSSalBmp ); if( rSalBmp.mpDIB ) { // TODO: reference counting... mpDIB = new BitmapBuffer( *rSalBmp.mpDIB ); // TODO: get rid of this when BitmapBuffer gets copy constructor mpDIB->mpBits = new BYTE[ mpDIB->mnScanlineSize * mpDIB->mnHeight ]; if( mpDIB ) memcpy( mpDIB->mpBits, rSalBmp.mpDIB->mpBits, mpDIB->mnScanlineSize * mpDIB->mnHeight ); } else if( rSalBmp.mpDDB ) { #ifndef _USE_PRINT_EXTENSION_ ImplCreateFromDrawable( rSalBmp.mpDDB->ImplGetPixmap(), rSalBmp.mpDDB->ImplGetDepth(), 0, 0, rSalBmp.mpDDB->ImplGetWidth(), rSalBmp.mpDDB->ImplGetHeight() ); #endif } return( ( !rSalBmp.mpDIB && !rSalBmp.mpDDB ) || ( rSalBmp.mpDIB && ( mpDIB != NULL ) ) || ( rSalBmp.mpDDB && ( mpDDB != NULL ) ) ); } // ----------------------------------------------------------------------------- bool X11SalBitmap::Create( const SalBitmap&, SalGraphics* ) { return FALSE; } // ----------------------------------------------------------------------------- bool X11SalBitmap::Create( const SalBitmap&, USHORT ) { return FALSE; } // ----------------------------------------------------------------------------- void X11SalBitmap::Destroy() { if( mpDIB ) { delete[] mpDIB->mpBits; delete mpDIB, mpDIB = NULL; } if( mpDDB ) delete mpDDB, mpDDB = NULL; if( mpCache ) mpCache->ImplRemove( this ); } // ----------------------------------------------------------------------------- Size X11SalBitmap::GetSize() const { Size aSize; if( mpDIB ) aSize.Width() = mpDIB->mnWidth, aSize.Height() = mpDIB->mnHeight; else if( mpDDB ) aSize.Width() = mpDDB->ImplGetWidth(), aSize.Height() = mpDDB->ImplGetHeight(); return aSize; } // ----------------------------------------------------------------------------- USHORT X11SalBitmap::GetBitCount() const { USHORT nBitCount; if( mpDIB ) nBitCount = mpDIB->mnBitCount; else if( mpDDB ) { nBitCount = mpDDB->ImplGetDepth(); if( nBitCount && nBitCount > 1 ) { if( nBitCount <= 4 ) nBitCount = 4; else if( nBitCount <= 8 ) nBitCount = 8; else nBitCount = 24; } } else nBitCount = 0; return nBitCount; } // ----------------------------------------------------------------------------- BitmapBuffer* X11SalBitmap::AcquireBuffer( bool ) { if( !mpDIB && mpDDB ) { mpDIB = ImplCreateDIB( mpDDB->ImplGetPixmap(), mpDDB->ImplGetDepth(), 0, 0, mpDDB->ImplGetWidth(), mpDDB->ImplGetHeight() ); } return mpDIB; } // ----------------------------------------------------------------------------- void X11SalBitmap::ReleaseBuffer( BitmapBuffer*, bool bReadOnly ) { if( !bReadOnly ) { if( mpDDB ) delete mpDDB, mpDDB = NULL; if( mpCache ) mpCache->ImplRemove( this ); } } // ----------------------------------------------------------------------------- bool X11SalBitmap::GetSystemData( BitmapSystemData& rData ) { if( mpDDB ) { // Rename/retype pDummy to your likings (though X11 Pixmap is // prolly not a good idea, since it's accessed from // non-platform aware code in vcl/bitmap.hxx) rData.aPixmap = (void*)mpDDB->ImplGetPixmap(); return true; } return false; } // -------------- // - ImplSalDDB - // -------------- ImplSalDDB::ImplSalDDB( #ifdef _USE_PRINT_EXTENSION_ SalDisplay* pDisplay, #endif XImage* pImage, Drawable aDrawable, const SalTwoRect& rTwoRect ) : maPixmap ( 0 ), maTwoRect ( rTwoRect ), mnDepth ( pImage->depth ) { #if !defined(_USE_PRINT_EXTENSION_) SalDisplay* pSalDisp = GetSalData()->GetDisplay(); Display* pXDisp = pSalDisp->GetDisplay(); #else Display* pXDisp = pDisplay->GetDisplay(); #endif if( (maPixmap = XCreatePixmap( pXDisp, aDrawable, ImplGetWidth(), ImplGetHeight(), ImplGetDepth() )) ) { XGCValues aValues; GC aGC; int nValues = GCFunction; aValues.function = GXcopy; if( 1 == mnDepth ) { nValues |= ( GCForeground | GCBackground ); aValues.foreground = 1, aValues.background = 0; } aGC = XCreateGC( pXDisp, maPixmap, nValues, &aValues ); XPutImage( pXDisp, maPixmap, aGC, pImage, 0, 0, 0, 0, maTwoRect.mnDestWidth, maTwoRect.mnDestHeight ); XFreeGC( pXDisp, aGC ); } } // ----------------------------------------------------------------------------------------- // create from XImage ImplSalDDB::ImplSalDDB (Display* pDisplay, XLIB_Window hWindow, XImage* pImage) { maPixmap = XCreatePixmap (pDisplay, hWindow, pImage->width, pImage->height, pImage->depth); if (maPixmap != 0) { XGCValues aValues; GC aGC; int nValues = GCFunction; aValues.function = GXcopy; if (pImage->depth == 1) { nValues |= ( GCForeground | GCBackground ); aValues.foreground = 1; aValues.background = 0; } aGC = XCreateGC (pDisplay, maPixmap, nValues, &aValues); XPutImage (pDisplay, maPixmap, aGC, pImage, 0, 0, 0, 0, pImage->width, pImage->height); XFreeGC (pDisplay, aGC); maTwoRect.mnSrcX = 0; maTwoRect.mnSrcY = 0; maTwoRect.mnDestX = 0; maTwoRect.mnDestY = 0; maTwoRect.mnSrcWidth = pImage->width; maTwoRect.mnDestWidth = pImage->width; maTwoRect.mnSrcHeight = pImage->height; maTwoRect.mnDestHeight = pImage->height; mnDepth = pImage->depth; } } // ----------------------------------------------------------------------------- ImplSalDDB::ImplSalDDB( #ifdef _USE_PRINT_EXTENSION_ SalDisplay* pDisplay, #endif Drawable aDrawable, long nDrawableDepth, long nX, long nY, long nWidth, long nHeight ) : mnDepth( nDrawableDepth ) { #if !defined(_USE_PRINT_EXTENSION_) SalDisplay* pSalDisp = GetSalData()->GetDisplay(); Display* pXDisp = pSalDisp->GetDisplay(); #else Display* pXDisp = pDisplay->GetDisplay(); #endif if( (maPixmap = XCreatePixmap( pXDisp, aDrawable, nWidth, nHeight, nDrawableDepth )) ) { XGCValues aValues; GC aGC; int nValues = GCFunction; aValues.function = GXcopy; if( 1 == mnDepth ) { nValues |= ( GCForeground | GCBackground ); aValues.foreground = 1, aValues.background = 0; } aGC = XCreateGC( pXDisp, maPixmap, nValues, &aValues ); #ifdef _USE_PRINT_EXTENSION_ ImplDraw( pDisplay, aDrawable, nDrawableDepth, maPixmap, mnDepth, nX, nY, nWidth, nHeight, 0, 0, aGC ); #else ImplDraw( aDrawable, nDrawableDepth, maPixmap, mnDepth, nX, nY, nWidth, nHeight, 0, 0, aGC ); #endif XFreeGC( pXDisp, aGC ); maTwoRect.mnSrcX = maTwoRect.mnSrcY = maTwoRect.mnDestX = maTwoRect.mnDestY = 0; maTwoRect.mnSrcWidth = maTwoRect.mnDestWidth = nWidth; maTwoRect.mnSrcHeight = maTwoRect.mnDestHeight = nHeight; } } // ----------------------------------------------------------------------------- ImplSalDDB::~ImplSalDDB() { if( maPixmap && ImplGetSVData() ) XFreePixmap( GetSalData()->GetDisplay()->GetDisplay(), maPixmap ); } // ----------------------------------------------------------------------------- bool ImplSalDDB::ImplMatches( long nDepth, const SalTwoRect& rTwoRect ) const { bool bRet = FALSE; if( ( maPixmap != 0 ) && ( ( mnDepth == nDepth ) || ( 1 == mnDepth ) ) ) { if( rTwoRect.mnSrcX == maTwoRect.mnSrcX && rTwoRect.mnSrcY == maTwoRect.mnSrcY && rTwoRect.mnSrcWidth == maTwoRect.mnSrcWidth && rTwoRect.mnSrcHeight == maTwoRect.mnSrcHeight && rTwoRect.mnDestWidth == maTwoRect.mnDestWidth && rTwoRect.mnDestHeight == maTwoRect.mnDestHeight ) { // absolutely indentically bRet = TRUE; } else if( rTwoRect.mnSrcWidth == rTwoRect.mnDestWidth && rTwoRect.mnSrcHeight == rTwoRect.mnDestHeight && maTwoRect.mnSrcWidth == maTwoRect.mnDestWidth && maTwoRect.mnSrcHeight == maTwoRect.mnDestHeight && rTwoRect.mnSrcX >= maTwoRect.mnSrcX && rTwoRect.mnSrcY >= maTwoRect.mnSrcY && ( rTwoRect.mnSrcX + rTwoRect.mnSrcWidth ) <= ( maTwoRect.mnSrcX + maTwoRect.mnSrcWidth ) && ( rTwoRect.mnSrcY + rTwoRect.mnSrcHeight ) <= ( maTwoRect.mnSrcY + maTwoRect.mnSrcHeight ) ) { bRet = TRUE; } } return bRet; } // ----------------------------------------------------------------------------- void ImplSalDDB::ImplDraw( #ifdef _USE_PRINT_EXTENSION_ SalDisplay* pDisplay, #endif Drawable aDrawable, long nDrawableDepth, const SalTwoRect& rTwoRect, const GC& rGC ) const { #ifdef _USE_PRINT_EXTENSION_ ImplDraw( pDisplay, maPixmap, mnDepth, aDrawable, nDrawableDepth, rTwoRect.mnSrcX - maTwoRect.mnSrcX, rTwoRect.mnSrcY - maTwoRect.mnSrcY, rTwoRect.mnDestWidth, rTwoRect.mnDestHeight, rTwoRect.mnDestX, rTwoRect.mnDestY, rGC ); #else ImplDraw( maPixmap, mnDepth, aDrawable, nDrawableDepth, rTwoRect.mnSrcX - maTwoRect.mnSrcX, rTwoRect.mnSrcY - maTwoRect.mnSrcY, rTwoRect.mnDestWidth, rTwoRect.mnDestHeight, rTwoRect.mnDestX, rTwoRect.mnDestY, rGC ); #endif } // ----------------------------------------------------------------------------- void ImplSalDDB::ImplDraw( #ifdef _USE_PRINT_EXTENSION_ SalDisplay* pDisplay, #endif Drawable aSrcDrawable, long nSrcDrawableDepth, Drawable aDstDrawable, long, long nSrcX, long nSrcY, long nDestWidth, long nDestHeight, long nDestX, long nDestY, const GC& rGC ) { #if !defined(_USE_PRINT_EXTENSION_) SalDisplay* pSalDisp = GetSalData()->GetDisplay(); Display* pXDisp = pSalDisp->GetDisplay(); #else Display* pXDisp = pDisplay->GetDisplay(); #endif if( 1 == nSrcDrawableDepth ) { XCopyPlane( pXDisp, aSrcDrawable, aDstDrawable, rGC, nSrcX, nSrcY, nDestWidth, nDestHeight, nDestX, nDestY, 1 ); } else { XCopyArea( pXDisp, aSrcDrawable, aDstDrawable, rGC, nSrcX, nSrcY, nDestWidth, nDestHeight, nDestX, nDestY ); } } // ---------------------- // - ImplSalBitmapCache - // ---------------------- struct ImplBmpObj { X11SalBitmap* mpBmp; ULONG mnMemSize; ULONG mnFlags; ImplBmpObj( X11SalBitmap* pBmp, ULONG nMemSize, ULONG nFlags ) : mpBmp( pBmp ), mnMemSize( nMemSize ), mnFlags( nFlags ) {} }; // ----------------------------------------------------------------------------- ImplSalBitmapCache::ImplSalBitmapCache() : mnTotalSize( 0UL ) { } // ----------------------------------------------------------------------------- ImplSalBitmapCache::~ImplSalBitmapCache() { ImplClear(); } // ----------------------------------------------------------------------------- void ImplSalBitmapCache::ImplAdd( X11SalBitmap* pBmp, ULONG nMemSize, ULONG nFlags ) { ImplBmpObj* pObj; bool bFound = FALSE; for( pObj = (ImplBmpObj*) maBmpList.Last(); pObj && !bFound; pObj = (ImplBmpObj*) maBmpList.Prev() ) if( pObj->mpBmp == pBmp ) bFound = TRUE; mnTotalSize += nMemSize; if( bFound ) { mnTotalSize -= pObj->mnMemSize; pObj->mnMemSize = nMemSize, pObj->mnFlags = nFlags; } else maBmpList.Insert( new ImplBmpObj( pBmp, nMemSize, nFlags ), LIST_APPEND ); } // ----------------------------------------------------------------------------- void ImplSalBitmapCache::ImplRemove( X11SalBitmap* pBmp ) { for( ImplBmpObj* pObj = (ImplBmpObj*) maBmpList.Last(); pObj; pObj = (ImplBmpObj*) maBmpList.Prev() ) { if( pObj->mpBmp == pBmp ) { maBmpList.Remove( pObj ); pObj->mpBmp->ImplRemovedFromCache(); mnTotalSize -= pObj->mnMemSize; delete pObj; break; } } } // ----------------------------------------------------------------------------- void ImplSalBitmapCache::ImplClear() { for( ImplBmpObj* pObj = (ImplBmpObj*) maBmpList.First(); pObj; pObj = (ImplBmpObj*) maBmpList.Next() ) { pObj->mpBmp->ImplRemovedFromCache(); delete pObj; } maBmpList.Clear(); mnTotalSize = 0; }