When destroying the static vcl::CommandInfoProvider aProvider from vcl::CommandInfoProvider::Instance (vcl/source/helper/commandinfoprovider.cxx) during exit, it releases its mxCachedGlobalAcceleratorConfiguration reference on GlobalAcceleratorConfiguration (framework/source/accelerators/globalacceleratorconfiguration.cxx), which may get destroyed, whose base class framework::XCUBasedAcceleratorConfiguration (framework/source/inc/accelerators/acceleratorconfiguration.hxx) has a salhelper::SingletonRef<framework::KeyMapping> member, whose destructor (include/salhelper/singletonref.hxx) uses salhelper::SingletonRef<framework::KeyMapping>::SingletonLockInit::operator ()'s static osl::Mutex aInstance. If, during construction, the instantiation of salhelper::SingletonRef<framework::KeyMapping>::SingletonLockInit::operator ()'s static osl::Mutex aInstance finishes before the instantiation of vcl::CommandInfoProvider::Instance's static vcl::CommandInfoProvider aProvider, the corresponding atexit cleanup actions will be recorded in the right order, causing the above chain of calls to find the static Mutex still alive when used from within the static CommandInfoProvider's destruction. However, vcl::CommandInfoProvider's mxCachedGlobalAcceleratorConfiguration is only set to css::ui::GlobalAcceleratorConfiguration::create in vcl::CommandInfoProvider::GetGlobalAcceleratorConfiguration, so the instantiation of the static Mutex instance can finish after the instantiation of the static CommandInfoProvider instance, recording the atexit cleanup actions in the wrong order, causing the static Mutex to be used after destruction. This occasionally caused PythonTest_sfx2_python to hang during exit for me on Linux, where trying to lock a destroyed pthread mutex can apparently deadlock. rtl::Static does away with the need to do anything in the destructor, at the expense of always keeping the instance alive until exit (and not being able to recreate an already destroyed instance during exit, but code that would require that behavior would probably already be broken to begin with), so the order of creation of the CommandInfoProvider and GlobalAcceleratorConfiguration instances becomes less of a concern. Change-Id: Id6e3860ad9e5b7045980a0b9bf9eaef2e24129bb
391 lines
17 KiB
C++
391 lines
17 KiB
C++
/* -*- 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 .
|
|
*/
|
|
|
|
#ifndef INCLUDED_FRAMEWORK_SOURCE_INC_ACCELERATORS_ACCELERATORCONFIGURATION_HXX
|
|
#define INCLUDED_FRAMEWORK_SOURCE_INC_ACCELERATORS_ACCELERATORCONFIGURATION_HXX
|
|
|
|
#include <accelerators/presethandler.hxx>
|
|
#include <accelerators/acceleratorcache.hxx>
|
|
#include <macros/xinterface.hxx>
|
|
#include <macros/xtypeprovider.hxx>
|
|
#include <general.h>
|
|
#include <stdtypes.h>
|
|
|
|
#include <com/sun/star/container/XNameAccess.hpp>
|
|
#include <com/sun/star/lang/XTypeProvider.hpp>
|
|
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
|
|
#include <com/sun/star/ui/XAcceleratorConfiguration.hpp>
|
|
#include <com/sun/star/ui/XUIConfiguration.hpp>
|
|
#include <com/sun/star/ui/XUIConfigurationPersistence.hpp>
|
|
|
|
#include <com/sun/star/ui/XUIConfigurationStorage.hpp>
|
|
#include <com/sun/star/io/XStream.hpp>
|
|
#include <com/sun/star/io/XInputStream.hpp>
|
|
#include <com/sun/star/io/XOutputStream.hpp>
|
|
#include <com/sun/star/util/XChangesListener.hpp>
|
|
|
|
// TODO use XPresetHandler interface instead if available
|
|
#include <com/sun/star/form/XReset.hpp>
|
|
|
|
#include <cppuhelper/propshlp.hxx>
|
|
#include <cppuhelper/implbase.hxx>
|
|
|
|
// definition
|
|
|
|
namespace framework
|
|
{
|
|
|
|
const char CFG_ENTRY_PRIMARY[] = "PrimaryKeys";
|
|
const char CFG_ENTRY_GLOBAL[] = "Global";
|
|
const char CFG_ENTRY_MODULES[] = "Modules";
|
|
|
|
/**
|
|
implements a read/write access to the accelerator configuration.
|
|
*/
|
|
class XMLBasedAcceleratorConfiguration : public ::cppu::WeakImplHelper<
|
|
css::form::XReset, // TODO use XPresetHandler instead if available
|
|
css::ui::XAcceleratorConfiguration > // => css::ui::XUIConfigurationPersistence
|
|
// css::ui::XUIConfigurationStorage
|
|
// css::ui::XUIConfiguration
|
|
{
|
|
|
|
// member
|
|
|
|
protected:
|
|
|
|
/** the global uno service manager.
|
|
Must be used to create own needed services. */
|
|
css::uno::Reference< css::uno::XComponentContext > m_xContext;
|
|
|
|
/** used to:
|
|
i ) copy configuration files from the share to the user layer
|
|
ii ) provide access to these config files
|
|
iii) cache all sub storages on the path from the top to the bottom(!)
|
|
iv ) provide commit for changes. */
|
|
PresetHandler m_aPresetHandler;
|
|
|
|
/** contains the cached configuration data */
|
|
AcceleratorCache m_aReadCache;
|
|
|
|
/** used to implement the copy on write pattern! */
|
|
AcceleratorCache* m_pWriteCache;
|
|
|
|
// native interface!
|
|
|
|
public:
|
|
|
|
XMLBasedAcceleratorConfiguration( const css::uno::Reference< css::uno::XComponentContext >& xContext);
|
|
virtual ~XMLBasedAcceleratorConfiguration( );
|
|
|
|
// uno interface!
|
|
|
|
public:
|
|
|
|
// XAcceleratorConfiguration
|
|
virtual css::uno::Sequence< css::awt::KeyEvent > SAL_CALL getAllKeyEvents()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual OUString SAL_CALL getCommandByKeyEvent(const css::awt::KeyEvent& aKeyEvent)
|
|
throw(css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
virtual void SAL_CALL setKeyEvent(const css::awt::KeyEvent& aKeyEvent,
|
|
const OUString& sCommand )
|
|
throw(css::lang::IllegalArgumentException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
virtual void SAL_CALL removeKeyEvent(const css::awt::KeyEvent& aKeyEvent)
|
|
throw(css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
virtual css::uno::Sequence< css::awt::KeyEvent > SAL_CALL getKeyEventsByCommand(const OUString& sCommand)
|
|
throw(css::lang::IllegalArgumentException ,
|
|
css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
virtual css::uno::Sequence< css::uno::Any > SAL_CALL getPreferredKeyEventsForCommandList(const css::uno::Sequence< OUString >& lCommandList)
|
|
throw (css::lang::IllegalArgumentException, css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL removeCommandFromAllKeyEvents(const OUString& sCommand)
|
|
throw(css::lang::IllegalArgumentException ,
|
|
css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
// XUIConfigurationPersistence
|
|
virtual void SAL_CALL reload()
|
|
throw(css::uno::Exception ,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL store()
|
|
throw(css::uno::Exception ,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL storeToStorage(const css::uno::Reference< css::embed::XStorage >& xStorage)
|
|
throw(css::uno::Exception ,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual sal_Bool SAL_CALL isModified()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual sal_Bool SAL_CALL isReadOnly()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// XUIConfigurationStorage
|
|
virtual void SAL_CALL setStorage(const css::uno::Reference< css::embed::XStorage >& xStorage)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual sal_Bool SAL_CALL hasStorage()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// XUIConfiguration
|
|
virtual void SAL_CALL addConfigurationListener(const css::uno::Reference< css::ui::XUIConfigurationListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL removeConfigurationListener(const css::uno::Reference< css::ui::XUIConfigurationListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// XReset
|
|
// TODO use XPresetHandler instead if available
|
|
virtual void SAL_CALL reset()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL addResetListener(const css::uno::Reference< css::form::XResetListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL removeResetListener(const css::uno::Reference< css::form::XResetListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// called when changes occurred in the storage
|
|
void changesOccurred();
|
|
|
|
// helper for derived classes
|
|
|
|
protected:
|
|
|
|
/** @short return the current office locale.
|
|
|
|
@descr We do not cache this value, because we are not listen
|
|
for changes on the configuration layer ...
|
|
|
|
@return OUString
|
|
The current office locale as BCP47 string.
|
|
*/
|
|
OUString impl_ts_getLocale() const;
|
|
|
|
// helper
|
|
|
|
private:
|
|
|
|
/** @short load a configuration set, using the given stream.
|
|
|
|
@param xStream
|
|
provides the XML structure as stream.
|
|
*/
|
|
void impl_ts_load(const css::uno::Reference< css::io::XInputStream >& xStream);
|
|
|
|
/** @short save a configuration set, using the given stream.
|
|
|
|
@param xStream
|
|
the XML structure can be written there.
|
|
*/
|
|
void impl_ts_save(const css::uno::Reference< css::io::XOutputStream >& xStream);
|
|
|
|
/** @short returns a reference to one of our internal cache members.
|
|
|
|
@descr We implement the copy-on-write pattern. Doing so
|
|
we know two caches internally. The second one is used
|
|
only, if the container was changed.
|
|
|
|
This method here returns access to one of these
|
|
caches - depending on the change state of this
|
|
configuration service.
|
|
|
|
@param bWriteAccessRequested
|
|
if the outside code wish to change the container
|
|
it must call this method with "sal_True". So the internal
|
|
cache can be prepared for that (means copy-on-write ...).
|
|
|
|
@return [AcceleratorCache]
|
|
c++ reference(!) to one of our internal caches.
|
|
*/
|
|
AcceleratorCache& impl_getCFG(bool bWriteAccessRequested = false);
|
|
|
|
};
|
|
|
|
class XCUBasedAcceleratorConfiguration : public ::cppu::WeakImplHelper<
|
|
css::util::XChangesListener,
|
|
css::lang::XComponent,
|
|
css::form::XReset, // TODO use XPresetHandler instead if available
|
|
css::ui::XAcceleratorConfiguration > // => css::ui::XUIConfigurationPersistence
|
|
// css::ui::XUIConfigurationStorage
|
|
// css::ui::XUIConfiguration
|
|
{
|
|
|
|
// member
|
|
|
|
protected:
|
|
|
|
/** the global uno service manager.
|
|
Must be used to create own needed services. */
|
|
css::uno::Reference< css::uno::XComponentContext > m_xContext;
|
|
|
|
css::uno::Reference< css::container::XNameAccess > m_xCfg;
|
|
AcceleratorCache m_aPrimaryReadCache;
|
|
AcceleratorCache m_aSecondaryReadCache;
|
|
AcceleratorCache* m_pPrimaryWriteCache;
|
|
AcceleratorCache* m_pSecondaryWriteCache;
|
|
|
|
OUString m_sGlobalOrModules;
|
|
OUString m_sModuleCFG;
|
|
|
|
// native interface!
|
|
|
|
public:
|
|
|
|
XCUBasedAcceleratorConfiguration( const css::uno::Reference< css::uno::XComponentContext >& xContext );
|
|
virtual ~XCUBasedAcceleratorConfiguration( );
|
|
|
|
// uno interface!
|
|
|
|
public:
|
|
|
|
// XAcceleratorConfiguration
|
|
virtual css::uno::Sequence< css::awt::KeyEvent > SAL_CALL getAllKeyEvents()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual OUString SAL_CALL getCommandByKeyEvent(const css::awt::KeyEvent& aKeyEvent)
|
|
throw(css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
virtual void SAL_CALL setKeyEvent(const css::awt::KeyEvent& aKeyEvent,
|
|
const OUString& sCommand )
|
|
throw (css::lang::IllegalArgumentException, css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL removeKeyEvent(const css::awt::KeyEvent& aKeyEvent)
|
|
throw(css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
virtual css::uno::Sequence< css::awt::KeyEvent > SAL_CALL getKeyEventsByCommand(const OUString& sCommand)
|
|
throw(css::lang::IllegalArgumentException ,
|
|
css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
virtual css::uno::Sequence< css::uno::Any > SAL_CALL getPreferredKeyEventsForCommandList(const css::uno::Sequence< OUString >& lCommandList)
|
|
throw(css::lang::IllegalArgumentException, css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL removeCommandFromAllKeyEvents(const OUString& sCommand)
|
|
throw(css::lang::IllegalArgumentException ,
|
|
css::container::NoSuchElementException,
|
|
css::uno::RuntimeException, std::exception ) override;
|
|
|
|
// XUIConfigurationPersistence
|
|
virtual void SAL_CALL reload()
|
|
throw(css::uno::Exception ,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL store()
|
|
throw(css::uno::Exception ,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL storeToStorage(const css::uno::Reference< css::embed::XStorage >& xStorage)
|
|
throw(css::uno::Exception ,
|
|
css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual sal_Bool SAL_CALL isModified()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual sal_Bool SAL_CALL isReadOnly()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// XUIConfigurationStorage
|
|
virtual void SAL_CALL setStorage(const css::uno::Reference< css::embed::XStorage >& xStorage)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual sal_Bool SAL_CALL hasStorage()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// XUIConfiguration
|
|
virtual void SAL_CALL addConfigurationListener(const css::uno::Reference< css::ui::XUIConfigurationListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL removeConfigurationListener(const css::uno::Reference< css::ui::XUIConfigurationListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// XReset
|
|
// TODO use XPresetHandler instead if available
|
|
virtual void SAL_CALL reset()
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL addResetListener(const css::uno::Reference< css::form::XResetListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
virtual void SAL_CALL removeResetListener(const css::uno::Reference< css::form::XResetListener >& xListener)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// css.util.XChangesListener
|
|
virtual void SAL_CALL changesOccurred(const css::util::ChangesEvent& aEvent)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// css.lang.XEventListener
|
|
virtual void SAL_CALL disposing(const css::lang::EventObject& aEvent)
|
|
throw(css::uno::RuntimeException, std::exception) override;
|
|
|
|
// XComponent
|
|
virtual void SAL_CALL dispose() throw (css::uno::RuntimeException, std::exception) override;
|
|
virtual void SAL_CALL addEventListener( const css::uno::Reference< css::lang::XEventListener >& xListener ) throw (css::uno::RuntimeException, std::exception) override;
|
|
virtual void SAL_CALL removeEventListener( const css::uno::Reference< css::lang::XEventListener >& aListener ) throw (css::uno::RuntimeException, std::exception) override;
|
|
|
|
// helper for derived classes
|
|
|
|
protected:
|
|
|
|
/** @short return the current office locale.
|
|
|
|
@descr We do not cache this value, because we are not listen
|
|
for changes on the configuration layer ...
|
|
|
|
@return OUString
|
|
The current office locale as BCP47 string.
|
|
*/
|
|
OUString impl_ts_getLocale() const;
|
|
|
|
// helper
|
|
|
|
private:
|
|
|
|
void impl_ts_load(bool bPreferred, const css::uno::Reference< css::container::XNameAccess >& xCfg);
|
|
void impl_ts_save(bool bPreferred, const css::uno::Reference< css::container::XNameAccess >& xCfg);
|
|
|
|
void insertKeyToConfiguration(const css::awt::KeyEvent& aKeyEvent, const OUString& sCommand, const bool bPreferred);
|
|
void removeKeyFromConfiguration(const css::awt::KeyEvent& aKeyEvent, const bool bPreferred);
|
|
|
|
void reloadChanged(const OUString& sPrimarySecondary, const OUString& sGlobalModules, const OUString& sModule, const OUString& sKey);
|
|
AcceleratorCache& impl_getCFG(bool bPreferred, bool bWriteAccessRequested = false);
|
|
|
|
};
|
|
|
|
} // namespace framework
|
|
|
|
#endif // INCLUDED_FRAMEWORK_SOURCE_INC_ACCELERATORS_ACCELERATORCONFIGURATION_HXX
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|