/* -*- 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 namespace svt { namespace { class AsyncAccelExec : public cppu::WeakImplHelper { private: css::uno::Reference m_xFrame; css::uno::Reference< css::frame::XDispatch > m_xDispatch; css::util::URL m_aURL; vcl::EventPoster m_aAsyncCallback; public: /** creates a new instance of this class, which can be used one times only! This instance can be forced to execute its internal set request asynchronous. After that it deletes itself! */ static rtl::Reference createOneShotInstance(const css::uno::Reference& xFrame, const css::uno::Reference& xDispatch, const css::util::URL& rURL); void execAsync(); private: virtual void SAL_CALL disposing(const css::lang::EventObject&) override { m_xFrame->removeEventListener(this); m_xFrame.clear(); m_xDispatch.clear(); } /** @short allow creation of instances of this class by using our factory only! */ AsyncAccelExec(css::uno::Reference xFrame, css::uno::Reference< css::frame::XDispatch > xDispatch, css::util::URL aURL); DECL_LINK(impl_ts_asyncCallback, LinkParamNone*, void); }; } AcceleratorExecute::AcceleratorExecute() { } AcceleratorExecute::~AcceleratorExecute() { // does nothing real } std::unique_ptr AcceleratorExecute::createAcceleratorHelper() { return std::unique_ptr(new AcceleratorExecute); } void AcceleratorExecute::init(const css::uno::Reference< css::uno::XComponentContext >& rxContext, const css::uno::Reference< css::frame::XFrame >& xEnv ) { // SAFE -> ---------------------------------- std::unique_lock aLock(m_aLock); // take over the uno service manager m_xContext = rxContext; // specify our internal dispatch provider // frame or desktop?! => document or global config. bool bDesktopIsUsed = false; m_xDispatcher.set(xEnv, css::uno::UNO_QUERY); if (!m_xDispatcher.is()) { aLock.unlock(); // <- SAFE ------------------------------ css::uno::Reference< css::frame::XDispatchProvider > xDispatcher(css::frame::Desktop::create(rxContext), css::uno::UNO_QUERY_THROW); // SAFE -> ------------------------------ aLock.lock(); m_xDispatcher = std::move(xDispatcher); bDesktopIsUsed = true; } aLock.unlock(); // <- SAFE ---------------------------------- // open all needed configuration objects css::uno::Reference< css::ui::XAcceleratorConfiguration > xGlobalCfg; css::uno::Reference< css::ui::XAcceleratorConfiguration > xModuleCfg; css::uno::Reference< css::ui::XAcceleratorConfiguration > xDocCfg ; // global cfg xGlobalCfg = css::ui::GlobalAcceleratorConfiguration::create(rxContext); if (!bDesktopIsUsed) { // module cfg xModuleCfg = AcceleratorExecute::st_openModuleConfig(rxContext, xEnv); // doc cfg css::uno::Reference< css::frame::XController > xController; css::uno::Reference< css::frame::XModel > xModel; xController = xEnv->getController(); if (xController.is()) xModel = xController->getModel(); if (xModel.is()) xDocCfg = AcceleratorExecute::st_openDocConfig(xModel); } // SAFE -> ------------------------------ aLock.lock(); m_xGlobalCfg = std::move(xGlobalCfg); m_xModuleCfg = std::move(xModuleCfg); m_xDocCfg = std::move(xDocCfg); aLock.unlock(); // <- SAFE ---------------------------------- } bool AcceleratorExecute::execute(const vcl::KeyCode& aVCLKey) { css::awt::KeyEvent aAWTKey = AcceleratorExecute::st_VCLKey2AWTKey(aVCLKey); return execute(aAWTKey); } bool AcceleratorExecute::execute(const css::awt::KeyEvent& aAWTKey) { OUString sCommand = impl_ts_findCommand(aAWTKey); // No Command found? Do nothing! User is not interested on any error handling .-) if (sCommand.isEmpty()) return false; // SAFE -> ---------------------------------- std::unique_lock aLock(m_aLock); // or for some reason m_xContext is NULL (which would crash impl_ts_getURLParser() if (!m_xContext.is()) return false; css::uno::Reference< css::frame::XDispatchProvider > xProvider = m_xDispatcher; aLock.unlock(); // <- SAFE ---------------------------------- // convert command in URL structure css::uno::Reference< css::util::XURLTransformer > xParser = impl_ts_getURLParser(); css::util::URL aURL; aURL.Complete = sCommand; xParser->parseStrict(aURL); // ask for dispatch object css::uno::Reference< css::frame::XDispatch > xDispatch = xProvider->queryDispatch(aURL, OUString(), 0); bool bRet = xDispatch.is(); if ( bRet ) { // Note: Such instance can be used one times only and destroy itself afterwards .-) css::uno::Reference xFrame(xProvider, css::uno::UNO_QUERY); if (vcl::lok::isUnipoll()) { // tdf#130382 - all synchronous really. try { xDispatch->dispatch (aURL, css::uno::Sequence< css::beans::PropertyValue >()); } catch(const css::uno::Exception&ev) { SAL_INFO("svtools", "exception on key emission: " << ev.Message); } } else { rtl::Reference pExec = AsyncAccelExec::createOneShotInstance(xFrame, xDispatch, aURL); pExec->execAsync(); } } return bRet; } css::awt::KeyEvent AcceleratorExecute::st_VCLKey2AWTKey(const vcl::KeyCode& aVCLKey) { css::awt::KeyEvent aAWTKey; aAWTKey.Modifiers = 0; aAWTKey.KeyCode = static_cast(aVCLKey.GetCode()); if (aVCLKey.IsShift()) aAWTKey.Modifiers |= css::awt::KeyModifier::SHIFT; if (aVCLKey.IsMod1()) aAWTKey.Modifiers |= css::awt::KeyModifier::MOD1; if (aVCLKey.IsMod2()) aAWTKey.Modifiers |= css::awt::KeyModifier::MOD2; if (aVCLKey.IsMod3()) aAWTKey.Modifiers |= css::awt::KeyModifier::MOD3; return aAWTKey; } vcl::KeyCode AcceleratorExecute::st_AWTKey2VCLKey(const css::awt::KeyEvent& aAWTKey) { bool bShift = ((aAWTKey.Modifiers & css::awt::KeyModifier::SHIFT) == css::awt::KeyModifier::SHIFT ); bool bMod1 = ((aAWTKey.Modifiers & css::awt::KeyModifier::MOD1 ) == css::awt::KeyModifier::MOD1 ); bool bMod2 = ((aAWTKey.Modifiers & css::awt::KeyModifier::MOD2 ) == css::awt::KeyModifier::MOD2 ); bool bMod3 = ((aAWTKey.Modifiers & css::awt::KeyModifier::MOD3 ) == css::awt::KeyModifier::MOD3 ); sal_uInt16 nKey = static_cast(aAWTKey.KeyCode); return vcl::KeyCode(nKey, bShift, bMod1, bMod2, bMod3); } OUString AcceleratorExecute::findCommand(const css::awt::KeyEvent& aKey) { return impl_ts_findCommand(aKey); } OUString AcceleratorExecute::impl_ts_findCommand(const css::awt::KeyEvent& aKey) { // SAFE -> ---------------------------------- std::unique_lock aLock(m_aLock); css::uno::Reference< css::ui::XAcceleratorConfiguration > xGlobalCfg = m_xGlobalCfg; css::uno::Reference< css::ui::XAcceleratorConfiguration > xModuleCfg = m_xModuleCfg; css::uno::Reference< css::ui::XAcceleratorConfiguration > xDocCfg = m_xDocCfg ; aLock.unlock(); // <- SAFE ---------------------------------- OUString sCommand; try { if (xDocCfg.is()) sCommand = xDocCfg->getCommandByKeyEvent(aKey); if (!sCommand.isEmpty()) return sCommand; } catch(const css::container::NoSuchElementException&) {} try { if (xModuleCfg.is()) sCommand = xModuleCfg->getCommandByKeyEvent(aKey); if (!sCommand.isEmpty()) return sCommand; } catch(const css::container::NoSuchElementException&) {} try { if (xGlobalCfg.is()) sCommand = xGlobalCfg->getCommandByKeyEvent(aKey); if (!sCommand.isEmpty()) return sCommand; } catch(const css::container::NoSuchElementException&) {} // fall back to functional key codes if( aKey.Modifiers == 0 ) { switch( aKey.KeyCode ) { case css::awt::Key::DELETE_TO_BEGIN_OF_LINE: return u".uno:DelToStartOfLine"_ustr; case css::awt::Key::DELETE_TO_END_OF_LINE: return u".uno:DelToEndOfLine"_ustr; case css::awt::Key::DELETE_TO_BEGIN_OF_PARAGRAPH: return u".uno:DelToStartOfPara"_ustr; case css::awt::Key::DELETE_TO_END_OF_PARAGRAPH: return u".uno:DelToEndOfPara"_ustr; case css::awt::Key::DELETE_WORD_BACKWARD: return u".uno:DelToStartOfWord"_ustr; case css::awt::Key::DELETE_WORD_FORWARD: return u".uno:DelToEndOfWord"_ustr; case css::awt::Key::INSERT_LINEBREAK: return u".uno:InsertLinebreak"_ustr; case css::awt::Key::INSERT_PARAGRAPH: return u".uno:InsertPara"_ustr; case css::awt::Key::MOVE_WORD_BACKWARD: return u".uno:GoToPrevWord"_ustr; case css::awt::Key::MOVE_WORD_FORWARD: return u".uno:GoToNextWord"_ustr; case css::awt::Key::MOVE_TO_BEGIN_OF_LINE: return u".uno:GoToStartOfLine"_ustr; case css::awt::Key::MOVE_TO_END_OF_LINE: return u".uno:GoToEndOfLine"_ustr; case css::awt::Key::MOVE_TO_BEGIN_OF_PARAGRAPH: return u".uno:GoToStartOfPara"_ustr; case css::awt::Key::MOVE_TO_END_OF_PARAGRAPH: return u".uno:GoToEndOfPara"_ustr; case css::awt::Key::MOVE_TO_BEGIN_OF_DOCUMENT: return u".uno:GoToStartOfDoc"_ustr; case css::awt::Key::MOVE_TO_END_OF_DOCUMENT: return u".uno:GoToEndOfDoc"_ustr; case css::awt::Key::SELECT_BACKWARD: return u".uno:CharLeftSel"_ustr; case css::awt::Key::SELECT_FORWARD: return u".uno:CharRightSel"_ustr; case css::awt::Key::SELECT_WORD_BACKWARD: return u".uno:WordLeftSel"_ustr; case css::awt::Key::SELECT_WORD_FORWARD: return u".uno:WordRightSel"_ustr; case css::awt::Key::SELECT_WORD: return u".uno:SelectWord"_ustr; case css::awt::Key::SELECT_LINE: return OUString(); case css::awt::Key::SELECT_PARAGRAPH: return u".uno:SelectText"_ustr; case css::awt::Key::SELECT_TO_BEGIN_OF_LINE: return u".uno:StartOfLineSel"_ustr; case css::awt::Key::SELECT_TO_END_OF_LINE: return u".uno:EndOfLineSel"_ustr; case css::awt::Key::SELECT_TO_BEGIN_OF_PARAGRAPH: return u".uno:StartOfParaSel"_ustr; case css::awt::Key::SELECT_TO_END_OF_PARAGRAPH: return u".uno:EndOfParaSel"_ustr; case css::awt::Key::SELECT_TO_BEGIN_OF_DOCUMENT: return u".uno:StartOfDocumentSel"_ustr; case css::awt::Key::SELECT_TO_END_OF_DOCUMENT: return u".uno:EndOfDocumentSel"_ustr; case css::awt::Key::SELECT_ALL: return u".uno:SelectAll"_ustr; default: break; } } return OUString(); } css::uno::Reference< css::ui::XAcceleratorConfiguration > AcceleratorExecute::st_openModuleConfig(const css::uno::Reference< css::uno::XComponentContext >& rxContext, const css::uno::Reference< css::frame::XFrame >& xFrame) { css::uno::Reference< css::frame::XModuleManager2 > xModuleDetection( css::frame::ModuleManager::create(rxContext)); OUString sModule; try { sModule = xModuleDetection->identify(xFrame); } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) { return css::uno::Reference< css::ui::XAcceleratorConfiguration >(); } css::uno::Reference< css::ui::XModuleUIConfigurationManagerSupplier > xUISupplier( css::ui::theModuleUIConfigurationManagerSupplier::get(rxContext) ); css::uno::Reference< css::ui::XAcceleratorConfiguration > xAccCfg; try { css::uno::Reference< css::ui::XUIConfigurationManager > xUIManager = xUISupplier->getUIConfigurationManager(sModule); xAccCfg = xUIManager->getShortCutManager(); } catch(const css::container::NoSuchElementException&) {} return xAccCfg; } css::uno::Reference AcceleratorExecute::lok_createNewAcceleratorConfiguration(const css::uno::Reference< css::uno::XComponentContext >& rxContext, const OUString& sModule) { css::uno::Reference< css::ui::XModuleUIConfigurationManagerSupplier > xUISupplier(css::ui::theModuleUIConfigurationManagerSupplier::get(rxContext)); try { css::uno::Reference xUIManager = xUISupplier->getUIConfigurationManager(sModule); css::ui::XModuleUIConfigurationManager2* t = static_cast(xUIManager.get()); // Return new short cut manager in case current view's language is different from previous ones. return t->createShortCutManager(); } catch(const css::container::NoSuchElementException&) {} return css::uno::Reference(); } void AcceleratorExecute::lok_setModuleConfig(const css::uno::Reference& acceleratorConfig) { this->m_xModuleCfg = acceleratorConfig; } css::uno::Reference< css::ui::XAcceleratorConfiguration > AcceleratorExecute::st_openDocConfig(const css::uno::Reference< css::frame::XModel >& xModel) { css::uno::Reference< css::ui::XAcceleratorConfiguration > xAccCfg; css::uno::Reference< css::ui::XUIConfigurationManagerSupplier > xUISupplier(xModel, css::uno::UNO_QUERY); if (xUISupplier.is()) { css::uno::Reference< css::ui::XUIConfigurationManager > xUIManager = xUISupplier->getUIConfigurationManager(); xAccCfg = xUIManager->getShortCutManager(); } return xAccCfg; } css::uno::Reference< css::util::XURLTransformer > AcceleratorExecute::impl_ts_getURLParser() { // SAFE -> ---------------------------------- std::unique_lock aLock(m_aLock); if (m_xURLParser.is()) return m_xURLParser; css::uno::Reference< css::uno::XComponentContext > xContext = m_xContext; aLock.unlock(); // <- SAFE ---------------------------------- css::uno::Reference< css::util::XURLTransformer > xParser = css::util::URLTransformer::create( xContext ); // SAFE -> ---------------------------------- aLock.lock(); m_xURLParser = xParser; aLock.unlock(); // <- SAFE ---------------------------------- return xParser; } AsyncAccelExec::AsyncAccelExec(css::uno::Reference xFrame, css::uno::Reference xDispatch, css::util::URL aURL) : m_xFrame(std::move(xFrame)) , m_xDispatch(std::move(xDispatch)) , m_aURL(std::move(aURL)) , m_aAsyncCallback(LINK(this, AsyncAccelExec, impl_ts_asyncCallback)) { acquire(); } rtl::Reference AsyncAccelExec::createOneShotInstance(const css::uno::Reference &xFrame, const css::uno::Reference< css::frame::XDispatch >& xDispatch, const css::util::URL& rURL) { rtl::Reference pExec = new AsyncAccelExec(xFrame, xDispatch, rURL); return pExec; } void AsyncAccelExec::execAsync() { if (m_xFrame.is()) m_xFrame->addEventListener(this); m_aAsyncCallback.Post(); } IMPL_LINK_NOARG(AsyncAccelExec, impl_ts_asyncCallback, LinkParamNone*, void) { if (m_xDispatch.is()) { try { if (m_xFrame.is()) m_xFrame->removeEventListener(this); m_xDispatch->dispatch(m_aURL, css::uno::Sequence< css::beans::PropertyValue >()); } catch(const css::uno::Exception&) { } } release(); } } // namespace svt /* vim:set shiftwidth=4 softtabstop=4 expandtab: */