/* -*- 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 namespace framework{ WindowCommandDispatch::WindowCommandDispatch(const css::uno::Reference< css::uno::XComponentContext >& xContext , const css::uno::Reference< css::frame::XFrame >& xFrame) : m_xContext (xContext ) , m_xFrame (xFrame ) , m_xWindow (xFrame->getContainerWindow()) { impl_startListening(); } WindowCommandDispatch::~WindowCommandDispatch() { impl_stopListening(); m_xContext.clear(); } void WindowCommandDispatch::impl_startListening() { osl::ClearableMutexGuard aReadLock(m_mutex); css::uno::Reference< css::awt::XWindow > xWindow( m_xWindow.get(), css::uno::UNO_QUERY ); aReadLock.clear(); if ( ! xWindow.is()) return; { SolarMutexGuard aSolarLock; vcl::Window* pWindow = VCLUnoHelper::GetWindow(xWindow); if ( ! pWindow) return; pWindow->AddEventListener( LINK(this, WindowCommandDispatch, impl_notifyCommand) ); } } void WindowCommandDispatch::impl_stopListening() { osl::ClearableMutexGuard aReadLock(m_mutex); css::uno::Reference< css::awt::XWindow > xWindow( m_xWindow.get(), css::uno::UNO_QUERY ); aReadLock.clear(); if (!xWindow.is()) return; { SolarMutexGuard aSolarLock; vcl::Window* pWindow = VCLUnoHelper::GetWindow(xWindow); if (!pWindow) return; pWindow->RemoveEventListener( LINK(this, WindowCommandDispatch, impl_notifyCommand) ); m_xWindow.clear(); } } IMPL_LINK_TYPED(WindowCommandDispatch, impl_notifyCommand, VclWindowEvent&, rEvent, void) { if (rEvent.GetId() == VCLEVENT_OBJECT_DYING) { impl_stopListening(); return; } if (rEvent.GetId() != VCLEVENT_WINDOW_COMMAND) return; const CommandEvent* pCommand = static_cast(rEvent.GetData()); if (pCommand->GetCommand() != CommandEventId::ShowDialog) return; const CommandDialogData* pData = pCommand->GetDialogData(); if ( ! pData) return; const ShowDialogId nCommand = pData->GetDialogId(); OUString sCommand; switch (nCommand) { case ShowDialogId::Preferences : sCommand = ".uno:OptionsTreeDialog"; break; case ShowDialogId::About : sCommand = ".uno:About"; break; default : return; } impl_dispatchCommand(sCommand); } void WindowCommandDispatch::impl_dispatchCommand(const OUString& sCommand) { // ignore all errors here. It's clicking a menu entry only ... // The user will try it again, in case nothing happens .-) try { // SYNCHRONIZED -> osl::ClearableMutexGuard aReadLock(m_mutex); css::uno::Reference< css::frame::XDispatchProvider > xProvider(m_xFrame.get(), css::uno::UNO_QUERY_THROW); css::uno::Reference< css::uno::XComponentContext > xContext = m_xContext; aReadLock.clear(); // <- SYNCHRONIZED // check provider ... we know it's weak reference only if ( ! xProvider.is()) return; css::uno::Reference< css::util::XURLTransformer > xParser(css::util::URLTransformer::create(xContext)); css::util::URL aCommand; aCommand.Complete = sCommand; xParser->parseStrict(aCommand); css::uno::Reference< css::frame::XDispatch > xDispatch = xProvider->queryDispatch(aCommand, SPECIALTARGET_SELF, 0); if (xDispatch.is()) xDispatch->dispatch(aCommand, css::uno::Sequence< css::beans::PropertyValue >()); } catch(const css::uno::Exception&) {} } } // namespace framework /* vim:set shiftwidth=4 softtabstop=4 expandtab: */