/* -*- 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 using namespace css; MenuContentHandler::MenuContentHandler(uno::Reference const& xFrame) : m_xContext(comphelper::getProcessComponentContext()) , m_xFrame(xFrame) , m_xCharacterClassification(i18n::CharacterClassification::create(m_xContext)) , m_xURLTransformer(util::URLTransformer::create(m_xContext)) , m_sModuleLongName(vcl::CommandInfoProvider::GetModuleIdentifier(xFrame)) { uno::Reference xModuleConfigSupplier; xModuleConfigSupplier.set(ui::theModuleUIConfigurationManagerSupplier::get(m_xContext)); uno::Reference xConfigurationManager; xConfigurationManager = xModuleConfigSupplier->getUIConfigurationManager(m_sModuleLongName); uno::Reference xConfigData; xConfigData = xConfigurationManager->getSettings(u"private:resource/menubar/menubar"_ustr, false); gatherMenuContent(xConfigData, m_aMenuContent); } void MenuContentHandler::gatherMenuContent( uno::Reference const& xIndexAccess, MenuContent& rMenuContent) { std::u16string_view aMenuLabelSeparator = AllSettings::GetLayoutRTL() ? u" ◂ " : u" ▸ "; for (sal_Int32 n = 0; n < xIndexAccess->getCount(); n++) { MenuContent aNewContent; uno::Sequence aProperties; uno::Reference xIndexContainer; if (!(xIndexAccess->getByIndex(n) >>= aProperties)) continue; bool bIsVisible = true; bool bIsEnabled = true; for (auto const& rProperty : aProperties) { OUString aPropertyName = rProperty.Name; if (aPropertyName == "CommandURL") rProperty.Value >>= aNewContent.m_aCommandURL; else if (aPropertyName == "ItemDescriptorContainer") rProperty.Value >>= xIndexContainer; else if (aPropertyName == "IsVisible") rProperty.Value >>= bIsVisible; else if (aPropertyName == "Enabled") rProperty.Value >>= bIsEnabled; } if (!bIsEnabled || !bIsVisible) continue; auto aCommandProperties = vcl::CommandInfoProvider::GetCommandProperties( aNewContent.m_aCommandURL, m_sModuleLongName); aNewContent.m_aMenuLabel = vcl::CommandInfoProvider::GetLabelForCommand(aCommandProperties); if (!rMenuContent.m_aFullLabelWithPath.isEmpty()) aNewContent.m_aFullLabelWithPath = rMenuContent.m_aFullLabelWithPath + aMenuLabelSeparator; aNewContent.m_aFullLabelWithPath += aNewContent.m_aMenuLabel; aNewContent.m_aSearchableMenuLabel = toLower(aNewContent.m_aFullLabelWithPath); aNewContent.m_aTooltip = vcl::CommandInfoProvider::GetTooltipForCommand( aNewContent.m_aCommandURL, aCommandProperties, m_xFrame); if (xIndexContainer.is()) gatherMenuContent(xIndexContainer, aNewContent); rMenuContent.m_aSubMenuContent.push_back(aNewContent); } } void MenuContentHandler::findInMenu(OUString const& rText, std::unique_ptr& rpCommandTreeView, std::vector& rCommandList) { m_aAdded.clear(); OUString aLowerCaseText = toLower(rText); // find submenus and menu items that start with the searched text auto aTextStartCriterium = [](MenuContent const& rMenuContent, OUString const& rSearchText) { OUString aSearchText = " / " + rSearchText; return rMenuContent.m_aSearchableMenuLabel.indexOf(aSearchText) > 0; }; findInMenuRecursive(m_aMenuContent, aLowerCaseText, rpCommandTreeView, rCommandList, aTextStartCriterium); // find submenus and menu items that contain the searched text auto aTextAllCriterium = [](MenuContent const& rMenuContent, OUString const& rSearchText) { return rMenuContent.m_aSearchableMenuLabel.indexOf(rSearchText) > 0; }; findInMenuRecursive(m_aMenuContent, aLowerCaseText, rpCommandTreeView, rCommandList, aTextAllCriterium); } void MenuContentHandler::findInMenuRecursive( MenuContent const& rMenuContent, OUString const& rText, std::unique_ptr& rpCommandTreeView, std::vector& rCommandList, std::function const& rSearchCriterium) { for (MenuContent const& aSubContent : rMenuContent.m_aSubMenuContent) { if (rSearchCriterium(aSubContent, rText)) { addCommandIfPossible(aSubContent, rpCommandTreeView, rCommandList); } findInMenuRecursive(aSubContent, rText, rpCommandTreeView, rCommandList, rSearchCriterium); } } void MenuContentHandler::addCommandIfPossible( MenuContent const& rMenuContent, const std::unique_ptr& rpCommandTreeView, std::vector& rCommandList) { if (m_aAdded.find(rMenuContent.m_aFullLabelWithPath) != m_aAdded.end()) return; OUString sCommandURL = rMenuContent.m_aCommandURL; util::URL aCommandURL; aCommandURL.Complete = sCommandURL; if (!m_xURLTransformer->parseStrict(aCommandURL)) return; auto* pViewFrame = SfxViewFrame::Current(); if (!pViewFrame) return; SfxSlotPool& rSlotPool = SfxSlotPool::GetSlotPool(pViewFrame); const SfxSlot* pSlot = rSlotPool.GetUnoSlot(aCommandURL.Path); if (!pSlot) return; std::unique_ptr pState; SfxItemState eState = pViewFrame->GetBindings().QueryState(pSlot->GetSlotId(), pState); if (eState == SfxItemState::DISABLED) return; auto xGraphic = vcl::CommandInfoProvider::GetXGraphicForCommand(sCommandURL, m_xFrame); rCommandList.emplace_back(sCommandURL, rMenuContent.m_aTooltip); auto pIter = rpCommandTreeView->make_iterator(); rpCommandTreeView->insert(nullptr, -1, &rMenuContent.m_aFullLabelWithPath, nullptr, nullptr, nullptr, false, pIter.get()); rpCommandTreeView->set_image(*pIter, xGraphic); m_aAdded.insert(rMenuContent.m_aFullLabelWithPath); } OUString MenuContentHandler::toLower(OUString const& rString) { const css::lang::Locale& rLocale = Application::GetSettings().GetUILanguageTag().getLocale(); return m_xCharacterClassification->toLower(rString, 0, rString.getLength(), rLocale); } CommandListBox::CommandListBox(weld::Window* pParent, uno::Reference const& xFrame) : mxBuilder(Application::CreateBuilder(pParent, u"sfx/ui/commandpopup.ui"_ustr)) , mxPopover(mxBuilder->weld_popover(u"CommandPopup"_ustr)) , mpEntry(mxBuilder->weld_entry(u"command_entry"_ustr)) , mpCommandTreeView(mxBuilder->weld_tree_view(u"command_treeview"_ustr)) , mpMenuContentHandler(std::make_unique(xFrame)) { mpEntry->connect_changed(LINK(this, CommandListBox, ModifyHdl)); mpEntry->connect_key_press(LINK(this, CommandListBox, TreeViewKeyPress)); mpCommandTreeView->connect_query_tooltip(LINK(this, CommandListBox, QueryTooltip)); mpCommandTreeView->connect_row_activated(LINK(this, CommandListBox, RowActivated)); Size aFrameSize = pParent->get_size(); // Set size of the pop-over window tools::Long nWidth = std::max(tools::Long(400), aFrameSize.Width() / 3); mpCommandTreeView->set_size_request(nWidth, 400); // Set the location of the pop-over window tools::Rectangle aRect(Point(aFrameSize.Width() / 2, 0), Size(0, 0)); mxPopover->popup_at_rect(pParent, aRect); mpEntry->grab_focus(); } IMPL_LINK_NOARG(CommandListBox, QueryTooltip, const weld::TreeIter&, OUString) { size_t nSelected = mpCommandTreeView->get_selected_index(); if (nSelected < maCommandList.size()) { auto const& rCurrent = maCommandList[nSelected]; return rCurrent.m_aTooltip; } return OUString(); } IMPL_LINK_NOARG(CommandListBox, RowActivated, weld::TreeView&, bool) { OUString aCommandURL; int nSelected = mpCommandTreeView->get_selected_index(); if (nSelected != -1 && nSelected < int(maCommandList.size())) { auto const& rCurrent = maCommandList[nSelected]; aCommandURL = rCurrent.m_aCommandURL; } dispatchCommandAndClose(aCommandURL); return true; } IMPL_LINK(CommandListBox, TreeViewKeyPress, const KeyEvent&, rKeyEvent, bool) { if (rKeyEvent.GetKeyCode().GetCode() == KEY_DOWN || rKeyEvent.GetKeyCode().GetCode() == KEY_UP) { int nDirection = rKeyEvent.GetKeyCode().GetCode() == KEY_DOWN ? 1 : -1; int nNewIndex = mpCommandTreeView->get_selected_index() + nDirection; nNewIndex = std::clamp(nNewIndex, 0, mpCommandTreeView->n_children() - 1); mpCommandTreeView->select(nNewIndex); mpCommandTreeView->set_cursor(nNewIndex); return true; } else if (rKeyEvent.GetKeyCode().GetCode() == KEY_RETURN) { RowActivated(*mpCommandTreeView); return true; } return false; } IMPL_LINK_NOARG(CommandListBox, ModifyHdl, weld::Entry&, void) { mpCommandTreeView->clear(); maCommandList.clear(); OUString sText = mpEntry->get_text(); if (sText.isEmpty()) return; mpCommandTreeView->freeze(); mpMenuContentHandler->findInMenu(sText, mpCommandTreeView, maCommandList); mpCommandTreeView->thaw(); if (mpCommandTreeView->n_children() > 0) { mpCommandTreeView->set_cursor(0); mpCommandTreeView->select(0); } mpEntry->grab_focus(); } void CommandListBox::dispatchCommandAndClose(OUString const& rCommand) { mxPopover->popdown(); if (!rCommand.isEmpty()) comphelper::dispatchCommand(rCommand, uno::Sequence()); } void CommandPopupHandler::showPopup(weld::Window* pParent, css::uno::Reference const& xFrame) { auto pCommandListBox = std::make_unique(pParent, xFrame); pCommandListBox->connect_closed(LINK(this, CommandPopupHandler, PopupModeEnd)); mpListBox = std::move(pCommandListBox); } IMPL_LINK_NOARG(CommandPopupHandler, PopupModeEnd, weld::Popover&, void) { mpListBox.reset(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */