/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::io; using namespace ::com::sun::star::datatransfer; namespace sd { #define CHECK_FORMAT_TRANS( _def_Type ) ( ( nFormat == (_def_Type) || nFormat == SotClipboardFormatId::NONE ) && aDataHelper.HasFormat( _def_Type ) ) /************************************************************************* |* |* Paste |* \************************************************************************/ namespace { struct ImpRememberOrigAndClone { SdrObject* pOrig; SdrObject* pClone; }; } static SdrObject* ImpGetClone(std::vector& aConnectorContainer, SdrObject const * pConnObj) { for(const ImpRememberOrigAndClone& rImp : aConnectorContainer) { if(pConnObj == rImp.pOrig) return rImp.pClone; } return nullptr; } // restrict movement to WorkArea static void ImpCheckInsertPos(Point& rPos, const Size& rSize, const ::tools::Rectangle& rWorkArea) { if(rWorkArea.IsEmpty()) return; ::tools::Rectangle aMarkRect(Point(rPos.X() - (rSize.Width() / 2), rPos.Y() - (rSize.Height() / 2)), rSize); if(aMarkRect.Contains(rWorkArea)) return; if(aMarkRect.Left() < rWorkArea.Left()) { rPos.AdjustX(rWorkArea.Left() - aMarkRect.Left() ); } if(aMarkRect.Right() > rWorkArea.Right()) { rPos.AdjustX( -(aMarkRect.Right() - rWorkArea.Right()) ); } if(aMarkRect.Top() < rWorkArea.Top()) { rPos.AdjustY(rWorkArea.Top() - aMarkRect.Top() ); } if(aMarkRect.Bottom() > rWorkArea.Bottom()) { rPos.AdjustY( -(aMarkRect.Bottom() - rWorkArea.Bottom()) ); } } bool View::InsertMetaFile( const TransferableDataHelper& rDataHelper, const Point& rPos, ImageMap const * pImageMap, bool bOptimize ) { GDIMetaFile aMtf; if( !rDataHelper.GetGDIMetaFile( SotClipboardFormatId::GDIMETAFILE, aMtf ) ) return false; bool bVector = false; Graphic aGraphic; // check if metafile only contains a pixel image, if so insert a bitmap instead if( bOptimize ) { MetaAction* pAction = aMtf.FirstAction(); while( pAction && !bVector ) { switch( pAction->GetType() ) { case MetaActionType::POINT: case MetaActionType::LINE: case MetaActionType::RECT: case MetaActionType::ROUNDRECT: case MetaActionType::ELLIPSE: case MetaActionType::ARC: case MetaActionType::PIE: case MetaActionType::CHORD: case MetaActionType::POLYLINE: case MetaActionType::POLYGON: case MetaActionType::POLYPOLYGON: case MetaActionType::TEXT: case MetaActionType::TEXTARRAY: case MetaActionType::STRETCHTEXT: case MetaActionType::TEXTRECT: case MetaActionType::GRADIENT: case MetaActionType::HATCH: case MetaActionType::WALLPAPER: case MetaActionType::EPS: case MetaActionType::TEXTLINE: case MetaActionType::FLOATTRANSPARENT: case MetaActionType::GRADIENTEX: case MetaActionType::BMPSCALEPART: case MetaActionType::BMPEXSCALEPART: bVector = true; break; case MetaActionType::BMP: case MetaActionType::BMPSCALE: case MetaActionType::BMPEX: case MetaActionType::BMPEXSCALE: if( aGraphic.GetType() != GraphicType::NONE ) { bVector = true; } else switch( pAction->GetType() ) { case MetaActionType::BMP: { MetaBmpAction* pBmpAction = dynamic_cast< MetaBmpAction* >( pAction ); if( pBmpAction ) aGraphic = Graphic(BitmapEx(pBmpAction->GetBitmap())); } break; case MetaActionType::BMPSCALE: { MetaBmpScaleAction* pBmpScaleAction = dynamic_cast< MetaBmpScaleAction* >( pAction ); if( pBmpScaleAction ) aGraphic = Graphic(BitmapEx(pBmpScaleAction->GetBitmap())); } break; case MetaActionType::BMPEX: { MetaBmpExAction* pBmpExAction = dynamic_cast< MetaBmpExAction* >( pAction ); if( pBmpExAction ) aGraphic = Graphic(pBmpExAction->GetBitmapEx() ); } break; case MetaActionType::BMPEXSCALE: { MetaBmpExScaleAction* pBmpExScaleAction = dynamic_cast< MetaBmpExScaleAction* >( pAction ); if( pBmpExScaleAction ) aGraphic = Graphic( pBmpExScaleAction->GetBitmapEx() ); } break; default: break; } break; default: break; } pAction = aMtf.NextAction(); } } // it is not a vector metafile but it also has no graphic? if( !bVector && (aGraphic.GetType() == GraphicType::NONE) ) bVector = true; // restrict movement to WorkArea Point aInsertPos( rPos ); Size aImageSize = bVector ? aMtf.GetPrefSize() : aGraphic.GetSizePixel(); ImpCheckInsertPos(aInsertPos, aImageSize, GetWorkArea()); if( bVector ) aGraphic = Graphic( aMtf ); aGraphic.SetPrefMapMode( aMtf.GetPrefMapMode() ); aGraphic.SetPrefSize( aMtf.GetPrefSize() ); InsertGraphic( aGraphic, mnAction, aInsertPos, nullptr, pImageMap ); return true; } bool View::InsertData( const TransferableDataHelper& rDataHelper, const Point& rPos, sal_Int8& rDnDAction, bool bDrag, SotClipboardFormatId nFormat, sal_uInt16 nPage, SdrLayerID nLayer ) { maDropPos = rPos; mnAction = rDnDAction; mbIsDropAllowed = false; TransferableDataHelper aDataHelper( rDataHelper ); SdrObject* pPickObj = nullptr; SdPage* pPage = nullptr; std::unique_ptr pImageMap; bool bReturn = false; bool bLink = ( ( mnAction & DND_ACTION_LINK ) != 0 ); bool bCopy = ( ( ( mnAction & DND_ACTION_COPY ) != 0 ) || bLink ); SdrInsertFlags nPasteOptions = SdrInsertFlags::SETDEFLAYER; if (mpViewSh != nullptr) { OSL_ASSERT (mpViewSh->GetViewShell()!=nullptr); SfxInPlaceClient* pIpClient = mpViewSh->GetViewShell()->GetIPClient(); if( dynamic_cast< ::sd::slidesorter::SlideSorterViewShell *>( mpViewSh ) != nullptr || (pIpClient!=nullptr && pIpClient->IsObjectInPlaceActive())) nPasteOptions |= SdrInsertFlags::DONTMARK; } if( bDrag ) { SdrPageView* pPV = nullptr; pPickObj = PickObj(rPos, getHitTolLog(), pPV); } if( nPage != SDRPAGE_NOTFOUND ) pPage = static_cast( mrDoc.GetPage( nPage ) ); SdTransferable* pOwnData = nullptr; SdTransferable* pImplementation = SdTransferable::getImplementation( aDataHelper.GetTransferable() ); if(pImplementation && (rDnDAction & DND_ACTION_LINK)) { // suppress own data when it's intention is to use it as fill information pImplementation = nullptr; } bool bSelfDND = false; // try to get own transfer data if( pImplementation ) { if( SD_MOD()->pTransferClip == pImplementation ) pOwnData = SD_MOD()->pTransferClip; else if( SD_MOD()->pTransferDrag == pImplementation ) { pOwnData = SD_MOD()->pTransferDrag; bSelfDND = true; } else if( SD_MOD()->pTransferSelection == pImplementation ) pOwnData = SD_MOD()->pTransferSelection; } const bool bGroupUndoFromDragWithDrop = bSelfDND && mpDragSrcMarkList && IsUndoEnabled(); if (bGroupUndoFromDragWithDrop) { OUString aStr(SdResId(STR_UNDO_DRAGDROP)); BegUndo(aStr + " " + mpDragSrcMarkList->GetMarkDescription()); } // ImageMap? if( !pOwnData && aDataHelper.HasFormat( SotClipboardFormatId::SVIM ) ) { if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( SotClipboardFormatId::SVIM ) ) { pImageMap.reset(new ImageMap); // mba: clipboard always must contain absolute URLs (could be from alien source) pImageMap->Read( *xStm ); } } bool bTable = false; // check special cases for pasting table formats as RTL if( !bLink && (nFormat == SotClipboardFormatId::NONE || (nFormat == SotClipboardFormatId::RTF) || (nFormat == SotClipboardFormatId::RICHTEXT)) ) { // if the object supports rtf and there is a table involved, default is to create a table bool bIsRTF = aDataHelper.HasFormat( SotClipboardFormatId::RTF ); if( ( bIsRTF || aDataHelper.HasFormat( SotClipboardFormatId::RICHTEXT ) ) && ! aDataHelper.HasFormat( SotClipboardFormatId::DRAWING ) ) { auto nFormatId = bIsRTF ? SotClipboardFormatId::RTF : SotClipboardFormatId::RICHTEXT; if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( nFormatId )) { xStm->Seek( 0 ); OStringBuffer aLine; while (xStm->ReadLine(aLine)) { size_t x = std::string_view(aLine).find( "\\trowd" ); if (x != std::string_view::npos) { bTable = true; nFormat = bIsRTF ? SotClipboardFormatId::RTF : SotClipboardFormatId::RICHTEXT; break; } } } } } // Changed the whole decision tree to be dependent of bReturn as a flag that // the work was done; this allows to check multiple formats and not just fail // when a CHECK_FORMAT_TRANS(*format*) detected format does not work. This is // e.g. necessary for SotClipboardFormatId::BITMAP if (!bReturn && pOwnData) { // Paste only if SfxClassificationHelper recommends so. const SfxObjectShellRef& pSource = pOwnData->GetDocShell(); SfxObjectShell* pDestination = mrDoc.GetDocSh(); if (pSource.is() && pDestination) { SfxClassificationCheckPasteResult eResult = SfxClassificationHelper::CheckPaste(pSource->getDocProperties(), pDestination->getDocProperties()); if (!SfxClassificationHelper::ShowPasteInfo(eResult)) bReturn = true; } } if( !bReturn && pOwnData && nFormat == SotClipboardFormatId::NONE ) { const View* pSourceView = pOwnData->GetView(); if( pOwnData->GetDocShell().is() && pOwnData->IsPageTransferable() ) { mpClipboard->HandlePageDrop (*pOwnData); bReturn = true; } else if( pSourceView ) { if( pSourceView == this ) { // same view if( nLayer != SDRLAYER_NOTFOUND ) { // drop on layer tab bar SdrLayerAdmin& rLayerAdmin = mrDoc.GetLayerAdmin(); SdrLayer* pLayer = rLayerAdmin.GetLayerPerID( nLayer ); SdrPageView* pPV = GetSdrPageView(); OUString aLayer = pLayer->GetName(); if( !pPV->IsLayerLocked( aLayer ) ) { pOwnData->SetInternalMove( true ); const SdrMarkList& rMarkList = GetMarkedObjectList(); rMarkList.ForceSort(); for( size_t nM = 0; nM < rMarkList.GetMarkCount(); ++nM ) { SdrMark* pM = rMarkList.GetMark( nM ); SdrObject* pO = pM->GetMarkedSdrObj(); if( pO ) { // #i11702# if( IsUndoEnabled() ) { BegUndo(SdResId(STR_MODIFYLAYER)); AddUndo(GetModel().GetSdrUndoFactory().CreateUndoObjectLayerChange(*pO, pO->GetLayer(), nLayer)); EndUndo(); } pO->SetLayer( nLayer ); } } bReturn = true; } } else { SdrPageView* pPV = GetSdrPageView(); bool bDropOnTabBar = true; if( !pPage && pPV->GetPage()->GetPageNum() != mnDragSrcPgNum ) { pPage = static_cast( pPV->GetPage() ); bDropOnTabBar = false; } if( pPage ) { // drop on other page OUString aActiveLayer = GetActiveLayer(); if( !pPV->IsLayerLocked( aActiveLayer ) ) { if( !IsPresObjSelected() ) { SdrMarkList* pMarkList; if( (mnDragSrcPgNum != SDRPAGE_NOTFOUND) && (mnDragSrcPgNum != pPV->GetPage()->GetPageNum()) ) { pMarkList = mpDragSrcMarkList.get(); } else { // actual mark list is used pMarkList = new SdrMarkList( GetMarkedObjectList()); } pMarkList->ForceSort(); // stuff to remember originals and clones std::vector aConnectorContainer; size_t nConnectorCount = 0; Point aCurPos; // calculate real position of current // source objects, if necessary (#103207) if( pOwnData == SD_MOD()->pTransferSelection ) { ::tools::Rectangle aCurBoundRect; if( pMarkList->TakeBoundRect( pPV, aCurBoundRect ) ) aCurPos = aCurBoundRect.TopLeft(); else aCurPos = pOwnData->GetStartPos(); } else aCurPos = pOwnData->GetStartPos(); const Size aVector( maDropPos.X() - aCurPos.X(), maDropPos.Y() - aCurPos.Y() ); std::unordered_set aNameSet; for(size_t a = 0; a < pMarkList->GetMarkCount(); ++a) { SdrMark* pM = pMarkList->GetMark(a); rtl::Reference pObj(pM->GetMarkedSdrObj()->CloneSdrObject(pPage->getSdrModelFromSdrPage())); if(pObj) { if(!bDropOnTabBar) { // do a NbcMove(...) instead of setting SnapRects here pObj->NbcMove(aVector); } SdrObject* pMarkParent = pM->GetMarkedSdrObj()->getParentSdrObjectFromSdrObject(); if (bCopy || (pMarkParent && pMarkParent->IsGroupObject())) pPage->InsertObjectThenMakeNameUnique(pObj.get(), aNameSet); else pPage->InsertObject(pObj.get()); if( IsUndoEnabled() ) { BegUndo(SdResId(STR_UNDO_DRAGDROP)); AddUndo(GetModel().GetSdrUndoFactory().CreateUndoNewObject(*pObj)); EndUndo(); } ImpRememberOrigAndClone aRem; aRem.pOrig = pM->GetMarkedSdrObj(); aRem.pClone = pObj.get(); aConnectorContainer.push_back(aRem); if(dynamic_cast< SdrEdgeObj *>( pObj.get() ) != nullptr) nConnectorCount++; } } // try to re-establish connections at clones if(nConnectorCount) { for(size_t a = 0; a < aConnectorContainer.size(); ++a) { ImpRememberOrigAndClone* pRem = &aConnectorContainer[a]; if(auto pCloneEdge = dynamic_cast( pRem->pClone )) { SdrEdgeObj* pOrigEdge = static_cast(pRem->pOrig); // test first connection SdrObjConnection& rConn0 = pOrigEdge->GetConnection(false); SdrObject* pConnObj = rConn0.GetSdrObject(); if(pConnObj) { SdrObject* pConnClone = ImpGetClone(aConnectorContainer, pConnObj); if(pConnClone) { // if dest obj was cloned, too, re-establish connection pCloneEdge->ConnectToNode(false, pConnClone); pCloneEdge->GetConnection(false).SetConnectorId(rConn0.GetConnectorId()); } else { // set position of connection point of original connected object const SdrGluePointList* pGlueList = pConnObj->GetGluePointList(); if(pGlueList) { sal_uInt16 nInd = pGlueList->FindGluePoint(rConn0.GetConnectorId()); if(SDRGLUEPOINT_NOTFOUND != nInd) { const SdrGluePoint& rGluePoint = (*pGlueList)[nInd]; Point aPosition = rGluePoint.GetAbsolutePos(*pConnObj); aPosition.AdjustX(aVector.Width() ); aPosition.AdjustY(aVector.Height() ); pCloneEdge->SetTailPoint(false, aPosition); } } } } // test second connection SdrObjConnection& rConn1 = pOrigEdge->GetConnection(true); pConnObj = rConn1.GetSdrObject(); if(pConnObj) { SdrObject* pConnClone = ImpGetClone(aConnectorContainer, pConnObj); if(pConnClone) { // if dest obj was cloned, too, re-establish connection pCloneEdge->ConnectToNode(true, pConnClone); pCloneEdge->GetConnection(true).SetConnectorId(rConn1.GetConnectorId()); } else { // set position of connection point of original connected object const SdrGluePointList* pGlueList = pConnObj->GetGluePointList(); if(pGlueList) { sal_uInt16 nInd = pGlueList->FindGluePoint(rConn1.GetConnectorId()); if(SDRGLUEPOINT_NOTFOUND != nInd) { const SdrGluePoint& rGluePoint = (*pGlueList)[nInd]; Point aPosition = rGluePoint.GetAbsolutePos(*pConnObj); aPosition.AdjustX(aVector.Width() ); aPosition.AdjustY(aVector.Height() ); pCloneEdge->SetTailPoint(true, aPosition); } } } } } } } if( pMarkList != mpDragSrcMarkList.get() ) delete pMarkList; bReturn = true; } else { maDropErrorIdle.Start(); bReturn = false; } } } else { pOwnData->SetInternalMove( true ); MoveAllMarked( Size( maDropPos.X() - pOwnData->GetStartPos().X(), maDropPos.Y() - pOwnData->GetStartPos().Y() ), bCopy ); bReturn = true; } } } else { // different views if( !pSourceView->IsPresObjSelected() ) { // model is owned by from AllocModel() created DocShell SdDrawDocument* pSourceDoc = static_cast(&pSourceView->GetModel()); pSourceDoc->CreatingDataObj( pOwnData ); SdDrawDocument* pModel = static_cast( pSourceView->CreateMarkedObjModel().release() ); bReturn = Paste(*pModel, maDropPos, pPage, nPasteOptions); if( !pPage ) pPage = static_cast( GetSdrPageView()->GetPage() ); OUString aLayout = pPage->GetLayoutName(); sal_Int32 nPos = aLayout.indexOf(SD_LT_SEPARATOR); if (nPos != -1) aLayout = aLayout.copy(0, nPos); pPage->SetPresentationLayout( aLayout, false, false ); pSourceDoc->CreatingDataObj( nullptr ); } else { maDropErrorIdle.Start(); bReturn = false; } } } else { SdDrawDocument* pWorkModel = const_cast(pOwnData->GetWorkDocument()); SdPage* pWorkPage = pWorkModel->GetSdPage( 0, PageKind::Standard ); pWorkPage->SetSdrObjListRectsDirty(); // #i120393# Clipboard data uses full object geometry range const Size aSize( pWorkPage->GetAllObjBoundRect().GetSize() ); maDropPos.setX( pOwnData->GetStartPos().X() + ( aSize.Width() >> 1 ) ); maDropPos.setY( pOwnData->GetStartPos().Y() + ( aSize.Height() >> 1 ) ); // delete pages, that are not of any interest for us for( ::tools::Long i = pWorkModel->GetPageCount() - 1; i >= 0; i-- ) { SdPage* pP = static_cast< SdPage* >( pWorkModel->GetPage( static_cast(i) ) ); if( pP->GetPageKind() != PageKind::Standard ) pWorkModel->DeletePage( static_cast(i) ); } bReturn = Paste(*pWorkModel, maDropPos, pPage, nPasteOptions); if( !pPage ) pPage = static_cast( GetSdrPageView()->GetPage() ); OUString aLayout = pPage->GetLayoutName(); sal_Int32 nPos = aLayout.indexOf(SD_LT_SEPARATOR); if (nPos != -1) aLayout = aLayout.copy(0, nPos); pPage->SetPresentationLayout( aLayout, false, false ); } } if(!bReturn && CHECK_FORMAT_TRANS( SotClipboardFormatId::PDF )) { if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( SotClipboardFormatId::PDF )) { Point aInsertPos(rPos); Graphic aGraphic; if (vcl::ImportPDF(*xStm, aGraphic)) { const sal_uInt64 nGraphicContentSize(xStm->Tell()); xStm->Seek(0); BinaryDataContainer aGraphicContent(*xStm, nGraphicContentSize); aGraphic.SetGfxLink(std::make_shared(aGraphicContent, GfxLinkType::NativePdf)); InsertGraphic(aGraphic, mnAction, aInsertPos, nullptr, nullptr); bReturn = true; } } } if(!bReturn && CHECK_FORMAT_TRANS( SotClipboardFormatId::DRAWING )) { if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( SotClipboardFormatId::DRAWING )) { DrawDocShellRef xShell = new DrawDocShell(SfxObjectCreateMode::INTERNAL, false, DocumentType::Impress); xShell->DoInitNew(); SdDrawDocument* pModel = xShell->GetDoc(); pModel->InsertPage(pModel->AllocPage(false).get()); Reference< XComponent > xComponent = xShell->GetModel(); xStm->Seek( 0 ); css::uno::Reference< css::io::XInputStream > xInputStream( new utl::OInputStreamWrapper( *xStm ) ); bReturn = SvxDrawingLayerImport( pModel, xInputStream, xComponent, "com.sun.star.comp.Impress.XMLOasisImporter" ); if( pModel->GetPageCount() == 0 ) { OSL_FAIL("empty or invalid drawing xml document on clipboard!" ); } else { bool bChanged = false; if( bReturn ) { if( pModel->GetSdPage( 0, PageKind::Standard )->GetObjCount() == 1 ) { // only one object SdrObject* pObj = pModel->GetSdPage( 0, PageKind::Standard )->GetObj( 0 ); SdrPageView* pPV = nullptr; SdrObject* pPickObj2 = PickObj(rPos, getHitTolLog(), pPV); if( ( mnAction & DND_ACTION_MOVE ) && pPickObj2 && pObj ) { // replace object SdrPage* pWorkPage = GetSdrPageView()->GetPage(); rtl::Reference pNewObj(pObj->CloneSdrObject(pWorkPage->getSdrModelFromSdrPage())); ::tools::Rectangle aPickObjRect( pPickObj2->GetCurrentBoundRect() ); Size aPickObjSize( aPickObjRect.GetSize() ); Point aVec( aPickObjRect.TopLeft() ); ::tools::Rectangle aObjRect( pNewObj->GetCurrentBoundRect() ); Size aObjSize( aObjRect.GetSize() ); Fraction aScaleWidth( aPickObjSize.Width(), aObjSize.Width() ); Fraction aScaleHeight( aPickObjSize.Height(), aObjSize.Height() ); pNewObj->NbcResize( aObjRect.TopLeft(), aScaleWidth, aScaleHeight ); aVec -= aObjRect.TopLeft(); pNewObj->NbcMove( Size( aVec.X(), aVec.Y() ) ); const bool bUndo = IsUndoEnabled(); if( bUndo ) BegUndo(SdResId(STR_UNDO_DRAGDROP)); pNewObj->NbcSetLayer( pPickObj->GetLayer() ); pWorkPage->InsertObject( pNewObj.get() ); if( bUndo ) { AddUndo( mrDoc.GetSdrUndoFactory().CreateUndoNewObject( *pNewObj ) ); AddUndo( mrDoc.GetSdrUndoFactory().CreateUndoDeleteObject( *pPickObj2 ) ); } pWorkPage->RemoveObject( pPickObj2->GetOrdNum() ); if( bUndo ) { EndUndo(); } bChanged = true; mnAction = DND_ACTION_COPY; } else if( ( mnAction & DND_ACTION_LINK ) && pPickObj && pObj && dynamic_cast< const SdrGrafObj *>( pPickObj ) == nullptr && dynamic_cast< const SdrOle2Obj *>( pPickObj ) == nullptr ) { SfxItemSet aSet( mrDoc.GetPool() ); // set new attributes to object const bool bUndo = IsUndoEnabled(); if( bUndo ) { BegUndo( SdResId(STR_UNDO_DRAGDROP) ); AddUndo( mrDoc.GetSdrUndoFactory().CreateUndoAttrObject( *pPickObj ) ); } aSet.Put( pObj->GetMergedItemSet() ); /* Do not take over corner radius. There are gradients (rectangles) in the gallery with corner radius of 0. We should not use that on the object. */ aSet.ClearItem( SDRATTR_CORNER_RADIUS ); const SdrGrafObj* pSdrGrafObj = dynamic_cast< const SdrGrafObj* >(pObj); if(pSdrGrafObj) { // If we have a graphic as source object, use its graphic // content as fill style aSet.Put(XFillStyleItem(drawing::FillStyle_BITMAP)); aSet.Put(XFillBitmapItem(pSdrGrafObj->GetGraphic())); } pPickObj->SetMergedItemSetAndBroadcast( aSet ); if( DynCastE3dObject( pPickObj ) && DynCastE3dObject( pObj ) ) { // handle 3D attribute in addition SfxItemSetFixed aNewSet( mrDoc.GetPool() ); SfxItemSetFixed aOldSet( mrDoc.GetPool() ); aOldSet.Put(pPickObj->GetMergedItemSet()); aNewSet.Put( pObj->GetMergedItemSet() ); if( bUndo ) AddUndo( std::make_unique( *static_cast< E3dObject* >(pPickObj), aNewSet, aOldSet)); pPickObj->SetMergedItemSetAndBroadcast( aNewSet ); } if( bUndo ) EndUndo(); bChanged = true; } } } if( !bChanged ) { SdrPage* pWorkPage = pModel->GetSdPage( 0, PageKind::Standard ); pWorkPage->SetSdrObjListRectsDirty(); if( pOwnData ) { // #i120393# Clipboard data uses full object geometry range const Size aSize( pWorkPage->GetAllObjBoundRect().GetSize() ); maDropPos.setX( pOwnData->GetStartPos().X() + ( aSize.Width() >> 1 ) ); maDropPos.setY( pOwnData->GetStartPos().Y() + ( aSize.Height() >> 1 ) ); } bReturn = Paste(*pModel, maDropPos, pPage, nPasteOptions); } xShell->DoClose(); } } } if(!bReturn && CHECK_FORMAT_TRANS(SotClipboardFormatId::SBA_FIELDDATAEXCHANGE)) { OUString aOUString; if( aDataHelper.GetString( SotClipboardFormatId::SBA_FIELDDATAEXCHANGE, aOUString ) ) { rtl::Reference pObj = CreateFieldControl( aOUString ); if( pObj ) { ::tools::Rectangle aRect( pObj->GetLogicRect() ); Size aSize( aRect.GetSize() ); maDropPos.AdjustX( -( aSize.Width() >> 1 ) ); maDropPos.AdjustY( -( aSize.Height() >> 1 ) ); aRect.SetPos( maDropPos ); pObj->SetLogicRect( aRect ); InsertObjectAtView( pObj.get(), *GetSdrPageView(), SdrInsertFlags::SETDEFLAYER ); bReturn = true; } } } if(!bReturn && !bLink && (CHECK_FORMAT_TRANS(SotClipboardFormatId::EMBED_SOURCE) || CHECK_FORMAT_TRANS(SotClipboardFormatId::EMBEDDED_OBJ)) && aDataHelper.HasFormat(SotClipboardFormatId::OBJECTDESCRIPTOR)) { //TODO/LATER: is it possible that this format is binary?! (from old versions of SO) uno::Reference < io::XInputStream > xStm; TransferableObjectDescriptor aObjDesc; if (aDataHelper.GetTransferableObjectDescriptor(SotClipboardFormatId::OBJECTDESCRIPTOR, aObjDesc)) { OUString aDocShellID = SfxObjectShell::CreateShellID(mrDoc.GetDocSh()); xStm = aDataHelper.GetInputStream(nFormat != SotClipboardFormatId::NONE ? nFormat : SotClipboardFormatId::EMBED_SOURCE, aDocShellID); if (!xStm.is()) xStm = aDataHelper.GetInputStream(SotClipboardFormatId::EMBEDDED_OBJ, aDocShellID); } if (xStm.is()) { if( mrDoc.GetDocSh() && ( mrDoc.GetDocSh()->GetClassName() == aObjDesc.maClassName ) ) { uno::Reference < embed::XStorage > xStore( ::comphelper::OStorageHelper::GetStorageFromInputStream( xStm ) ); ::sd::DrawDocShellRef xDocShRef( new ::sd::DrawDocShell( SfxObjectCreateMode::EMBEDDED, true, mrDoc.GetDocumentType() ) ); // mba: BaseURL doesn't make sense for clipboard functionality SfxMedium *pMedium = new SfxMedium( xStore, OUString() ); if( xDocShRef->DoLoad( pMedium ) ) { SdDrawDocument* pModel = xDocShRef->GetDoc(); SdPage* pWorkPage = pModel->GetSdPage( 0, PageKind::Standard ); pWorkPage->SetSdrObjListRectsDirty(); if( pOwnData ) { // #i120393# Clipboard data uses full object geometry range const Size aSize( pWorkPage->GetAllObjBoundRect().GetSize() ); maDropPos.setX( pOwnData->GetStartPos().X() + ( aSize.Width() >> 1 ) ); maDropPos.setY( pOwnData->GetStartPos().Y() + ( aSize.Height() >> 1 ) ); } // delete pages, that are not of any interest for us for( ::tools::Long i = pModel->GetPageCount() - 1; i >= 0; i-- ) { SdPage* pP = static_cast< SdPage* >( pModel->GetPage( static_cast(i) ) ); if( pP->GetPageKind() != PageKind::Standard ) pModel->DeletePage( static_cast(i) ); } bReturn = Paste(*pModel, maDropPos, pPage, nPasteOptions); if( !pPage ) pPage = static_cast(GetSdrPageView()->GetPage()); OUString aLayout = pPage->GetLayoutName(); sal_Int32 nPos = aLayout.indexOf(SD_LT_SEPARATOR); if (nPos != -1) aLayout = aLayout.copy(0, nPos); pPage->SetPresentationLayout( aLayout, false, false ); } xDocShRef->DoClose(); xDocShRef.clear(); } else { OUString aName; uno::Reference < embed::XEmbeddedObject > xObj = mpDocSh->GetEmbeddedObjectContainer().InsertEmbeddedObject( xStm, aName ); if ( xObj.is() ) { svt::EmbeddedObjectRef aObjRef( xObj, aObjDesc.mnViewAspect ); Size aSize; if ( aObjDesc.mnViewAspect == embed::Aspects::MSOLE_ICON ) { if( aObjDesc.maSize.Width() && aObjDesc.maSize.Height() ) aSize = aObjDesc.maSize; else { MapMode aMapMode( MapUnit::Map100thMM ); aSize = aObjRef.GetSize( &aMapMode ); } } else { awt::Size aSz; MapUnit aMapUnit = VCLUnoHelper::UnoEmbed2VCLMapUnit( xObj->getMapUnit( aObjDesc.mnViewAspect ) ); if( aObjDesc.maSize.Width() && aObjDesc.maSize.Height() ) { Size aTmp(OutputDevice::LogicToLogic(aObjDesc.maSize, MapMode(MapUnit::Map100thMM), MapMode(aMapUnit))); aSz.Width = aTmp.Width(); aSz.Height = aTmp.Height(); xObj->setVisualAreaSize( aObjDesc.mnViewAspect, aSz ); } try { aSz = xObj->getVisualAreaSize( aObjDesc.mnViewAspect ); } catch( embed::NoVisualAreaSizeException& ) { // if the size still was not set the default size will be set later } aSize = Size( aSz.Width, aSz.Height ); if( !aSize.Width() || !aSize.Height() ) { aSize.setWidth( 14100 ); aSize.setHeight( 10000 ); aSize = OutputDevice::LogicToLogic(Size(14100, 10000), MapMode(MapUnit::Map100thMM), MapMode(aMapUnit)); aSz.Width = aSize.Width(); aSz.Height = aSize.Height(); xObj->setVisualAreaSize( aObjDesc.mnViewAspect, aSz ); } aSize = OutputDevice::LogicToLogic(aSize, MapMode(aMapUnit), MapMode(MapUnit::Map100thMM)); } Size aMaxSize( mrDoc.GetMaxObjSize() ); maDropPos.AdjustX( -(std::min( aSize.Width(), aMaxSize.Width() ) >> 1) ); maDropPos.AdjustY( -(std::min( aSize.Height(), aMaxSize.Height() ) >> 1) ); ::tools::Rectangle aRect( maDropPos, aSize ); rtl::Reference pObj = new SdrOle2Obj( getSdrModelFromSdrView(), aObjRef, aName, aRect); SdrPageView* pPV = GetSdrPageView(); SdrInsertFlags nOptions = SdrInsertFlags::SETDEFLAYER; if (mpViewSh!=nullptr) { OSL_ASSERT (mpViewSh->GetViewShell()!=nullptr); SfxInPlaceClient* pIpClient = mpViewSh->GetViewShell()->GetIPClient(); if (pIpClient!=nullptr && pIpClient->IsObjectInPlaceActive()) nOptions |= SdrInsertFlags::DONTMARK; } // bInserted of false means that pObj has been deleted bool bInserted = InsertObjectAtView( pObj.get(), *pPV, nOptions ); if (bInserted && pImageMap) pObj->AppendUserData( std::unique_ptr(new SvxIMapInfo( *pImageMap )) ); if (bInserted && pObj->IsChart()) { bool bDisableDataTableDialog = false; svt::EmbeddedObjectRef::TryRunningState( xObj ); uno::Reference< beans::XPropertySet > xProps( xObj->getComponent(), uno::UNO_QUERY ); if ( xProps.is() && ( xProps->getPropertyValue( u"DisableDataTableDialog"_ustr ) >>= bDisableDataTableDialog ) && bDisableDataTableDialog ) { xProps->setPropertyValue( u"DisableDataTableDialog"_ustr , uno::Any( false ) ); xProps->setPropertyValue( u"DisableComplexChartTypes"_ustr , uno::Any( false ) ); uno::Reference< util::XModifiable > xModifiable( xProps, uno::UNO_QUERY ); if ( xModifiable.is() ) { xModifiable->setModified( true ); } } } bReturn = true; } } } } if(!bReturn && !bLink && (CHECK_FORMAT_TRANS(SotClipboardFormatId::EMBEDDED_OBJ_OLE) || CHECK_FORMAT_TRANS(SotClipboardFormatId::EMBED_SOURCE_OLE)) && aDataHelper.HasFormat(SotClipboardFormatId::OBJECTDESCRIPTOR_OLE)) { // online insert ole if format is forced or no gdi metafile is available if( (nFormat != SotClipboardFormatId::NONE) || !aDataHelper.HasFormat( SotClipboardFormatId::GDIMETAFILE ) ) { uno::Reference < io::XInputStream > xStm; TransferableObjectDescriptor aObjDesc; if ( aDataHelper.GetTransferableObjectDescriptor( SotClipboardFormatId::OBJECTDESCRIPTOR_OLE, aObjDesc ) ) { uno::Reference < embed::XEmbeddedObject > xObj; OUString aName; xStm = aDataHelper.GetInputStream(nFormat != SotClipboardFormatId::NONE ? nFormat : SotClipboardFormatId::EMBED_SOURCE_OLE, OUString()); if (!xStm.is()) xStm = aDataHelper.GetInputStream(SotClipboardFormatId::EMBEDDED_OBJ_OLE, OUString()); if (xStm.is()) { xObj = mpDocSh->GetEmbeddedObjectContainer().InsertEmbeddedObject( xStm, aName ); } else { try { uno::Reference< embed::XStorage > xTmpStor = ::comphelper::OStorageHelper::GetTemporaryStorage(); uno::Reference < embed::XEmbedObjectClipboardCreator > xClipboardCreator = embed::MSOLEObjectSystemCreator::create( ::comphelper::getProcessComponentContext() ); embed::InsertedObjectInfo aInfo = xClipboardCreator->createInstanceInitFromClipboard( xTmpStor, u"DummyName"_ustr , uno::Sequence< beans::PropertyValue >() ); // TODO/LATER: in future InsertedObjectInfo will be used to get container related information // for example whether the object should be an iconified one xObj = aInfo.Object; if ( xObj.is() ) mpDocSh->GetEmbeddedObjectContainer().InsertEmbeddedObject( xObj, aName ); } catch( uno::Exception& ) {} } if ( xObj.is() ) { svt::EmbeddedObjectRef aObjRef( xObj, aObjDesc.mnViewAspect ); // try to get the replacement image from the clipboard Graphic aGraphic; SotClipboardFormatId nGrFormat = SotClipboardFormatId::NONE; // (for Selection Manager in Trusted Solaris) #ifndef __sun if( aDataHelper.GetGraphic( SotClipboardFormatId::SVXB, aGraphic ) ) nGrFormat = SotClipboardFormatId::SVXB; else if( aDataHelper.GetGraphic( SotClipboardFormatId::GDIMETAFILE, aGraphic ) ) nGrFormat = SotClipboardFormatId::GDIMETAFILE; else if( aDataHelper.GetGraphic( SotClipboardFormatId::BITMAP, aGraphic ) ) nGrFormat = SotClipboardFormatId::BITMAP; #endif // insert replacement image ( if there is one ) into the object helper if ( nGrFormat != SotClipboardFormatId::NONE ) { datatransfer::DataFlavor aDataFlavor; SotExchange::GetFormatDataFlavor( nGrFormat, aDataFlavor ); aObjRef.SetGraphic( aGraphic, aDataFlavor.MimeType ); } Size aSize; if ( aObjDesc.mnViewAspect == embed::Aspects::MSOLE_ICON ) { if( aObjDesc.maSize.Width() && aObjDesc.maSize.Height() ) aSize = aObjDesc.maSize; else { MapMode aMapMode( MapUnit::Map100thMM ); aSize = aObjRef.GetSize( &aMapMode ); } } else { MapUnit aMapUnit = VCLUnoHelper::UnoEmbed2VCLMapUnit( xObj->getMapUnit( aObjDesc.mnViewAspect ) ); awt::Size aSz; try{ aSz = xObj->getVisualAreaSize( aObjDesc.mnViewAspect ); } catch( embed::NoVisualAreaSizeException& ) { // the default size will be set later } if( aObjDesc.maSize.Width() && aObjDesc.maSize.Height() ) { Size aTmp(OutputDevice::LogicToLogic(aObjDesc.maSize, MapMode(MapUnit::Map100thMM), MapMode(aMapUnit))); if ( aSz.Width != aTmp.Width() || aSz.Height != aTmp.Height() ) { aSz.Width = aTmp.Width(); aSz.Height = aTmp.Height(); xObj->setVisualAreaSize( aObjDesc.mnViewAspect, aSz ); } } aSize = Size( aSz.Width, aSz.Height ); if( !aSize.Width() || !aSize.Height() ) { aSize = OutputDevice::LogicToLogic(Size(14100, 10000), MapMode(MapUnit::Map100thMM), MapMode(aMapUnit)); aSz.Width = aSize.Width(); aSz.Height = aSize.Height(); xObj->setVisualAreaSize( aObjDesc.mnViewAspect, aSz ); } aSize = OutputDevice::LogicToLogic(aSize, MapMode(aMapUnit), MapMode(MapUnit::Map100thMM)); } Size aMaxSize( mrDoc.GetMaxObjSize() ); maDropPos.AdjustX( -(std::min( aSize.Width(), aMaxSize.Width() ) >> 1) ); maDropPos.AdjustY( -(std::min( aSize.Height(), aMaxSize.Height() ) >> 1) ); ::tools::Rectangle aRect( maDropPos, aSize ); rtl::Reference pObj = new SdrOle2Obj( getSdrModelFromSdrView(), aObjRef, aName, aRect); SdrPageView* pPV = GetSdrPageView(); SdrInsertFlags nOptions = SdrInsertFlags::SETDEFLAYER; if (mpViewSh!=nullptr) { OSL_ASSERT (mpViewSh->GetViewShell()!=nullptr); SfxInPlaceClient* pIpClient = mpViewSh->GetViewShell()->GetIPClient(); if (pIpClient!=nullptr && pIpClient->IsObjectInPlaceActive()) nOptions |= SdrInsertFlags::DONTMARK; } bReturn = InsertObjectAtView( pObj.get(), *pPV, nOptions ); if (bReturn) { if( pImageMap ) pObj->AppendUserData( std::unique_ptr(new SvxIMapInfo( *pImageMap )) ); // let the object stay in loaded state after insertion pObj->Unload(); } } } } if( !bReturn && aDataHelper.HasFormat( SotClipboardFormatId::GDIMETAFILE ) ) { // if no object was inserted, insert a picture InsertMetaFile( aDataHelper, rPos, pImageMap.get(), true ); bReturn = true; } } if(!bReturn && (!bLink || pPickObj) && CHECK_FORMAT_TRANS(SotClipboardFormatId::SVXB)) { if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( SotClipboardFormatId::SVXB )) { Point aInsertPos( rPos ); Graphic aGraphic; TypeSerializer aSerializer(*xStm); aSerializer.readGraphic(aGraphic); if( pOwnData && pOwnData->GetWorkDocument() ) { const SdDrawDocument* pWorkModel = pOwnData->GetWorkDocument(); SdrPage* pWorkPage = const_cast( ( pWorkModel->GetPageCount() > 1 ) ? pWorkModel->GetSdPage( 0, PageKind::Standard ) : pWorkModel->GetPage( 0 ) ); pWorkPage->SetSdrObjListRectsDirty(); // #i120393# Clipboard data uses full object geometry range const Size aSize( pWorkPage->GetAllObjBoundRect().GetSize() ); aInsertPos.setX( pOwnData->GetStartPos().X() + ( aSize.Width() >> 1 ) ); aInsertPos.setY( pOwnData->GetStartPos().Y() + ( aSize.Height() >> 1 ) ); } // restrict movement to WorkArea Size aImageMapSize = OutputDevice::LogicToLogic(aGraphic.GetPrefSize(), aGraphic.GetPrefMapMode(), MapMode(MapUnit::Map100thMM)); ImpCheckInsertPos(aInsertPos, aImageMapSize, GetWorkArea()); InsertGraphic( aGraphic, mnAction, aInsertPos, nullptr, pImageMap.get() ); bReturn = true; } } if(!bReturn && (!bLink || pPickObj) && CHECK_FORMAT_TRANS(SotClipboardFormatId::GDIMETAFILE)) { Point aInsertPos( rPos ); if( pOwnData && pOwnData->GetWorkDocument() ) { const SdDrawDocument* pWorkModel = pOwnData->GetWorkDocument(); SdrPage* pWorkPage = const_cast( ( pWorkModel->GetPageCount() > 1 ) ? pWorkModel->GetSdPage( 0, PageKind::Standard ) : pWorkModel->GetPage( 0 ) ); pWorkPage->SetSdrObjListRectsDirty(); // #i120393# Clipboard data uses full object geometry range const Size aSize( pWorkPage->GetAllObjBoundRect().GetSize() ); aInsertPos.setX( pOwnData->GetStartPos().X() + ( aSize.Width() >> 1 ) ); aInsertPos.setY( pOwnData->GetStartPos().Y() + ( aSize.Height() >> 1 ) ); } bReturn = InsertMetaFile( aDataHelper, aInsertPos, pImageMap.get(), nFormat == SotClipboardFormatId::NONE ); } if(!bReturn && (!bLink || pPickObj) && CHECK_FORMAT_TRANS(SotClipboardFormatId::BITMAP)) { BitmapEx aBmpEx; // get basic Bitmap data aDataHelper.GetBitmapEx(SotClipboardFormatId::BITMAP, aBmpEx); if(aBmpEx.IsEmpty()) { // if this did not work, try to get graphic formats and convert these to bitmap Graphic aGraphic; if(aDataHelper.GetGraphic(SotClipboardFormatId::GDIMETAFILE, aGraphic)) { aBmpEx = aGraphic.GetBitmapEx(); } else if(aDataHelper.GetGraphic(SotClipboardFormatId::SVXB, aGraphic)) { aBmpEx = aGraphic.GetBitmapEx(); } else if(aDataHelper.GetGraphic(SotClipboardFormatId::BITMAP, aGraphic)) { aBmpEx = aGraphic.GetBitmapEx(); } } if(!aBmpEx.IsEmpty()) { Point aInsertPos( rPos ); if( pOwnData && pOwnData->GetWorkDocument() ) { const SdDrawDocument* pWorkModel = pOwnData->GetWorkDocument(); SdrPage* pWorkPage = const_cast( ( pWorkModel->GetPageCount() > 1 ) ? pWorkModel->GetSdPage( 0, PageKind::Standard ) : pWorkModel->GetPage( 0 ) ); pWorkPage->SetSdrObjListRectsDirty(); // #i120393# Clipboard data uses full object geometry range const Size aSize( pWorkPage->GetAllObjBoundRect().GetSize() ); aInsertPos.setX( pOwnData->GetStartPos().X() + ( aSize.Width() >> 1 ) ); aInsertPos.setY( pOwnData->GetStartPos().Y() + ( aSize.Height() >> 1 ) ); } // restrict movement to WorkArea Size aImageMapSize(aBmpEx.GetPrefSize()); ImpCheckInsertPos(aInsertPos, aImageMapSize, GetWorkArea()); InsertGraphic( aBmpEx, mnAction, aInsertPos, nullptr, pImageMap.get() ); bReturn = true; } } if(!bReturn && pPickObj && CHECK_FORMAT_TRANS( SotClipboardFormatId::XFA ) ) { uno::Any const data(aDataHelper.GetAny(SotClipboardFormatId::XFA, u""_ustr)); uno::Sequence props; if (data >>= props) { if( IsUndoEnabled() ) { BegUndo( SdResId(STR_UNDO_DRAGDROP) ); AddUndo(GetModel().GetSdrUndoFactory().CreateUndoAttrObject(*pPickObj)); EndUndo(); } ::comphelper::SequenceAsHashMap const map(props); drawing::FillStyle eFill(drawing::FillStyle_BITMAP); // default to something that's ignored Color aColor(COL_BLACK); auto it = map.find(u"FillStyle"_ustr); if (it != map.end()) { XFillStyleItem style; style.PutValue(it->second, 0); eFill = style.GetValue(); } it = map.find(u"FillColor"_ustr); if (it != map.end()) { XFillColorItem color; color.PutValue(it->second, 0); aColor = color.GetColorValue(); } if( eFill == drawing::FillStyle_SOLID || eFill == drawing::FillStyle_NONE ) { SfxItemSet aSet( mrDoc.GetPool() ); bool bClosed = pPickObj->IsClosedObj(); ::sd::Window* pWin = mpViewSh->GetActiveWindow(); double fHitLog = pWin->PixelToLogic(Size(FuPoor::HITPIX, 0 ) ).Width(); const ::tools::Long n2HitLog = fHitLog * 2; Point aHitPosR( rPos ); Point aHitPosL( rPos ); Point aHitPosT( rPos ); Point aHitPosB( rPos ); const SdrLayerIDSet* pVisiLayer = &GetSdrPageView()->GetVisibleLayers(); aHitPosR.AdjustX(n2HitLog ); aHitPosL.AdjustX( -n2HitLog ); aHitPosT.AdjustY(n2HitLog ); aHitPosB.AdjustY( -n2HitLog ); if( bClosed && SdrObjectPrimitiveHit(*pPickObj, aHitPosR, {fHitLog, fHitLog}, *GetSdrPageView(), pVisiLayer, false) && SdrObjectPrimitiveHit(*pPickObj, aHitPosL, {fHitLog, fHitLog}, *GetSdrPageView(), pVisiLayer, false) && SdrObjectPrimitiveHit(*pPickObj, aHitPosT, {fHitLog, fHitLog}, *GetSdrPageView(), pVisiLayer, false) && SdrObjectPrimitiveHit(*pPickObj, aHitPosB, {fHitLog, fHitLog}, *GetSdrPageView(), pVisiLayer, false) ) { // area fill if(eFill == drawing::FillStyle_SOLID ) aSet.Put(XFillColorItem(u""_ustr, aColor)); aSet.Put( XFillStyleItem( eFill ) ); } else aSet.Put( XLineColorItem( u""_ustr, aColor ) ); // add text color pPickObj->SetMergedItemSetAndBroadcast( aSet ); } bReturn = true; } } if(!bReturn && !bLink && CHECK_FORMAT_TRANS(SotClipboardFormatId::HTML)) { if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( SotClipboardFormatId::HTML )) { xStm->Seek( 0 ); OStringBuffer aLine; while (xStm->ReadLine(aLine)) { if (std::string_view(aLine).find( "" ) != std::string_view::npos || std::string_view(aLine).find( "
Seek( 0 ); if (bTable) { bReturn = PasteHTMLTable(*xStm, pPage, nPasteOptions); } else { // mba: clipboard always must contain absolute URLs (could be from alien source) bReturn = SdrView::Paste( *xStm, EETextFormat::Html, maDropPos, pPage, nPasteOptions ); } } } if(!bReturn && !bLink && CHECK_FORMAT_TRANS(SotClipboardFormatId::EDITENGINE_ODF_TEXT_FLAT)) { if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( SotClipboardFormatId::EDITENGINE_ODF_TEXT_FLAT )) { OutlinerView* pOLV = GetTextEditOutlinerView(); xStm->Seek( 0 ); if( pOLV ) { ::tools::Rectangle aRect( pOLV->GetOutputArea() ); Point aPos( pOLV->GetWindow()->PixelToLogic( maDropPos ) ); if( aRect.Contains( aPos ) || ( !bDrag && IsTextEdit() ) ) { // mba: clipboard always must contain absolute URLs (could be from alien source) pOLV->Read( *xStm, EETextFormat::Xml, mpDocSh->GetHeaderAttributes() ); bReturn = true; } } if( !bReturn ) // mba: clipboard always must contain absolute URLs (could be from alien source) bReturn = SdrView::Paste( *xStm, EETextFormat::Xml, maDropPos, pPage, nPasteOptions ); } } if(!bReturn && !bLink) { bool bIsHtmlSimple = CHECK_FORMAT_TRANS(SotClipboardFormatId::HTML_SIMPLE); if (bIsHtmlSimple) { if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream(SotClipboardFormatId::HTML_SIMPLE)) { xStm->Seek(0); OutlinerView* pOLV = GetTextEditOutlinerView(); MSE40HTMLClipFormatObj aMSE40HTMLClipFormatObj; SvStream* pHtmlStream = aMSE40HTMLClipFormatObj.IsValid(*xStm); if (pOLV) { ::tools::Rectangle aRect(pOLV->GetOutputArea()); Point aPos(pOLV->GetWindow()->PixelToLogic(maDropPos)); if (aRect.Contains(aPos) || (!bDrag && IsTextEdit())) { // mba: clipboard always must contain absolute URLs (could be from alien source) pOLV->Read(*pHtmlStream, EETextFormat::Html, mpDocSh->GetHeaderAttributes()); bReturn = true; } } if (!bReturn) // mba: clipboard always must contain absolute URLs (could be from alien source) bReturn = SdrView::Paste(*pHtmlStream, EETextFormat::Html, maDropPos, pPage, nPasteOptions); } } bool bIsRTF = CHECK_FORMAT_TRANS(SotClipboardFormatId::RTF); if (!bReturn && (bIsRTF || CHECK_FORMAT_TRANS(SotClipboardFormatId::RICHTEXT))) { auto nFormatId = bIsRTF ? SotClipboardFormatId::RTF : SotClipboardFormatId::RICHTEXT; if (std::unique_ptr xStm = aDataHelper.GetSotStorageStream( nFormatId )) { xStm->Seek( 0 ); if( bTable ) { bReturn = PasteRTFTable( *xStm, pPage, nPasteOptions ); } else { OutlinerView* pOLV = GetTextEditOutlinerView(); if( pOLV ) { ::tools::Rectangle aRect( pOLV->GetOutputArea() ); Point aPos( pOLV->GetWindow()->PixelToLogic( maDropPos ) ); if( aRect.Contains( aPos ) || ( !bDrag && IsTextEdit() ) ) { // mba: clipboard always must contain absolute URLs (could be from alien source) pOLV->Read( *xStm, EETextFormat::Rtf, mpDocSh->GetHeaderAttributes() ); bReturn = true; } } if( !bReturn ) // mba: clipboard always must contain absolute URLs (could be from alien source) bReturn = SdrView::Paste( *xStm, EETextFormat::Rtf, maDropPos, pPage, nPasteOptions ); } } } } if(!bReturn && CHECK_FORMAT_TRANS(SotClipboardFormatId::FILE_LIST)) { FileList aDropFileList; if( aDataHelper.GetFileList( SotClipboardFormatId::FILE_LIST, aDropFileList ) ) { maDropFileVector.clear(); for( sal_uLong i = 0, nCount = aDropFileList.Count(); i < nCount; i++ ) maDropFileVector.push_back( aDropFileList.GetFile( i ) ); maDropInsertFileIdle.Start(); } bReturn = true; } if(!bReturn && CHECK_FORMAT_TRANS(SotClipboardFormatId::SIMPLE_FILE)) { OUString aDropFile; if( aDataHelper.GetString( SotClipboardFormatId::SIMPLE_FILE, aDropFile ) ) { maDropFileVector.clear(); maDropFileVector.push_back( aDropFile ); maDropInsertFileIdle.Start(); } bReturn = true; } if(!bReturn && !bLink && CHECK_FORMAT_TRANS(SotClipboardFormatId::STRING)) { if( ( SotClipboardFormatId::STRING == nFormat ) || ( !aDataHelper.HasFormat( SotClipboardFormatId::SOLK ) && !aDataHelper.HasFormat( SotClipboardFormatId::NETSCAPE_BOOKMARK ) && !aDataHelper.HasFormat( SotClipboardFormatId::FILENAME ) ) ) { OUString aOUString; if( aDataHelper.GetString( SotClipboardFormatId::STRING, aOUString ) ) { OutlinerView* pOLV = GetTextEditOutlinerView(); if( pOLV ) { pOLV->InsertText( aOUString ); bReturn = true; } if( !bReturn ) bReturn = SdrView::Paste( aOUString, maDropPos, pPage, nPasteOptions ); } } } MarkListHasChanged(); mbIsDropAllowed = true; rDnDAction = mnAction; if (bGroupUndoFromDragWithDrop) { // this is called eventually by the underlying toolkit anyway in the case of a self-dnd // but we call it early in this case to group its undo actions into this open dnd undo group // and rely on that repeated calls to View::DragFinished are safe to do DragFinished(mnAction); EndUndo(); } return bReturn; } bool View::PasteRTFTable( SvStream& rStm, SdrPage* pPage, SdrInsertFlags nPasteOptions ) { DrawDocShellRef xShell = new DrawDocShell(SfxObjectCreateMode::INTERNAL, false, DocumentType::Impress); xShell->DoInitNew(); SdDrawDocument* pModel = xShell->GetDoc(); pModel->GetItemPool().SetDefaultMetric(MapUnit::Map100thMM); pModel->InsertPage(pModel->AllocPage(false).get()); CreateTableFromRTF(rStm, pModel); bool bRet = Paste(*pModel, maDropPos, pPage, nPasteOptions); xShell->DoClose(); return bRet; } bool View::PasteHTMLTable( SvStream& rStm, SdrPage* pPage, SdrInsertFlags nPasteOptions ) { DrawDocShellRef xShell = new DrawDocShell(SfxObjectCreateMode::INTERNAL, false, DocumentType::Impress); xShell->DoInitNew(); SdDrawDocument* pModel = xShell->GetDoc(); pModel->GetItemPool().SetDefaultMetric(MapUnit::Map100thMM); pModel->InsertPage(pModel->AllocPage(false).get()); CreateTableFromHTML(rStm, pModel); bool bRet = Paste(*pModel, maDropPos, pPage, nPasteOptions); xShell->DoClose(); return bRet; } } // end of namespace sd /* vim:set shiftwidth=4 softtabstop=4 expandtab: */