/* -*- 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 "BasicPaneFactory.hxx" #include "ChildWindowPane.hxx" #include "FrameWindowPane.hxx" #include "FullScreenPane.hxx" #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::drawing::framework; using ::sd::framework::FrameworkHelper; namespace { enum PaneId { CenterPaneId, FullScreenPaneId, LeftImpressPaneId, LeftDrawPaneId }; const sal_Int32 gnConfigurationUpdateStartEvent(0); const sal_Int32 gnConfigurationUpdateEndEvent(1); } namespace sd::framework { /** Store URL, XPane reference and (local) PaneId for every pane factory that is registered at the PaneController. */ class BasicPaneFactory::PaneDescriptor { public: OUString msPaneURL; Reference mxPane; PaneId mePaneId; /** The mbReleased flag is set when the pane has been released. Some panes are just hidden and destroyed. When the pane is reused this flag is reset. */ bool mbIsReleased; bool CompareURL(std::u16string_view rsPaneURL) const { return msPaneURL == rsPaneURL; } bool ComparePane(const Reference& rxPane) const { return mxPane == rxPane; } }; class BasicPaneFactory::PaneContainer : public ::std::vector { public: PaneContainer() {} }; //===== PaneFactory =========================================================== BasicPaneFactory::BasicPaneFactory ( const Reference& rxContext) : mxComponentContext(rxContext), mpViewShellBase(nullptr), mpPaneContainer(new PaneContainer) { } BasicPaneFactory::~BasicPaneFactory() { } void BasicPaneFactory::disposing(std::unique_lock&) { Reference xCC (mxConfigurationControllerWeak); if (xCC.is()) { xCC->removeResourceFactoryForReference(this); xCC->removeConfigurationChangeListener(this); mxConfigurationControllerWeak.clear(); } for (const auto& rDescriptor : *mpPaneContainer) { if (rDescriptor.mbIsReleased) { Reference xComponent (rDescriptor.mxPane, UNO_QUERY); if (xComponent.is()) { xComponent->removeEventListener(this); xComponent->dispose(); } } } } void SAL_CALL BasicPaneFactory::initialize (const Sequence& aArguments) { if (!aArguments.hasElements()) return; try { // Get the XController from the first argument. Reference xController (aArguments[0], UNO_QUERY_THROW); // Tunnel through the controller to obtain access to the ViewShellBase. try { Reference xTunnel (xController, UNO_QUERY_THROW); if (auto pController = comphelper::getFromUnoTunnel(xTunnel)) mpViewShellBase = pController->GetViewShellBase(); } catch(RuntimeException&) {} Reference xCM (xController, UNO_QUERY_THROW); Reference xCC (xCM->getConfigurationController()); mxConfigurationControllerWeak = xCC; // Add pane factories for the two left panes (one for Impress and one for // Draw) and the center pane. if (xController.is() && xCC.is()) { PaneDescriptor aDescriptor; aDescriptor.msPaneURL = FrameworkHelper::msCenterPaneURL; aDescriptor.mePaneId = CenterPaneId; aDescriptor.mbIsReleased = false; mpPaneContainer->push_back(aDescriptor); xCC->addResourceFactory(aDescriptor.msPaneURL, this); aDescriptor.msPaneURL = FrameworkHelper::msFullScreenPaneURL; aDescriptor.mePaneId = FullScreenPaneId; mpPaneContainer->push_back(aDescriptor); xCC->addResourceFactory(aDescriptor.msPaneURL, this); aDescriptor.msPaneURL = FrameworkHelper::msLeftImpressPaneURL; aDescriptor.mePaneId = LeftImpressPaneId; mpPaneContainer->push_back(aDescriptor); xCC->addResourceFactory(aDescriptor.msPaneURL, this); aDescriptor.msPaneURL = FrameworkHelper::msLeftDrawPaneURL; aDescriptor.mePaneId = LeftDrawPaneId; mpPaneContainer->push_back(aDescriptor); xCC->addResourceFactory(aDescriptor.msPaneURL, this); } // Register as configuration change listener. if (xCC.is()) { xCC->addConfigurationChangeListener( this, FrameworkHelper::msConfigurationUpdateStartEvent, Any(gnConfigurationUpdateStartEvent)); xCC->addConfigurationChangeListener( this, FrameworkHelper::msConfigurationUpdateEndEvent, Any(gnConfigurationUpdateEndEvent)); } } catch (RuntimeException&) { Reference xCC (mxConfigurationControllerWeak); if (xCC.is()) xCC->removeResourceFactoryForReference(this); } } //===== XPaneFactory ========================================================== Reference SAL_CALL BasicPaneFactory::createResource ( const Reference& rxPaneId) { ThrowIfDisposed(); Reference xPane; // Based on the ResourceURL of the given ResourceId look up the // corresponding factory descriptor. PaneContainer::iterator iDescriptor ( ::std::find_if ( mpPaneContainer->begin(), mpPaneContainer->end(), [&] (PaneDescriptor const& rPane) { return rPane.CompareURL(rxPaneId->getResourceURL()); } )); if (iDescriptor == mpPaneContainer->end()) { // The requested pane can not be created by any of the factories // managed by the called BasicPaneFactory object. throw lang::IllegalArgumentException("BasicPaneFactory::createPane() called for unknown resource id", nullptr, 0); } if (iDescriptor->mxPane.is()) { // The pane has already been created and is still active (has // not yet been released). This should not happen. xPane = iDescriptor->mxPane; } else { // Create a new pane. switch (iDescriptor->mePaneId) { case CenterPaneId: xPane = CreateFrameWindowPane(rxPaneId); break; case FullScreenPaneId: xPane = CreateFullScreenPane(mxComponentContext, rxPaneId); break; case LeftImpressPaneId: case LeftDrawPaneId: xPane = CreateChildWindowPane( rxPaneId, *iDescriptor); break; } iDescriptor->mxPane = xPane; // Listen for the pane being disposed. Reference xComponent (xPane, UNO_QUERY); if (xComponent.is()) xComponent->addEventListener(this); } iDescriptor->mbIsReleased = false; return xPane; } void SAL_CALL BasicPaneFactory::releaseResource ( const Reference& rxPane) { ThrowIfDisposed(); // Based on the given XPane reference look up the corresponding factory // descriptor. PaneContainer::iterator iDescriptor ( ::std::find_if( mpPaneContainer->begin(), mpPaneContainer->end(), [&] (PaneDescriptor const& rPane) { return rPane.ComparePane(rxPane); } )); if (iDescriptor == mpPaneContainer->end()) { // The given XPane reference is either empty or the pane was not // created by any of the factories managed by the called // BasicPaneFactory object. throw lang::IllegalArgumentException("BasicPaneFactory::releasePane() called for pane that was not created by same factory.", nullptr, 0); } // The given pane was created by one of the factories. Child // windows are just hidden and will be reused when requested later. // Other windows are disposed and their reference is reset so that // on the next createPane() call for the same pane type the pane is // created anew. ChildWindowPane* pChildWindowPane = dynamic_cast(rxPane.get()); if (pChildWindowPane != nullptr) { iDescriptor->mbIsReleased = true; pChildWindowPane->Hide(); } else { iDescriptor->mxPane = nullptr; Reference xComponent (rxPane, UNO_QUERY); if (xComponent.is()) { // We are disposing the pane and do not have to be informed of // that. xComponent->removeEventListener(this); xComponent->dispose(); } } } //===== XConfigurationChangeListener ========================================== void SAL_CALL BasicPaneFactory::notifyConfigurationChange ( const ConfigurationChangeEvent& /* rEvent */ ) { // FIXME: nothing to do } //===== lang::XEventListener ================================================== void SAL_CALL BasicPaneFactory::disposing ( const lang::EventObject& rEventObject) { if (mxConfigurationControllerWeak.get() == rEventObject.Source) { mxConfigurationControllerWeak.clear(); } else { // Has one of the panes been disposed? If so, then release the // reference to that pane, but not the pane descriptor. Reference xPane (rEventObject.Source, UNO_QUERY); PaneContainer::iterator iDescriptor ( ::std::find_if ( mpPaneContainer->begin(), mpPaneContainer->end(), [&] (PaneDescriptor const& rPane) { return rPane.ComparePane(xPane); } )); if (iDescriptor != mpPaneContainer->end()) { iDescriptor->mxPane = nullptr; } } } Reference BasicPaneFactory::CreateFrameWindowPane ( const Reference& rxPaneId) { Reference xPane; if (mpViewShellBase != nullptr) { xPane = new FrameWindowPane(rxPaneId, mpViewShellBase->GetViewWindow()); } return xPane; } Reference BasicPaneFactory::CreateFullScreenPane ( const Reference& rxComponentContext, const Reference& rxPaneId) { Reference xPane ( new FullScreenPane( rxComponentContext, rxPaneId, mpViewShellBase->GetViewWindow())); return xPane; } Reference BasicPaneFactory::CreateChildWindowPane ( const Reference& rxPaneId, const PaneDescriptor& rDescriptor) { Reference xPane; if (mpViewShellBase != nullptr) { // Create the corresponding shell and determine the id of the child window. sal_uInt16 nChildWindowId = 0; ::std::unique_ptr pShell; switch (rDescriptor.mePaneId) { case LeftImpressPaneId: pShell.reset(new LeftImpressPaneShell()); nChildWindowId = ::sd::LeftPaneImpressChildWindow::GetChildWindowId(); break; case LeftDrawPaneId: pShell.reset(new LeftDrawPaneShell()); nChildWindowId = ::sd::LeftPaneDrawChildWindow::GetChildWindowId(); break; default: break; } // With shell and child window id create the ChildWindowPane // wrapper. if (pShell != nullptr) { xPane = new ChildWindowPane( rxPaneId, nChildWindowId, *mpViewShellBase, std::move(pShell)); } } return xPane; } void BasicPaneFactory::ThrowIfDisposed() const { if (m_bDisposed) { throw lang::DisposedException ("BasicPaneFactory object has already been disposed", const_cast(static_cast(this))); } } } // end of namespace sd::framework extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface* com_sun_star_comp_Draw_framework_BasicPaneFactory_get_implementation(css::uno::XComponentContext* context, css::uno::Sequence const &) { return cppu::acquire(new sd::framework::BasicPaneFactory(context)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */