libreoffice/sd/source/ui/view/ViewTabBar.cxx
Stephan Bergmann e57ca02849 Remove dynamic exception specifications
...(for now, from LIBO_INTERNAL_CODE only).  See the mail thread starting at
<https://lists.freedesktop.org/archives/libreoffice/2017-January/076665.html>
"Dynamic Exception Specifications" for details.

Most changes have been done automatically by the rewriting loplugin:dynexcspec
(after enabling the rewriting mode, to be committed shortly).  The way it only
removes exception specs from declarations if it also sees a definition, it
identified some dead declarations-w/o-definitions (that have been removed
manually) and some cases where a definition appeared in multiple include files
(which have also been cleaned up manually).  There's also been cases of macro
paramters (that were used to abstract over exception specs) that have become
unused now (and been removed).

Furthermore, some code needed to be cleaned up manually
(avmedia/source/quicktime/ and connectivity/source/drivers/kab/), as I had no
configurations available that would actually build that code.  Missing @throws
documentation has not been applied in such manual clean-up.

Change-Id: I3408691256c9b0c12bc5332de976743626e13960
Reviewed-on: https://gerrit.libreoffice.org/33574
Tested-by: Jenkins <ci@libreoffice.org>
Reviewed-by: Stephan Bergmann <sbergman@redhat.com>
2017-01-26 12:54:43 +00:00

585 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 .
*/
#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)
{
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)
{
if (rEvent.Source == mxConfigurationController)
{
mxConfigurationController = nullptr;
mxController = nullptr;
}
}
//----- XTabBar ---------------------------------------------------------------
void SAL_CALL ViewTabBar::addTabBarButtonAfter (
const TabBarButton& rButton,
const TabBarButton& rAnchor)
{
const SolarMutexGuard aSolarGuard;
AddTabBarButton(rButton, rAnchor);
}
void SAL_CALL ViewTabBar::appendTabBarButton (const TabBarButton& rButton)
{
const SolarMutexGuard aSolarGuard;
AddTabBarButton(rButton);
}
void SAL_CALL ViewTabBar::removeTabBarButton (const TabBarButton& rButton)
{
const SolarMutexGuard aSolarGuard;
RemoveTabBarButton(rButton);
}
sal_Bool SAL_CALL ViewTabBar::hasTabBarButton (const TabBarButton& rButton)
{
const SolarMutexGuard aSolarGuard;
return HasTabBarButton(rButton);
}
Sequence<TabBarButton> SAL_CALL ViewTabBar::getTabBarButtons()
{
const SolarMutexGuard aSolarGuard;
return GetTabBarButtons();
}
//----- XResource -------------------------------------------------------------
Reference<XResourceId> SAL_CALL ViewTabBar::getResourceId()
{
return mxViewTabBarId;
}
sal_Bool SAL_CALL ViewTabBar::isAnchorOnly()
{
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)
{
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: */