/* -*- 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 "sddll.hxx" #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 "annotationmanager.hxx" #include "annotationmanagerimpl.hxx" #include "annotationwindow.hxx" #include "annotations.hrc" #include "Annotation.hxx" #include "ToolBarManager.hxx" #include "DrawDocShell.hxx" #include "DrawViewShell.hxx" #include "DrawController.hxx" #include "glob.hrc" #include "sdresid.hxx" #include "EventMultiplexer.hxx" #include "ViewShellManager.hxx" #include "helpids.h" #include "sdpage.hxx" #include "drawdoc.hxx" #include "textapi.hxx" #include "optsitem.hxx" #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::drawing; using namespace ::com::sun::star::document; using namespace ::com::sun::star::geometry; using namespace ::com::sun::star::container; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::text; using namespace ::com::sun::star::view; using namespace ::com::sun::star::style; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::ui; using namespace ::com::sun::star::task; using namespace ::com::sun::star::office; namespace { enum class CommentNotificationType { Add, Modify, Remove }; void lcl_CommentNotification(CommentNotificationType nType, const SfxViewShell* pViewShell, Reference& rxAnnotation) { // callbacks only if tiled annotations are explicitly turned off by LOK client if (!comphelper::LibreOfficeKit::isActive() || comphelper::LibreOfficeKit::isTiledAnnotations()) return; boost::property_tree::ptree aAnnotation; aAnnotation.put("action", (nType == CommentNotificationType::Add ? "Add" : (nType == CommentNotificationType::Remove ? "Remove" : (nType == CommentNotificationType::Modify ? "Modify" : "???")))); aAnnotation.put("id", sd::getAnnotationId(rxAnnotation)); if (nType != CommentNotificationType::Remove && rxAnnotation.is()) { aAnnotation.put("id", sd::getAnnotationId(rxAnnotation)); aAnnotation.put("author", rxAnnotation->getAuthor()); aAnnotation.put("dateTime", utl::toISO8601(rxAnnotation->getDateTime())); uno::Reference xText(rxAnnotation->getTextRange()); aAnnotation.put("text", xText->getString()); const SdPage* pPage = sd::getAnnotationPage(rxAnnotation); aAnnotation.put("parthash", pPage ? OString::number(pPage->GetHashCode()) : OString()); } boost::property_tree::ptree aTree; aTree.add_child("comment", aAnnotation); std::stringstream aStream; boost::property_tree::write_json(aStream, aTree); std::string aPayload = aStream.str(); pViewShell->libreOfficeKitViewCallback(LOK_CALLBACK_COMMENT, aPayload.c_str()); } } // anonymous ns namespace sd { SfxItemPool* GetAnnotationPool() { static SfxItemPool* s_pAnnotationPool = nullptr; if( s_pAnnotationPool == nullptr ) { s_pAnnotationPool = EditEngine::CreatePool( false ); s_pAnnotationPool->SetPoolDefaultItem(SvxFontHeightItem(423,100,EE_CHAR_FONTHEIGHT)); vcl::Font aAppFont( Application::GetSettings().GetStyleSettings().GetAppFont() ); s_pAnnotationPool->SetPoolDefaultItem(SvxFontItem(aAppFont.GetFamilyType(),aAppFont.GetFamilyName(),"",PITCH_DONTKNOW,RTL_TEXTENCODING_DONTKNOW,EE_CHAR_FONTINFO)); } return s_pAnnotationPool; } static SfxBindings* getBindings( ViewShellBase& rBase ) { if( rBase.GetMainViewShell().get() && rBase.GetMainViewShell()->GetViewFrame() ) return &rBase.GetMainViewShell()->GetViewFrame()->GetBindings(); return nullptr; } static SfxDispatcher* getDispatcher( ViewShellBase& rBase ) { if( rBase.GetMainViewShell().get() && rBase.GetMainViewShell()->GetViewFrame() ) return rBase.GetMainViewShell()->GetViewFrame()->GetDispatcher(); return nullptr; } css::util::DateTime getCurrentDateTime() { DateTime aCurrentDate( DateTime::SYSTEM ); return css::util::DateTime( 0, aCurrentDate.GetSec(), aCurrentDate.GetMin(), aCurrentDate.GetHour(), aCurrentDate.GetDay(), aCurrentDate.GetMonth(), aCurrentDate.GetYear(), false ); } OUString getAnnotationDateTimeString( const Reference< XAnnotation >& xAnnotation ) { OUString sRet; if( xAnnotation.is() ) { const SvtSysLocale aSysLocale; const LocaleDataWrapper& rLocalData = aSysLocale.GetLocaleData(); css::util::DateTime aDateTime( xAnnotation->getDateTime() ); Date aSysDate( Date::SYSTEM ); Date aDate( aDateTime.Day, aDateTime.Month, aDateTime.Year ); if (aDate==aSysDate) sRet = SdResId(STR_ANNOTATION_TODAY); else if (aDate == Date(aSysDate-1)) sRet = SdResId(STR_ANNOTATION_YESTERDAY); else if (aDate.IsValidAndGregorian() ) sRet = rLocalData.getDate(aDate); ::tools::Time aTime( aDateTime ); if(aTime.GetTime() != 0) sRet = sRet + " " + rLocalData.getTime( aTime,false ); } return sRet; } AnnotationManagerImpl::AnnotationManagerImpl( ViewShellBase& rViewShellBase ) : AnnotationManagerImplBase( m_aMutex ) , mrBase( rViewShellBase ) , mpDoc( rViewShellBase.GetDocument() ) , mbShowAnnotations( true ) , mbPopupMenuActive( false ) , mnUpdateTagsEvent( nullptr ) { SdOptions* pOptions = SD_MOD()->GetSdOptions(mpDoc->GetDocumentType()); if( pOptions ) mbShowAnnotations = pOptions->IsShowComments(); } void AnnotationManagerImpl::init() { // get current controller and initialize listeners try { addListener(); mxView.set(mrBase.GetController(), UNO_QUERY); } catch( Exception& ) { OSL_FAIL( "sd::AnnotationManagerImpl::AnnotationManagerImpl(), Exception caught!" ); } try { Reference xModel (mrBase.GetDocShell()->GetModel(), UNO_QUERY_THROW ); Reference xListener( this ); xModel->addEventListener( xListener ); } catch( Exception& ) { } } // WeakComponentImplHelper void SAL_CALL AnnotationManagerImpl::disposing () { try { Reference xModel (mrBase.GetDocShell()->GetModel(), UNO_QUERY_THROW ); Reference xListener( this ); xModel->removeEventListener( xListener ); } catch( Exception& ) { } removeListener(); DisposeTags(); if( mnUpdateTagsEvent ) { Application::RemoveUserEvent( mnUpdateTagsEvent ); mnUpdateTagsEvent = nullptr; } mxView.clear(); mxCurrentPage.clear(); } // XEventListener void SAL_CALL AnnotationManagerImpl::notifyEvent( const css::document::EventObject& aEvent ) { if( aEvent.EventName == "OnAnnotationInserted" || aEvent.EventName == "OnAnnotationRemoved" || aEvent.EventName == "OnAnnotationChanged" ) { // AnnotationInsertion and modification is not handled here because when // a new annotation is inserted, it consists of OnAnnotationInserted // followed by a chain of OnAnnotationChanged (called for setting each // of the annotation attributes - author, text etc.). This is not what a // LOK client wants. So only handle removal here as annotation removal // consists of only one event - 'OnAnnotationRemoved' if ( aEvent.EventName == "OnAnnotationRemoved" ) { Reference< XAnnotation > xAnnotation( aEvent.Source, uno::UNO_QUERY ); if ( xAnnotation.is() ) { lcl_CommentNotification(CommentNotificationType::Remove, &mrBase, xAnnotation); } } UpdateTags(); } } void SAL_CALL AnnotationManagerImpl::disposing( const css::lang::EventObject& /*Source*/ ) { } Reference AnnotationManagerImpl::GetAnnotationById(sal_uInt32 nAnnotationId) { SdPage* pPage = nullptr; do { pPage = GetNextPage(pPage, true); if( pPage && !pPage->getAnnotations().empty() ) { AnnotationVector aAnnotations(pPage->getAnnotations()); for( AnnotationVector::iterator iter = aAnnotations.begin(); iter != aAnnotations.end(); ++iter ) { Reference xAnnotation(*iter); if( sd::getAnnotationId(xAnnotation) == nAnnotationId ) { return xAnnotation; } } } } while( pPage ); Reference xAnnotationEmpty; return xAnnotationEmpty; } void AnnotationManagerImpl::ShowAnnotations( bool bShow ) { // enforce show annotations if a new annotation is inserted if( mbShowAnnotations != bShow ) { mbShowAnnotations = bShow; SdOptions* pOptions = SD_MOD()->GetSdOptions(mpDoc->GetDocumentType()); if( pOptions ) pOptions->SetShowComments( mbShowAnnotations ); UpdateTags(); } } void AnnotationManagerImpl::ExecuteAnnotation(SfxRequest& rReq ) { switch( rReq.GetSlot() ) { case SID_INSERT_POSTIT: ExecuteInsertAnnotation( rReq ); break; case SID_DELETE_POSTIT: case SID_DELETEALL_POSTIT: case SID_DELETEALLBYAUTHOR_POSTIT: ExecuteDeleteAnnotation( rReq ); break; case SID_EDIT_POSTIT: ExecuteEditAnnotation( rReq ); break; case SID_PREVIOUS_POSTIT: case SID_NEXT_POSTIT: SelectNextAnnotation( rReq.GetSlot() == SID_NEXT_POSTIT ); break; case SID_REPLYTO_POSTIT: ExecuteReplyToAnnotation( rReq ); break; case SID_TOGGLE_NOTES: ShowAnnotations( !mbShowAnnotations ); break; } } void AnnotationManagerImpl::ExecuteInsertAnnotation(SfxRequest& rReq) { if (!comphelper::LibreOfficeKit::isActive() || comphelper::LibreOfficeKit::isTiledAnnotations()) ShowAnnotations(true); const SfxItemSet* pArgs = rReq.GetArgs(); OUString sText; if (pArgs) { const SfxPoolItem* pPoolItem = nullptr; if (SfxItemState::SET == pArgs->GetItemState(SID_ATTR_POSTIT_TEXT, true, &pPoolItem)) { sText = static_cast(pPoolItem)->GetValue(); } } InsertAnnotation(sText); } void AnnotationManagerImpl::ExecuteDeleteAnnotation(SfxRequest& rReq) { ShowAnnotations( true ); const SfxItemSet* pArgs = rReq.GetArgs(); switch( rReq.GetSlot() ) { case SID_DELETEALL_POSTIT: DeleteAllAnnotations(); break; case SID_DELETEALLBYAUTHOR_POSTIT: if( pArgs ) { const SfxPoolItem* pPoolItem = nullptr; if( SfxItemState::SET == pArgs->GetItemState( SID_DELETEALLBYAUTHOR_POSTIT, true, &pPoolItem ) ) { OUString sAuthor( static_cast( pPoolItem )->GetValue() ); DeleteAnnotationsByAuthor( sAuthor ); } } break; case SID_DELETE_POSTIT: { Reference< XAnnotation > xAnnotation; sal_uInt32 nId = 0; if( pArgs ) { const SfxPoolItem* pPoolItem = nullptr; if( SfxItemState::SET == pArgs->GetItemState( SID_DELETE_POSTIT, true, &pPoolItem ) ) static_cast(pPoolItem)->GetValue() >>= xAnnotation; if( SfxItemState::SET == pArgs->GetItemState( SID_ATTR_POSTIT_ID, true, &pPoolItem ) ) nId = static_cast(pPoolItem)->GetValue().toUInt32(); } if (nId != 0) xAnnotation = GetAnnotationById(nId); else if( !xAnnotation.is() ) GetSelectedAnnotation( xAnnotation ); DeleteAnnotation( xAnnotation ); } break; } UpdateTags(); } void AnnotationManagerImpl::ExecuteEditAnnotation(SfxRequest& rReq) { const SfxItemSet* pArgs = rReq.GetArgs(); Reference< XAnnotation > xAnnotation; sal_uInt32 nId = 0; OUString sText; if (pArgs) { const SfxPoolItem* pPoolItem = nullptr; if (SfxItemState::SET == pArgs->GetItemState(SID_ATTR_POSTIT_ID, true, &pPoolItem)) { nId = static_cast(pPoolItem)->GetValue().toUInt32(); xAnnotation = GetAnnotationById(nId); } if (SfxItemState::SET == pArgs->GetItemState(SID_ATTR_POSTIT_TEXT, true, &pPoolItem)) sText = static_cast(pPoolItem)->GetValue(); if (xAnnotation.is() && !sText.isEmpty()) { // TODO: Not allow other authors to change others' comments ? Reference xText(xAnnotation->getTextRange()); xText->setString(sText); const SfxViewShell* pViewShell = SfxViewShell::GetFirst(); while (pViewShell) { lcl_CommentNotification(CommentNotificationType::Modify, pViewShell, xAnnotation); pViewShell = SfxViewShell::GetNext(*pViewShell); } } } UpdateTags(true); } void AnnotationManagerImpl::InsertAnnotation(const OUString& rText) { SdPage* pPage = GetCurrentPage(); if( pPage ) { if( mpDoc->IsUndoEnabled() ) mpDoc->BegUndo( SD_RESSTR( STR_ANNOTATION_UNDO_INSERT ) ); // find free space for new annotation int y = 0, x = 0; AnnotationVector aAnnotations( pPage->getAnnotations() ); if( !aAnnotations.empty() ) { const int page_width = pPage->GetSize().Width(); const int width = 1000; const int height = 800; ::tools::Rectangle aTagRect; while( true ) { ::tools::Rectangle aNewRect( x, y, x + width - 1, y + height - 1 ); bool bFree = true; for( AnnotationVector::iterator iter = aAnnotations.begin(); iter != aAnnotations.end(); ++iter ) { RealPoint2D aPoint( (*iter)->getPosition() ); aTagRect.Left() = sal::static_int_cast< long >( aPoint.X * 100.0 ); aTagRect.Top() = sal::static_int_cast< long >( aPoint.Y * 100.0 ); aTagRect.Right() = aTagRect.Left() + width - 1; aTagRect.Bottom() = aTagRect.Top() + height - 1; if( aNewRect.IsOver( aTagRect ) ) { bFree = false; break; } } if( !bFree ) { x += width; if( x > page_width ) { x = 0; y += height; } } else { break; } } } Reference< XAnnotation > xAnnotation; pPage->createAnnotation( xAnnotation ); OUString sAuthor; if (comphelper::LibreOfficeKit::isActive()) sAuthor = mrBase.GetMainViewShell()->GetView()->GetAuthor(); else { SvtUserOptions aUserOptions; sAuthor = aUserOptions.GetFullName(); xAnnotation->setInitials( aUserOptions.GetID() ); } if (!rText.isEmpty()) { Reference xText(xAnnotation->getTextRange()); xText->setString(rText); } // set current author to new annotation xAnnotation->setAuthor( sAuthor ); // set current time to new annotation xAnnotation->setDateTime( getCurrentDateTime() ); // set position RealPoint2D aPos( ((double)x) / 100.0, ((double)y) / 100.0 ); xAnnotation->setPosition( aPos ); if( mpDoc->IsUndoEnabled() ) mpDoc->EndUndo(); // Tell our LOK clients about new comment added const SfxViewShell* pViewShell = SfxViewShell::GetFirst(); while (pViewShell) { lcl_CommentNotification(CommentNotificationType::Add, pViewShell, xAnnotation); pViewShell = SfxViewShell::GetNext(*pViewShell); } UpdateTags(true); SelectAnnotation( xAnnotation, true ); } } void AnnotationManagerImpl::ExecuteReplyToAnnotation( SfxRequest& rReq ) { Reference< XAnnotation > xAnnotation; const SfxItemSet* pArgs = rReq.GetArgs(); sal_uInt32 nReplyId = 0; // Id of the comment to reply to OUString sReplyText; if( pArgs ) { const SfxPoolItem* pPoolItem = nullptr; if( SfxItemState::SET == pArgs->GetItemState( SID_ATTR_POSTIT_ID, true, &pPoolItem ) ) { nReplyId = static_cast(pPoolItem)->GetValue().toUInt32(); xAnnotation = GetAnnotationById(nReplyId); } else if( SfxItemState::SET == pArgs->GetItemState( rReq.GetSlot(), true, &pPoolItem ) ) static_cast( pPoolItem )->GetValue() >>= xAnnotation; if( SfxItemState::SET == pArgs->GetItemState( SID_ATTR_POSTIT_TEXT, true, &pPoolItem ) ) sReplyText = static_cast( pPoolItem )->GetValue(); } TextApiObject* pTextApi = getTextApiObject( xAnnotation ); if( pTextApi ) { std::unique_ptr< ::Outliner > pOutliner( new ::Outliner(GetAnnotationPool(),OutlinerMode::TextObject) ); SdDrawDocument::SetCalcFieldValueHdl( pOutliner.get() ); pOutliner->SetUpdateMode( true ); OUString aStr(SD_RESSTR(STR_ANNOTATION_REPLY)); OUString sAuthor( xAnnotation->getAuthor() ); if( sAuthor.isEmpty() ) sAuthor = SD_RESSTR( STR_ANNOTATION_NOAUTHOR ); aStr = aStr.replaceFirst("%1", sAuthor); aStr += " (" + getAnnotationDateTimeString( xAnnotation ) + "): \""; OUString sQuote( pTextApi->GetText() ); if( sQuote.isEmpty() ) sQuote = "..."; aStr += sQuote + "\"\n"; sal_Int32 nParaCount = comphelper::string::getTokenCount(aStr, '\n'); for( sal_Int32 nPara = 0; nPara < nParaCount; nPara++ ) pOutliner->Insert( aStr.getToken( nPara, '\n' ), EE_PARA_APPEND, -1 ); if( pOutliner->GetParagraphCount() > 1 ) { SfxItemSet aAnswerSet( pOutliner->GetEmptyItemSet() ); aAnswerSet.Put(SvxPostureItem(ITALIC_NORMAL,EE_CHAR_ITALIC)); ESelection aSel; aSel.nEndPara = pOutliner->GetParagraphCount()-2; aSel.nEndPos = pOutliner->GetText( pOutliner->GetParagraph( aSel.nEndPara ) ).getLength(); pOutliner->QuickSetAttribs( aAnswerSet, aSel ); } if (!sReplyText.isEmpty()) pOutliner->Insert(sReplyText); std::unique_ptr< OutlinerParaObject > pOPO( pOutliner->CreateParaObject() ); pTextApi->SetText( *pOPO.get() ); OUString sReplyAuthor; if (comphelper::LibreOfficeKit::isActive()) sReplyAuthor = mrBase.GetMainViewShell()->GetView()->GetAuthor(); else { SvtUserOptions aUserOptions; sReplyAuthor = aUserOptions.GetFullName(); xAnnotation->setInitials( aUserOptions.GetID() ); } xAnnotation->setAuthor( sReplyAuthor ); // set current time to reply xAnnotation->setDateTime( getCurrentDateTime() ); // Tell our LOK clients about this (comment modification) const SfxViewShell* pViewShell = SfxViewShell::GetFirst(); while (pViewShell) { lcl_CommentNotification(CommentNotificationType::Modify, pViewShell, xAnnotation); pViewShell = SfxViewShell::GetNext(*pViewShell); } UpdateTags(true); SelectAnnotation( xAnnotation, true ); } } void AnnotationManagerImpl::DeleteAnnotation( const Reference< XAnnotation >& xAnnotation ) { SdPage* pPage = GetCurrentPage(); if( xAnnotation.is() && pPage ) { if( mpDoc->IsUndoEnabled() ) mpDoc->BegUndo( SD_RESSTR( STR_ANNOTATION_UNDO_DELETE ) ); pPage->removeAnnotation( xAnnotation ); if( mpDoc->IsUndoEnabled() ) mpDoc->EndUndo(); UpdateTags(); } } void AnnotationManagerImpl::DeleteAnnotationsByAuthor( const OUString& sAuthor ) { if( mpDoc->IsUndoEnabled() ) mpDoc->BegUndo( SD_RESSTR( STR_ANNOTATION_UNDO_DELETE ) ); SdPage* pPage = nullptr; do { pPage = GetNextPage( pPage, true ); if( pPage && !pPage->getAnnotations().empty() ) { AnnotationVector aAnnotations( pPage->getAnnotations() ); for( AnnotationVector::iterator iter = aAnnotations.begin(); iter != aAnnotations.end(); ++iter ) { Reference< XAnnotation > xAnnotation( *iter ); if( xAnnotation->getAuthor() == sAuthor ) { if( mxSelectedAnnotation == xAnnotation ) mxSelectedAnnotation.clear(); pPage->removeAnnotation( xAnnotation ); } } } } while( pPage ); if( mpDoc->IsUndoEnabled() ) mpDoc->EndUndo(); } void AnnotationManagerImpl::DeleteAllAnnotations() { if( mpDoc->IsUndoEnabled() ) mpDoc->BegUndo( SD_RESSTR( STR_ANNOTATION_UNDO_DELETE ) ); SdPage* pPage = nullptr; do { pPage = GetNextPage( pPage, true ); if( pPage && !pPage->getAnnotations().empty() ) { AnnotationVector aAnnotations( pPage->getAnnotations() ); for( AnnotationVector::iterator iter = aAnnotations.begin(); iter != aAnnotations.end(); ++iter ) { pPage->removeAnnotation( (*iter) ); } } } while( pPage ); mxSelectedAnnotation.clear(); if( mpDoc->IsUndoEnabled() ) mpDoc->EndUndo(); } void AnnotationManagerImpl::GetAnnotationState(SfxItemSet& rSet) { SdPage* pCurrentPage = GetCurrentPage(); const bool bReadOnly = mrBase.GetDocShell()->IsReadOnly(); const bool bWrongPageKind = (pCurrentPage == nullptr) || (pCurrentPage->GetPageKind() != PageKind::Standard); const SvtSaveOptions::ODFDefaultVersion nCurrentODFVersion( SvtSaveOptions().GetODFDefaultVersion() ); if( bReadOnly || bWrongPageKind || (nCurrentODFVersion <= SvtSaveOptions::ODFVER_012) ) rSet.DisableItem( SID_INSERT_POSTIT ); rSet.Put(SfxBoolItem(SID_TOGGLE_NOTES, mbShowAnnotations)); Reference< XAnnotation > xAnnotation; GetSelectedAnnotation( xAnnotation ); // Don't disable these slot in case of LOK, as postit doesn't need to // selected before doing an operation on it in LOK if( (!xAnnotation.is() && !comphelper::LibreOfficeKit::isActive()) || bReadOnly ) { rSet.DisableItem( SID_DELETE_POSTIT ); rSet.DisableItem( SID_EDIT_POSTIT ); } SdPage* pPage = nullptr; bool bHasAnnotations = false; do { pPage = GetNextPage( pPage, true ); if( pPage && !pPage->getAnnotations().empty() ) bHasAnnotations = true; } while( pPage && !bHasAnnotations ); if( !bHasAnnotations || bReadOnly ) { rSet.DisableItem( SID_DELETEALL_POSTIT ); } if( bWrongPageKind || !bHasAnnotations ) { rSet.DisableItem( SID_PREVIOUS_POSTIT ); rSet.DisableItem( SID_NEXT_POSTIT ); } } void AnnotationManagerImpl::SelectNextAnnotation(bool bForeward) { ShowAnnotations( true ); Reference< XAnnotation > xCurrent; GetSelectedAnnotation( xCurrent ); SdPage* pPage = GetCurrentPage(); if( !pPage ) return; AnnotationVector aAnnotations( pPage->getAnnotations() ); if( bForeward ) { if( xCurrent.is() ) { for( AnnotationVector::iterator iter = aAnnotations.begin(); iter != aAnnotations.end(); ++iter ) { if( (*iter) == xCurrent ) { ++iter; if( iter != aAnnotations.end() ) { SelectAnnotation( (*iter) ); return; } break; } } } else if( !aAnnotations.empty() ) { SelectAnnotation( *(aAnnotations.begin()) ); return; } } else { if( xCurrent.is() ) { for( AnnotationVector::iterator iter = aAnnotations.begin(); iter != aAnnotations.end(); ++iter ) { if( (*iter) == xCurrent ) { if( iter != aAnnotations.begin() ) { --iter; SelectAnnotation( (*iter) ); return; } break; } } } else if( !aAnnotations.empty() ) { AnnotationVector::iterator iter( aAnnotations.end() ); SelectAnnotation( *(--iter) ); return; } } mxSelectedAnnotation.clear(); do { do { pPage = GetNextPage( pPage, bForeward ); if( pPage && !pPage->getAnnotations().empty() ) { // switch to next/previous slide with annotations std::shared_ptr pDrawViewShell(std::dynamic_pointer_cast(mrBase.GetMainViewShell())); if (pDrawViewShell.get() != nullptr) { pDrawViewShell->ChangeEditMode(pPage->IsMasterPage() ? EditMode::MasterPage : EditMode::Page, false); pDrawViewShell->SwitchPage((pPage->GetPageNum() - 1) >> 1); SfxDispatcher* pDispatcher = getDispatcher( mrBase ); if( pDispatcher ) pDispatcher->Execute( bForeward ? SID_NEXT_POSTIT : SID_PREVIOUS_POSTIT ); return; } } } while( pPage ); // The question text depends on the search direction. bool bImpress = mpDoc->GetDocumentType() == DocumentType::Impress; sal_uInt16 nStringId; if(bForeward) nStringId = bImpress ? STR_ANNOTATION_WRAP_FORWARD : STR_ANNOTATION_WRAP_FORWARD_DRAW; else nStringId = bImpress ? STR_ANNOTATION_WRAP_BACKWARD : STR_ANNOTATION_WRAP_BACKWARD_DRAW; // Pop up question box that asks the user whether to wrap around. // The dialog is made modal with respect to the whole application. ScopedVclPtrInstance< QueryBox > aQuestionBox( nullptr, (WB_YES_NO | WB_DEF_YES), SD_RESSTR(nStringId)); aQuestionBox->SetImage( QueryBox::GetStandardImage() ); if (aQuestionBox->Execute() != RET_YES) break; } while( true ); } void AnnotationManagerImpl::onTagSelected( AnnotationTag& rTag ) { mxSelectedAnnotation = rTag.GetAnnotation(); invalidateSlots(); } void AnnotationManagerImpl::onTagDeselected( AnnotationTag& rTag ) { if( rTag.GetAnnotation() == mxSelectedAnnotation ) { mxSelectedAnnotation.clear(); invalidateSlots(); } } void AnnotationManagerImpl::SelectAnnotation( const css::uno::Reference< css::office::XAnnotation >& xAnnotation, bool bEdit /* = sal_False */ ) { mxSelectedAnnotation = xAnnotation; const AnnotationTagVector::const_iterator aEnd( maTagVector.end() ); for( AnnotationTagVector::const_iterator iter( maTagVector.begin() ); iter != aEnd; ++iter) { if( (*iter)->GetAnnotation() == xAnnotation ) { SmartTagReference xTag( (*iter).get() ); mrBase.GetMainViewShell()->GetView()->getSmartTags().select( xTag ); (*iter)->OpenPopup( bEdit ); break; } } } void AnnotationManagerImpl::GetSelectedAnnotation( css::uno::Reference< css::office::XAnnotation >& xAnnotation ) { xAnnotation = mxSelectedAnnotation; } void AnnotationManagerImpl::invalidateSlots() { SfxBindings* pBindings = getBindings( mrBase ); if( pBindings ) { pBindings->Invalidate( SID_INSERT_POSTIT ); pBindings->Invalidate( SID_DELETE_POSTIT ); pBindings->Invalidate( SID_DELETEALL_POSTIT ); pBindings->Invalidate( SID_PREVIOUS_POSTIT ); pBindings->Invalidate( SID_NEXT_POSTIT ); pBindings->Invalidate( SID_UNDO ); pBindings->Invalidate( SID_REDO ); } } void AnnotationManagerImpl::onSelectionChanged() { if( mxView.is() && mrBase.GetDrawView() ) try { Reference< XAnnotationAccess > xPage( mxView->getCurrentPage(), UNO_QUERY ); if( xPage != mxCurrentPage ) { mxCurrentPage = xPage; UpdateTags(true); } } catch( Exception& ) { OSL_FAIL( "sd::AnnotationManagerImpl::onSelectionChanged(), exception caught!" ); } } void AnnotationManagerImpl::UpdateTags( bool bSynchron ) { if( bSynchron ) { if( mnUpdateTagsEvent ) Application::RemoveUserEvent( mnUpdateTagsEvent ); UpdateTagsHdl(nullptr); } else { if( !mnUpdateTagsEvent && mxView.is() ) mnUpdateTagsEvent = Application::PostUserEvent( LINK( this, AnnotationManagerImpl, UpdateTagsHdl ) ); } } IMPL_LINK_NOARG(AnnotationManagerImpl, UpdateTagsHdl, void*, void) { mnUpdateTagsEvent = nullptr; DisposeTags(); if( mbShowAnnotations ) CreateTags(); if( mrBase.GetDrawView() ) static_cast< ::sd::View* >( mrBase.GetDrawView() )->updateHandles(); invalidateSlots(); } void AnnotationManagerImpl::CreateTags() { if( mxCurrentPage.is() && mpDoc ) try { int nIndex = 1; maFont = Application::GetSettings().GetStyleSettings().GetAppFont(); rtl::Reference< AnnotationTag > xSelectedTag; Reference< XAnnotationEnumeration > xEnum( mxCurrentPage->createAnnotationEnumeration() ); while( xEnum->hasMoreElements() ) { Reference< XAnnotation > xAnnotation( xEnum->nextElement() ); Color aColor( GetColorLight( mpDoc->GetAnnotationAuthorIndex( xAnnotation->getAuthor() ) ) ); rtl::Reference< AnnotationTag > xTag( new AnnotationTag( *this, *mrBase.GetMainViewShell()->GetView(), xAnnotation, aColor, nIndex++, maFont ) ); maTagVector.push_back(xTag); if( xAnnotation == mxSelectedAnnotation ) { xSelectedTag = xTag; } } if( xSelectedTag.is() ) { SmartTagReference xTag( xSelectedTag.get() ); mrBase.GetMainViewShell()->GetView()->getSmartTags().select( xTag ); } else { // no tag, no selection! mxSelectedAnnotation.clear(); } } catch( Exception& ) { OSL_FAIL( "sd::AnnotationManagerImpl::onSelectionChanged(), exception caught!" ); } } void AnnotationManagerImpl::DisposeTags() { if( !maTagVector.empty() ) { AnnotationTagVector::iterator iter = maTagVector.begin(); do { (*iter++)->Dispose(); } while( iter != maTagVector.end() ); maTagVector.clear(); } } void AnnotationManagerImpl::addListener() { Link aLink( LINK(this,AnnotationManagerImpl,EventMultiplexerListener) ); mrBase.GetEventMultiplexer()->AddEventListener(aLink); } void AnnotationManagerImpl::removeListener() { Link aLink( LINK(this,AnnotationManagerImpl,EventMultiplexerListener) ); mrBase.GetEventMultiplexer()->RemoveEventListener( aLink ); } IMPL_LINK(AnnotationManagerImpl,EventMultiplexerListener, tools::EventMultiplexerEvent&, rEvent, void) { switch (rEvent.meEventId) { case EventMultiplexerEventId::CurrentPageChanged: case EventMultiplexerEventId::EditViewSelection: onSelectionChanged(); break; case EventMultiplexerEventId::MainViewRemoved: mxView.clear(); onSelectionChanged(); break; case EventMultiplexerEventId::MainViewAdded: mxView.set( mrBase.GetController(), UNO_QUERY ); onSelectionChanged(); break; default: break; } } namespace { sal_uInt16 IdentToSID(const OString& rIdent) { if (rIdent == "reply") return SID_REPLYTO_POSTIT; else if (rIdent == "delete") return SID_DELETE_POSTIT; else if (rIdent == "deleteby") return SID_DELETEALLBYAUTHOR_POSTIT; else if (rIdent == "deleteall") return SID_DELETEALL_POSTIT; else if (rIdent == "copy") return SID_COPY; else if (rIdent == "paste") return SID_PASTE; else if (rIdent == "bold") return SID_ATTR_CHAR_WEIGHT; else if (rIdent == "italic") return SID_ATTR_CHAR_POSTURE; else if (rIdent == "underline") return SID_ATTR_CHAR_UNDERLINE; else if (rIdent == "strike") return SID_ATTR_CHAR_STRIKEOUT; return 0; } } void AnnotationManagerImpl::ExecuteAnnotationContextMenu( const Reference< XAnnotation >& xAnnotation, vcl::Window* pParent, const ::tools::Rectangle& rContextRect, bool bButtonMenu /* = false */ ) { SfxDispatcher* pDispatcher( getDispatcher( mrBase ) ); if( !pDispatcher ) return; const bool bReadOnly = mrBase.GetDocShell()->IsReadOnly(); AnnotationWindow* pAnnotationWindow = bButtonMenu ? nullptr : dynamic_cast< AnnotationWindow* >( pParent ); if( bReadOnly && !pAnnotationWindow ) return; OUString sUIFile; if (pAnnotationWindow) sUIFile = "modules/simpress/ui/annotationmenu.ui"; else sUIFile = "modules/simpress/ui/annotationtagmenu.ui"; VclBuilder aBuilder(nullptr, VclBuilderContainer::getUIRootDir(), sUIFile, ""); VclPtr pMenu(aBuilder.get_menu("menu")); SvtUserOptions aUserOptions; OUString sCurrentAuthor( aUserOptions.GetFullName() ); OUString sAuthor( xAnnotation->getAuthor() ); OUString aStr(pMenu->GetItemText(pMenu->GetItemId("deleteby"))); OUString aReplace( sAuthor ); if( aReplace.isEmpty() ) aReplace = SD_RESSTR( STR_ANNOTATION_NOAUTHOR ); aStr = aStr.replaceFirst("%1", aReplace); pMenu->SetItemText(pMenu->GetItemId("deleteby"), aStr); pMenu->EnableItem(pMenu->GetItemId("reply"), (sAuthor != sCurrentAuthor) && !bReadOnly); pMenu->EnableItem(pMenu->GetItemId("delete"), xAnnotation.is() && !bReadOnly); pMenu->EnableItem(pMenu->GetItemId("deleteby"), !bReadOnly); pMenu->EnableItem(pMenu->GetItemId("deleteall"), !bReadOnly); if( pAnnotationWindow ) { if( pAnnotationWindow->IsProtected() || bReadOnly ) { pMenu->EnableItem(pMenu->GetItemId("bold"), false); pMenu->EnableItem(pMenu->GetItemId("italic"), false); pMenu->EnableItem(pMenu->GetItemId("underline"), false); pMenu->EnableItem(pMenu->GetItemId("strike"), false); pMenu->EnableItem(pMenu->GetItemId("paste"), false); } else { SfxItemSet aSet(pAnnotationWindow->getView()->GetAttribs()); if ( aSet.GetItemState( EE_CHAR_WEIGHT ) == SfxItemState::SET ) { if( static_cast(aSet.Get( EE_CHAR_WEIGHT )).GetWeight() == WEIGHT_BOLD ) pMenu->CheckItem(pMenu->GetItemId("bold")); } if ( aSet.GetItemState( EE_CHAR_ITALIC ) == SfxItemState::SET ) { if( static_cast(aSet.Get( EE_CHAR_ITALIC )).GetPosture() != ITALIC_NONE ) pMenu->CheckItem(pMenu->GetItemId("italic")); } if ( aSet.GetItemState( EE_CHAR_UNDERLINE ) == SfxItemState::SET ) { if( static_cast(aSet.Get( EE_CHAR_UNDERLINE )).GetLineStyle() != LINESTYLE_NONE ) pMenu->CheckItem(pMenu->GetItemId("underline")); } if ( aSet.GetItemState( EE_CHAR_STRIKEOUT ) == SfxItemState::SET ) { if( static_cast(aSet.Get( EE_CHAR_STRIKEOUT )).GetStrikeout() != STRIKEOUT_NONE ) pMenu->CheckItem(pMenu->GetItemId("strike")); } TransferableDataHelper aDataHelper( TransferableDataHelper::CreateFromSystemClipboard( pAnnotationWindow ) ); pMenu->EnableItem(pMenu->GetItemId("paste"), aDataHelper.GetFormatCount() != 0); } pMenu->EnableItem(pMenu->GetItemId("copy"), pAnnotationWindow->getView()->HasSelection()); } // set slot images Reference< css::frame::XFrame > xFrame( mrBase.GetMainViewShell()->GetViewFrame()->GetFrame().GetFrameInterface() ); if( xFrame.is() ) { for( sal_uInt16 nPos = 0; nPos < pMenu->GetItemCount(); nPos++ ) { sal_uInt16 nId = pMenu->GetItemId( nPos ); if (!pMenu->IsItemEnabled(nId)) continue; Image aImage( vcl::CommandInfoProvider::GetImageForCommand( pMenu->GetItemCommand( nId ), xFrame ) ); if( !!aImage ) pMenu->SetItemImage( nId, aImage ); } } // tdf#99388 and tdf#99712 make known that PopupMenu is active at parent to // allow suppressing closing of that window if needed setPopupMenuActive(true); sal_uInt16 nId = pMenu->Execute( pParent, rContextRect, PopupMenuFlags::ExecuteDown|PopupMenuFlags::NoMouseUpClose ); nId = IdentToSID(pMenu->GetItemIdent(nId)); // tdf#99388 and tdf#99712 reset flag, need to be done before reacting // since closing it is one possible reaction setPopupMenuActive(false); switch( nId ) { case SID_REPLYTO_POSTIT: { const SfxUnoAnyItem aItem( SID_REPLYTO_POSTIT, Any( xAnnotation ) ); pDispatcher->ExecuteList(SID_REPLYTO_POSTIT, SfxCallMode::ASYNCHRON, { &aItem }); break; } case SID_DELETE_POSTIT: { const SfxUnoAnyItem aItem( SID_DELETE_POSTIT, Any( xAnnotation ) ); pDispatcher->ExecuteList(SID_DELETE_POSTIT, SfxCallMode::ASYNCHRON, { &aItem }); break; } case SID_DELETEALLBYAUTHOR_POSTIT: { const SfxStringItem aItem( SID_DELETEALLBYAUTHOR_POSTIT, sAuthor ); pDispatcher->ExecuteList( SID_DELETEALLBYAUTHOR_POSTIT, SfxCallMode::ASYNCHRON, { &aItem }); break; } case SID_DELETEALL_POSTIT: pDispatcher->Execute( SID_DELETEALL_POSTIT ); break; case SID_COPY: case SID_PASTE: case SID_ATTR_CHAR_WEIGHT: case SID_ATTR_CHAR_POSTURE: case SID_ATTR_CHAR_UNDERLINE: case SID_ATTR_CHAR_STRIKEOUT: if( pAnnotationWindow ) pAnnotationWindow->ExecuteSlot( nId ); break; } } Color AnnotationManagerImpl::GetColor(sal_uInt16 aAuthorIndex) { if (!Application::GetSettings().GetStyleSettings().GetHighContrastMode()) { static const Color aArrayNormal[] = { COL_AUTHOR1_NORMAL, COL_AUTHOR2_NORMAL, COL_AUTHOR3_NORMAL, COL_AUTHOR4_NORMAL, COL_AUTHOR5_NORMAL, COL_AUTHOR6_NORMAL, COL_AUTHOR7_NORMAL, COL_AUTHOR8_NORMAL, COL_AUTHOR9_NORMAL }; return Color( aArrayNormal[ aAuthorIndex % SAL_N_ELEMENTS( aArrayNormal ) ] ); } return Color(COL_WHITE); } Color AnnotationManagerImpl::GetColorLight(sal_uInt16 aAuthorIndex) { if (!Application::GetSettings().GetStyleSettings().GetHighContrastMode()) { static const Color aArrayLight[] = { COL_AUTHOR1_LIGHT, COL_AUTHOR2_LIGHT, COL_AUTHOR3_LIGHT, COL_AUTHOR4_LIGHT, COL_AUTHOR5_LIGHT, COL_AUTHOR6_LIGHT, COL_AUTHOR7_LIGHT, COL_AUTHOR8_LIGHT, COL_AUTHOR9_LIGHT }; return Color( aArrayLight[ aAuthorIndex % SAL_N_ELEMENTS( aArrayLight ) ] ); } return Color(COL_WHITE); } Color AnnotationManagerImpl::GetColorDark(sal_uInt16 aAuthorIndex) { if (!Application::GetSettings().GetStyleSettings().GetHighContrastMode()) { static const Color aArrayAnkor[] = { COL_AUTHOR1_DARK, COL_AUTHOR2_DARK, COL_AUTHOR3_DARK, COL_AUTHOR4_DARK, COL_AUTHOR5_DARK, COL_AUTHOR6_DARK, COL_AUTHOR7_DARK, COL_AUTHOR8_DARK, COL_AUTHOR9_DARK }; return Color( aArrayAnkor[ aAuthorIndex % SAL_N_ELEMENTS( aArrayAnkor ) ] ); } return Color(COL_WHITE); } SdPage* AnnotationManagerImpl::GetNextPage( SdPage* pPage, bool bForward ) { if( pPage == nullptr ) { if (bForward) return mpDoc->GetSdPage(0, PageKind::Standard ); // first page else return mpDoc->GetMasterSdPage( mpDoc->GetMasterSdPageCount(PageKind::Standard) - 1, PageKind::Standard ); // last page } sal_uInt16 nPageNum = (pPage->GetPageNum() - 1) >> 1; // first all non master pages if( !pPage->IsMasterPage() ) { if( bForward ) { if( nPageNum >= mpDoc->GetSdPageCount(PageKind::Standard)-1 ) { // we reached end of draw pages, start with master pages (skip handout master for draw) return mpDoc->GetMasterSdPage( (mpDoc->GetDocumentType() == DocumentType::Impress) ? 0 : 1, PageKind::Standard ); } nPageNum++; } else { if( nPageNum == 0 ) return nullptr; // we are already on the first draw page, finished nPageNum--; } return mpDoc->GetSdPage(nPageNum, PageKind::Standard); } else { if( bForward ) { if( nPageNum >= mpDoc->GetMasterSdPageCount(PageKind::Standard)-1 ) { return nullptr; // we reached the end, there is nothing more to see here } nPageNum++; } else { if( nPageNum == (mpDoc->GetDocumentType() == DocumentType::Impress ? 0 : 1) ) { // we reached beginning of master pages, start with end if pages return mpDoc->GetSdPage( mpDoc->GetSdPageCount(PageKind::Standard)-1, PageKind::Standard ); } nPageNum--; } return mpDoc->GetMasterSdPage(nPageNum,PageKind::Standard); } } SdPage* AnnotationManagerImpl::GetCurrentPage() { if (mrBase.GetMainViewShell().get()) return mrBase.GetMainViewShell()->getCurrentPage(); return nullptr; } AnnotationManager::AnnotationManager( ViewShellBase& rViewShellBase ) : mxImpl( new AnnotationManagerImpl( rViewShellBase ) ) { mxImpl->init(); } AnnotationManager::~AnnotationManager() { mxImpl->dispose(); } void AnnotationManager::ExecuteAnnotation(SfxRequest& rRequest) { mxImpl->ExecuteAnnotation( rRequest ); } void AnnotationManager::GetAnnotationState(SfxItemSet& rItemSet) { mxImpl->GetAnnotationState(rItemSet); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */