597 lines
18 KiB
C++
597 lines
18 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 "EventMultiplexer.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;
|
|
using ::sd::tools::EventMultiplexerEvent;
|
|
|
|
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) SAL_OVERRIDE;
|
|
virtual void ActivatePage() SAL_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(NULL),
|
|
mxViewTabBarId(rxViewTabBarId),
|
|
mpViewShellBase(NULL)
|
|
{
|
|
// 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 != NULL
|
|
&& rxViewTabBarId->isBoundToURL(
|
|
FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT))
|
|
{
|
|
mpViewShellBase->SetViewTabBar(this);
|
|
}
|
|
}
|
|
|
|
ViewTabBar::~ViewTabBar()
|
|
{
|
|
}
|
|
|
|
void ViewTabBar::disposing()
|
|
{
|
|
if (mpViewShellBase != NULL
|
|
&& mxViewTabBarId->isBoundToURL(
|
|
FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT))
|
|
{
|
|
mpViewShellBase->SetViewTabBar(NULL);
|
|
}
|
|
|
|
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 = NULL;
|
|
}
|
|
|
|
{
|
|
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, NULL);
|
|
mpTabPage.disposeAndClear();
|
|
mpTabControl.disposeAndClear();
|
|
}
|
|
|
|
mxController = NULL;
|
|
}
|
|
|
|
vcl::Window* ViewTabBar::GetAnchorWindow(
|
|
const Reference<XResourceId>& rxViewTabBarId,
|
|
const Reference<frame::XController>& rxController)
|
|
{
|
|
vcl::Window* pWindow = NULL;
|
|
ViewShellBase* pBase = NULL;
|
|
|
|
// 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 != NULL && pBase->GetViewFrame() != NULL)
|
|
pWindow = &pBase->GetViewFrame()->GetWindow();
|
|
}
|
|
|
|
// The rest is (at the moment) just for the emergency case.
|
|
if (pWindow == NULL)
|
|
{
|
|
Reference<XPane> xPane;
|
|
try
|
|
{
|
|
Reference<XControllerManager> xControllerManager (rxController, UNO_QUERY_THROW);
|
|
Reference<XConfigurationController> xCC (
|
|
xControllerManager->getConfigurationController());
|
|
if (xCC.is())
|
|
xPane = Reference<XPane>(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 != NULL)
|
|
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 = NULL;
|
|
mxController = NULL;
|
|
}
|
|
}
|
|
|
|
//----- XTabBar ---------------------------------------------------------------
|
|
|
|
void SAL_CALL ViewTabBar::addTabBarButtonAfter (
|
|
const TabBarButton& rButton,
|
|
const TabBarButton& rAnchor)
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
const SolarMutexGuard aSolarGuard;
|
|
AddTabBarButton(rButton, rAnchor);
|
|
}
|
|
|
|
void SAL_CALL ViewTabBar::appendTabBarButton (const TabBarButton& rButton)
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
const SolarMutexGuard aSolarGuard;
|
|
AddTabBarButton(rButton);
|
|
}
|
|
|
|
void SAL_CALL ViewTabBar::removeTabBarButton (const TabBarButton& rButton)
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
const SolarMutexGuard aSolarGuard;
|
|
RemoveTabBarButton(rButton);
|
|
}
|
|
|
|
sal_Bool SAL_CALL ViewTabBar::hasTabBarButton (const TabBarButton& rButton)
|
|
throw (::com::sun::star::uno::RuntimeException, std::exception)
|
|
{
|
|
const SolarMutexGuard aSolarGuard;
|
|
return HasTabBarButton(rButton);
|
|
}
|
|
|
|
Sequence<TabBarButton> SAL_CALL ViewTabBar::getTabBarButtons()
|
|
throw (::com::sun::star::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 = Reference<XView>(xConfigurationController->getResource(
|
|
ResourceId::create(
|
|
::comphelper::getProcessComponentContext(),
|
|
FrameworkHelper::msCenterPaneURL)),
|
|
UNO_QUERY);
|
|
}
|
|
catch (const DeploymentException&)
|
|
{
|
|
}
|
|
|
|
Client* pIPClient = NULL;
|
|
if (mpViewShellBase != NULL)
|
|
pIPClient = dynamic_cast<Client*>(mpViewShellBase->GetIPClient());
|
|
if (pIPClient==NULL || ! 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!=NULL && 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 ::com::sun::star::drawing::framework::TabBarButton& rButton,
|
|
const ::com::sun::star::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 ::com::sun::star::drawing::framework::TabBarButton& rButton)
|
|
{
|
|
AddTabBarButton(rButton, maTabBarButtons.size());
|
|
}
|
|
|
|
void ViewTabBar::AddTabBarButton (
|
|
const ::com::sun::star::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 ::com::sun::star::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 ::com::sun::star::drawing::framework::TabBarButton& rButton)
|
|
{
|
|
bool bResult (false);
|
|
|
|
for (size_t nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex)
|
|
{
|
|
if (IsEqual(maTabBarButtons[nIndex], rButton))
|
|
{
|
|
bResult = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
::com::sun::star::uno::Sequence<com::sun::star::drawing::framework::TabBarButton>
|
|
ViewTabBar::GetTabBarButtons()
|
|
{
|
|
sal_uInt32 nCount (maTabBarButtons.size());
|
|
::com::sun::star::uno::Sequence<com::sun::star::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 != NULL)
|
|
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: */
|