/* -*- 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/. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define MNI_OPEN 1 #define MNI_EDIT 2 //#define MNI_DEFAULT_TEMPLATE 3 #define MNI_DELETE 4 #define MNI_RENAME 5 using namespace ::com::sun::star; bool ViewFilter_Application::isFilteredExtension(FILTER_APPLICATION filter, const OUString &rExt) { bool bRet = rExt == "ott" || rExt == "stw" || rExt == "oth" || rExt == "dot" || rExt == "dotx" || rExt == "otm" || rExt == "ots" || rExt == "stc" || rExt == "xlt" || rExt == "xltm" || rExt == "xltx" || rExt == "otp" || rExt == "sti" || rExt == "pot" || rExt == "potm" || rExt == "potx" || rExt == "otg" || rExt == "std"; if (filter == FILTER_APPLICATION::WRITER) { bRet = rExt == "ott" || rExt == "stw" || rExt == "oth" || rExt == "dot" || rExt == "dotx" || rExt == "otm"; } else if (filter == FILTER_APPLICATION::CALC) { bRet = rExt == "ots" || rExt == "stc" || rExt == "xlt" || rExt == "xltm" || rExt == "xltx"; } else if (filter == FILTER_APPLICATION::IMPRESS) { bRet = rExt == "otp" || rExt == "sti" || rExt == "pot" || rExt == "potm" || rExt == "potx"; } else if (filter == FILTER_APPLICATION::DRAW) { bRet = rExt == "otg" || rExt == "std"; } return bRet; } bool ViewFilter_Application::isValid (const OUString &rPath) const { INetURLObject aUrl(rPath); return isFilteredExtension(mApp, aUrl.getExtension()); } bool ViewFilter_Application::operator () (const ThumbnailViewItem *pItem) { const TemplateViewItem *pTempItem = dynamic_cast(pItem); if (pTempItem) return isValid(pTempItem->getPath()); return true; } TemplateLocalView::TemplateLocalView ( vcl::Window* pParent) : ThumbnailView(pParent, WB_TABSTOP), mnCurRegionId(0), maSelectedItem(nullptr), mnThumbnailWidth(TEMPLATE_THUMBNAIL_MAX_WIDTH), mnThumbnailHeight(TEMPLATE_THUMBNAIL_MAX_HEIGHT), maPosition(0,0), mpDocTemplates(new SfxDocumentTemplates) { } TemplateLocalView::~TemplateLocalView() { disposeOnce(); } void TemplateLocalView::dispose() { maRegions.clear(); maAllTemplates.clear(); mpDocTemplates.reset(); ThumbnailView::dispose(); } void TemplateLocalView::Populate () { maRegions.clear(); maAllTemplates.clear(); sal_uInt16 nCount = mpDocTemplates->GetRegionCount(); for (sal_uInt16 i = 0; i < nCount; ++i) { OUString aRegionName(mpDocTemplates->GetFullRegionName(i)); std::unique_ptr pItem(new TemplateContainerItem( i+1 )); pItem->mnRegionId = i; pItem->maTitle = aRegionName; sal_uInt16 nEntries = mpDocTemplates->GetCount(i); for (sal_uInt16 j = 0; j < nEntries; ++j) { OUString aName = mpDocTemplates->GetName(i,j); OUString aURL = mpDocTemplates->GetPath(i,j); TemplateItemProperties aProperties; aProperties.nId = j+1; aProperties.nDocId = j; aProperties.nRegionId = i; aProperties.aName = aName; aProperties.aPath = aURL; aProperties.aRegionName = aRegionName; aProperties.aThumbnail = TemplateLocalView::fetchThumbnail(aURL, mnThumbnailWidth, mnThumbnailHeight); pItem->maTemplates.push_back(aProperties); maAllTemplates.push_back(aProperties); } maRegions.push_back(std::move(pItem)); } } void TemplateLocalView::reload () { mpDocTemplates->Update(); Populate(); // Check if we are currently browsing a region or root folder if (mnCurRegionId) { sal_uInt16 nRegionId = mnCurRegionId - 1; //Is offset by 1 for (auto const & pRegion : maRegions) { if (pRegion->mnRegionId == nRegionId) { showRegion(pRegion.get()); break; } } } else showAllTemplates(); //No items should be selected by default deselectItems(); } void TemplateLocalView::showAllTemplates() { mnCurRegionId = 0; insertItems(maAllTemplates, false, true); } void TemplateLocalView::showRegion(TemplateContainerItem const *pItem) { mnCurRegionId = pItem->mnRegionId+1; insertItems(pItem->maTemplates); } IMPL_LINK(TemplateLocalView, ContextMenuSelectHdl, Menu*, pMenu, bool) { sal_uInt16 nMenuId = pMenu->GetCurItemId(); switch(nMenuId) { case MNI_OPEN: maOpenTemplateHdl.Call(maSelectedItem); break; case MNI_EDIT: maEditTemplateHdl.Call(maSelectedItem); break; case MNI_RENAME: { InputDialog aTitleEditDlg(GetFrameWeld(), SfxResId(STR_RENAME_TEMPLATE)); OUString sOldTitle = maSelectedItem->getTitle(); aTitleEditDlg.SetEntryText(sOldTitle); aTitleEditDlg.HideHelpBtn(); if (!aTitleEditDlg.run()) break; OUString sNewTitle = comphelper::string::strip(aTitleEditDlg.GetEntryText(), ' '); if ( !sNewTitle.isEmpty() && sNewTitle != sOldTitle ) { maSelectedItem->setTitle(sNewTitle); } } break; case MNI_DELETE: { std::unique_ptr xQueryDlg(Application::CreateMessageDialog(GetFrameWeld(), VclMessageType::Question, VclButtonsType::YesNo, SfxResId(STR_QMSG_SEL_TEMPLATE_DELETE))); if (xQueryDlg->run() != RET_YES) break; reload(); } break; default: break; } return false; } bool TemplateLocalView::renameItem(ThumbnailViewItem* pItem, const OUString& sNewTitle) { sal_uInt16 nRegionId = 0; sal_uInt16 nDocId = USHRT_MAX; TemplateViewItem* pDocItem = dynamic_cast( pItem ); if ( pDocItem ) { nRegionId = pDocItem->mnRegionId; nDocId = pDocItem->mnDocId; } return mpDocTemplates->SetName( sNewTitle, nRegionId, nDocId ); } void TemplateLocalView::insertItems(const std::vector &rTemplates, bool isRegionSelected, bool bShowCategoryInTooltip) { std::vector> aItems(rTemplates.size()); for (size_t i = 0, n = rTemplates.size(); i < n; ++i ) { const TemplateItemProperties *pCur = &rTemplates[i]; std::unique_ptr pChild; if(isRegionSelected) pChild.reset(new TemplateViewItem(*this, pCur->nId)); else pChild.reset(new TemplateViewItem(*this, i+1)); pChild->mnDocId = pCur->nDocId; pChild->mnRegionId = pCur->nRegionId; pChild->maTitle = pCur->aName; pChild->setPath(pCur->aPath); if(!bShowCategoryInTooltip) pChild->setHelpText(pCur->aName); else { OUString sHelpText = SfxResId(STR_TEMPLATE_TOOLTIP); sHelpText = (sHelpText.replaceFirst("$1", pCur->aName)).replaceFirst("$2", pCur->aRegionName); pChild->setHelpText(sHelpText); } pChild->maPreview1 = pCur->aThumbnail; if(IsDefaultTemplate(pCur->aPath)) pChild->showDefaultIcon(true); if ( pCur->aThumbnail.IsEmpty() ) { // Use the default thumbnail if we have nothing else pChild->maPreview1 = TemplateLocalView::getDefaultThumbnail(pCur->aPath); } aItems[i] = std::move(pChild); } updateItems(std::move(aItems)); } void TemplateLocalView::updateThumbnailDimensions(long itemMaxSize) { mnThumbnailWidth = itemMaxSize; mnThumbnailHeight = itemMaxSize; } void TemplateLocalView::MouseButtonDown( const MouseEvent& rMEvt ) { GrabFocus(); ThumbnailView::MouseButtonDown(rMEvt); } void TemplateLocalView::RequestHelp( const HelpEvent& rHEvt ) { if ( rHEvt.GetMode() & HelpEventMode::QUICK ) { tools::Rectangle aRect( OutputToScreenPixel( GetPosPixel() ), GetSizePixel() ); Help::ShowQuickHelp( this, aRect, GetQuickHelpText(), QuickHelpFlags::CtrlText | QuickHelpFlags::TipStyleBalloon ); return; } ThumbnailView::RequestHelp( rHEvt ); } void TemplateLocalView::Command( const CommandEvent& rCEvt ) { if ( rCEvt.GetCommand() == CommandEventId::ContextMenu ) { if(rCEvt.IsMouseEvent()) { deselectItems(); size_t nPos = ImplGetItem(rCEvt.GetMousePosPixel()); Point aPosition (rCEvt.GetMousePosPixel()); maPosition = aPosition; ThumbnailViewItem* pItem = ImplGetItem(nPos); const TemplateViewItem *pViewItem = dynamic_cast(pItem); if(pViewItem) { maSelectedItem = dynamic_cast(pItem); maCreateContextMenuHdl.Call(pItem); } } else { for (ThumbnailViewItem* pItem : mFilteredItemList) { //create context menu for the first selected item if (pItem->isSelected()) { deselectItems(); pItem->setSelection(true); tools::Rectangle aRect = pItem->getDrawArea(); maPosition = aRect.Center(); maSelectedItem = dynamic_cast(pItem); maCreateContextMenuHdl.Call(pItem); break; } } } } ThumbnailView::Command(rCEvt); } void TemplateLocalView::KeyInput( const KeyEvent& rKEvt ) { vcl::KeyCode aKeyCode = rKEvt.GetKeyCode(); if(aKeyCode == ( KEY_MOD1 | KEY_A ) ) { for (ThumbnailViewItem* pItem : mFilteredItemList) { if (!pItem->isSelected()) { pItem->setSelection(true); } } if (IsReallyVisible() && IsUpdateMode()) Invalidate(); return; } else if( aKeyCode == KEY_DELETE && !mFilteredItemList.empty()) { std::unique_ptr xQueryDlg(Application::CreateMessageDialog(GetFrameWeld(), VclMessageType::Question, VclButtonsType::YesNo, SfxResId(STR_QMSG_SEL_TEMPLATE_DELETE))); if (xQueryDlg->run() != RET_YES) return; reload(); } ThumbnailView::KeyInput(rKEvt); } void TemplateLocalView::setCreateContextMenuHdl(const Link &rLink) { maCreateContextMenuHdl = rLink; } void TemplateLocalView::setOpenTemplateHdl(const Link &rLink) { maOpenTemplateHdl = rLink; } void TemplateLocalView::setEditTemplateHdl(const Link &rLink) { maEditTemplateHdl = rLink; } BitmapEx TemplateLocalView::scaleImg (const BitmapEx &rImg, long width, long height) { BitmapEx aImg = rImg; if (!rImg.IsEmpty()) { Size aSize = rImg.GetSizePixel(); if (aSize.Width() == 0) aSize.setWidth( 1 ); if (aSize.Height() == 0) aSize.setHeight( 1 ); // make the picture fit the given width/height constraints double nRatio = std::min(double(width)/double(aSize.Width()), double(height)/double(aSize.Height())); aImg.Scale(Size(aSize.Width() * nRatio, aSize.Height() * nRatio)); } return aImg; } bool TemplateLocalView::IsDefaultTemplate(const OUString& rPath) { SvtModuleOptions aModOpt; const css::uno::Sequence &aServiceNames = aModOpt.GetAllServiceNames(); return std::any_of(aServiceNames.begin(), aServiceNames.end(), [&rPath](const OUString& rName) { return SfxObjectFactory::GetStandardTemplate(rName).match(rPath); }); } BitmapEx TemplateLocalView::getDefaultThumbnail( const OUString& rPath ) { BitmapEx aImg; INetURLObject aUrl(rPath); OUString aExt = aUrl.getExtension(); if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::WRITER, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_TEXT); else if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::CALC, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_SHEET); else if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::IMPRESS, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_PRESENTATION); else if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::DRAW, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_DRAWING); return aImg; } BitmapEx TemplateLocalView::fetchThumbnail (const OUString &msURL, long width, long height) { return TemplateLocalView::scaleImg(ThumbnailView::readThumbnail(msURL), width, height); } void TemplateLocalView::OnItemDblClicked (ThumbnailViewItem *pItem) { TemplateViewItem* pViewItem = dynamic_cast(pItem); if( pViewItem ) maOpenTemplateHdl.Call(pViewItem); } SfxTemplateLocalView::SfxTemplateLocalView(std::unique_ptr xWindow, std::unique_ptr xMenu) : SfxThumbnailView(std::move(xWindow), std::move(xMenu)) , mnCurRegionId(0) , maSelectedItem(nullptr) , maPosition(0,0) , mpDocTemplates(new SfxDocumentTemplates) { } SfxTemplateLocalView::~SfxTemplateLocalView() { } void SfxTemplateLocalView::Populate() { maRegions.clear(); maAllTemplates.clear(); sal_uInt16 nCount = mpDocTemplates->GetRegionCount(); for (sal_uInt16 i = 0; i < nCount; ++i) { OUString aRegionName(mpDocTemplates->GetFullRegionName(i)); std::unique_ptr pItem(new TemplateContainerItem( i+1 )); pItem->mnRegionId = i; pItem->maTitle = aRegionName; sal_uInt16 nEntries = mpDocTemplates->GetCount(i); for (sal_uInt16 j = 0; j < nEntries; ++j) { OUString aName = mpDocTemplates->GetName(i,j); OUString aURL = mpDocTemplates->GetPath(i,j); TemplateItemProperties aProperties; aProperties.nId = j+1; aProperties.nDocId = j; aProperties.nRegionId = i; aProperties.aName = aName; aProperties.aPath = aURL; aProperties.aRegionName = aRegionName; aProperties.aThumbnail = TemplateLocalView::fetchThumbnail(aURL, TEMPLATE_THUMBNAIL_MAX_WIDTH, TEMPLATE_THUMBNAIL_MAX_HEIGHT); pItem->maTemplates.push_back(aProperties); maAllTemplates.push_back(aProperties); } maRegions.push_back(std::move(pItem)); } } void SfxTemplateLocalView::reload() { mpDocTemplates->Update(); Populate(); // Check if we are currently browsing a region or root folder if (mnCurRegionId) { sal_uInt16 nRegionId = mnCurRegionId - 1; //Is offset by 1 for (auto const & pRegion : maRegions) { if (pRegion->mnRegionId == nRegionId) { showRegion(pRegion.get()); break; } } } else showAllTemplates(); //No items should be selected by default deselectItems(); } void SfxTemplateLocalView::showAllTemplates() { mnCurRegionId = 0; insertItems(maAllTemplates, false, true); maOpenRegionHdl.Call(nullptr); } void SfxTemplateLocalView::showRegion(TemplateContainerItem const *pItem) { mnCurRegionId = pItem->mnRegionId+1; insertItems(pItem->maTemplates); maOpenRegionHdl.Call(nullptr); } void SfxTemplateLocalView::showRegion(const OUString &rName) { for (auto const & pRegion : maRegions) { if (pRegion->maTitle == rName) { showRegion(pRegion.get()); break; } } } TemplateContainerItem* SfxTemplateLocalView::getRegion(OUString const & rName) { for (auto const & pRegion : maRegions) if (pRegion->maTitle == rName) return pRegion.get(); return nullptr; } void SfxTemplateLocalView::createContextMenu(const bool bIsDefault) { mxContextMenu->clear(); mxContextMenu->append("open",SfxResId(STR_OPEN)); mxContextMenu->append("edit",SfxResId(STR_EDIT_TEMPLATE)); if(!bIsDefault) mxContextMenu->append("default",SfxResId(STR_DEFAULT_TEMPLATE)); else mxContextMenu->append("default",SfxResId(STR_RESET_DEFAULT)); mxContextMenu->append_separator("separator"); mxContextMenu->append("rename",SfxResId(STR_SFX_RENAME)); mxContextMenu->append("delete",SfxResId(STR_DELETE)); deselectItems(); maSelectedItem->setSelection(true); maItemStateHdl.Call(maSelectedItem); ContextMenuSelectHdl(mxContextMenu->popup_at_rect(GetDrawingArea(), tools::Rectangle(maPosition, Size(1,1)))); Invalidate(); } void SfxTemplateLocalView::ContextMenuSelectHdl(const OString& rIdent) { if (rIdent == "open") maOpenTemplateHdl.Call(maSelectedItem); else if (rIdent == "edit") maEditTemplateHdl.Call(maSelectedItem); else if (rIdent == "rename") { InputDialog aTitleEditDlg(GetDrawingArea(), SfxResId(STR_RENAME_TEMPLATE)); OUString sOldTitle = maSelectedItem->getTitle(); aTitleEditDlg.SetEntryText(sOldTitle); aTitleEditDlg.HideHelpBtn(); if (!aTitleEditDlg.run()) return; OUString sNewTitle = comphelper::string::strip(aTitleEditDlg.GetEntryText(), ' '); if ( !sNewTitle.isEmpty() && sNewTitle != sOldTitle ) { maSelectedItem->setTitle(sNewTitle); } } else if (rIdent == "delete") { std::unique_ptr xQueryDlg(Application::CreateMessageDialog(GetDrawingArea(), VclMessageType::Question, VclButtonsType::YesNo, SfxResId(STR_QMSG_SEL_TEMPLATE_DELETE))); if (xQueryDlg->run() != RET_YES) return; maDeleteTemplateHdl.Call(maSelectedItem); reload(); } else if (rIdent == "default") maDefaultTemplateHdl.Call(maSelectedItem); } sal_uInt16 SfxTemplateLocalView::getRegionId(size_t pos) const { assert(pos < maRegions.size()); return maRegions[pos]->mnId; } sal_uInt16 SfxTemplateLocalView::getRegionId(OUString const & sRegion) const { for (auto const & pRegion : maRegions) { if (pRegion->maTitle == sRegion) return pRegion->mnId; } return 0; } OUString SfxTemplateLocalView::getRegionName(const sal_uInt16 nRegionId) const { return mpDocTemplates->GetRegionName(nRegionId); } OUString SfxTemplateLocalView::getRegionItemName(const sal_uInt16 nItemId) const { for (auto const & pRegion : maRegions) { if (pRegion->mnId == nItemId) return pRegion->maTitle; } return OUString(); } std::vector SfxTemplateLocalView::getFolderNames() { size_t n = maRegions.size(); std::vector ret(n); for (size_t i = 0; i < n; ++i) ret[i] = maRegions[i]->maTitle; return ret; } std::vector SfxTemplateLocalView::getFilteredItems(const std::function &rFunc) const { std::vector aItems; if (mnCurRegionId) { TemplateContainerItem *pFolderItem = maRegions[mnCurRegionId-1].get(); for (const TemplateItemProperties & rItemProps : pFolderItem->maTemplates) { if (rFunc(rItemProps)) aItems.push_back(rItemProps); } } else { for (auto const & pFolderItem : maRegions) { for (const TemplateItemProperties & rItemProps : pFolderItem->maTemplates) { if (rFunc(rItemProps)) aItems.push_back(rItemProps); } } } return aItems; } sal_uInt16 SfxTemplateLocalView::createRegion(const OUString &rName) { sal_uInt16 nRegionId = mpDocTemplates->GetRegionCount(); // Next regionId sal_uInt16 nItemId = getNextItemId(); if (!mpDocTemplates->InsertDir(rName,nRegionId)) return 0; // Insert to the region cache list and to the thumbnail item list std::unique_ptr pItem(new TemplateContainerItem( nItemId )); pItem->mnRegionId = nRegionId; pItem->maTitle = rName; maRegions.push_back(std::move(pItem)); return nItemId; } bool SfxTemplateLocalView::renameRegion(const OUString &rTitle, const OUString &rNewTitle) { TemplateContainerItem *pRegion = getRegion(rTitle); if(pRegion) { sal_uInt16 nRegionId = pRegion->mnRegionId; return mpDocTemplates->SetName( rNewTitle, nRegionId, USHRT_MAX/*nDocId*/ ); } return false; } bool SfxTemplateLocalView::removeRegion(const sal_uInt16 nItemId) { sal_uInt16 nRegionId = USHRT_MAX; // Remove from the region cache list for (auto pRegionIt = maRegions.begin(); pRegionIt != maRegions.end();) { if ( (*pRegionIt)->mnId == nItemId ) { if (!mpDocTemplates->Delete((*pRegionIt)->mnRegionId,USHRT_MAX)) return false; nRegionId = (*pRegionIt)->mnRegionId; pRegionIt = maRegions.erase(pRegionIt); } else { // Synchronize regions cache ids with SfxDocumentTemplates if (nRegionId != USHRT_MAX && (*pRegionIt)->mnRegionId > nRegionId) --(*pRegionIt)->mnRegionId; ++pRegionIt; } } if (nRegionId == USHRT_MAX) return false; // Synchronize view regions ids with SfxDocumentTemplates for (auto const& region : maRegions) { if (region->mnRegionId > nRegionId) --region->mnRegionId; } return true; } bool SfxTemplateLocalView::removeTemplate (const sal_uInt16 nItemId, const sal_uInt16 nSrcItemId) { for (auto const & pRegion : maRegions) { if (pRegion->mnId == nSrcItemId) { TemplateContainerItem *pItem = pRegion.get(); auto pIter = std::find_if(pItem->maTemplates.begin(), pItem->maTemplates.end(), [nItemId](const TemplateItemProperties& rTemplate) { return rTemplate.nId == nItemId; }); if (pIter != pItem->maTemplates.end()) { if (!mpDocTemplates->Delete(pItem->mnRegionId,pIter->nDocId)) return false; pIter = pItem->maTemplates.erase(pIter); if (pRegion->mnRegionId == mnCurRegionId-1) { RemoveItem(nItemId); Invalidate(); } // Update Doc Idx for all templates that follow for (; pIter != pItem->maTemplates.end(); ++pIter) pIter->nDocId = pIter->nDocId - 1; } CalculateItemPositions(); break; } } return true; } bool SfxTemplateLocalView::moveTemplate (const ThumbnailViewItem *pItem, const sal_uInt16 nSrcItem, const sal_uInt16 nTargetItem) { TemplateContainerItem *pTarget = nullptr; TemplateContainerItem *pSrc = nullptr; for (auto const & pRegion : maRegions) { if (pRegion->mnId == nTargetItem) pTarget = pRegion.get(); else if (pRegion->mnId == nSrcItem) pSrc = pRegion.get(); } if (pTarget && pSrc) { sal_uInt16 nSrcRegionId = pSrc->mnRegionId; sal_uInt16 nTargetRegion = pTarget->mnRegionId; sal_uInt16 nTargetIdx = mpDocTemplates->GetCount(nTargetRegion); // Next Idx const TemplateViewItem *pViewItem = static_cast(pItem); bool bCopy = !mpDocTemplates->Move(nTargetRegion,nTargetIdx,nSrcRegionId,pViewItem->mnDocId); if (bCopy) { OUString sQuery = SfxResId(STR_MSG_QUERY_COPY).replaceFirst("$1", pViewItem->maTitle).replaceFirst("$2", getRegionName(nTargetRegion)); std::unique_ptr xQueryDlg(Application::CreateMessageDialog(GetDrawingArea(), VclMessageType::Question, VclButtonsType::YesNo, sQuery)); if (xQueryDlg->run() != RET_YES) return false; if (!mpDocTemplates->Copy(nTargetRegion,nTargetIdx,nSrcRegionId,pViewItem->mnDocId)) return false; } // move template to destination TemplateItemProperties aTemplateItem; aTemplateItem.nId = nTargetIdx + 1; aTemplateItem.nDocId = nTargetIdx; aTemplateItem.nRegionId = nTargetRegion; aTemplateItem.aName = pViewItem->maTitle; aTemplateItem.aPath = mpDocTemplates->GetPath(nTargetRegion,nTargetIdx); aTemplateItem.aRegionName = pViewItem->maHelpText; aTemplateItem.aThumbnail = pViewItem->maPreview1; pTarget->maTemplates.push_back(aTemplateItem); if (!bCopy) { // remove template from region cached data std::vector::iterator aIter; for (aIter = pSrc->maTemplates.begin(); aIter != pSrc->maTemplates.end();) { if (aIter->nDocId == pViewItem->mnDocId) { aIter = pSrc->maTemplates.erase(aIter); } else { // Keep region document id synchronized with SfxDocumentTemplates if (aIter->nDocId > pViewItem->mnDocId) --aIter->nDocId; ++aIter; } } // Keep view document id synchronized with SfxDocumentTemplates for (auto const& item : mItemList) { auto pTemplateViewItem = static_cast(item.get()); if (pTemplateViewItem->mnDocId > pViewItem->mnDocId) --pTemplateViewItem->mnDocId; } } CalculateItemPositions(); Invalidate(); return true; } return false; } void SfxTemplateLocalView::moveTemplates(const std::set &rItems, const sal_uInt16 nTargetItem) { TemplateContainerItem *pTarget = nullptr; TemplateContainerItem *pSrc = nullptr; for (auto const & pRegion : maRegions) { if (pRegion->mnId == nTargetItem) pTarget = pRegion.get(); } if (!pTarget) return; bool refresh = false; sal_uInt16 nTargetRegion = pTarget->mnRegionId; sal_uInt16 nTargetIdx = mpDocTemplates->GetCount(nTargetRegion); // Next Idx std::vector aItemIds; // List of moved items ids (also prevents the invalidation of rItems iterators when we remove them as we go) std::set::const_iterator aSelIter; for ( aSelIter = rItems.begin(); aSelIter != rItems.end(); ++aSelIter, ++nTargetIdx ) { const TemplateViewItem *pViewItem = static_cast(*aSelIter); sal_uInt16 nSrcRegionId = pViewItem->mnRegionId; for (auto const & pRegion : maRegions) { if (pRegion->mnRegionId == nSrcRegionId) pSrc = pRegion.get(); } if(pSrc) { bool bCopy = !mpDocTemplates->Move(nTargetRegion,nTargetIdx,nSrcRegionId,pViewItem->mnDocId); if (bCopy) { OUString sQuery = SfxResId(STR_MSG_QUERY_COPY).replaceFirst("$1", pViewItem->maTitle).replaceFirst("$2", getRegionName(nTargetRegion)); std::unique_ptr xQueryDlg(Application::CreateMessageDialog(GetDrawingArea(), VclMessageType::Question, VclButtonsType::YesNo, sQuery)); if (xQueryDlg->run() != RET_YES) { OUString sMsg(SfxResId(STR_MSG_ERROR_LOCAL_MOVE)); sMsg = sMsg.replaceFirst("$1",getRegionName(nTargetRegion)); std::unique_ptr xBox(Application::CreateMessageDialog(GetDrawingArea(), VclMessageType::Warning, VclButtonsType::Ok, sMsg.replaceFirst( "$2",pViewItem->maTitle))); xBox->run(); return; //return if any single move operation fails } if (!mpDocTemplates->Copy(nTargetRegion,nTargetIdx,nSrcRegionId,pViewItem->mnDocId)) { continue; } } // move template to destination TemplateItemProperties aTemplateItem; aTemplateItem.nId = nTargetIdx + 1; aTemplateItem.nDocId = nTargetIdx; aTemplateItem.nRegionId = nTargetRegion; aTemplateItem.aName = pViewItem->maTitle; aTemplateItem.aPath = mpDocTemplates->GetPath(nTargetRegion,nTargetIdx); aTemplateItem.aRegionName = pViewItem->maHelpText; aTemplateItem.aThumbnail = pViewItem->maPreview1; pTarget->maTemplates.push_back(aTemplateItem); if (!bCopy) { // remove template from region cached data std::vector::iterator pPropIter; for (pPropIter = pSrc->maTemplates.begin(); pPropIter != pSrc->maTemplates.end();) { if (pPropIter->nDocId == pViewItem->mnDocId) { pPropIter = pSrc->maTemplates.erase(pPropIter); aItemIds.push_back(pViewItem->mnDocId + 1);//mnid } else { // Keep region document id synchronized with SfxDocumentTemplates if (pPropIter->nDocId > pViewItem->mnDocId) --pPropIter->nDocId; ++pPropIter; } } // Keep view document id synchronized with SfxDocumentTemplates for (auto const& item : mItemList) { auto pTemplateViewItem = static_cast(item.get()); if (pTemplateViewItem->mnDocId > pViewItem->mnDocId) --pTemplateViewItem->mnDocId; } } } refresh = true; } // Remove items from the current view for (auto const& itemId : aItemIds) RemoveItem(itemId); if (refresh) { CalculateItemPositions(); Invalidate(); } } bool SfxTemplateLocalView::copyFrom (TemplateContainerItem *pItem, const OUString &rPath) { sal_uInt16 nId = 1; sal_uInt16 nDocId = 0; sal_uInt16 nRegionId = pItem->mnRegionId; OUString aPath(rPath); if (!pItem->maTemplates.empty()) { nId = pItem->maTemplates.back().nId+1; nDocId = pItem->maTemplates.back().nDocId+1; } if (mpDocTemplates->CopyFrom(nRegionId,nDocId,aPath)) { TemplateItemProperties aTemplate; aTemplate.nId = nId; aTemplate.nDocId = nDocId; aTemplate.nRegionId = nRegionId; aTemplate.aName = aPath; aTemplate.aThumbnail = SfxTemplateLocalView::fetchThumbnail(rPath, TEMPLATE_THUMBNAIL_MAX_WIDTH, TEMPLATE_THUMBNAIL_MAX_HEIGHT); aTemplate.aPath = rPath; aTemplate.aRegionName = getRegionName(nRegionId); pItem->maTemplates.push_back(aTemplate); CalculateItemPositions(); return true; } return false; } bool SfxTemplateLocalView::exportTo(const sal_uInt16 nItemId, const sal_uInt16 nRegionItemId, const OUString &rName) { for (auto const & pRegItem : maRegions) { if (pRegItem->mnId == nRegionItemId) { for (auto const& elem : pRegItem->maTemplates) { if (elem.nId == nItemId) { return mpDocTemplates->CopyTo(pRegItem->mnRegionId,elem.nDocId,rName); } } break; } } return false; } bool SfxTemplateLocalView::renameItem(ThumbnailViewItem* pItem, const OUString& sNewTitle) { sal_uInt16 nRegionId = 0; sal_uInt16 nDocId = USHRT_MAX; TemplateViewItem* pDocItem = dynamic_cast( pItem ); if ( pDocItem ) { nRegionId = pDocItem->mnRegionId; nDocId = pDocItem->mnDocId; } return mpDocTemplates->SetName( sNewTitle, nRegionId, nDocId ); } void SfxTemplateLocalView::insertItems(const std::vector &rTemplates, bool isRegionSelected, bool bShowCategoryInTooltip) { std::vector> aItems(rTemplates.size()); for (size_t i = 0, n = rTemplates.size(); i < n; ++i ) { const TemplateItemProperties *pCur = &rTemplates[i]; std::unique_ptr pChild; if(isRegionSelected) pChild.reset(new TemplateViewItem(*this, pCur->nId)); else pChild.reset(new TemplateViewItem(*this, i+1)); pChild->mnDocId = pCur->nDocId; pChild->mnRegionId = pCur->nRegionId; pChild->maTitle = pCur->aName; pChild->setPath(pCur->aPath); if(!bShowCategoryInTooltip) pChild->setHelpText(pCur->aName); else { OUString sHelpText = SfxResId(STR_TEMPLATE_TOOLTIP); sHelpText = (sHelpText.replaceFirst("$1", pCur->aName)).replaceFirst("$2", pCur->aRegionName); pChild->setHelpText(sHelpText); } pChild->maPreview1 = pCur->aThumbnail; if(IsDefaultTemplate(pCur->aPath)) pChild->showDefaultIcon(true); if ( pCur->aThumbnail.IsEmpty() ) { // Use the default thumbnail if we have nothing else pChild->maPreview1 = SfxTemplateLocalView::getDefaultThumbnail(pCur->aPath); } aItems[i] = std::move(pChild); } updateItems(std::move(aItems)); } bool SfxTemplateLocalView::MouseButtonDown( const MouseEvent& rMEvt ) { GrabFocus(); return SfxThumbnailView::MouseButtonDown(rMEvt); } bool SfxTemplateLocalView::Command(const CommandEvent& rCEvt) { if (rCEvt.GetCommand() != CommandEventId::ContextMenu) return CustomWidgetController::Command(rCEvt); if (rCEvt.IsMouseEvent()) { deselectItems(); size_t nPos = ImplGetItem(rCEvt.GetMousePosPixel()); Point aPosition(rCEvt.GetMousePosPixel()); maPosition = aPosition; ThumbnailViewItem* pItem = ImplGetItem(nPos); const TemplateViewItem *pViewItem = dynamic_cast(pItem); if(pViewItem) { maSelectedItem = dynamic_cast(pItem); maCreateContextMenuHdl.Call(pItem); } } else { for (ThumbnailViewItem* pItem : mFilteredItemList) { //create context menu for the first selected item if (pItem->isSelected()) { deselectItems(); pItem->setSelection(true); maItemStateHdl.Call(pItem); tools::Rectangle aRect = pItem->getDrawArea(); maPosition = aRect.Center(); maSelectedItem = dynamic_cast(pItem); maCreateContextMenuHdl.Call(pItem); break; } } } return true; } bool SfxTemplateLocalView::KeyInput( const KeyEvent& rKEvt ) { vcl::KeyCode aKeyCode = rKEvt.GetKeyCode(); if(aKeyCode == ( KEY_MOD1 | KEY_A ) ) { for (ThumbnailViewItem* pItem : mFilteredItemList) { if (!pItem->isSelected()) { pItem->setSelection(true); maItemStateHdl.Call(pItem); } } if (IsReallyVisible() && IsUpdateMode()) Invalidate(); return true; } else if( aKeyCode == KEY_DELETE && !mFilteredItemList.empty()) { std::unique_ptr xQueryDlg(Application::CreateMessageDialog(GetDrawingArea(), VclMessageType::Question, VclButtonsType::YesNo, SfxResId(STR_QMSG_SEL_TEMPLATE_DELETE))); if (xQueryDlg->run() != RET_YES) return true; //copy to avoid changing filtered item list during deletion ThumbnailValueItemList mFilteredItemListCopy = mFilteredItemList; for (ThumbnailViewItem* pItem : mFilteredItemListCopy) { if (pItem->isSelected()) { maDeleteTemplateHdl.Call(pItem); } } reload(); } return SfxThumbnailView::KeyInput(rKEvt); } void SfxTemplateLocalView::setOpenRegionHdl(const Link &rLink) { maOpenRegionHdl = rLink; } void SfxTemplateLocalView::setCreateContextMenuHdl(const Link &rLink) { maCreateContextMenuHdl = rLink; } void SfxTemplateLocalView::setOpenTemplateHdl(const Link &rLink) { maOpenTemplateHdl = rLink; } void SfxTemplateLocalView::setEditTemplateHdl(const Link &rLink) { maEditTemplateHdl = rLink; } void SfxTemplateLocalView::setDeleteTemplateHdl(const Link &rLink) { maDeleteTemplateHdl = rLink; } void SfxTemplateLocalView::setDefaultTemplateHdl(const Link &rLink) { maDefaultTemplateHdl = rLink; } BitmapEx SfxTemplateLocalView::scaleImg (const BitmapEx &rImg, long width, long height) { BitmapEx aImg = rImg; if (!rImg.IsEmpty()) { Size aSize = rImg.GetSizePixel(); if (aSize.Width() == 0) aSize.setWidth( 1 ); if (aSize.Height() == 0) aSize.setHeight( 1 ); // make the picture fit the given width/height constraints double nRatio = std::min(double(width)/double(aSize.Width()), double(height)/double(aSize.Height())); aImg.Scale(Size(aSize.Width() * nRatio, aSize.Height() * nRatio)); } return aImg; } bool SfxTemplateLocalView::IsDefaultTemplate(const OUString& rPath) { SvtModuleOptions aModOpt; const css::uno::Sequence &aServiceNames = aModOpt.GetAllServiceNames(); return std::any_of(aServiceNames.begin(), aServiceNames.end(), [&rPath](const OUString& rName) { return SfxObjectFactory::GetStandardTemplate(rName).match(rPath); }); } void SfxTemplateLocalView::RemoveDefaultTemplateIcon(const OUString& rPath) { for (const std::unique_ptr& pItem : mItemList) { TemplateViewItem* pViewItem = dynamic_cast(pItem.get()); if (pViewItem && pViewItem->getPath().match(rPath)) { pViewItem->showDefaultIcon(false); Invalidate(); return; } } } BitmapEx SfxTemplateLocalView::getDefaultThumbnail( const OUString& rPath ) { BitmapEx aImg; INetURLObject aUrl(rPath); OUString aExt = aUrl.getExtension(); if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::WRITER, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_TEXT); else if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::CALC, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_SHEET); else if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::IMPRESS, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_PRESENTATION); else if ( ViewFilter_Application::isFilteredExtension( FILTER_APPLICATION::DRAW, aExt) ) aImg = BitmapEx(SFX_THUMBNAIL_DRAWING); return aImg; } BitmapEx SfxTemplateLocalView::fetchThumbnail (const OUString &msURL, long width, long height) { return SfxTemplateLocalView::scaleImg(ThumbnailView::readThumbnail(msURL), width, height); } void SfxTemplateLocalView::OnItemDblClicked (ThumbnailViewItem *pItem) { TemplateViewItem* pViewItem = dynamic_cast(pItem); if( pViewItem ) maOpenTemplateHdl.Call(pViewItem); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */