/************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2008 by Sun Microsystems, Inc. * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: scanunx.cxx,v $ * $Revision: 1.10 $ * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org 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 version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_extensions.hxx" #include #include #include #include #if OSL_DEBUG_LEVEL > 1 #include #endif BitmapTransporter::BitmapTransporter() { #if OSL_DEBUG_LEVEL > 1 fprintf( stderr, "BitmapTransporter\n" ); #endif } BitmapTransporter::~BitmapTransporter() { #if OSL_DEBUG_LEVEL > 1 fprintf( stderr, "~BitmapTransporter\n" ); #endif } // ----------------------------------------------------------------------------- ANY SAL_CALL BitmapTransporter::queryInterface( const Type& rType ) throw( RuntimeException ) { const ANY aRet( cppu::queryInterface( rType, static_cast< AWT::XBitmap* >( this ) ) ); return( aRet.hasValue() ? aRet : OWeakObject::queryInterface( rType ) ); } // ----------------------------------------------------------------------------- AWT::Size BitmapTransporter::getSize() throw() { vos::OGuard aGuard( m_aProtector ); int nPreviousPos = m_aStream.Tell(); AWT::Size aRet; // ensure that there is at least a header m_aStream.Seek( STREAM_SEEK_TO_END ); int nLen = m_aStream.Tell(); if( nLen > 15 ) { m_aStream.Seek( 4 ); m_aStream >> aRet.Width >> aRet.Height; } else aRet.Width = aRet.Height = 0; m_aStream.Seek( nPreviousPos ); return aRet; } // ----------------------------------------------------------------------------- SEQ( sal_Int8 ) BitmapTransporter::getDIB() throw() { vos::OGuard aGuard( m_aProtector ); int nPreviousPos = m_aStream.Tell(); // create return value m_aStream.Seek( STREAM_SEEK_TO_END ); int nBytes = m_aStream.Tell(); m_aStream.Seek( 0 ); SEQ( sal_Int8 ) aValue( nBytes ); m_aStream.Read( aValue.getArray(), nBytes ); m_aStream.Seek( nPreviousPos ); return aValue; } // -------------- // - SaneHolder - // -------------- struct SaneHolder { Sane m_aSane; REF( AWT::XBitmap ) m_xBitmap; vos::OMutex m_aProtector; ScanError m_nError; bool m_bBusy; }; DECLARE_LIST( SaneHolderList, SaneHolder* ) static SaneHolderList allSanes; static vos::OMutex aSaneProtector; // ----------------- // - ScannerThread - // ----------------- class ScannerThread : public vos::OThread { SaneHolder* m_pHolder; REF( com::sun::star::lang::XEventListener ) m_xListener; ScannerManager* m_pManager; // just for the disposing call public: virtual void run(); virtual void onTerminated() { delete this; } public: ScannerThread( SaneHolder* pHolder, const REF( com::sun::star::lang::XEventListener )& listener, ScannerManager* pManager ); virtual ~ScannerThread(); }; // ----------------------------------------------------------------------------- ScannerThread::ScannerThread( SaneHolder* pHolder, const REF( com::sun::star::lang::XEventListener )& listener, ScannerManager* pManager ) : m_pHolder( pHolder ), m_xListener( listener ), m_pManager( pManager ) { #if OSL_DEBUG_LEVEL > 1 fprintf( stderr, "ScannerThread\n" ); #endif } ScannerThread::~ScannerThread() { #if OSL_DEBUG_LEVEL > 1 fprintf( stderr, "~ScannerThread\n" ); #endif } void ScannerThread::run() { vos::OGuard aGuard( m_pHolder->m_aProtector ); BitmapTransporter* pTransporter = new BitmapTransporter; REF( XInterface ) aIf( static_cast< OWeakObject* >( pTransporter ) ); m_pHolder->m_xBitmap = REF( AWT::XBitmap )( aIf, UNO_QUERY ); m_pHolder->m_bBusy = true; if( m_pHolder->m_aSane.IsOpen() ) { int nOption = m_pHolder->m_aSane.GetOptionByName( "preview" ); if( nOption != -1 ) m_pHolder->m_aSane.SetOptionValue( nOption, (BOOL)FALSE ); m_pHolder->m_nError = m_pHolder->m_aSane.Start( *pTransporter ) ? ScanError_ScanErrorNone : ScanError_ScanCanceled; } else m_pHolder->m_nError = ScanError_ScannerNotAvailable; REF( XInterface ) xXInterface( static_cast< OWeakObject* >( m_pManager ) ); m_xListener->disposing( com::sun::star::lang::EventObject(xXInterface) ); m_pHolder->m_bBusy = false; } // ------------------ // - ScannerManager - // ------------------ void ScannerManager::DestroyData() { // was unused, now because of i99835: "Scanning interface not SANE API compliant" // delete all SaneHolder to get Sane Dtor called int i; for ( i = allSanes.Count(); i > 0; i-- ) { SaneHolder *pSaneHolder = allSanes.GetObject(i-1); if ( pSaneHolder ) delete pSaneHolder; } } // ----------------------------------------------------------------------------- AWT::Size ScannerManager::getSize() throw() { AWT::Size aRet; aRet.Width = aRet.Height = 0; return aRet; } // ----------------------------------------------------------------------------- SEQ( sal_Int8 ) ScannerManager::getDIB() throw() { return SEQ( sal_Int8 )(); } // ----------------------------------------------------------------------------- SEQ( ScannerContext ) ScannerManager::getAvailableScanners() throw() { vos::OGuard aGuard( aSaneProtector ); if( ! allSanes.Count() ) { SaneHolder* pSaneHolder = new SaneHolder; pSaneHolder->m_nError = ScanError_ScanErrorNone; pSaneHolder->m_bBusy = false; if( Sane::IsSane() ) allSanes.Insert( pSaneHolder ); else delete pSaneHolder; } if( Sane::IsSane() ) { SEQ( ScannerContext ) aRet(1); aRet.getArray()[0].ScannerName = ::rtl::OUString::createFromAscii( "SANE" ); aRet.getArray()[0].InternalData = 0; return aRet; } return SEQ( ScannerContext )(); } // ----------------------------------------------------------------------------- BOOL ScannerManager::configureScanner( ScannerContext& scanner_context ) throw( ScannerException ) { vos::OGuard aGuard( aSaneProtector ); #if OSL_DEBUG_LEVEL > 1 fprintf( stderr, "ScannerManager::configureScanner\n" ); #endif if( scanner_context.InternalData < 0 || (ULONG)scanner_context.InternalData >= allSanes.Count() ) throw ScannerException( ::rtl::OUString::createFromAscii( "Scanner does not exist" ), REF( XScannerManager )( this ), ScanError_InvalidContext ); SaneHolder* pHolder = allSanes.GetObject( scanner_context.InternalData ); if( pHolder->m_bBusy ) throw ScannerException( ::rtl::OUString::createFromAscii( "Scanner is busy" ), REF( XScannerManager )( this ), ScanError_ScanInProgress ); pHolder->m_bBusy = true; SaneDlg aDlg( NULL, pHolder->m_aSane ); BOOL bRet = (BOOL)aDlg.Execute(); pHolder->m_bBusy = false; return bRet; } // ----------------------------------------------------------------------------- void ScannerManager::startScan( const ScannerContext& scanner_context, const REF( com::sun::star::lang::XEventListener )& listener ) throw( ScannerException ) { vos::OGuard aGuard( aSaneProtector ); #if OSL_DEBUG_LEVEL > 1 fprintf( stderr, "ScannerManager::startScan\n" ); #endif if( scanner_context.InternalData < 0 || (ULONG)scanner_context.InternalData >= allSanes.Count() ) throw ScannerException( ::rtl::OUString::createFromAscii( "Scanner does not exist" ), REF( XScannerManager )( this ), ScanError_InvalidContext ); SaneHolder* pHolder = allSanes.GetObject( scanner_context.InternalData ); if( pHolder->m_bBusy ) throw ScannerException( ::rtl::OUString::createFromAscii( "Scanner is busy" ), REF( XScannerManager )( this ), ScanError_ScanInProgress ); pHolder->m_bBusy = true; ScannerThread* pThread = new ScannerThread( pHolder, listener, this ); pThread->create(); } // ----------------------------------------------------------------------------- ScanError ScannerManager::getError( const ScannerContext& scanner_context ) throw( ScannerException ) { vos::OGuard aGuard( aSaneProtector ); if( scanner_context.InternalData < 0 || (ULONG)scanner_context.InternalData >= allSanes.Count() ) throw ScannerException( ::rtl::OUString::createFromAscii( "Scanner does not exist" ), REF( XScannerManager )( this ), ScanError_InvalidContext ); SaneHolder* pHolder = allSanes.GetObject( scanner_context.InternalData ); return pHolder->m_nError; } // ----------------------------------------------------------------------------- REF( AWT::XBitmap ) ScannerManager::getBitmap( const ScannerContext& scanner_context ) throw( ScannerException ) { vos::OGuard aGuard( aSaneProtector ); if( scanner_context.InternalData < 0 || (ULONG)scanner_context.InternalData >= allSanes.Count() ) throw ScannerException( ::rtl::OUString::createFromAscii( "Scanner does not exist" ), REF( XScannerManager )( this ), ScanError_InvalidContext ); SaneHolder* pHolder = allSanes.GetObject( scanner_context.InternalData ); vos::OGuard aProtGuard( pHolder->m_aProtector ); REF( AWT::XBitmap ) xRet( pHolder->m_xBitmap ); pHolder->m_xBitmap = REF( AWT::XBitmap )(); return xRet; }