Files
loongoffice/chart2/source/controller/main/ChartWindow.cxx
Noel Grandin 1b1afc10f4 Fraction: make conversion operators and constructor explicit
and simplify some of the calculations that needed to be changed.

Which resulted in one unit test needing to change by one pixel,
let's hope not an indication of a real problem.

Change-Id: Ie56434f35f4e58d21ee6f671392e93dc7542fca3
Reviewed-on: https://gerrit.libreoffice.org/42240
Tested-by: Jenkins <ci@libreoffice.org>
Reviewed-by: Noel Grandin <noel.grandin@collabora.co.uk>
2017-09-13 15:46:47 +02:00

429 lines
12 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 "ChartWindow.hxx"
#include "ChartController.hxx"
#include "helpids.h"
#include "uiobject.hxx"
#include <vcl/help.hxx>
#include <vcl/openglwin.hxx>
#include <vcl/settings.hxx>
#include <config_features.h>
#include <com/sun/star/chart2/X3DChartWindowProvider.hpp>
#include <sfx2/ipclient.hxx>
#include <sfx2/viewsh.hxx>
#include <sfx2/lokhelper.hxx>
#include <comphelper/lok.hxx>
#define TWIPS_PER_PIXEL 15
using namespace ::com::sun::star;
namespace
{
::tools::Rectangle lcl_AWTRectToVCLRect( const css::awt::Rectangle & rAWTRect )
{
::tools::Rectangle aResult;
aResult.setX( rAWTRect.X );
aResult.setY( rAWTRect.Y );
aResult.setWidth( rAWTRect.Width );
aResult.setHeight( rAWTRect.Height );
return aResult;
}
} // anonymous namespace
namespace chart
{
ChartWindow::ChartWindow( ChartController* pController, vcl::Window* pParent, WinBits nStyle )
: Window(pParent, nStyle)
, m_pWindowController( pController )
, m_bInPaint(false)
, m_pViewShellWindow( nullptr )
#if HAVE_FEATURE_OPENGL
, m_pOpenGLWindow(VclPtr<OpenGLWindow>::Create(this, false))
#else
, m_pOpenGLWindow(nullptr)
#endif
{
set_id("chart_window");
SetHelpId( HID_SCH_WIN_DOCUMENT );
SetMapMode( MapMode(MapUnit::Map100thMM) );
adjustHighContrastMode();
// chart does not depend on exact pixel painting => enable antialiased drawing
SetAntialiasing( AntialiasingFlags::EnableB2dDraw | GetAntialiasing() );
EnableRTL( false );
if( pParent )
pParent->EnableRTL( false );// #i96215# necessary for a correct position of the context menu in rtl mode
if( m_pOpenGLWindow )
{
m_pOpenGLWindow->Show();
uno::Reference< chart2::X3DChartWindowProvider > x3DWindowProvider(pController->getModel(), uno::UNO_QUERY_THROW);
sal_uInt64 nWindowPtr = reinterpret_cast<sal_uInt64>(m_pOpenGLWindow.get());
x3DWindowProvider->setWindow(nWindowPtr);
uno::Reference<util::XUpdatable> const xUpdatable(x3DWindowProvider,
uno::UNO_QUERY_THROW);
xUpdatable->update();
}
}
ChartWindow::~ChartWindow()
{
disposeOnce();
}
void ChartWindow::dispose()
{
if (m_pWindowController && m_pWindowController->getModel().is())
{
uno::Reference< chart2::X3DChartWindowProvider > x3DWindowProvider(m_pWindowController->getModel(), uno::UNO_QUERY_THROW);
x3DWindowProvider->setWindow(0);
uno::Reference<util::XUpdatable> const xUpdatable(x3DWindowProvider,
uno::UNO_QUERY_THROW);
xUpdatable->update();
}
m_pOpenGLWindow.disposeAndClear();
m_pViewShellWindow.clear();
vcl::Window::dispose();
}
void ChartWindow::PrePaint(vcl::RenderContext& )
{
// forward VCLs PrePaint window event to DrawingLayer
if (m_pWindowController)
{
m_pWindowController->PrePaint();
}
}
void ChartWindow::Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle& rRect)
{
if (comphelper::LibreOfficeKit::isActive()
&& rRenderContext.GetOutDevType() != OutDevType::OUTDEV_VIRDEV)
return;
m_bInPaint = true;
if (m_pOpenGLWindow && m_pOpenGLWindow->IsVisible())
{
m_pOpenGLWindow->Paint(rRenderContext, rRect);
}
else if (m_pWindowController)
{
m_pWindowController->execute_Paint(rRenderContext, rRect);
}
else
{
Window::Paint(rRenderContext, rRect);
}
m_bInPaint = false;
}
void ChartWindow::MouseButtonDown(const MouseEvent& rMEvt)
{
if( m_pWindowController )
m_pWindowController->execute_MouseButtonDown(rMEvt);
else
Window::MouseButtonDown(rMEvt);
}
void ChartWindow::MouseMove( const MouseEvent& rMEvt )
{
if( m_pWindowController )
m_pWindowController->execute_MouseMove( rMEvt );
else
Window::MouseMove( rMEvt );
}
void ChartWindow::Tracking( const TrackingEvent& rTEvt )
{
if( !m_pWindowController )
Window::Tracking( rTEvt );
}
void ChartWindow::MouseButtonUp( const MouseEvent& rMEvt )
{
if( m_pWindowController )
m_pWindowController->execute_MouseButtonUp( rMEvt );
else
Window::MouseButtonUp( rMEvt );
}
void ChartWindow::Resize()
{
if( m_pWindowController )
m_pWindowController->execute_Resize();
else
Window::Resize();
if( m_pOpenGLWindow )
m_pOpenGLWindow->SetSizePixel(GetSizePixel());
}
void ChartWindow::Activate()
{
if( !m_pWindowController )
Window::Activate();
}
void ChartWindow::Deactivate()
{
if( !m_pWindowController )
Window::Deactivate();
}
void ChartWindow::GetFocus()
{
if( !m_pWindowController )
Window::GetFocus();
}
void ChartWindow::LoseFocus()
{
if( !m_pWindowController )
Window::LoseFocus();
}
void ChartWindow::Command( const CommandEvent& rCEvt )
{
if( m_pWindowController )
m_pWindowController->execute_Command( rCEvt );
else
Window::Command( rCEvt );
}
void ChartWindow::KeyInput( const KeyEvent& rKEvt )
{
if( m_pWindowController )
{
if( !m_pWindowController->execute_KeyInput(rKEvt) )
Window::KeyInput(rKEvt);
}
else
Window::KeyInput( rKEvt );
}
uno::Reference< css::accessibility::XAccessible > ChartWindow::CreateAccessible()
{
if( m_pWindowController )
return m_pWindowController->CreateAccessible();
else
return Window::CreateAccessible();
}
void ChartWindow::DataChanged( const DataChangedEvent& rDCEvt )
{
vcl::Window::DataChanged( rDCEvt );
if ( (rDCEvt.GetType() == DataChangedEventType::SETTINGS) &&
(rDCEvt.GetFlags() & AllSettingsFlags::STYLE) )
{
adjustHighContrastMode();
}
}
void ChartWindow::RequestHelp( const HelpEvent& rHEvt )
{
bool bHelpHandled = false;
if( ( rHEvt.GetMode() & HelpEventMode::QUICK ) &&
m_pWindowController )
{
// Point aLogicHitPos = PixelToLogic( rHEvt.GetMousePosPixel()); // old chart: GetPointerPosPixel()
Point aLogicHitPos = PixelToLogic( GetPointerPosPixel());
OUString aQuickHelpText;
awt::Rectangle aHelpRect;
bool bIsBalloonHelp( Help::IsBalloonHelpEnabled() );
bHelpHandled = m_pWindowController->requestQuickHelp( aLogicHitPos, bIsBalloonHelp, aQuickHelpText, aHelpRect );
if( bHelpHandled )
{
if( bIsBalloonHelp )
Help::ShowBalloon(
this, rHEvt.GetMousePosPixel(), lcl_AWTRectToVCLRect( aHelpRect ), aQuickHelpText );
else
Help::ShowQuickHelp(
this, lcl_AWTRectToVCLRect( aHelpRect ), aQuickHelpText );
}
}
if( !bHelpHandled )
vcl::Window::RequestHelp( rHEvt );
}
void ChartWindow::adjustHighContrastMode()
{
static const DrawModeFlags nContrastMode =
DrawModeFlags::SettingsLine | DrawModeFlags::SettingsFill |
DrawModeFlags::SettingsText | DrawModeFlags::SettingsGradient;
bool bUseContrast = GetSettings().GetStyleSettings().GetHighContrastMode();
SetDrawMode( bUseContrast ? nContrastMode : DrawModeFlags::Default );
}
void ChartWindow::ForceInvalidate()
{
vcl::Window::Invalidate();
if(m_pOpenGLWindow)
{
m_pOpenGLWindow->Invalidate();
}
}
void ChartWindow::Invalidate( InvalidateFlags nFlags )
{
if( m_bInPaint ) // #i101928# superfluous paint calls while entering and editing charts"
return;
vcl::Window::Invalidate( nFlags );
if(m_pOpenGLWindow)
{
m_pOpenGLWindow->Invalidate( nFlags );
}
}
void ChartWindow::Invalidate( const tools::Rectangle& rRect, InvalidateFlags nFlags )
{
if( m_bInPaint ) // #i101928# superfluous paint calls while entering and editing charts"
return;
vcl::Window::Invalidate( rRect, nFlags );
if(m_pOpenGLWindow)
{
m_pOpenGLWindow->Invalidate( rRect, nFlags );
}
}
void ChartWindow::Invalidate( const vcl::Region& rRegion, InvalidateFlags nFlags )
{
if( m_bInPaint ) // #i101928# superfluous paint calls while entering and editing charts"
return;
vcl::Window::Invalidate( rRegion, nFlags );
if(m_pOpenGLWindow)
{
m_pOpenGLWindow->Invalidate( rRegion, nFlags );
}
}
void ChartWindow::LogicInvalidate(const tools::Rectangle* pRectangle)
{
SfxViewShell* pCurrentShell = SfxViewShell::Current();
if ( nullptr == pCurrentShell )
return;
OString sRectangle;
if (!pRectangle)
{
// we have to invalidate the whole chart area not the whole document
sRectangle = GetBoundingBox().toString();
}
else
{
tools::Rectangle aRectangle(*pRectangle);
// When dragging shapes the map mode is disabled.
if (IsMapModeEnabled())
{
if (GetMapMode().GetMapUnit() == MapUnit::Map100thMM)
aRectangle = OutputDevice::LogicToLogic(aRectangle, MapUnit::Map100thMM, MapUnit::MapTwip);
}
else
{
aRectangle = PixelToLogic(aRectangle, MapMode(MapUnit::MapTwip));
}
vcl::Window* pEditWin = GetParentEditWin();
if (pEditWin)
{
MapMode aCWMapMode = GetMapMode();
double fXScale( aCWMapMode.GetScaleX() );
double fYScale( aCWMapMode.GetScaleY() );
if (!IsMapModeEnabled())
{
aRectangle.Left() /= fXScale;
aRectangle.Right() /= fXScale;
aRectangle.Top() /= fYScale;
aRectangle.Bottom() /= fYScale;
}
Point aOffset = this->GetOffsetPixelFrom(*pEditWin);
aOffset.X() *= (TWIPS_PER_PIXEL / fXScale);
aOffset.Y() *= (TWIPS_PER_PIXEL / fYScale);
aRectangle = tools::Rectangle(aRectangle.TopLeft() + aOffset, aRectangle.GetSize());
}
sRectangle = aRectangle.toString();
}
SfxLokHelper::notifyInvalidation(pCurrentShell, sRectangle);
}
FactoryFunction ChartWindow::GetUITestFactory() const
{
return ChartWindowUIObject::create;
}
ChartController* ChartWindow::GetController()
{
return m_pWindowController;
}
vcl::Window* ChartWindow::GetParentEditWin()
{
if (m_pViewShellWindow)
return m_pViewShellWindow.get();
// So, you are thinking, why do not invoke pCurrentShell->GetWindow() ?
// Because in Impress the parent edit win is not view shell window.
SfxViewShell* pCurrentShell = SfxViewShell::Current();
if( pCurrentShell )
{
SfxInPlaceClient* pIPClient = pCurrentShell->GetIPClient();
if (pIPClient)
{
vcl::Window* pRootWin = pIPClient->GetEditWin();
if(pRootWin && pRootWin->IsAncestorOf(*this))
{
m_pViewShellWindow = pRootWin;
return m_pViewShellWindow.get();
}
}
}
return nullptr;
}
tools::Rectangle ChartWindow::GetBoundingBox()
{
tools::Rectangle aBBox;
vcl::Window* pRootWin = GetParentEditWin();
if (pRootWin)
{
// In all cases, the following code fragment
// returns the chart bounding box in twips.
MapMode aCWMapMode = GetMapMode();
double fXScale( aCWMapMode.GetScaleX() );
double fYScale( aCWMapMode.GetScaleY() );
Point aOffset = GetOffsetPixelFrom(*pRootWin);
aOffset.X() *= (TWIPS_PER_PIXEL / fXScale);
aOffset.Y() *= (TWIPS_PER_PIXEL / fYScale);
Size aSize = GetSizePixel();
aSize.Width() *= (TWIPS_PER_PIXEL / fXScale);
aSize.Height() *= (TWIPS_PER_PIXEL / fYScale);
aBBox = tools::Rectangle(aOffset, aSize);
}
return aBBox;
}
} //namespace chart
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */