1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <sdr/contact/viewcontactofsdrobjcustomshape.hxx>
21 #include <svx/svdoashp.hxx>
22 #include <svx/sdooitm.hxx>
23 #include <sdr/primitive2d/sdrattributecreator.hxx>
24 #include <sdr/primitive2d/sdrcustomshapeprimitive2d.hxx>
25 #include <basegfx/matrix/b2dhommatrixtools.hxx>
26 #include <svx/obj3d.hxx>
27 #include <vcl/canvastools.hxx>
30 namespace sdr::contact
32 ViewContactOfSdrObjCustomShape::ViewContactOfSdrObjCustomShape(SdrObjCustomShape
& rCustomShape
)
33 : ViewContactOfTextObj(rCustomShape
)
37 ViewContactOfSdrObjCustomShape::~ViewContactOfSdrObjCustomShape()
41 basegfx::B2DRange
ViewContactOfSdrObjCustomShape::getCorrectedTextBoundRect() const
43 const tools::Rectangle
aObjectBound(GetCustomShapeObj().GetGeoRect());
44 tools::Rectangle
aTextBound(aObjectBound
);
45 GetCustomShapeObj().GetTextBounds(aTextBound
);
46 basegfx::B2DRange aTextRange
= vcl::unotools::b2DRectangleFromRectangle(aTextBound
);
47 const basegfx::B2DRange aObjectRange
= vcl::unotools::b2DRectangleFromRectangle(aObjectBound
);
49 // no need to correct if no extra text range
50 if(aTextRange
!= aObjectRange
)
52 const GeoStat
& rGeoStat(GetCustomShapeObj().GetGeoStat());
54 // only correct when rotation and/or shear is used
55 if(rGeoStat
.m_nShearAngle
|| rGeoStat
.m_nRotationAngle
)
57 // text range needs to be corrected by
58 // aObjectRange.getCenter() - aRotObjectRange.getCenter() since it's
59 // defined differently by using rotation around object center. Start
61 basegfx::B2DVector
aTranslation(aObjectRange
.getCenter());
63 // get rotated and sheared object's range
64 basegfx::B2DRange
aRotObjectRange(aObjectRange
);
65 basegfx::B2DHomMatrix aRotMatrix
;
67 aRotMatrix
.translate(-aObjectRange
.getMinimum().getX(), -aObjectRange
.getMinimum().getY());
69 if(rGeoStat
.m_nShearAngle
)
71 aRotMatrix
.shearX(-rGeoStat
.mfTanShearAngle
);
74 if(rGeoStat
.m_nRotationAngle
)
76 aRotMatrix
.rotate(toRadians(36000_deg100
- rGeoStat
.m_nRotationAngle
));
79 aRotMatrix
.translate(aObjectRange
.getMinimum().getX(), aObjectRange
.getMinimum().getY());
80 aRotObjectRange
.transform(aRotMatrix
);
82 // add negative translation part
83 aTranslation
-= aRotObjectRange
.getCenter();
86 aTextRange
= basegfx::B2DRange(
87 aTextRange
.getMinX() + aTranslation
.getX(), aTextRange
.getMinY() + aTranslation
.getY(),
88 aTextRange
.getMaxX() + aTranslation
.getX(), aTextRange
.getMaxY() + aTranslation
.getY());
91 // NbcMirror() of SdrTextObj (from which SdrObjCustomShape is derived), adds a
92 // 180deg rotation around the shape center to GeoStat.nRotationAngle. So remove here the
93 // 180° rotation, which was added by GetTextBounds().
94 if(GetCustomShapeObj().IsMirroredY())
96 basegfx::B2DHomMatrix
aRotMatrix(basegfx::utils::createRotateAroundPoint(
97 aObjectRange
.getCenterX(), aObjectRange
.getCenterY(), M_PI
));
98 aTextRange
.transform(aRotMatrix
);
105 void ViewContactOfSdrObjCustomShape::createViewIndependentPrimitive2DSequence(drawinglayer::primitive2d::Primitive2DDecompositionVisitor
& rVisitor
) const
107 const SfxItemSet
& rItemSet
= GetCustomShapeObj().GetMergedItemSet();
109 // #i98072# Get shadow and text; eventually suppress the text if it's
110 // a TextPath FontworkGallery object
111 const drawinglayer::attribute::SdrEffectsTextAttribute
aAttribute(
112 drawinglayer::primitive2d::createNewSdrEffectsTextAttribute(
114 GetCustomShapeObj().getText(0),
115 GetCustomShapeObj().IsTextPath()));
116 drawinglayer::primitive2d::Primitive2DContainer xGroup
;
117 bool bHasText(!aAttribute
.getText().isDefault());
119 // create Primitive2DContainer from sub-geometry
120 const SdrObject
* pSdrObjRepresentation
= GetCustomShapeObj().GetSdrObjectFromCustomShape();
121 bool b3DShape(false);
123 if(pSdrObjRepresentation
)
125 // tdf#118498 The processing of SdrObjListIter for SdrIterMode::DeepNoGroups
126 // did change for 3D-Objects, it now correctly enters and iterates the
127 // SdrObjects in the E3dScene (same as for SdrObjGroup). This is more correct
128 // as the old version which just checked for dynamic_cast<const SdrObjGroup*>
129 // and *only* entered these, ignoring E3dScene as grouping-object.
130 // But how to fix that? Taking back the SdrObjListIter change would be easy, but
131 // not correct. After checking ViewContactOfE3dScene and ViewContactOfGroup
132 // I see that both traverse their children by themselves (on VC-Level,
133 // see createViewIndependentPrimitive2DSequence implementations and usage of
134 // GetObjectCount()). Thus in principle iterating here (esp. 'deep') seems to
135 // be wrong anyways, it might have even created wrong and double geometries
136 // (only with complex CustomShapes with multiple representation SdrObjects and
137 // only visible when transparency involved, but runtime-expensive).
138 // Thus: Just do not iterate, will check behaviour deeply.
139 b3DShape
= (nullptr != DynCastE3dObject(pSdrObjRepresentation
));
140 pSdrObjRepresentation
->GetViewContact().getViewIndependentPrimitive2DContainer(xGroup
);
143 if(bHasText
|| !xGroup
.empty())
145 // prepare text box geometry
146 basegfx::B2DHomMatrix aTextBoxMatrix
;
147 bool bWordWrap(false);
149 // take unrotated snap rect as default, then get the
150 // unrotated text box. Rotation needs to be done centered
151 const tools::Rectangle
aObjectBound(GetCustomShapeObj().GetGeoRect());
152 const basegfx::B2DRange aObjectRange
= vcl::unotools::b2DRectangleFromRectangle(aObjectBound
);
156 // #i101684# get the text range unrotated and absolute to the object range
157 const basegfx::B2DRange
aTextRange(getCorrectedTextBoundRect());
159 // Rotation before scaling
160 if(!basegfx::fTools::equalZero(GetCustomShapeObj().GetExtraTextRotation(true)))
162 basegfx::B2DVector
aTranslation(0.5, 0.5);
163 aTextBoxMatrix
.translate( -aTranslation
.getX(), -aTranslation
.getY() );
164 aTextBoxMatrix
.rotate(basegfx::deg2rad(
165 360.0 - GetCustomShapeObj().GetExtraTextRotation(true)));
166 aTextBoxMatrix
.translate( aTranslation
.getX(), aTranslation
.getY() );
168 // give text object a size
169 aTextBoxMatrix
.scale(aTextRange
.getWidth(), aTextRange
.getHeight());
171 // check if we have a rotation/shear at all to take care of
172 const double fExtraTextRotation(GetCustomShapeObj().GetExtraTextRotation());
173 const GeoStat
& rGeoStat(GetCustomShapeObj().GetGeoStat());
175 if(rGeoStat
.m_nShearAngle
|| rGeoStat
.m_nRotationAngle
|| !basegfx::fTools::equalZero(fExtraTextRotation
))
177 if(aObjectRange
!= aTextRange
)
179 // move relative to unrotated object range
180 aTextBoxMatrix
.translate(
181 aTextRange
.getMinX() - aObjectRange
.getMinimum().getX(),
182 aTextRange
.getMinY() - aObjectRange
.getMinimum().getY());
185 if(!basegfx::fTools::equalZero(fExtraTextRotation
))
187 basegfx::B2DVector
aTranslation(
188 ( aTextRange
.getWidth() / 2 ) + ( aTextRange
.getMinX() - aObjectRange
.getMinimum().getX() ),
189 ( aTextRange
.getHeight() / 2 ) + ( aTextRange
.getMinY() - aObjectRange
.getMinimum().getY() ) );
190 aTextBoxMatrix
.translate( -aTranslation
.getX(), -aTranslation
.getY() );
191 aTextBoxMatrix
.rotate(basegfx::deg2rad(360.0 - fExtraTextRotation
));
192 aTextBoxMatrix
.translate( aTranslation
.getX(), aTranslation
.getY() );
195 if(rGeoStat
.m_nShearAngle
)
197 aTextBoxMatrix
.shearX(-rGeoStat
.mfTanShearAngle
);
200 if(rGeoStat
.m_nRotationAngle
)
202 aTextBoxMatrix
.rotate(toRadians(36000_deg100
- rGeoStat
.m_nRotationAngle
));
205 // give text it's target position
206 aTextBoxMatrix
.translate(aObjectRange
.getMinimum().getX(), aObjectRange
.getMinimum().getY());
210 aTextBoxMatrix
.translate(aTextRange
.getMinX(), aTextRange
.getMinY());
213 // check if SdrTextWordWrapItem is set
214 bWordWrap
= GetCustomShapeObj().GetMergedItem(SDRATTR_TEXT_WORDWRAP
).GetValue();
217 // fill object matrix
218 const basegfx::B2DHomMatrix
aObjectMatrix(basegfx::utils::createScaleShearXRotateTranslateB2DHomMatrix(
219 aObjectRange
.getWidth(), aObjectRange
.getHeight(),
220 /*fShearX=*/0, /*fRotate=*/0,
221 aObjectRange
.getMinX(), aObjectRange
.getMinY()));
224 const drawinglayer::primitive2d::Primitive2DReference
xReference(
225 new drawinglayer::primitive2d::SdrCustomShapePrimitive2D(
232 rVisitor
.visit(xReference
);
236 } // end of namespace
238 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */