Files
libreoffice/svx/source/svdraw/svdsnpv.cxx
Norbert Thiebaud af75d7a4c9 SdrSnapView: initializae members in the constructor
remove ClearVars() which was only
use by the constructor
instead initialize properly things in the constructor itself

Change-Id: I20a0ea26c5dce680844f81cf85ee72e1809a7068
Reviewed-on: https://gerrit.libreoffice.org/15890
Tested-by: Jenkins <ci@libreoffice.org>
Reviewed-by: Norbert Thiebaud <nthiebaud@gmail.com>
2015-05-26 00:19:19 +00:00

666 lines
20 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 <svx/svdsnpv.hxx>
#include <math.h>
#include <svx/svdetc.hxx>
#include <svx/svdobj.hxx>
#include <svx/svdpagv.hxx>
#include <svx/svdpage.hxx>
#include "svx/svditer.hxx"
#include <svx/sdr/overlay/overlayobjectlist.hxx>
#include <sdr/overlay/overlaycrosshair.hxx>
#include <sdr/overlay/overlayhelpline.hxx>
#include <svx/sdr/overlay/overlaymanager.hxx>
#include <basegfx/matrix/b2dhommatrix.hxx>
#include <svx/sdrpaintwindow.hxx>
class ImplPageOriginOverlay
{
// The OverlayObjects
sdr::overlay::OverlayObjectList maObjects;
// The current position in logical coodinates
basegfx::B2DPoint maPosition;
public:
ImplPageOriginOverlay(const SdrPaintView& rView, const basegfx::B2DPoint& rStartPos);
~ImplPageOriginOverlay();
void SetPosition(const basegfx::B2DPoint& rNewPosition);
};
ImplPageOriginOverlay::ImplPageOriginOverlay(const SdrPaintView& rView, const basegfx::B2DPoint& rStartPos)
: maPosition(rStartPos)
{
for(sal_uInt32 a(0L); a < rView.PaintWindowCount(); a++)
{
SdrPaintWindow* pCandidate = rView.GetPaintWindow(a);
rtl::Reference< sdr::overlay::OverlayManager > xTargetOverlay = pCandidate->GetOverlayManager();
if (xTargetOverlay.is())
{
sdr::overlay::OverlayCrosshairStriped* aNew = new sdr::overlay::OverlayCrosshairStriped(
maPosition);
xTargetOverlay->add(*aNew);
maObjects.append(*aNew);
}
}
}
ImplPageOriginOverlay::~ImplPageOriginOverlay()
{
// The OverlayObjects are cleared using the destructor of OverlayObjectList.
// That destructor calls clear() at the list which removes all objects from the
// OverlayManager and deletes them.
}
void ImplPageOriginOverlay::SetPosition(const basegfx::B2DPoint& rNewPosition)
{
if(rNewPosition != maPosition)
{
// apply to OverlayObjects
for(sal_uInt32 a(0); a < maObjects.count(); a++)
{
sdr::overlay::OverlayCrosshairStriped* pCandidate =
static_cast< sdr::overlay::OverlayCrosshairStriped* >(&maObjects.getOverlayObject(a));
if(pCandidate)
{
pCandidate->setBasePosition(rNewPosition);
}
}
// remember new position
maPosition = rNewPosition;
}
}
class ImplHelpLineOverlay
{
// The OverlayObjects
sdr::overlay::OverlayObjectList maObjects;
// The current position in logical coodinates
basegfx::B2DPoint maPosition;
// HelpLine specific stuff
SdrPageView* mpPageView;
sal_uInt16 mnHelpLineNumber;
SdrHelpLineKind meHelpLineKind;
public:
ImplHelpLineOverlay(const SdrPaintView& rView, const basegfx::B2DPoint& rStartPos,
SdrPageView* pPageView, sal_uInt16 nHelpLineNumber, SdrHelpLineKind eKind);
~ImplHelpLineOverlay();
void SetPosition(const basegfx::B2DPoint& rNewPosition);
// access to HelpLine specific stuff
SdrPageView* GetPageView() const { return mpPageView; }
sal_uInt16 GetHelpLineNumber() const { return mnHelpLineNumber; }
SdrHelpLineKind GetHelpLineKind() const { return meHelpLineKind; }
};
ImplHelpLineOverlay::ImplHelpLineOverlay(
const SdrPaintView& rView, const basegfx::B2DPoint& rStartPos,
SdrPageView* pPageView, sal_uInt16 nHelpLineNumber, SdrHelpLineKind eKind)
: maPosition(rStartPos),
mpPageView(pPageView),
mnHelpLineNumber(nHelpLineNumber),
meHelpLineKind(eKind)
{
for(sal_uInt32 a(0L); a < rView.PaintWindowCount(); a++)
{
SdrPaintWindow* pCandidate = rView.GetPaintWindow(a);
rtl::Reference< sdr::overlay::OverlayManager > xTargetOverlay = pCandidate->GetOverlayManager();
if (xTargetOverlay.is())
{
sdr::overlay::OverlayHelplineStriped* aNew = new sdr::overlay::OverlayHelplineStriped(
maPosition, meHelpLineKind);
xTargetOverlay->add(*aNew);
maObjects.append(*aNew);
}
}
}
ImplHelpLineOverlay::~ImplHelpLineOverlay()
{
// The OverlayObjects are cleared using the destructor of OverlayObjectList.
// That destructor calls clear() at the list which removes all objects from the
// OverlayManager and deletes them.
}
void ImplHelpLineOverlay::SetPosition(const basegfx::B2DPoint& rNewPosition)
{
if(rNewPosition != maPosition)
{
// apply to OverlayObjects
for(sal_uInt32 a(0); a < maObjects.count(); a++)
{
sdr::overlay::OverlayHelplineStriped* pCandidate =
static_cast< sdr::overlay::OverlayHelplineStriped* >(&maObjects.getOverlayObject(a));
if(pCandidate)
{
pCandidate->setBasePosition(rNewPosition);
}
}
// remember new position
maPosition = rNewPosition;
}
}
SdrSnapView::SdrSnapView(SdrModel* pModel1, OutputDevice* pOut)
: SdrPaintView(pModel1,pOut)
, mpPageOriginOverlay(0L)
, mpHelpLineOverlay(0L)
, nMagnSizPix(4)
, nSnapAngle(1500)
, nEliminatePolyPointLimitAngle(0)
, eCrookMode(SDRCROOK_ROTATE)
, bSnapEnab(true)
, bGridSnap(true)
, bSnapTo1Pix(true)
, bBordSnap(true)
, bHlplSnap(true)
, bOFrmSnap(true)
, bOPntSnap(false)
, bOConSnap(true)
, bMoveMFrmSnap(true)
, bMoveOFrmSnap(true)
, bMoveOPntSnap(true)
, bMoveOConSnap(true)
, bMoveSnapOnlyTopLeft(false)
, bOrtho(false)
, bBigOrtho(true)
, bAngleSnapEnab(false)
, bMoveOnlyDragging(false)
, bSlantButShear(false)
, bCrookNoContortion(false)
, bHlplFixed(false)
, bEliminatePolyPoints(false)
{
}
SdrSnapView::~SdrSnapView()
{
BrkSetPageOrg();
BrkDragHelpLine();
}
bool SdrSnapView::IsAction() const
{
return IsSetPageOrg() || IsDragHelpLine() || SdrPaintView::IsAction();
}
void SdrSnapView::MovAction(const Point& rPnt)
{
SdrPaintView::MovAction(rPnt);
if (IsSetPageOrg()) {
MovSetPageOrg(rPnt);
}
if (IsDragHelpLine()) {
MovDragHelpLine(rPnt);
}
}
void SdrSnapView::EndAction()
{
if (IsSetPageOrg()) {
EndSetPageOrg();
}
if (IsDragHelpLine()) {
EndDragHelpLine();
}
SdrPaintView::EndAction();
}
void SdrSnapView::BckAction()
{
BrkSetPageOrg();
BrkDragHelpLine();
SdrPaintView::BckAction();
}
void SdrSnapView::BrkAction()
{
BrkSetPageOrg();
BrkDragHelpLine();
SdrPaintView::BrkAction();
}
void SdrSnapView::TakeActionRect(Rectangle& rRect) const
{
if (IsSetPageOrg() || IsDragHelpLine()) {
rRect=Rectangle(aDragStat.GetNow(),aDragStat.GetNow());
} else {
SdrPaintView::TakeActionRect(rRect);
}
}
Point SdrSnapView::GetSnapPos(const Point& rPnt, const SdrPageView* pPV) const
{
Point aPt(rPnt);
SnapPos(aPt,pPV);
return aPt;
}
#define NOT_SNAPPED 0x7FFFFFFF
SdrSnap SdrSnapView::SnapPos(Point& rPnt, const SdrPageView* pPV) const
{
if (!bSnapEnab) return SdrSnap::NOTSNAPPED;
long x=rPnt.X();
long y=rPnt.Y();
if (pPV==NULL) {
pPV=GetSdrPageView();
if (pPV==NULL) return SdrSnap::NOTSNAPPED;
}
long dx=NOT_SNAPPED;
long dy=NOT_SNAPPED;
long dx1,dy1;
long mx=aMagnSiz.Width();
long my=aMagnSiz.Height();
if (bHlplVisible && bHlplSnap && !IsDragHelpLine())
{
const SdrHelpLineList& rHLL=pPV->GetHelpLines();
sal_uInt16 nCount=rHLL.GetCount();
for (sal_uInt16 i=nCount; i>0;) {
i--;
const SdrHelpLine& rHL=rHLL[i];
const Point& rPos=rHL.GetPos();
switch (rHL.GetKind()) {
case SDRHELPLINE_VERTICAL: {
long a=x-rPos.X();
if (std::abs(a)<=mx) { dx1=-a; if (std::abs(dx1)<std::abs(dx)) dx=dx1; }
} break;
case SDRHELPLINE_HORIZONTAL: {
long b=y-rPos.Y();
if (std::abs(b)<=my) { dy1=-b; if (std::abs(dy1)<std::abs(dy)) dy=dy1; }
} break;
case SDRHELPLINE_POINT: {
long a=x-rPos.X();
long b=y-rPos.Y();
if (std::abs(a)<=mx && std::abs(b)<=my) {
dx1=-a; dy1=-b;
if (std::abs(dx1)<std::abs(dx) && std::abs(dy1)<std::abs(dy)) { dx=dx1; dy=dy1; }
}
} break;
} // switch
}
}
if (bBordVisible && bBordSnap) {
SdrPage* pPage=pPV->GetPage();
long xs=pPage->GetWdt();
long ys=pPage->GetHgt();
long lft=pPage->GetLftBorder();
long rgt=pPage->GetRgtBorder();
long upp=pPage->GetUppBorder();
long lwr=pPage->GetLwrBorder();
long a;
a=x- lft ; if (std::abs(a)<=mx) { dx1=-a; if (std::abs(dx1)<std::abs(dx)) dx=dx1; } // left margin
a=x-(xs-rgt); if (std::abs(a)<=mx) { dx1=-a; if (std::abs(dx1)<std::abs(dx)) dx=dx1; } // right margin
a=x ; if (std::abs(a)<=mx) { dx1=-a; if (std::abs(dx1)<std::abs(dx)) dx=dx1; } // left edge of paper
a=x- xs ; if (std::abs(a)<=mx) { dx1=-a; if (std::abs(dx1)<std::abs(dx)) dx=dx1; } // right edge of paper
a=y- upp ; if (std::abs(a)<=my) { dy1=-a; if (std::abs(dy1)<std::abs(dy)) dy=dy1; } // left margin
a=y-(ys-lwr); if (std::abs(a)<=my) { dy1=-a; if (std::abs(dy1)<std::abs(dy)) dy=dy1; } // right margin
a=y ; if (std::abs(a)<=my) { dy1=-a; if (std::abs(dy1)<std::abs(dy)) dy=dy1; } // left edge of paper
a=y- ys ; if (std::abs(a)<=my) { dy1=-a; if (std::abs(dy1)<std::abs(dy)) dy=dy1; } // right edge of paper
}
if (bOFrmSnap || bOPntSnap) {
sal_uIntPtr nMaxPointSnapCount=200;
sal_uIntPtr nMaxFrameSnapCount=200;
// go back to IM_DEEPNOGROUPS runthrough for snap to object comparisons
SdrObjListIter aIter(*pPV->GetPage(),IM_DEEPNOGROUPS,true);
while (aIter.IsMore() && (nMaxPointSnapCount>0 || nMaxFrameSnapCount>0)) {
SdrObject* pO=aIter.Next();
Rectangle aRect(pO->GetCurrentBoundRect());
aRect.Left ()-=mx;
aRect.Right ()+=mx;
aRect.Top ()-=my;
aRect.Bottom()+=my;
if (aRect.IsInside(rPnt)) {
if (bOPntSnap && nMaxPointSnapCount>0)
{
sal_uInt32 nCount(pO->GetSnapPointCount());
for (sal_uInt32 i(0L); i < nCount && nMaxPointSnapCount > 0L; i++)
{
Point aP(pO->GetSnapPoint(i));
dx1=x-aP.X();
dy1=y-aP.Y();
if (std::abs(dx1)<=mx && std::abs(dy1)<=my && std::abs(dx1)<std::abs(dx) && std::abs(dy1)<std::abs(dy)) {
dx=-dx1;
dy=-dy1;
}
nMaxPointSnapCount--;
}
}
if (bOFrmSnap && nMaxFrameSnapCount>0) {
Rectangle aLog(pO->GetSnapRect());
Rectangle aR1(aLog);
aR1.Left ()-=mx;
aR1.Right ()+=mx;
aR1.Top ()-=my;
aR1.Bottom()+=my;
if (aR1.IsInside(rPnt)) {
if (std::abs(x-aLog.Left ())<=mx) { dx1=-(x-aLog.Left ()); if (std::abs(dx1)<std::abs(dx)) dx=dx1; }
if (std::abs(x-aLog.Right ())<=mx) { dx1=-(x-aLog.Right ()); if (std::abs(dx1)<std::abs(dx)) dx=dx1; }
if (std::abs(y-aLog.Top ())<=my) { dy1=-(y-aLog.Top ()); if (std::abs(dy1)<std::abs(dy)) dy=dy1; }
if (std::abs(y-aLog.Bottom())<=my) { dy1=-(y-aLog.Bottom()); if (std::abs(dy1)<std::abs(dy)) dy=dy1; }
}
nMaxFrameSnapCount--;
}
}
}
}
if(bGridSnap)
{
double fSnapWidth = aSnapWdtX;
if(dx == NOT_SNAPPED && fSnapWidth != 0.0)
{
double fx = (double)x;
// round instead of trunc
if(fx - (double)pPV->GetPageOrigin().X() >= 0.0)
fx += fSnapWidth / 2.0;
else
fx -= fSnapWidth / 2.0;
x = (long)((fx - (double)pPV->GetPageOrigin().X()) / fSnapWidth);
x = (long)((double)x * fSnapWidth + (double)pPV->GetPageOrigin().X());
dx = 0;
}
fSnapWidth = aSnapWdtY;
if(dy == NOT_SNAPPED && fSnapWidth)
{
double fy = (double)y;
// round instead of trunc
if(fy - (double)pPV->GetPageOrigin().Y() >= 0.0)
fy += fSnapWidth / 2.0;
else
fy -= fSnapWidth / 2.0;
y = (long)((fy - (double)pPV->GetPageOrigin().Y()) / fSnapWidth);
y = (long)((double)y * fSnapWidth + (double)pPV->GetPageOrigin().Y());
dy = 0;
}
}
SdrSnap bRet=SdrSnap::NOTSNAPPED;
if (dx==NOT_SNAPPED) dx=0; else bRet|=SdrSnap::XSNAPPED;
if (dy==NOT_SNAPPED) dy=0; else bRet|=SdrSnap::YSNAPPED;
rPnt.X()=x+dx;
rPnt.Y()=y+dy;
return bRet;
}
void SdrSnapView::CheckSnap(const Point& rPt, const SdrPageView* pPV, long& nBestXSnap, long& nBestYSnap, bool& bXSnapped, bool& bYSnapped) const
{
Point aPt(rPt);
SdrSnap nRet=SnapPos(aPt,pPV);
aPt-=rPt;
if (nRet & SdrSnap::XSNAPPED) {
if (bXSnapped) {
if (std::abs(aPt.X())<std::abs(nBestXSnap)) {
nBestXSnap=aPt.X();
}
} else {
nBestXSnap=aPt.X();
bXSnapped=true;
}
}
if (nRet & SdrSnap::YSNAPPED) {
if (bYSnapped) {
if (std::abs(aPt.Y())<std::abs(nBestYSnap)) {
nBestYSnap=aPt.Y();
}
} else {
nBestYSnap=aPt.Y();
bYSnapped=true;
}
}
}
bool SdrSnapView::BegSetPageOrg(const Point& rPnt)
{
BrkAction();
DBG_ASSERT(0L == mpPageOriginOverlay, "SdrSnapView::BegSetPageOrg: There exists a ImplPageOriginOverlay (!)");
basegfx::B2DPoint aStartPos(rPnt.X(), rPnt.Y());
mpPageOriginOverlay = new ImplPageOriginOverlay(*this, aStartPos);
aDragStat.Reset(GetSnapPos(rPnt,NULL));
return true;
}
void SdrSnapView::MovSetPageOrg(const Point& rPnt)
{
if(IsSetPageOrg())
{
aDragStat.NextMove(GetSnapPos(rPnt,NULL));
DBG_ASSERT(mpPageOriginOverlay, "SdrSnapView::MovSetPageOrg: no ImplPageOriginOverlay (!)");
basegfx::B2DPoint aNewPos(aDragStat.GetNow().X(), aDragStat.GetNow().Y());
mpPageOriginOverlay->SetPosition(aNewPos);
}
}
bool SdrSnapView::EndSetPageOrg()
{
bool bRet(false);
if(IsSetPageOrg())
{
SdrPageView* pPV = GetSdrPageView();
if(pPV)
{
Point aPnt(aDragStat.GetNow());
pPV->SetPageOrigin(aPnt);
bRet = true;
}
// cleanup
BrkSetPageOrg();
}
return bRet;
}
void SdrSnapView::BrkSetPageOrg()
{
if(IsSetPageOrg())
{
DBG_ASSERT(mpPageOriginOverlay, "SdrSnapView::MovSetPageOrg: no ImplPageOriginOverlay (!)");
delete mpPageOriginOverlay;
mpPageOriginOverlay = 0L;
}
}
bool SdrSnapView::PickHelpLine(const Point& rPnt, short nTol, const OutputDevice& rOut, sal_uInt16& rnHelpLineNum, SdrPageView*& rpPV) const
{
rpPV=NULL;
nTol=ImpGetHitTolLogic(nTol,&rOut);
SdrPageView* pPV = GetSdrPageView();
if(pPV)
{
Point aPnt(rPnt);
sal_uInt16 nIndex=pPV->GetHelpLines().HitTest(aPnt,sal_uInt16(nTol),rOut);
if (nIndex!=SDRHELPLINE_NOTFOUND) {
rpPV=pPV;
rnHelpLineNum=nIndex;
return true;
}
}
return false;
}
// start HelpLine drag for new HelpLine
bool SdrSnapView::BegDragHelpLine(sal_uInt16 nHelpLineNum, SdrPageView* pPV)
{
bool bRet(false);
if(!bHlplFixed)
{
BrkAction();
if(pPV && nHelpLineNum < pPV->GetHelpLines().GetCount())
{
const SdrHelpLineList& rHelpLines = pPV->GetHelpLines();
const SdrHelpLine& rHelpLine = rHelpLines[nHelpLineNum];
Point aHelpLinePos = rHelpLine.GetPos();
basegfx::B2DPoint aStartPos(aHelpLinePos.X(), aHelpLinePos.Y());
DBG_ASSERT(0L == mpHelpLineOverlay, "SdrSnapView::BegDragHelpLine: There exists a ImplHelpLineOverlay (!)");
mpHelpLineOverlay = new ImplHelpLineOverlay(*this, aStartPos, pPV, nHelpLineNum, rHelpLine.GetKind());
aDragStat.Reset(GetSnapPos(aHelpLinePos, pPV));
aDragStat.SetMinMove(ImpGetMinMovLogic(-3, 0L));
bRet = true;
}
}
return bRet;
}
// start HelpLine drag with existing HelpLine
bool SdrSnapView::BegDragHelpLine(const Point& rPnt, SdrHelpLineKind eNewKind)
{
bool bRet(false);
BrkAction();
if(GetSdrPageView())
{
DBG_ASSERT(0L == mpHelpLineOverlay, "SdrSnapView::BegDragHelpLine: There exists a ImplHelpLineOverlay (!)");
basegfx::B2DPoint aStartPos(rPnt.X(), rPnt.Y());
mpHelpLineOverlay = new ImplHelpLineOverlay(*this, aStartPos, 0L, 0, eNewKind);
aDragStat.Reset(GetSnapPos(rPnt, 0L));
bRet = true;
}
return bRet;
}
Pointer SdrSnapView::GetDraggedHelpLinePointer() const
{
if(IsDragHelpLine())
{
switch(mpHelpLineOverlay->GetHelpLineKind())
{
case SDRHELPLINE_VERTICAL : return Pointer(PointerStyle::ESize);
case SDRHELPLINE_HORIZONTAL: return Pointer(PointerStyle::SSize);
default : return Pointer(PointerStyle::Move);
}
}
return Pointer(PointerStyle::Move);
}
void SdrSnapView::MovDragHelpLine(const Point& rPnt)
{
if(IsDragHelpLine() && aDragStat.CheckMinMoved(rPnt))
{
Point aPnt(GetSnapPos(rPnt, 0L));
if(aPnt != aDragStat.GetNow())
{
aDragStat.NextMove(aPnt);
DBG_ASSERT(mpHelpLineOverlay, "SdrSnapView::MovDragHelpLine: no ImplHelpLineOverlay (!)");
basegfx::B2DPoint aNewPos(aDragStat.GetNow().X(), aDragStat.GetNow().Y());
mpHelpLineOverlay->SetPosition(aNewPos);
}
}
}
bool SdrSnapView::EndDragHelpLine()
{
bool bRet(false);
if(IsDragHelpLine())
{
if(aDragStat.IsMinMoved())
{
SdrPageView* pPageView = mpHelpLineOverlay->GetPageView();
if(pPageView)
{
// moved existing one
Point aPnt(aDragStat.GetNow());
const SdrHelpLineList& rHelpLines = pPageView->GetHelpLines();
SdrHelpLine aChangedHelpLine = rHelpLines[mpHelpLineOverlay->GetHelpLineNumber()];
aChangedHelpLine.SetPos(aPnt);
pPageView->SetHelpLine(mpHelpLineOverlay->GetHelpLineNumber(), aChangedHelpLine);
bRet = true;
}
else
{
// create new one
pPageView = GetSdrPageView();
if(pPageView)
{
Point aPnt(aDragStat.GetNow());
SdrHelpLine aNewHelpLine(mpHelpLineOverlay->GetHelpLineKind(), aPnt);
pPageView->InsertHelpLine(aNewHelpLine);
bRet = true;
}
}
}
// cleanup
BrkDragHelpLine();
}
return bRet;
}
void SdrSnapView::BrkDragHelpLine()
{
if(IsDragHelpLine())
{
DBG_ASSERT(mpHelpLineOverlay, "SdrSnapView::EndDragHelpLine: no ImplHelpLineOverlay (!)");
delete mpHelpLineOverlay;
mpHelpLineOverlay = 0L;
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */