/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: imgprod.cxx,v $ * * $Revision: 1.7 $ * * last change: $Author: obo $ $Date: 2006-10-12 11:13:26 $ * * 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 * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_forms.hxx" #include "imgprod.hxx" #include #include #include #include #include #ifndef _COM_SUN_STAR_IO_XINPUTSTREAM_HPP_ #include #endif #ifndef SVTOOLS_SOURCE_MISC_IMAGERESOURCEACCESS_HXX #include "svtools/imageresourceaccess.hxx" #endif #ifndef _COMPHELPER_PROCESSFACTORY_HXX_ #include #endif // -------------------- // - ImgProdLockBytes - // -------------------- class ImgProdLockBytes : public SvLockBytes { ::com::sun::star::uno::Reference< ::com::sun::star::io::XInputStream > xStmRef; ::com::sun::star::uno::Sequence maSeq; ImgProdLockBytes() {}; public: ImgProdLockBytes( SvStream* pStm, sal_Bool bOwner ); ImgProdLockBytes( ::com::sun::star::uno::Reference< ::com::sun::star::io::XInputStream > & rStreamRef ); virtual ~ImgProdLockBytes(); virtual ErrCode ReadAt( sal_uInt32 nPos, void* pBuffer, sal_uInt32 nCount, sal_Size* pRead ) const; virtual ErrCode WriteAt( sal_uInt32 nPos, const void* pBuffer, sal_uInt32 nCount, sal_Size* pWritten ); virtual ErrCode Flush() const; virtual ErrCode SetSize( sal_uInt32 nSize ); virtual ErrCode Stat( SvLockBytesStat*, SvLockBytesStatFlag ) const; }; // ------------------------------------------------------------------------ ImgProdLockBytes::ImgProdLockBytes( SvStream* pStm, sal_Bool bOwner ) : SvLockBytes( pStm, bOwner ) { } // ------------------------------------------------------------------------ ImgProdLockBytes::ImgProdLockBytes( ::com::sun::star::uno::Reference< ::com::sun::star::io::XInputStream > & rStmRef ) : xStmRef( rStmRef ) { if( xStmRef.is() ) { const sal_uInt32 nBytesToRead = 65535; sal_uInt32 nRead; do { ::com::sun::star::uno::Sequence< sal_Int8 > aReadSeq; nRead = xStmRef->readSomeBytes( aReadSeq, nBytesToRead ); if( nRead ) { const sal_uInt32 nOldLength = maSeq.getLength(); maSeq.realloc( nOldLength + nRead ); rtl_copyMemory( maSeq.getArray() + nOldLength, aReadSeq.getConstArray(), aReadSeq.getLength() ); } } while( nBytesToRead == nRead ); } } // ------------------------------------------------------------------------ ImgProdLockBytes::~ImgProdLockBytes() { } // ------------------------------------------------------------------------ ErrCode ImgProdLockBytes::ReadAt( sal_uInt32 nPos, void* pBuffer, sal_uInt32 nCount, sal_Size* pRead ) const { if( GetStream() ) { ( (SvStream*) GetStream() )->ResetError(); const ErrCode nErr = SvLockBytes::ReadAt( nPos, pBuffer, nCount, pRead ); ( (SvStream*) GetStream() )->ResetError(); return nErr; } else { const sal_uInt32 nSeqLen = maSeq.getLength(); ErrCode nErr = ERRCODE_NONE; if( nPos < nSeqLen ) { if( ( nPos + nCount ) > nSeqLen ) nCount = nSeqLen - nPos; memcpy( pBuffer, maSeq.getConstArray() + nPos, nCount ); *pRead = nCount; } else *pRead = 0UL; return nErr; } } // ------------------------------------------------------------------------ ErrCode ImgProdLockBytes::WriteAt( sal_uInt32 nPos, const void* pBuffer, sal_uInt32 nCount, sal_Size* pWritten ) { if( GetStream() ) return SvLockBytes::WriteAt( nPos, pBuffer, nCount, pWritten ); else { DBG_ASSERT( xStmRef.is(), "ImgProdLockBytes::WriteAt: xInputStream has no reference..." ); return ERRCODE_IO_CANTWRITE; } } // ------------------------------------------------------------------------ ErrCode ImgProdLockBytes::Flush() const { return ERRCODE_NONE; } // ------------------------------------------------------------------------ ErrCode ImgProdLockBytes::SetSize( sal_uInt32 nSize ) { if( GetStream() ) return SvLockBytes::SetSize( nSize ); else { DBG_ERROR( "ImgProdLockBytes::SetSize not supported for xInputStream..." ); return ERRCODE_IO_CANTWRITE; } } // ------------------------------------------------------------------------ ErrCode ImgProdLockBytes::Stat( SvLockBytesStat* pStat, SvLockBytesStatFlag eFlag ) const { if( GetStream() ) return SvLockBytes::Stat( pStat, eFlag ); else { DBG_ASSERT( xStmRef.is(), "ImgProdLockBytes::Stat: xInputStream has no reference..." ); pStat->nSize = maSeq.getLength(); return ERRCODE_NONE; } } // ----------------- // - ImageProducer - // ----------------- ImageProducer::ImageProducer() : mpStm ( NULL ), mpFilter ( NULL ), mnStatus ( 0UL ), mbConsInit ( sal_False ), mnLastError ( 0UL ), mbAsync ( sal_False ) { mpGraphic = new Graphic; DBG_ASSERT( Application::GetFilterHdl().IsSet(), "ImageProducer::ImageProducer(): No filter handler set" ); } // ------------------------------------------------------------ ImageProducer::~ImageProducer() { delete mpGraphic; mpGraphic = NULL; delete mpFilter; mpFilter = NULL; delete mpStm; mpStm = NULL; for( void* pCons = maConsList.First(); pCons; pCons = maConsList.Next() ) delete (::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons; } // ------------------------------------------------------------ // ::com::sun::star::uno::XInterface ::com::sun::star::uno::Any ImageProducer::queryInterface( const ::com::sun::star::uno::Type & rType ) throw(::com::sun::star::uno::RuntimeException) { ::com::sun::star::uno::Any aRet = ::cppu::queryInterface( rType, SAL_STATIC_CAST( ::com::sun::star::lang::XInitialization*, this ), SAL_STATIC_CAST( ::com::sun::star::awt::XImageProducer*, this ) ); return (aRet.hasValue() ? aRet : OWeakObject::queryInterface( rType )); } // ------------------------------------------------------------ void ImageProducer::addConsumer( const ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer >& rxConsumer ) throw(::com::sun::star::uno::RuntimeException) { DBG_ASSERT( rxConsumer.is(), "::AddConsumer(...): No consumer referenced!" ); if( rxConsumer.is() ) maConsList.Insert( new ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > ( rxConsumer ), LIST_APPEND ); } // ------------------------------------------------------------ void ImageProducer::removeConsumer( const ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer >& rxConsumer ) throw(::com::sun::star::uno::RuntimeException) { for( sal_uInt32 n = maConsList.Count(); n; ) { ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > * pRef = (::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) maConsList.GetObject( --n ); if( *pRef == rxConsumer ) { delete pRef; maConsList.Remove( n ); break; } } } // ------------------------------------------------------------ void ImageProducer::SetImage( const ::rtl::OUString& rPath ) { maURL = rPath; mpGraphic->Clear(); mbConsInit = sal_False; mbAsync = sal_False; delete mpStm; if ( ::svt::ImageResourceAccess::isImageResourceURL( maURL ) ) { mpStm = ::svt::ImageResourceAccess::getImageStream( ::comphelper::getProcessServiceFactory(), maURL ); } else if( maURL.getLength() ) { SvStream* pIStm = ::utl::UcbStreamHelper::CreateStream( maURL, STREAM_STD_READ ); mpStm = pIStm ? new SvStream( new ImgProdLockBytes( pIStm, sal_True ) ) : NULL; } else mpStm = NULL; } // ------------------------------------------------------------ void ImageProducer::SetImage( SvStream& rStm ) { maURL = ::rtl::OUString(); mpGraphic->Clear(); mbConsInit = sal_False; mbAsync = sal_False; delete mpStm; mpStm = new SvStream( new ImgProdLockBytes( &rStm, sal_False ) ); } // ------------------------------------------------------------ void ImageProducer::SetImage( const Graphic& rGraphic ) { maURL = ::rtl::OUString(); *mpGraphic = rGraphic; mbConsInit = sal_False; mbAsync = sal_False; delete mpStm; mpStm = NULL; } // ------------------------------------------------------------ void ImageProducer::setImage( ::com::sun::star::uno::Reference< ::com::sun::star::io::XInputStream > & rInputStmRef ) { maURL = ::rtl::OUString(); mpGraphic->Clear(); mbConsInit = sal_False; mbAsync = sal_False; delete mpStm; if( rInputStmRef.is() ) mpStm = new SvStream( new ImgProdLockBytes( rInputStmRef ) ); else mpStm = NULL; } // ------------------------------------------------------------ void ImageProducer::NewDataAvailable() { if( ( GRAPHIC_NONE == mpGraphic->GetType() ) || mpGraphic->GetContext() ) startProduction(); } // ------------------------------------------------------------ void ImageProducer::startProduction() throw(::com::sun::star::uno::RuntimeException) { ResetLastError(); if( maConsList.Count() ) { bool bNotifyEmptyGraphics = false; // valid stream or filled graphic? => update consumers if( mpStm || ( mpGraphic->GetType() != GRAPHIC_NONE ) ) { // if we already have a graphic, we don't have to import again; // graphic is cleared if a new Stream is set if( ( mpGraphic->GetType() == GRAPHIC_NONE ) || mpGraphic->GetContext() ) { if( !ImplImportGraphic( *mpGraphic ) && maErrorHdl.IsSet() ) maErrorHdl.Call( this ); } if( mpGraphic->GetType() != GRAPHIC_NONE ) ImplUpdateData( *mpGraphic ); else bNotifyEmptyGraphics = true; } else bNotifyEmptyGraphics = true; if ( bNotifyEmptyGraphics ) { // reset image List aTmp; void* pCons; // create temporary list to hold interfaces for( pCons = maConsList.First(); pCons; pCons = maConsList.Next() ) aTmp.Insert( new ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons ), LIST_APPEND ); // iterate through interfaces for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) { ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->init( 0, 0 ); ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->complete( ::com::sun::star::awt::ImageStatus::IMAGESTATUS_STATICIMAGEDONE, this ); } // delete interfaces in temporary list for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) delete (::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons; } } } // ------------------------------------------------------------ sal_Bool ImageProducer::ImplImportGraphic( Graphic& rGraphic ) { USHORT nFilter = GRFILTER_FORMAT_DONTKNOW; short nRet; sal_Bool bRet = sal_False; if( ERRCODE_IO_PENDING == mpStm->GetError() ) mpStm->ResetError(); mpStm->Seek( 0UL ); if( mpFilter ) nRet = mpFilter->ImportGraphic( rGraphic, String(), *mpStm, nFilter ); else { if( GraphicConverter::Import( *mpStm, rGraphic ) == ERRCODE_NONE ) nRet = GRFILTER_OK; else nRet = GRFILTER_FILTERERROR; } if( ERRCODE_IO_PENDING == mpStm->GetError() ) mpStm->ResetError(); if( nRet == GRFILTER_OK ) bRet = sal_True; else mnLastError = nRet; return bRet; } // ------------------------------------------------------------ void ImageProducer::ImplUpdateData( const Graphic& rGraphic ) { // asynchronous? if( mpGraphic->GetContext() ) mbAsync = sal_True; ImplInitConsumer( rGraphic ); if( mbConsInit && maConsList.Count() ) { List aTmp; void* pCons; ImplUpdateConsumer( rGraphic ); mbConsInit = sal_False; // create temporary list to hold interfaces for( pCons = maConsList.First(); pCons; pCons = maConsList.Next() ) aTmp.Insert( new ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons ), LIST_APPEND ); // iterate through interfaces for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->complete( mnStatus = ::com::sun::star::awt::ImageStatus::IMAGESTATUS_STATICIMAGEDONE, this ); // delete interfaces in temporary list for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) delete (::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons; } } // ------------------------------------------------------------ void ImageProducer::ImplInitConsumer( const Graphic& rGraphic ) { Bitmap aBmp( rGraphic.GetBitmapEx().GetBitmap() ); BitmapReadAccess* pBmpAcc = aBmp.AcquireReadAccess(); if( pBmpAcc ) { List aTmp; void * pCons; sal_uInt16 nPalCount = 0; sal_uInt32 nRMask = 0; sal_uInt32 nGMask = 0; sal_uInt32 nBMask = 0; sal_uInt32 nAMask = 0; ::com::sun::star::uno::Sequence< sal_Int32 > aRGBPal; if( pBmpAcc->HasPalette() ) { nPalCount = pBmpAcc->GetPaletteEntryCount(); if( nPalCount ) { aRGBPal = ::com::sun::star::uno::Sequence< sal_Int32 >( nPalCount + 1 ); sal_Int32* pTmp = aRGBPal.getArray(); for( sal_uInt32 i = 0; i < nPalCount; i++, pTmp++ ) { const BitmapColor& rCol = pBmpAcc->GetPaletteColor( (sal_uInt16) i ); *pTmp = ( (sal_Int32) rCol.GetRed() ) << 24L; *pTmp |= ( (sal_Int32) rCol.GetGreen() ) << 16L; *pTmp |= ( (sal_Int32) rCol.GetBlue() ) << 8L; *pTmp |= 0x000000ffL; } if( rGraphic.IsTransparent() ) { // append transparent entry *pTmp = 0xffffff00L; mnTransIndex = nPalCount; nPalCount++; } else mnTransIndex = 0; } } else { nRMask = 0xff000000UL; nGMask = 0x00ff0000UL; nBMask = 0x0000ff00UL; nAMask = 0x000000ffUL; } // create temporary list to hold interfaces for( pCons = maConsList.First(); pCons; pCons = maConsList.Next() ) aTmp.Insert( new ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons ), LIST_APPEND ); // iterate through interfaces for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) { ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->init( pBmpAcc->Width(), pBmpAcc->Height() ); ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->setColorModel( pBmpAcc->GetBitCount(), aRGBPal, nRMask, nGMask, nBMask, nAMask ); } // delete interfaces in temporary list for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) delete (::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons; aBmp.ReleaseAccess( pBmpAcc ); mbConsInit = sal_True; } } // ------------------------------------------------------------ void ImageProducer::ImplUpdateConsumer( const Graphic& rGraphic ) { BitmapEx aBmpEx( rGraphic.GetBitmapEx() ); Bitmap aBmp( aBmpEx.GetBitmap() ); BitmapReadAccess* pBmpAcc = aBmp.AcquireReadAccess(); if( pBmpAcc ) { List aTmp; void* pCons; Bitmap aMask( aBmpEx.GetMask() ); BitmapReadAccess* pMskAcc = !!aMask ? aMask.AcquireReadAccess() : NULL; const long nWidth = pBmpAcc->Width(); const long nHeight = pBmpAcc->Height(); const long nStartX = 0L; const long nEndX = nWidth - 1L; const long nStartY = 0L; const long nEndY = nHeight - 1L; const long nPartWidth = nEndX - nStartX + 1; const long nPartHeight = nEndY - nStartY + 1; if( !pMskAcc ) { aMask = Bitmap( aBmp.GetSizePixel(), 1 ); aMask.Erase( COL_BLACK ); pMskAcc = aMask.AcquireReadAccess(); } // create temporary list to hold interfaces for( pCons = maConsList.First(); pCons; pCons = maConsList.Next() ) aTmp.Insert( new ::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons ), LIST_APPEND ); if( pBmpAcc->HasPalette() ) { const BitmapColor aWhite( pMskAcc->GetBestMatchingColor( Color( COL_WHITE ) ) ); if( mnTransIndex < 256 ) { ::com::sun::star::uno::Sequence aData( nPartWidth * nPartHeight ); sal_Int8* pTmp = aData.getArray(); for( long nY = nStartY; nY <= nEndY; nY++ ) { for( long nX = nStartX; nX <= nEndX; nX++ ) { if( pMskAcc->GetPixel( nY, nX ) == aWhite ) *pTmp++ = sal::static_int_cast< sal_Int8 >( mnTransIndex ); else *pTmp++ = pBmpAcc->GetPixel( nY, nX ).GetIndex(); } } // iterate through interfaces for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->setPixelsByBytes( nStartX, nStartY, nPartWidth, nPartHeight, aData, 0UL, nPartWidth ); } else { ::com::sun::star::uno::Sequence aData( nPartWidth * nPartHeight ); sal_Int32* pTmp = aData.getArray(); for( long nY = nStartY; nY <= nEndY; nY++ ) { for( long nX = nStartX; nX <= nEndX; nX++ ) { if( pMskAcc->GetPixel( nY, nX ) == aWhite ) *pTmp++ = mnTransIndex; else *pTmp++ = pBmpAcc->GetPixel( nY, nX ).GetIndex(); } } // iterate through interfaces for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->setPixelsByLongs( nStartX, nStartY, nPartWidth, nPartHeight, aData, 0UL, nPartWidth ); } } else { ::com::sun::star::uno::Sequence aData( nPartWidth * nPartHeight ); const BitmapColor aWhite( pMskAcc->GetBestMatchingColor( Color( COL_WHITE ) ) ); sal_Int32* pTmp = aData.getArray(); for( long nY = nStartY; nY <= nEndY; nY++ ) { for( long nX = nStartX; nX <= nEndX; nX++, pTmp++ ) { const BitmapColor aCol( pBmpAcc->GetPixel( nY, nX ) ); *pTmp = ( (sal_Int32) aCol.GetRed() ) << 24L; *pTmp |= ( (sal_Int32) aCol.GetGreen() ) << 16L; *pTmp |= ( (sal_Int32) aCol.GetBlue() ) << 8L; if( pMskAcc->GetPixel( nY, nX ) != aWhite ) *pTmp |= 0x000000ffUL; } } // iterate through interfaces for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) ( *(::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons )->setPixelsByLongs( nStartX, nStartY, nPartWidth, nPartHeight, aData, 0UL, nPartWidth ); } // delete interfaces in temporary list for( pCons = aTmp.First(); pCons; pCons = aTmp.Next() ) delete (::com::sun::star::uno::Reference< ::com::sun::star::awt::XImageConsumer > *) pCons; aBmp.ReleaseAccess( pBmpAcc ); aMask.ReleaseAccess( pMskAcc ); } } void ImageProducer::initialize( const ::com::sun::star::uno::Sequence< ::com::sun::star::uno::Any >& aArguments ) throw (::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException) { if ( aArguments.getLength() == 1 ) { ::com::sun::star::uno::Any aArg = aArguments.getConstArray()[0]; rtl::OUString aURL; if ( aArg >>= aURL ) { SetImage( aURL ); } } } namespace frm { ::com::sun::star::uno::Reference< ::com::sun::star::uno::XInterface > SAL_CALL ImageProducer_CreateInstance( const ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory >& ) { return ::com::sun::star::uno::Reference < ::com::sun::star::uno::XInterface >( ( ::cppu::OWeakObject* ) new ImageProducer ); } } // namespace frm