/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "codec.hxx" #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; BitmapEx GalleryResGetBitmapEx(const OUString &rId) { BitmapEx aBmpEx(rId); //TODO, check if any of these have no transparency layer if (!aBmpEx.IsTransparent()) aBmpEx = BitmapEx(aBmpEx.GetBitmap(), COL_LIGHTMAGENTA); return aBmpEx; } GalleryGraphicImportRet GalleryGraphicImport( const INetURLObject& rURL, Graphic& rGraphic, OUString& rFilterName, bool bShowProgress ) { GalleryGraphicImportRet nRet = GalleryGraphicImportRet::IMPORT_NONE; SfxMedium aMedium( rURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), StreamMode::READ ); aMedium.Download(); SvStream* pIStm = aMedium.GetInStream(); if( pIStm ) { GraphicFilter& rGraphicFilter = GraphicFilter::GetGraphicFilter(); std::unique_ptr pProgress(bShowProgress ? new GalleryProgress( &rGraphicFilter ) : nullptr); sal_uInt16 nFormat; if( !rGraphicFilter.ImportGraphic( rGraphic, rURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), *pIStm, GRFILTER_FORMAT_DONTKNOW, &nFormat ) ) { rFilterName = rGraphicFilter.GetImportFormatName( nFormat ); nRet = GalleryGraphicImportRet::IMPORT_FILE; } } return nRet; } bool GallerySvDrawImport( SvStream& rIStm, SdrModel& rModel ) { sal_uInt32 nVersion; bool bRet = false; if( GalleryCodec::IsCoded( rIStm, nVersion ) ) { SvMemoryStream aMemStm( 65535, 65535 ); GalleryCodec aCodec( rIStm ); aCodec.Read( aMemStm ); aMemStm.Seek( 0 ); if( 1 == nVersion ) { OSL_FAIL( "staroffice binary file formats are no longer supported inside the gallery!" ); bRet = false; } else if( 2 == nVersion ) { // recall to read as XML bRet = GallerySvDrawImport( aMemStm, rModel ); } } else { // read as XML uno::Reference< io::XInputStream > xInputStream( new utl::OInputStreamWrapper( rIStm ) ); rModel.GetItemPool().SetDefaultMetric( MapUnit::Map100thMM ); uno::Reference< lang::XComponent > xComponent; bRet = SvxDrawingLayerImport( &rModel, xInputStream, xComponent, "com.sun.star.comp.Draw.XMLOasisImporter" ); if( !bRet || (rModel.GetPageCount() == 0) ) { rIStm.Seek(0); bRet = SvxDrawingLayerImport( &rModel, xInputStream, xComponent, "com.sun.star.comp.Draw.XMLImporter" ); } } return bRet; } bool CreateIMapGraphic( const FmFormModel& rModel, Graphic& rGraphic, ImageMap& rImageMap ) { bool bRet = false; if ( rModel.GetPageCount() ) { const SdrPage* pPage = rModel.GetPage( 0 ); const SdrObject* pObj = pPage->GetObj( 0 ); if ( pPage->GetObjCount() == 1 && dynamic_cast( pObj) != nullptr ) { const sal_uInt16 nCount = pObj->GetUserDataCount(); // Exist in the user data an IMap information? for ( sal_uInt16 i = 0; i < nCount; i++ ) { const SdrObjUserData* pUserData = pObj->GetUserData( i ); if ( ( pUserData->GetInventor() == SdrInventor::SgaImap ) && ( pUserData->GetId() == ID_IMAPINFO ) ) { rGraphic = static_cast( pObj )->GetGraphic(); rImageMap = static_cast( pUserData )->GetImageMap(); bRet = true; break; } } } } return bRet; } OUString GetReducedString( const INetURLObject& rURL, sal_Int32 nMaxLen ) { OUString aReduced( rURL.GetMainURL( INetURLObject::DecodeMechanism::Unambiguous ) ); aReduced = aReduced.getToken( comphelper::string::getTokenCount(aReduced, '/') - 1, '/' ); if( INetProtocol::PrivSoffice != rURL.GetProtocol() ) { sal_Unicode aDelimiter; const OUString aPath( rURL.getFSysPath( FSysStyle::Detect, &aDelimiter ) ); const OUString aName( aReduced ); if( aPath.getLength() > nMaxLen ) { sal_Int32 nPathPrefixLen = nMaxLen - aName.getLength() - 4; if (nPathPrefixLen >= 0) { aReduced = aPath.copy(0, nPathPrefixLen) + "..." + OUStringLiteral1(aDelimiter) + aName; } else { aReduced += "..." + OUStringLiteral1(aDelimiter) + "..." + aName.copy( aName.getLength() - (nMaxLen - 7) ); } } else aReduced = aPath; } return aReduced; } OUString GetSvDrawStreamNameFromURL( const INetURLObject& rSvDrawObjURL ) { OUString aRet; if( rSvDrawObjURL.GetProtocol() == INetProtocol::PrivSoffice && comphelper::string::getTokenCount(rSvDrawObjURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), '/') == 3 ) { aRet = rSvDrawObjURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ).getToken( 2, '/' ); } return aRet; } bool FileExists( const INetURLObject& rURL ) { bool bRet = false; if( rURL.GetProtocol() != INetProtocol::NotValid ) { try { ::ucbhelper::Content aCnt( rURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), uno::Reference< ucb::XCommandEnvironment >(), comphelper::getProcessComponentContext() ); OUString aTitle; aCnt.getPropertyValue("Title") >>= aTitle; bRet = ( !aTitle.isEmpty() ); } catch( const ucb::ContentCreationException& ) { } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } } return bRet; } bool CreateDir( const INetURLObject& rURL ) { bool bRet = FileExists( rURL ); if( !bRet ) { try { uno::Reference< ucb::XCommandEnvironment > aCmdEnv; INetURLObject aParentURL( rURL ); aParentURL.removeSegment(); ::ucbhelper::Content aParent( aParentURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), aCmdEnv, comphelper::getProcessComponentContext() ); uno::Sequence< OUString > aProps( 1 ); uno::Sequence< uno::Any > aValues( 1 ); aProps[0] = "Title"; aValues[0] <<= rURL.GetName(); ::ucbhelper::Content aContent( rURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), aCmdEnv, comphelper::getProcessComponentContext() ); bRet = aParent.insertNewContent( "application/vnd.sun.staroffice.fsys-folder", aProps, aValues, aContent ); } catch( const ucb::ContentCreationException& ) { } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } } return bRet; } bool CopyFile( const INetURLObject& rSrcURL, const INetURLObject& rDstURL ) { bool bRet = false; try { ::ucbhelper::Content aDestPath( rDstURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), uno::Reference< ucb::XCommandEnvironment >(), comphelper::getProcessComponentContext() ); aDestPath.executeCommand( "transfer", uno::makeAny( ucb::TransferInfo( false, rSrcURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), rDstURL.GetName(), ucb::NameClash::OVERWRITE ) ) ); bRet = true; } catch( const ucb::ContentCreationException& ) { } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } return bRet; } bool KillFile( const INetURLObject& rURL ) { bool bRet = FileExists( rURL ); if( bRet ) { try { ::ucbhelper::Content aCnt( rURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ), uno::Reference< ucb::XCommandEnvironment >(), comphelper::getProcessComponentContext() ); aCnt.executeCommand( "delete", uno::makeAny( true ) ); } catch( const ucb::ContentCreationException& ) { bRet = false; } catch( const uno::RuntimeException& ) { bRet = false; } catch( const uno::Exception& ) { bRet = false; } } return bRet; } GalleryProgress::GalleryProgress( GraphicFilter* pFilter ) : mpFilter( pFilter ) { uno::Reference< lang::XMultiServiceFactory > xMgr( ::comphelper::getProcessServiceFactory() ); uno::Reference< awt::XProgressMonitor > xMonitor( xMgr->createInstance( "com.sun.star.awt.XProgressMonitor" ), uno::UNO_QUERY ); if ( xMonitor.is() ) { mxProgressBar.set( xMonitor, uno::UNO_QUERY ); if( mxProgressBar.is() ) { OUString aProgressText; if( mpFilter ) { aProgressText = SvxResId(RID_SVXSTR_GALLERY_FILTER); // mpFilter->SetUpdatePercentHdl( LINK( this, GalleryProgress, Update ) ); // sj: progress wasn't working up from SO7 at all // // so I am removing this. The gallery progress should // // be changed to use the XStatusIndicator instead of XProgressMonitor } else aProgressText = "Gallery"; xMonitor->addText( "Gallery", aProgressText, false ) ; mxProgressBar->setRange( 0, GALLERY_PROGRESS_RANGE ); } } } GalleryProgress::~GalleryProgress() { } void GalleryProgress::Update( sal_uIntPtr nVal, sal_uIntPtr nMaxVal ) { if( mxProgressBar.is() && nMaxVal ) mxProgressBar->setValue( std::min( (sal_uIntPtr)( (double) nVal / nMaxVal * GALLERY_PROGRESS_RANGE ), (sal_uIntPtr) GALLERY_PROGRESS_RANGE ) ); } GalleryTransferable::GalleryTransferable( GalleryTheme* pTheme, sal_uIntPtr nObjectPos, bool bLazy ) : mpTheme( pTheme ), meObjectKind( mpTheme->GetObjectKind( nObjectPos ) ), mnObjectPos( nObjectPos ), mpGraphicObject( nullptr ), mpImageMap( nullptr ), mpURL( nullptr ) { InitData( bLazy ); } GalleryTransferable::~GalleryTransferable() { } void GalleryTransferable::InitData( bool bLazy ) { switch( meObjectKind ) { case SgaObjKind::SvDraw: { if( !bLazy ) { if( !mpGraphicObject ) { Graphic aGraphic; if( mpTheme->GetGraphic( mnObjectPos, aGraphic ) ) mpGraphicObject = new GraphicObject( aGraphic ); } if( !mxModelStream.is() ) { mxModelStream = new SotStorageStream( "" ); mxModelStream->SetBufferSize( 16348 ); if( !mpTheme->GetModelStream( mnObjectPos, mxModelStream ) ) mxModelStream.clear(); else mxModelStream->Seek( 0 ); } } } break; case SgaObjKind::Animation: case SgaObjKind::Bitmap: case SgaObjKind::Inet: case SgaObjKind::Sound: { if( !mpURL ) { mpURL = new INetURLObject; if( !mpTheme->GetURL( mnObjectPos, *mpURL ) ) { delete mpURL; mpURL = nullptr; } } if( ( SgaObjKind::Sound != meObjectKind ) && !mpGraphicObject ) { Graphic aGraphic; if( mpTheme->GetGraphic( mnObjectPos, aGraphic ) ) mpGraphicObject = new GraphicObject( aGraphic ); } } break; default: OSL_FAIL( "GalleryTransferable::GalleryTransferable: invalid object type" ); break; } } void GalleryTransferable::AddSupportedFormats() { if( SgaObjKind::SvDraw == meObjectKind ) { AddFormat( SotClipboardFormatId::DRAWING ); AddFormat( SotClipboardFormatId::SVXB ); AddFormat( SotClipboardFormatId::GDIMETAFILE ); AddFormat( SotClipboardFormatId::BITMAP ); } else { if( mpURL ) AddFormat( SotClipboardFormatId::SIMPLE_FILE ); if( mpGraphicObject ) { AddFormat( SotClipboardFormatId::SVXB ); if( mpGraphicObject->GetType() == GraphicType::GdiMetafile ) { AddFormat( SotClipboardFormatId::GDIMETAFILE ); AddFormat( SotClipboardFormatId::BITMAP ); } else { AddFormat( SotClipboardFormatId::BITMAP ); AddFormat( SotClipboardFormatId::GDIMETAFILE ); } } } } bool GalleryTransferable::GetData( const datatransfer::DataFlavor& rFlavor, const OUString& /*rDestDoc*/ ) { SotClipboardFormatId nFormat = SotExchange::GetFormat( rFlavor ); bool bRet = false; InitData( false ); if( ( SotClipboardFormatId::DRAWING == nFormat ) && ( SgaObjKind::SvDraw == meObjectKind ) ) { bRet = ( mxModelStream.is() && SetObject( mxModelStream.get(), SotClipboardFormatId::NONE, rFlavor ) ); } else if( ( SotClipboardFormatId::SVIM == nFormat ) && mpImageMap ) { // TODO/MBA: do we need a BaseURL here?! bRet = SetImageMap( *mpImageMap ); } else if( ( SotClipboardFormatId::SIMPLE_FILE == nFormat ) && mpURL ) { bRet = SetString( mpURL->GetMainURL( INetURLObject::DecodeMechanism::NONE ), rFlavor ); } else if( ( SotClipboardFormatId::SVXB == nFormat ) && mpGraphicObject ) { bRet = SetGraphic( mpGraphicObject->GetGraphic() ); } else if( ( SotClipboardFormatId::GDIMETAFILE == nFormat ) && mpGraphicObject ) { bRet = SetGDIMetaFile( mpGraphicObject->GetGraphic().GetGDIMetaFile() ); } else if( ( SotClipboardFormatId::BITMAP == nFormat ) && mpGraphicObject ) { bRet = SetBitmapEx( mpGraphicObject->GetGraphic().GetBitmapEx(), rFlavor ); } return bRet; } bool GalleryTransferable::WriteObject( tools::SvRef& rxOStm, void* pUserObject, SotClipboardFormatId, const datatransfer::DataFlavor& ) { bool bRet = false; if( pUserObject ) { rxOStm->WriteStream( *static_cast< SotStorageStream* >( pUserObject ) ); bRet = ( rxOStm->GetError() == ERRCODE_NONE ); } return bRet; } void GalleryTransferable::DragFinished( sal_Int8 nDropAction ) { mpTheme->SetDragging( false ); mpTheme->SetDragPos( 0 ); if ( nDropAction ) { vcl::Window *pFocusWindow = Application::GetFocusWindow(); if ( pFocusWindow ) pFocusWindow->GrabFocusToDocument(); } } void GalleryTransferable::ObjectReleased() { mxModelStream.clear(); delete mpGraphicObject; mpGraphicObject = nullptr; delete mpImageMap; mpImageMap = nullptr; delete mpURL; mpURL = nullptr; } void GalleryTransferable::StartDrag( vcl::Window* pWindow, sal_Int8 nDragSourceActions ) { INetURLObject aURL; if( mpTheme->GetURL( mnObjectPos, aURL ) && ( aURL.GetProtocol() != INetProtocol::NotValid ) ) { mpTheme->SetDragging( true ); mpTheme->SetDragPos( mnObjectPos ); TransferableHelper::StartDrag( pWindow, nDragSourceActions ); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */