/************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2008 by Sun Microsystems, Inc. * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: svdotextpathdecomposition.cxx,v $ * * $Revision: 1.2 $ * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_svx.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include ////////////////////////////////////////////////////////////////////////////// // primitive decomposition helpers #include #include #include #include #include #include #include #include #include #include #include #include ////////////////////////////////////////////////////////////////////////////// using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::i18n; ////////////////////////////////////////////////////////////////////////////// // PathTextPortion helper namespace { class impPathTextPortion { basegfx::B2DVector maOffset; String maText; xub_StrLen mnTextStart; xub_StrLen mnTextLength; sal_uInt16 mnParagraph; xub_StrLen mnIndex; SvxFont maFont; ::std::vector< double > maDblDXArray; // double DXArray, font size independent -> unit coordinate system ::com::sun::star::lang::Locale maLocale; // bitfield unsigned mbRTL : 1; public: impPathTextPortion(DrawPortionInfo& rInfo) : maOffset(rInfo.mrStartPos.X(), rInfo.mrStartPos.Y()), maText(rInfo.mrText), mnTextStart(rInfo.mnTextStart), mnTextLength(rInfo.mnTextLen), mnParagraph(rInfo.mnPara), mnIndex(rInfo.mnIndex), maFont(rInfo.mrFont), maDblDXArray(), maLocale(rInfo.mpLocale ? *rInfo.mpLocale : ::com::sun::star::lang::Locale()), mbRTL(rInfo.mrFont.IsVertical() ? false : rInfo.IsRTL()) { if(mnTextLength) { maDblDXArray.reserve(mnTextLength); const sal_Int32 nFontWidth(0L == maFont.GetWidth() ? maFont.GetHeight() : maFont.GetWidth()); const double fScaleFactor(0L != nFontWidth ? 1.0 / (double)nFontWidth : 1.0); for(xub_StrLen a(0); a < mnTextLength; a++) { maDblDXArray.push_back((double)rInfo.mpDXArray[a] * fScaleFactor); } } } // for ::std::sort bool operator<(const impPathTextPortion& rComp) const { if(mnParagraph < rComp.mnParagraph) { return true; } if(maOffset.getX() < rComp.maOffset.getX()) { return true; } return (maOffset.getY() < rComp.maOffset.getY()); } const basegfx::B2DVector& getOffset() const { return maOffset; } const String& getText() const { return maText; } xub_StrLen getTextStart() const { return mnTextStart; } xub_StrLen getTextLength() const { return mnTextLength; } sal_uInt16 getParagraph() const { return mnParagraph; } xub_StrLen getIndex() const { return mnIndex; } const SvxFont& getFont() const { return maFont; } bool isRTL() const { return mbRTL; } const ::std::vector< double >& getDoubleDXArray() const { return maDblDXArray; } const ::com::sun::star::lang::Locale& getLocale() const { return maLocale; } xub_StrLen getPortionIndex(xub_StrLen nIndex, xub_StrLen nLength) const { if(mbRTL) { return (mnTextStart + (mnTextLength - (nIndex + nLength))); } else { return (mnTextStart + nIndex); } } double getDisplayLength(xub_StrLen nIndex, xub_StrLen nLength) const { drawinglayer::primitive2d::TextLayouterDevice aTextLayouter; double fRetval(0.0); if(maFont.IsVertical()) { fRetval = aTextLayouter.getTextHeight() * (double)nLength; } else { fRetval = aTextLayouter.getTextWidth(maText, getPortionIndex(nIndex, nLength), nLength); } return fRetval; } }; } // end of anonymous namespace ////////////////////////////////////////////////////////////////////////////// // TextBreakup helper namespace { class impTextBreakupHandler { SdrOutliner& mrOutliner; ::std::vector< impPathTextPortion > maPathTextPortions; DECL_LINK(decompositionPathTextPrimitive, DrawPortionInfo* ); public: impTextBreakupHandler(SdrOutliner& rOutliner) : mrOutliner(rOutliner) { } const ::std::vector< impPathTextPortion >& decompositionPathTextPrimitive() { // strip portions to maPathTextPortions mrOutliner.SetDrawPortionHdl(LINK(this, impTextBreakupHandler, decompositionPathTextPrimitive)); mrOutliner.StripPortions(); if(maPathTextPortions.size()) { // sort portions by paragraph, x and y ::std::sort(maPathTextPortions.begin(), maPathTextPortions.end()); } return maPathTextPortions; } }; IMPL_LINK(impTextBreakupHandler, decompositionPathTextPrimitive, DrawPortionInfo*, pInfo) { maPathTextPortions.push_back(impPathTextPortion(*pInfo)); return 0; } } // end of anonymous namespace ////////////////////////////////////////////////////////////////////////////// // TextBreakup data helper namespace { class impFormTextValues { sal_Int32 mnFormTextDistance; // distance from line in upright direction sal_Int32 mnFormTextStart; // shift from polygon start sal_Int32 mnFormTextShdwXVal; // shadow distance or 10th degrees sal_Int32 mnFormTextShdwYVal; // shadow distance or scaling sal_uInt16 mnFormTextShdwTransp; // shadow transparence XFormTextStyle meFormTextStyle; // on/off and char orientation XFormTextAdjust meFormTextAdjust; // adjustment (left/right/center) and scale XFormTextShadow meFormTextShadow; // shadow mode Color maFormTextShdwColor; // shadow color // bitfield unsigned mbFormTextMirror : 1; // change orientation unsigned mbFormTextOutline : 1; // show contour of objects public: impFormTextValues(const SfxItemSet& rSet) : mnFormTextDistance(((const XFormTextDistanceItem&)rSet.Get(XATTR_FORMTXTDISTANCE)).GetValue()), mnFormTextStart(((const XFormTextStartItem&)rSet.Get(XATTR_FORMTXTSTART)).GetValue()), mnFormTextShdwXVal(((const XFormTextShadowXValItem&)rSet.Get(XATTR_FORMTXTSHDWXVAL)).GetValue()), mnFormTextShdwYVal(((const XFormTextShadowYValItem&)rSet.Get(XATTR_FORMTXTSHDWYVAL)).GetValue()), mnFormTextShdwTransp(((const XFormTextShadowTranspItem&)rSet.Get(XATTR_FORMTXTSHDWTRANSP)).GetValue()), meFormTextStyle(((const XFormTextStyleItem&)rSet.Get(XATTR_FORMTXTSTYLE)).GetValue()), meFormTextAdjust(((const XFormTextAdjustItem&)rSet.Get(XATTR_FORMTXTADJUST)).GetValue()), meFormTextShadow(((const XFormTextShadowItem&)rSet.Get(XATTR_FORMTXTSHADOW)).GetValue()), maFormTextShdwColor(((const XFormTextShadowColorItem&)rSet.Get(XATTR_FORMTXTSHDWCOLOR)).GetColorValue()), mbFormTextMirror(((const XFormTextMirrorItem&)rSet.Get(XATTR_FORMTXTMIRROR)).GetValue()), mbFormTextOutline(((const XFormTextOutlineItem&)rSet.Get(XATTR_FORMTXTOUTLINE)).GetValue()) { } // data access sal_Int32 getFormTextDistance() const { return mnFormTextDistance; } sal_Int32 getFormTextStart() const { return mnFormTextStart; } sal_Int32 getFormTextShdwXVal() const { return mnFormTextShdwXVal; } sal_Int32 getFormTextShdwYVal() const { return mnFormTextShdwYVal; } sal_uInt16 getFormTextShdwTransp() const { return mnFormTextShdwTransp; } XFormTextStyle getFormTextStyle() const { return meFormTextStyle; } XFormTextAdjust getFormTextAdjust() const { return meFormTextAdjust; } XFormTextShadow getFormTextShadow() const { return meFormTextShadow; } Color getFormTextShdwColor() const { return maFormTextShdwColor; } bool getFormTextMirror() const { return mbFormTextMirror; } bool getFormTextOutline() const { return mbFormTextOutline; } }; } // end of anonymous namespace ////////////////////////////////////////////////////////////////////////////// // TextBreakup one poly and one paragraph helper namespace { class impPolygonParagraphHandler { const impFormTextValues& mrVal; // FormText parameters std::vector< drawinglayer::primitive2d::BasePrimitive2D* >& mrDecomposition; // destination primitive list std::vector< drawinglayer::primitive2d::BasePrimitive2D* >& mrShadowDecomposition; // destination primitive list for shadow Reference < com::sun::star::i18n::XBreakIterator > mxBreak; // break iterator double getParagraphTextLength(const ::std::vector< const impPathTextPortion* >& rTextPortions) { drawinglayer::primitive2d::TextLayouterDevice aTextLayouter; double fRetval(0.0); for(sal_uInt32 a(0L); a < rTextPortions.size(); a++) { const impPathTextPortion* pCandidate = rTextPortions[a]; if(pCandidate && pCandidate->getTextLength()) { aTextLayouter.setFont(pCandidate->getFont()); fRetval += pCandidate->getDisplayLength(0L, pCandidate->getTextLength()); } } return fRetval; } xub_StrLen getNextGlyphLen(const impPathTextPortion* pCandidate, xub_StrLen nPosition, const ::com::sun::star::lang::Locale& rFontLocale) { xub_StrLen nNextGlyphLen(1); if(mxBreak.is()) { sal_Int32 nDone(0L); nNextGlyphLen = (xub_StrLen)mxBreak->nextCharacters(pCandidate->getText(), nPosition, rFontLocale, CharacterIteratorMode::SKIPCELL, 1, nDone) - nPosition; } return nNextGlyphLen; } public: impPolygonParagraphHandler( const impFormTextValues& rVal, std::vector< drawinglayer::primitive2d::BasePrimitive2D* >& rDecomposition, std::vector< drawinglayer::primitive2d::BasePrimitive2D* >& rShadowDecomposition) : mrVal(rVal), mrDecomposition(rDecomposition), mrShadowDecomposition(rShadowDecomposition) { // prepare BreakIterator Reference < XMultiServiceFactory > xMSF = ::comphelper::getProcessServiceFactory(); Reference < XInterface > xInterface = xMSF->createInstance(::rtl::OUString::createFromAscii("com.sun.star.i18n.BreakIterator")); if(xInterface.is()) { Any x = xInterface->queryInterface(::getCppuType((const Reference< XBreakIterator >*)0)); x >>= mxBreak; } } void HandlePair(const basegfx::B2DPolygon rPolygonCandidate, const ::std::vector< const impPathTextPortion* >& rTextPortions) { // prepare polygon geometry, take into account as many parameters as possible basegfx::B2DPolygon aPolygonCandidate(rPolygonCandidate); const double fPolyLength(basegfx::tools::getLength(aPolygonCandidate)); double fPolyEnd(fPolyLength); double fPolyStart(0.0); double fScaleFactor(1.0); if(mrVal.getFormTextMirror()) { aPolygonCandidate.flip(); } if(mrVal.getFormTextStart() && (XFT_LEFT == mrVal.getFormTextAdjust() || XFT_RIGHT == mrVal.getFormTextAdjust())) { if(XFT_LEFT == mrVal.getFormTextAdjust()) { fPolyStart += mrVal.getFormTextStart(); if(fPolyStart > fPolyEnd) { fPolyStart = fPolyEnd; } } else { fPolyEnd -= mrVal.getFormTextStart(); if(fPolyEnd < fPolyStart) { fPolyEnd = fPolyStart; } } } if(XFT_LEFT != mrVal.getFormTextAdjust()) { // calculate total text length of this paragraph, some layout needs to be done const double fParagraphTextLength(getParagraphTextLength(rTextPortions)); // check if text is too long for paragraph. If yes, handle as if left aligned (default), // but still take care of XFT_AUTOSIZE in that case const bool bTextTooLong(fParagraphTextLength > (fPolyEnd - fPolyStart)); if(XFT_RIGHT == mrVal.getFormTextAdjust()) { if(!bTextTooLong) { // if right aligned, add difference to polygon start fPolyStart += ((fPolyEnd - fPolyStart) - fParagraphTextLength); } } else if(XFT_CENTER == mrVal.getFormTextAdjust()) { if(!bTextTooLong) { // if centered, add half of difference to polygon start fPolyStart += ((fPolyEnd - fPolyStart) - fParagraphTextLength) / 2.0; } } else if(XFT_AUTOSIZE == mrVal.getFormTextAdjust()) { // if scale, prepare scale factor between curve length and text length if(0.0 != fParagraphTextLength) { fScaleFactor = (fPolyEnd - fPolyStart) / fParagraphTextLength; } } } // handle text portions for this paragraph for(sal_uInt32 a(0L); a < rTextPortions.size() && fPolyStart < fPolyEnd; a++) { const impPathTextPortion* pCandidate = rTextPortions[a]; basegfx::B2DVector aSize; const drawinglayer::primitive2d::FontAttributes aCandidateFontAttributes(drawinglayer::primitive2d::getFontAttributesFromVclFont( aSize, pCandidate->getFont(), pCandidate->isRTL(), false)); if(pCandidate && pCandidate->getTextLength()) { drawinglayer::primitive2d::TextLayouterDevice aTextLayouter; aTextLayouter.setFont(pCandidate->getFont()); xub_StrLen nUsedTextLength(0); while(nUsedTextLength < pCandidate->getTextLength() && fPolyStart < fPolyEnd) { xub_StrLen nNextGlyphLen(getNextGlyphLen(pCandidate, pCandidate->getTextStart() + nUsedTextLength, pCandidate->getLocale())); // prepare portion length. Takes RTL sections into account. double fPortionLength(pCandidate->getDisplayLength(nUsedTextLength, nNextGlyphLen)); if(XFT_AUTOSIZE == mrVal.getFormTextAdjust()) { // when scaling, expand portion length fPortionLength *= fScaleFactor; } // create transformation basegfx::B2DHomMatrix aNewTransformA, aNewTransformB, aNewShadowTransform; basegfx::B2DPoint aStartPos(basegfx::tools::getPositionAbsolute(aPolygonCandidate, fPolyStart, fPolyLength)); basegfx::B2DPoint aEndPos(aStartPos); // add font scaling aNewTransformA.scale(aSize.getX(), aSize.getY()); // prepare scaling of text primitive if(XFT_AUTOSIZE == mrVal.getFormTextAdjust()) { // when scaling, expand text primitive scaling aNewTransformA.scale(fScaleFactor, fScaleFactor); } // eventually create shadow primitives from aDecomposition and add to rDecomposition const bool bShadow(XFTSHADOW_NONE != mrVal.getFormTextShadow()); if(bShadow) { if(XFTSHADOW_NORMAL == mrVal.getFormTextShadow()) { aNewShadowTransform.translate(mrVal.getFormTextShdwXVal(), -mrVal.getFormTextShdwYVal()); } else // XFTSHADOW_SLANT { double fScaleValue(mrVal.getFormTextShdwYVal() / 100.0); double fShearValue(-mrVal.getFormTextShdwXVal() * F_PI1800); aNewShadowTransform.scale(1.0, fScaleValue); aNewShadowTransform.shearX(sin(fShearValue)); aNewShadowTransform.scale(1.0, cos(fShearValue)); } } switch(mrVal.getFormTextStyle()) { case XFT_ROTATE : { aEndPos = basegfx::tools::getPositionAbsolute(aPolygonCandidate, fPolyStart + fPortionLength, fPolyLength); const basegfx::B2DVector aDirection(aEndPos - aStartPos); aNewTransformB.rotate(atan2(aDirection.getY(), aDirection.getX())); aNewTransformB.translate(aStartPos.getX(), aStartPos.getY()); break; } case XFT_UPRIGHT : { aNewTransformB.translate(aStartPos.getX() - (fPortionLength / 2.0), aStartPos.getY()); break; } case XFT_SLANTX : { aEndPos = basegfx::tools::getPositionAbsolute(aPolygonCandidate, fPolyStart + fPortionLength, fPolyLength); const basegfx::B2DVector aDirection(aEndPos - aStartPos); const double fShearValue(atan2(aDirection.getY(), aDirection.getX())); const double fSin(sin(fShearValue)); const double fCos(cos(fShearValue)); aNewTransformB.shearX(-fSin); // Scale may lead to objects without height since fCos == 0.0 is possible. // Renderers need to handle that, it's not a forbidden value and does not // need to be avoided aNewTransformB.scale(1.0, fCos); aNewTransformB.translate(aStartPos.getX() - (fPortionLength / 2.0), aStartPos.getY()); break; } case XFT_SLANTY : { aEndPos = basegfx::tools::getPositionAbsolute(aPolygonCandidate, fPolyStart + fPortionLength, fPolyLength); const basegfx::B2DVector aDirection(aEndPos - aStartPos); const double fShearValue(atan2(aDirection.getY(), aDirection.getX())); const double fCos(cos(fShearValue)); const double fTan(tan(fShearValue)); // shear to 'stand' on the curve aNewTransformB.shearY(fTan); // scale in X to make as tight as needed. As with XFT_SLANT_X, this may // lead to primitives without width which the renderers will handle aNewTransformA.scale(fCos, 1.0); aNewTransformB.translate(aStartPos.getX(), aStartPos.getY()); break; } default : break; // XFT_NONE } // distance from path? if(mrVal.getFormTextDistance()) { if(aEndPos.equal(aStartPos)) { aEndPos = basegfx::tools::getPositionAbsolute(aPolygonCandidate, fPolyStart + fPortionLength, fPolyLength); } // use back vector (aStartPos - aEndPos) here to get mirrored perpendicular as in old stuff const basegfx::B2DVector aPerpendicular(basegfx::getNormalizedPerpendicular(aStartPos - aEndPos) * mrVal.getFormTextDistance()); aNewTransformB.translate(aPerpendicular.getX(), aPerpendicular.getY()); } // shadow primitive creation if(bShadow) { if(pCandidate->getText().Len() && nNextGlyphLen) { const Color aShadowColor(mrVal.getFormTextShdwColor()); const basegfx::BColor aRGBShadowColor(aShadowColor.getBColor()); const xub_StrLen nPortionIndex(pCandidate->getPortionIndex(nUsedTextLength, nNextGlyphLen)); const ::std::vector< double > aNewDXArray( pCandidate->getDoubleDXArray().begin() + nPortionIndex, pCandidate->getDoubleDXArray().begin() + nPortionIndex + nNextGlyphLen); drawinglayer::primitive2d::TextSimplePortionPrimitive2D* pNew = new drawinglayer::primitive2d::TextSimplePortionPrimitive2D( aNewTransformB * aNewShadowTransform * aNewTransformA, pCandidate->getText(), nPortionIndex, nNextGlyphLen, aNewDXArray, aCandidateFontAttributes, pCandidate->getLocale(), aRGBShadowColor); mrShadowDecomposition.push_back(pNew); } } // primitive creation if(pCandidate->getText().Len() && nNextGlyphLen) { const Color aColor(pCandidate->getFont().GetColor()); const basegfx::BColor aRGBColor(aColor.getBColor()); const xub_StrLen nPortionIndex(pCandidate->getPortionIndex(nUsedTextLength, nNextGlyphLen)); const ::std::vector< double > aNewDXArray( pCandidate->getDoubleDXArray().begin() + nPortionIndex, pCandidate->getDoubleDXArray().begin() + nPortionIndex + nNextGlyphLen); drawinglayer::primitive2d::TextSimplePortionPrimitive2D* pNew = new drawinglayer::primitive2d::TextSimplePortionPrimitive2D( aNewTransformB * aNewTransformA, pCandidate->getText(), nPortionIndex, nNextGlyphLen, aNewDXArray, aCandidateFontAttributes, pCandidate->getLocale(), aRGBColor); mrDecomposition.push_back(pNew); } // consume from portion // no += here, xub_StrLen is USHORT and the compiler will gererate a warning here nUsedTextLength = nUsedTextLength + nNextGlyphLen; // consume from polygon fPolyStart += fPortionLength; } } } } }; } // end of anonymous namespace ////////////////////////////////////////////////////////////////////////////// // primitive decomposition helpers namespace { basegfx::B2DLineJoin impGetB2DLineJoin(XLineJoint eLineJoint) { switch(eLineJoint) { case XLINEJOINT_MIDDLE : { return basegfx::B2DLINEJOIN_MIDDLE; } case XLINEJOINT_BEVEL : { return basegfx::B2DLINEJOIN_BEVEL; } case XLINEJOINT_MITER : { return basegfx::B2DLINEJOIN_MITER; } case XLINEJOINT_ROUND : { return basegfx::B2DLINEJOIN_ROUND; } default : { return basegfx::B2DLINEJOIN_NONE; // XLINEJOINT_NONE } } } } // end of anonymous namespace ////////////////////////////////////////////////////////////////////////////// namespace { sal_uInt8 impGetStrokeTransparence(bool bShadow, const SfxItemSet& rSet) { sal_uInt8 nRetval; if(bShadow) { nRetval = (sal_uInt8)((((SdrShadowTransparenceItem&)(rSet.Get(SDRATTR_SHADOWTRANSPARENCE))).GetValue() * 255) / 100); } else { nRetval = (sal_uInt8)((((XLineTransparenceItem&)(rSet.Get(XATTR_LINETRANSPARENCE))).GetValue() * 255) / 100); } return nRetval; } drawinglayer::attribute::LineAttribute impGetLineAttribute(bool bShadow, const SfxItemSet& rSet) { basegfx::BColor aColorAttribute; if(bShadow) { const Color aShadowColor(((SdrShadowColorItem&)(rSet.Get(SDRATTR_SHADOWCOLOR))).GetColorValue()); aColorAttribute = aShadowColor.getBColor(); } else { const Color aLineColor(((XLineColorItem&)(rSet.Get(XATTR_LINECOLOR))).GetColorValue()); aColorAttribute = aLineColor.getBColor(); } const sal_uInt32 nLineWidth = ((const XLineWidthItem&)(rSet.Get(XATTR_LINEWIDTH))).GetValue(); const XLineJoint eLineJoint = ((const XLineJointItem&)(rSet.Get(XATTR_LINEJOINT))).GetValue(); return drawinglayer::attribute::LineAttribute(aColorAttribute, (double)nLineWidth, impGetB2DLineJoin(eLineJoint)); } drawinglayer::attribute::StrokeAttribute impGetStrokeAttribute(const SfxItemSet& rSet) { const XLineStyle eLineStyle = ((XLineStyleItem&)(rSet.Get(XATTR_LINESTYLE))).GetValue(); double fFullDotDashLen(0.0); ::std::vector< double > aDotDashArray; if(XLINE_DASH == eLineStyle) { const XDash& rDash = ((const XLineDashItem&)(rSet.Get(XATTR_LINEDASH))).GetDashValue(); if(rDash.GetDots() || rDash.GetDashes()) { const sal_uInt32 nLineWidth = ((const XLineWidthItem&)(rSet.Get(XATTR_LINEWIDTH))).GetValue(); fFullDotDashLen = rDash.CreateDotDashArray(aDotDashArray, (double)nLineWidth); } } return drawinglayer::attribute::StrokeAttribute(aDotDashArray, fFullDotDashLen); } } // end of anonymous namespace ////////////////////////////////////////////////////////////////////////////// // primitive decomposition helpers namespace { void impAddPolygonStrokePrimitives( const basegfx::B2DPolyPolygonVector& rB2DPolyPolyVector, const basegfx::B2DHomMatrix& rTransform, const drawinglayer::attribute::LineAttribute& rLineAttribute, const drawinglayer::attribute::StrokeAttribute& rStrokeAttribute, std::vector< drawinglayer::primitive2d::BasePrimitive2D* >& rTarget) { for(basegfx::B2DPolyPolygonVector::const_iterator aPolygon(rB2DPolyPolyVector.begin()); aPolygon != rB2DPolyPolyVector.end(); aPolygon++) { // prepare PolyPolygons basegfx::B2DPolyPolygon aB2DPolyPolygon = *aPolygon; aB2DPolyPolygon.transform(rTransform); for(sal_uInt32 a(0L); a < aB2DPolyPolygon.count(); a++) { // create one primitive per polygon drawinglayer::primitive2d::PolygonStrokePrimitive2D* pNew = new drawinglayer::primitive2d::PolygonStrokePrimitive2D( aB2DPolyPolygon.getB2DPolygon(a), rLineAttribute, rStrokeAttribute); rTarget.push_back(pNew); } } } drawinglayer::primitive2d::Primitive2DSequence impAddPathTextOutlines( const std::vector< drawinglayer::primitive2d::BasePrimitive2D* >& rSource, bool bShadow, const SfxItemSet& rSet) { std::vector< drawinglayer::primitive2d::BasePrimitive2D* > aNewPrimitives; for(sal_uInt32 a(0L); a < rSource.size(); a++) { const drawinglayer::primitive2d::TextSimplePortionPrimitive2D* pTextCandidate = dynamic_cast< const drawinglayer::primitive2d::TextSimplePortionPrimitive2D* >(rSource[a]); if(pTextCandidate) { basegfx::B2DPolyPolygonVector aB2DPolyPolyVector; basegfx::B2DHomMatrix aPolygonTransform; // get text outlines and their object transformation pTextCandidate->getTextOutlinesAndTransformation(aB2DPolyPolyVector, aPolygonTransform); if(aB2DPolyPolyVector.size()) { // prepare Line and Stroke Attribute const drawinglayer::attribute::LineAttribute aLineAttribute(impGetLineAttribute(bShadow, rSet)); const drawinglayer::attribute::StrokeAttribute aStrokeAttribute(impGetStrokeAttribute(rSet)); const sal_uInt8 nTransparence(impGetStrokeTransparence(bShadow, rSet)); // create stroke primitives std::vector< drawinglayer::primitive2d::BasePrimitive2D* > aStrokePrimitives; impAddPolygonStrokePrimitives(aB2DPolyPolyVector, aPolygonTransform, aLineAttribute, aStrokeAttribute, aStrokePrimitives); const sal_uInt32 nStrokeCount(aStrokePrimitives.size()); if(nStrokeCount) { if(nTransparence) { // create UnifiedAlphaPrimitive2D drawinglayer::primitive2d::Primitive2DSequence aStrokePrimitiveSequence(nStrokeCount); for(sal_uInt32 b(0L); b < nStrokeCount; b++) { aStrokePrimitiveSequence[b] = drawinglayer::primitive2d::Primitive2DReference(aStrokePrimitives[b]); } drawinglayer::primitive2d::UnifiedAlphaPrimitive2D* pNew2 = new drawinglayer::primitive2d::UnifiedAlphaPrimitive2D(aStrokePrimitiveSequence, (double)nTransparence / 100.0); aNewPrimitives.push_back(pNew2); } else { // add polygons to rDecomposition as polygonStrokePrimitives aNewPrimitives.insert(aNewPrimitives.end(), aStrokePrimitives.begin(), aStrokePrimitives.end()); } } } } } const sal_uInt32 nNewCount(aNewPrimitives.size()); if(nNewCount) { drawinglayer::primitive2d::Primitive2DSequence aRetval(nNewCount); for(sal_uInt32 a(0L); a < nNewCount; a++) { aRetval[a] = drawinglayer::primitive2d::Primitive2DReference(aNewPrimitives[a]); } return aRetval; } else { return drawinglayer::primitive2d::Primitive2DSequence(); } } } // end of anonymous namespace ////////////////////////////////////////////////////////////////////////////// // primitive decomposition bool SdrTextObj::impDecomposePathTextPrimitive( drawinglayer::primitive2d::Primitive2DSequence& rTarget, const drawinglayer::primitive2d::SdrPathTextPrimitive2D& rSdrPathTextPrimitive, const drawinglayer::geometry::ViewInformation2D& aViewInformation) const { drawinglayer::primitive2d::Primitive2DSequence aRetvalA; drawinglayer::primitive2d::Primitive2DSequence aRetvalB; // prepare outliner SdrOutliner& rOutliner = ImpGetDrawOutliner(); rOutliner.SetUpdateMode(true); rOutliner.Clear(); rOutliner.SetPaperSize(Size(LONG_MAX,LONG_MAX)); rOutliner.SetText(*rSdrPathTextPrimitive.getSdrText().GetOutlinerParaObject()); // set visualizing page at Outliner; needed e.g. for PageNumberField decomposition rOutliner.setVisualizedPage(GetSdrPageFromXDrawPage(aViewInformation.getVisualizedPage())); // now break up to text portions impTextBreakupHandler aConverter(rOutliner); const ::std::vector< impPathTextPortion > rPathTextPortions = aConverter.decompositionPathTextPrimitive(); if(rPathTextPortions.size()) { // get FormText values const SfxItemSet& rSet = rSdrPathTextPrimitive.getSdrText().GetItemSet(); const impFormTextValues aVal(rSet); // get polygon values const basegfx::B2DPolyPolygon& rPathPolyPolygon(rSdrPathTextPrimitive.getPathPolyPolygon()); // get loop count sal_uInt32 nLoopCount(rPathPolyPolygon.count()); if(rOutliner.GetParagraphCount() < nLoopCount) { nLoopCount = rOutliner.GetParagraphCount(); } if(nLoopCount && XFT_NONE != aVal.getFormTextStyle()) { // prepare common decomposition stuff std::vector< drawinglayer::primitive2d::BasePrimitive2D* > aRegularDecomposition; std::vector< drawinglayer::primitive2d::BasePrimitive2D* > aShadowDecomposition; impPolygonParagraphHandler aPolygonParagraphHandler(aVal, aRegularDecomposition, aShadowDecomposition); sal_uInt32 a; for(a = 0L; a < nLoopCount; a++) { // filter text portions for this paragraph ::std::vector< const impPathTextPortion* > aParagraphTextPortions; for(sal_uInt32 b(0L); b < rPathTextPortions.size(); b++) { const impPathTextPortion& rCandidate = rPathTextPortions[b]; if(rCandidate.getParagraph() == a) { aParagraphTextPortions.push_back(&rCandidate); } } // handle data pair polygon/ParagraphTextPortions if(aParagraphTextPortions.size()) { aPolygonParagraphHandler.HandlePair(rPathPolyPolygon.getB2DPolygon(a), aParagraphTextPortions); } } const sal_uInt32 nShadowCount(aShadowDecomposition.size()); const sal_uInt32 nRegularCount(aRegularDecomposition.size()); if(nShadowCount) { // add shadow primitives to decomposition aRetvalA.realloc(nShadowCount); for(a = 0L; a < nShadowCount; a++) { aRetvalA[a] = drawinglayer::primitive2d::Primitive2DReference(aShadowDecomposition[a]); } // evtl. add shadow outlines if(aVal.getFormTextOutline()) { const SfxItemSet& rLocalSet = rSdrPathTextPrimitive.getSdrText().GetObject().GetMergedItemSet(); const drawinglayer::primitive2d::Primitive2DSequence aOutlines(impAddPathTextOutlines(aShadowDecomposition, true, rLocalSet)); drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(aRetvalA, aOutlines); } } if(nRegularCount) { // add normal primitives to decomposition aRetvalB.realloc(nRegularCount); for(a = 0L; a < nRegularCount; a++) { aRetvalB[a] = drawinglayer::primitive2d::Primitive2DReference(aRegularDecomposition[a]); } // evtl. add outlines if(aVal.getFormTextOutline()) { const SfxItemSet& rLocalSet = rSdrPathTextPrimitive.getSdrText().GetObject().GetMergedItemSet(); const drawinglayer::primitive2d::Primitive2DSequence aOutlines(impAddPathTextOutlines(aRegularDecomposition, false, rLocalSet)); drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(aRetvalB, aOutlines); } } } } // cleanup outliner rOutliner.SetDrawPortionHdl(Link()); rOutliner.Clear(); rOutliner.setVisualizedPage(0); // concatenate all results drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(rTarget, aRetvalA); drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(rTarget, aRetvalB); return false; } ////////////////////////////////////////////////////////////////////////////// // eof