/* -*- 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 "ViewTabBar.hxx" #include "ViewShell.hxx" #include "ViewShellBase.hxx" #include "DrawViewShell.hxx" #include "FrameView.hxx" #include "framework/FrameworkHelper.hxx" #include "framework/Pane.hxx" #include "DrawController.hxx" #include "sdresid.hxx" #include "strings.hrc" #include "helpids.h" #include "Client.hxx" #include <vcl/svapp.hxx> #include <vcl/tabpage.hxx> #include <vcl/settings.hxx> #include <osl/mutex.hxx> #include <sfx2/viewfrm.hxx> #include <com/sun/star/drawing/framework/ResourceId.hpp> #include <com/sun/star/drawing/framework/XControllerManager.hpp> #include <com/sun/star/lang/XUnoTunnel.hpp> #include <com/sun/star/lang/DisposedException.hpp> #include <comphelper/processfactory.hxx> #include <comphelper/servicehelper.hxx> #include <tools/diagnose_ex.h> using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::drawing::framework; using ::sd::framework::FrameworkHelper; namespace sd { namespace { bool IsEqual (const TabBarButton& rButton1, const TabBarButton& rButton2) { return ((rButton1.ResourceId.is() && rButton2.ResourceId.is() && rButton1.ResourceId->compareTo(rButton2.ResourceId) == 0) || rButton1.ButtonLabel == rButton2.ButtonLabel); } class TabBarControl : public ::TabControl { public: TabBarControl (vcl::Window* pParentWindow, const ::rtl::Reference<ViewTabBar>& rpViewTabBar); virtual void Paint (vcl::RenderContext& rRenderContext, const Rectangle& rRect) override; virtual void ActivatePage() override; private: ::rtl::Reference<ViewTabBar> mpViewTabBar; }; } // end of anonymous namespace ViewTabBar::ViewTabBar ( const Reference<XResourceId>& rxViewTabBarId, const Reference<frame::XController>& rxController) : ViewTabBarInterfaceBase(maMutex), mpTabControl(VclPtr<TabBarControl>::Create(GetAnchorWindow(rxViewTabBarId,rxController), this)), mxController(rxController), maTabBarButtons(), mpTabPage(nullptr), mxViewTabBarId(rxViewTabBarId), mpViewShellBase(nullptr) { // Set one new tab page for all tab entries. We need it only to // determine the height of the tab bar. mpTabPage.reset(VclPtr<TabPage>::Create(mpTabControl.get())); mpTabPage->Hide(); // add some space before the tabitems mpTabControl->SetItemsOffset(Point(5, 3)); // Tunnel through the controller and use the ViewShellBase to obtain the // view frame. try { Reference<lang::XUnoTunnel> xTunnel (mxController, UNO_QUERY_THROW); DrawController* pController = reinterpret_cast<DrawController*>( xTunnel->getSomething(DrawController::getUnoTunnelId())); mpViewShellBase = pController->GetViewShellBase(); } catch (const RuntimeException&) { } // Register as listener at XConfigurationController. Reference<XControllerManager> xControllerManager (mxController, UNO_QUERY); if (xControllerManager.is()) { mxConfigurationController = xControllerManager->getConfigurationController(); if (mxConfigurationController.is()) { mxConfigurationController->addConfigurationChangeListener( this, FrameworkHelper::msResourceActivationEvent, Any()); } } mpTabControl->Show(); if (mpViewShellBase != nullptr && rxViewTabBarId->isBoundToURL( FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT)) { mpViewShellBase->SetViewTabBar(this); } } ViewTabBar::~ViewTabBar() { } void ViewTabBar::disposing() { if (mpViewShellBase != nullptr && mxViewTabBarId->isBoundToURL( FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT)) { mpViewShellBase->SetViewTabBar(nullptr); } if (mxConfigurationController.is()) { // Unregister listener from XConfigurationController. try { mxConfigurationController->removeConfigurationChangeListener(this); } catch (const lang::DisposedException&) { // Receiving a disposed exception is the normal case. Is there // a way to avoid it? } mxConfigurationController = nullptr; } { const SolarMutexGuard aSolarGuard; // Set all references to the one tab page to NULL and delete the page. for (sal_uInt16 nIndex=0; nIndex<mpTabControl->GetPageCount(); ++nIndex) mpTabControl->SetTabPage(nIndex, nullptr); mpTabPage.disposeAndClear(); mpTabControl.disposeAndClear(); } mxController = nullptr; } vcl::Window* ViewTabBar::GetAnchorWindow( const Reference<XResourceId>& rxViewTabBarId, const Reference<frame::XController>& rxController) { vcl::Window* pWindow = nullptr; ViewShellBase* pBase = nullptr; // Tunnel through the controller and use the ViewShellBase to obtain the // view frame. try { Reference<lang::XUnoTunnel> xTunnel (rxController, UNO_QUERY_THROW); DrawController* pController = reinterpret_cast<DrawController*>( xTunnel->getSomething(DrawController::getUnoTunnelId())); pBase = pController->GetViewShellBase(); } catch (const RuntimeException&) { } // The ViewTabBar supports at the moment only the center pane. if (rxViewTabBarId.is() && rxViewTabBarId->isBoundToURL( FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT)) { if (pBase != nullptr && pBase->GetViewFrame() != nullptr) pWindow = &pBase->GetViewFrame()->GetWindow(); } // The rest is (at the moment) just for the emergency case. if (pWindow == nullptr) { Reference<XPane> xPane; try { Reference<XControllerManager> xControllerManager (rxController, UNO_QUERY_THROW); Reference<XConfigurationController> xCC ( xControllerManager->getConfigurationController()); if (xCC.is()) xPane.set(xCC->getResource(rxViewTabBarId->getAnchor()), UNO_QUERY); } catch (const RuntimeException&) { } // Tunnel through the XWindow to the VCL side. try { Reference<lang::XUnoTunnel> xTunnel (xPane, UNO_QUERY_THROW); framework::Pane* pPane = reinterpret_cast<framework::Pane*>( xTunnel->getSomething(framework::Pane::getUnoTunnelId())); if (pPane != nullptr) pWindow = pPane->GetWindow()->GetParent(); } catch (const RuntimeException&) { } } return pWindow; } //----- XConfigurationChangeListener ------------------------------------------ void SAL_CALL ViewTabBar::notifyConfigurationChange ( const ConfigurationChangeEvent& rEvent) throw (RuntimeException, std::exception) { if (rEvent.Type.equals(FrameworkHelper::msResourceActivationEvent) && rEvent.ResourceId->getResourceURL().match(FrameworkHelper::msViewURLPrefix) && rEvent.ResourceId->isBoundTo(mxViewTabBarId->getAnchor(), AnchorBindingMode_DIRECT)) { UpdateActiveButton(); } } //----- XEventListener -------------------------------------------------------- void SAL_CALL ViewTabBar::disposing( const lang::EventObject& rEvent) throw (RuntimeException, std::exception) { if (rEvent.Source == mxConfigurationController) { mxConfigurationController = nullptr; mxController = nullptr; } } //----- XTabBar --------------------------------------------------------------- void SAL_CALL ViewTabBar::addTabBarButtonAfter ( const TabBarButton& rButton, const TabBarButton& rAnchor) throw (css::uno::RuntimeException, std::exception) { const SolarMutexGuard aSolarGuard; AddTabBarButton(rButton, rAnchor); } void SAL_CALL ViewTabBar::appendTabBarButton (const TabBarButton& rButton) throw (css::uno::RuntimeException, std::exception) { const SolarMutexGuard aSolarGuard; AddTabBarButton(rButton); } void SAL_CALL ViewTabBar::removeTabBarButton (const TabBarButton& rButton) throw (css::uno::RuntimeException, std::exception) { const SolarMutexGuard aSolarGuard; RemoveTabBarButton(rButton); } sal_Bool SAL_CALL ViewTabBar::hasTabBarButton (const TabBarButton& rButton) throw (css::uno::RuntimeException, std::exception) { const SolarMutexGuard aSolarGuard; return HasTabBarButton(rButton); } Sequence<TabBarButton> SAL_CALL ViewTabBar::getTabBarButtons() throw (css::uno::RuntimeException, std::exception) { const SolarMutexGuard aSolarGuard; return GetTabBarButtons(); } //----- XResource ------------------------------------------------------------- Reference<XResourceId> SAL_CALL ViewTabBar::getResourceId() throw (RuntimeException, std::exception) { return mxViewTabBarId; } sal_Bool SAL_CALL ViewTabBar::isAnchorOnly() throw (RuntimeException, std::exception) { return false; } //----- XUnoTunnel ------------------------------------------------------------ namespace { class theViewTabBarUnoTunnelId : public rtl::Static< UnoTunnelIdInit, theViewTabBarUnoTunnelId > {}; } const Sequence<sal_Int8>& ViewTabBar::getUnoTunnelId() { return theViewTabBarUnoTunnelId::get().getSeq(); } sal_Int64 SAL_CALL ViewTabBar::getSomething (const Sequence<sal_Int8>& rId) throw (RuntimeException, std::exception) { sal_Int64 nResult = 0; if (rId.getLength() == 16 && memcmp(getUnoTunnelId().getConstArray(), rId.getConstArray(), 16) == 0) { nResult = reinterpret_cast<sal_Int64>(this); } return nResult; } bool ViewTabBar::ActivatePage() { try { Reference<XControllerManager> xControllerManager (mxController,UNO_QUERY_THROW); Reference<XConfigurationController> xConfigurationController ( xControllerManager->getConfigurationController()); if ( ! xConfigurationController.is()) throw RuntimeException(); Reference<XView> xView; try { xView.set(xConfigurationController->getResource( ResourceId::create( ::comphelper::getProcessComponentContext(), FrameworkHelper::msCenterPaneURL)), UNO_QUERY); } catch (const DeploymentException&) { } Client* pIPClient = nullptr; if (mpViewShellBase != nullptr) pIPClient = dynamic_cast<Client*>(mpViewShellBase->GetIPClient()); if (pIPClient==nullptr || ! pIPClient->IsObjectInPlaceActive()) { sal_uInt16 nIndex (mpTabControl->GetCurPageId() - 1); if (nIndex < maTabBarButtons.size()) { xConfigurationController->requestResourceActivation( maTabBarButtons[nIndex].ResourceId, ResourceActivationMode_REPLACE); } return true; } else { // When we run into this else branch then we have an active OLE // object. We ignore the request to switch views. Additionally // we put the active tab back to the one for the current view. UpdateActiveButton(); } } catch (const RuntimeException&) { DBG_UNHANDLED_EXCEPTION(); } return false; } int ViewTabBar::GetHeight() { int nHeight (0); if (!maTabBarButtons.empty()) { TabPage* pActivePage (mpTabControl->GetTabPage( mpTabControl->GetCurPageId())); if (pActivePage!=nullptr && mpTabControl->IsReallyVisible()) nHeight = pActivePage->GetPosPixel().Y(); if (nHeight <= 0) // Using a default when the real height can not be determined. // To get correct height this method should be called when the // control is visible. nHeight = 21; } return nHeight; } void ViewTabBar::AddTabBarButton ( const css::drawing::framework::TabBarButton& rButton, const css::drawing::framework::TabBarButton& rAnchor) { sal_uInt32 nIndex; if ( ! rAnchor.ResourceId.is() || (rAnchor.ResourceId->getResourceURL().isEmpty() && rAnchor.ButtonLabel.isEmpty())) { nIndex = 0; } else { for (nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex) { if (IsEqual(maTabBarButtons[nIndex], rAnchor)) { ++nIndex; break; } } } AddTabBarButton(rButton,nIndex); } void ViewTabBar::AddTabBarButton ( const css::drawing::framework::TabBarButton& rButton) { AddTabBarButton(rButton, maTabBarButtons.size()); } void ViewTabBar::AddTabBarButton ( const css::drawing::framework::TabBarButton& rButton, sal_Int32 nPosition) { if (nPosition>=0 && nPosition<=mpTabControl->GetPageCount()) { sal_uInt16 nIndex ((sal_uInt16)nPosition); // Insert the button into our local array. maTabBarButtons.insert(maTabBarButtons.begin()+nIndex, rButton); UpdateTabBarButtons(); UpdateActiveButton(); } } void ViewTabBar::RemoveTabBarButton ( const css::drawing::framework::TabBarButton& rButton) { sal_uInt16 nIndex; for (nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex) { if (IsEqual(maTabBarButtons[nIndex], rButton)) { maTabBarButtons.erase(maTabBarButtons.begin()+nIndex); UpdateTabBarButtons(); UpdateActiveButton(); break; } } } bool ViewTabBar::HasTabBarButton ( const css::drawing::framework::TabBarButton& rButton) { bool bResult (false); for (css::drawing::framework::TabBarButton & r : maTabBarButtons) { if (IsEqual(r, rButton)) { bResult = true; break; } } return bResult; } css::uno::Sequence<css::drawing::framework::TabBarButton> ViewTabBar::GetTabBarButtons() { sal_uInt32 nCount (maTabBarButtons.size()); css::uno::Sequence<css::drawing::framework::TabBarButton> aList (nCount); for (sal_uInt32 nIndex=0; nIndex<nCount; ++nIndex) aList[nIndex] = maTabBarButtons[nIndex]; return aList; } void ViewTabBar::UpdateActiveButton() { Reference<XView> xView; if (mpViewShellBase != nullptr) xView = FrameworkHelper::Instance(*mpViewShellBase)->GetView( mxViewTabBarId->getAnchor()); if (xView.is()) { Reference<XResourceId> xViewId (xView->getResourceId()); for (size_t nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex) { if (maTabBarButtons[nIndex].ResourceId->compareTo(xViewId) == 0) { mpTabControl->SetCurPageId(nIndex+1); mpTabControl->::TabControl::ActivatePage(); break; } } } } void ViewTabBar::UpdateTabBarButtons() { TabBarButtonList::const_iterator iTab; sal_uInt16 nPageCount (mpTabControl->GetPageCount()); sal_uInt16 nIndex; for (iTab=maTabBarButtons.begin(),nIndex=1; iTab!=maTabBarButtons.end(); ++iTab,++nIndex) { // Create a new tab when there are not enough. if (nPageCount < nIndex) mpTabControl->InsertPage(nIndex, iTab->ButtonLabel); // Update the tab. mpTabControl->SetPageText(nIndex, iTab->ButtonLabel); mpTabControl->SetHelpText(nIndex, iTab->HelpText); mpTabControl->SetTabPage(nIndex, mpTabPage.get()); } // Delete tabs that are no longer used. for (; nIndex<=nPageCount; ++nIndex) mpTabControl->RemovePage(nIndex); mpTabPage->Hide(); } //===== TabBarControl ========================================================= TabBarControl::TabBarControl ( vcl::Window* pParentWindow, const ::rtl::Reference<ViewTabBar>& rpViewTabBar) : ::TabControl(pParentWindow), mpViewTabBar(rpViewTabBar) { } void TabBarControl::Paint (vcl::RenderContext& rRenderContext, const Rectangle& rRect) { Color aOriginalFillColor(rRenderContext.GetFillColor()); Color aOriginalLineColor(rRenderContext.GetLineColor()); // Because the actual window background is transparent--to avoid // flickering due to multiple background paintings by this and by child // windows--we have to paint the background for this control explicitly: // the actual control is not painted over its whole bounding box. rRenderContext.SetFillColor(rRenderContext.GetSettings().GetStyleSettings().GetDialogColor()); rRenderContext.SetLineColor(); rRenderContext.DrawRect(rRect); ::TabControl::Paint(rRenderContext, rRect); rRenderContext.SetFillColor(aOriginalFillColor); rRenderContext.SetLineColor(aOriginalLineColor); } void TabBarControl::ActivatePage() { if (mpViewTabBar->ActivatePage()) { // Call the parent so that the correct tab is highlighted. this->::TabControl::ActivatePage(); } } } // end of namespace sd /* vim:set shiftwidth=4 softtabstop=4 expandtab: */