summaryrefslogtreecommitdiff
path: root/drawinglayer/source/processor2d
diff options
context:
space:
mode:
authorArmin Le Grand (Collabora) <Armin.Le.Grand@me.com>2024-08-13 14:50:24 +0200
committerArmin Le Grand <Armin.Le.Grand@me.com>2024-08-14 10:48:51 +0200
commit4a0cd8990467466a04bafd7bb1a780247535ab83 (patch)
tree0ad062668269da8435c78bebc766f0e1531afe32 /drawinglayer/source/processor2d
parentcf23025faabdae03f4d4706410c80c7c46c683e9 (diff)
CairoSDPR: Handle EmphasisMarks
Handling EmphasisMarks for direct text rendering was a hard task - the EmphasisMark stuff is deeply buried in vcl (would have needed to move quite some includes from vcl to public) and thus hard to use. It is also quite old (tools Polygon, Rectangle). It was missing in the decomposition of the TextDecoratedPortionPrimitive2D (for good reason), but is needed now. I found a way using a callback lambda function to create the needed geometry in vcl and hand back the needed data to the caller, in this case to the decomposition. Adding it to the decomposition of TextDecoratedPortionPrimitive2D also guarantees that other renderers/usages will do the correct thing and all will look identical. Interestingly EmphasisMarks were never added to Metafiles (see OutputDevice::ImplDrawEmphasisMarks) so they were not 'missing' in Metafile-based exports. But for SDPRs they have to work. I added another encapsulating TextHierarchyPrimitive to encapsulate primitives that do represent EmphasisMarks and added to ignore these in the VclMetafileProcessor2D (as needed). Change-Id: I5b5c1528d86a123df2beb57d8366f05aa71e77cf Reviewed-on: https://gerrit.libreoffice.org/c/core/+/171826 Tested-by: Jenkins Reviewed-by: Armin Le Grand <Armin.Le.Grand@me.com>
Diffstat (limited to 'drawinglayer/source/processor2d')
-rw-r--r--drawinglayer/source/processor2d/cairopixelprocessor2d.cxx81
-rw-r--r--drawinglayer/source/processor2d/vclmetafileprocessor2d.cxx7
2 files changed, 33 insertions, 55 deletions
diff --git a/drawinglayer/source/processor2d/cairopixelprocessor2d.cxx b/drawinglayer/source/processor2d/cairopixelprocessor2d.cxx
index 21a43e25dcfc..dfbce4af0c70 100644
--- a/drawinglayer/source/processor2d/cairopixelprocessor2d.cxx
+++ b/drawinglayer/source/processor2d/cairopixelprocessor2d.cxx
@@ -2913,9 +2913,8 @@ void CairoPixelProcessor2D::processTextDecoratedPortionPrimitive2D(
}
void CairoPixelProcessor2D::renderTextBackground(
- const primitive2d::TextSimplePortionPrimitive2D& rTextCandidate,
- const primitive2d::TextLayouterDevice& rTextLayouter, const basegfx::B2DHomMatrix& rTransform,
- double fTextWidth)
+ const primitive2d::TextSimplePortionPrimitive2D& rTextCandidate, double fAscent,
+ double fDescent, const basegfx::B2DHomMatrix& rTransform, double fTextWidth)
{
cairo_save(mpRT);
cairo_matrix_t aMatrix;
@@ -2925,8 +2924,7 @@ void CairoPixelProcessor2D::renderTextBackground(
const basegfx::BColor aFillColor(
maBColorModifierStack.getModifiedColor(rTextCandidate.getTextFillColor().getBColor()));
cairo_set_source_rgb(mpRT, aFillColor.getRed(), aFillColor.getGreen(), aFillColor.getBlue());
- cairo_rectangle(mpRT, 0.0, -rTextLayouter.getFontAscent(), fTextWidth,
- rTextLayouter.getTextHeight());
+ cairo_rectangle(mpRT, 0.0, -fAscent, fTextWidth, fAscent + fDescent);
cairo_fill(mpRT);
cairo_restore(mpRT);
}
@@ -2991,52 +2989,9 @@ void CairoPixelProcessor2D::renderTextSimpleOrDecoratedPortionPrimitive2D(
const primitive2d::TextSimplePortionPrimitive2D& rTextCandidate,
const primitive2d::TextDecoratedPortionPrimitive2D* pDecoratedCandidate)
{
- // decompose primitive-local matrix to get local font scaling
- const basegfx::utils::B2DHomMatrixBufferedOnDemandDecompose aDecTrans(
- rTextCandidate.getTextTransform());
-
- // create a TextLayouter to access encapsulated VCL Text/Font related tooling
primitive2d::TextLayouterDevice aTextLayouter;
- aTextLayouter.setFontAttribute(rTextCandidate.getFontAttribute(), aDecTrans.getScale().getX(),
- aDecTrans.getScale().getY(), rTextCandidate.getLocale());
- const basegfx::BColor aRGBFontColor(
- maBColorModifierStack.getModifiedColor(rTextCandidate.getFontColor()));
- aTextLayouter.setTextColor(aRGBFontColor);
-
- if (rTextCandidate.getFontAttribute().getRTL())
- {
- vcl::text::ComplexTextLayoutFlags nRTLLayoutMode(
- aTextLayouter.getLayoutMode() & ~vcl::text::ComplexTextLayoutFlags::BiDiStrong);
- nRTLLayoutMode |= vcl::text::ComplexTextLayoutFlags::BiDiRtl
- | vcl::text::ComplexTextLayoutFlags::TextOriginLeft;
- aTextLayouter.setLayoutMode(nRTLLayoutMode);
- }
- else
- {
- // tdf#101686: This is LTR text, but the output device may have RTL state.
- vcl::text::ComplexTextLayoutFlags nLTRLayoutMode(aTextLayouter.getLayoutMode());
- nLTRLayoutMode = nLTRLayoutMode & ~vcl::text::ComplexTextLayoutFlags::BiDiRtl;
- nLTRLayoutMode = nLTRLayoutMode & ~vcl::text::ComplexTextLayoutFlags::BiDiStrong;
- aTextLayouter.setLayoutMode(nLTRLayoutMode);
- }
-
- // create integer DXArray. As mentioned above we can act in the
- // Text's local coordinate system without transformation at all
- const ::std::vector<double>& rDXArray(rTextCandidate.getDXArray());
- KernArray aDXArray;
-
- if (!rDXArray.empty())
- {
- aDXArray.reserve(rDXArray.size());
- for (auto const& elem : rDXArray)
- aDXArray.push_back(basegfx::fround(elem));
- }
-
- // create SalLayout. No need for a position, as mentioned text can work
- // without transformations, so start point is always 0,0
- std::unique_ptr<SalLayout> pSalLayout(aTextLayouter.getSalLayout(
- rTextCandidate.getText(), rTextCandidate.getTextPosition(), rTextCandidate.getTextLength(),
- basegfx::B2DPoint(0.0, 0.0), aDXArray, rTextCandidate.getKashidaArray()));
+ rTextCandidate.createTextLayouter(aTextLayouter);
+ std::unique_ptr<SalLayout> pSalLayout(rTextCandidate.createSalLayout(aTextLayouter));
if (!pSalLayout)
{
@@ -3046,6 +3001,8 @@ void CairoPixelProcessor2D::renderTextSimpleOrDecoratedPortionPrimitive2D(
}
// prepare local transformations
+ basegfx::utils::B2DHomMatrixBufferedOnDemandDecompose aDecTrans(
+ rTextCandidate.getTextTransform());
const basegfx::B2DHomMatrix aObjTransformWithoutScale(
basegfx::utils::createShearXRotateTranslateB2DHomMatrix(
aDecTrans.getShearX(), aDecTrans.getRotate(), aDecTrans.getTranslate()));
@@ -3056,11 +3013,23 @@ void CairoPixelProcessor2D::renderTextSimpleOrDecoratedPortionPrimitive2D(
{
// render TextBackground first -> casts no shadow itself, so do independent of
// text shadow being activated
- renderTextBackground(rTextCandidate, aTextLayouter, aFullTextTransform,
+ double fAscent(aTextLayouter.getFontAscent());
+ double fDescent(aTextLayouter.getFontDescent());
+
+ if (nullptr != pDecoratedCandidate
+ && primitive2d::TEXT_FONT_EMPHASIS_MARK_NONE
+ != pDecoratedCandidate->getTextEmphasisMark())
+ {
+ if (pDecoratedCandidate->getEmphasisMarkAbove())
+ fAscent += aTextLayouter.getTextHeight() * (250.0 / 1000.0);
+ if (pDecoratedCandidate->getEmphasisMarkBelow())
+ fDescent += aTextLayouter.getTextHeight() * (250.0 / 1000.0);
+ }
+
+ renderTextBackground(rTextCandidate, fAscent, fDescent, aFullTextTransform,
pSalLayout->GetTextWidth());
}
- // basegfx::utils::B2DHomMatrixBufferedOnDemandDecompose aDecTrans(rTextCandidate.getTextTransform());
bool bHasTextRelief(false);
bool bHasShadow(false);
bool bHasOutline(false);
@@ -3077,7 +3046,9 @@ void CairoPixelProcessor2D::renderTextSimpleOrDecoratedPortionPrimitive2D(
bHasTextDecoration
= primitive2d::TEXT_LINE_NONE != pDecoratedCandidate->getFontOverline()
|| primitive2d::TEXT_LINE_NONE != pDecoratedCandidate->getFontUnderline()
- || primitive2d::TEXT_STRIKEOUT_NONE != pDecoratedCandidate->getTextStrikeout();
+ || primitive2d::TEXT_STRIKEOUT_NONE != pDecoratedCandidate->getTextStrikeout()
+ || primitive2d::TEXT_FONT_EMPHASIS_MARK_NONE
+ != pDecoratedCandidate->getTextEmphasisMark();
}
if (bHasShadow)
@@ -3122,11 +3093,11 @@ void CairoPixelProcessor2D::renderTextSimpleOrDecoratedPortionPrimitive2D(
if (bHasOutline)
{
// todo
- renderSalLayout(pSalLayout, aRGBFontColor, aFullTextTransform,
- getViewInformation2D().getUseAntiAliasing());
}
// render text
+ const basegfx::BColor aRGBFontColor(
+ maBColorModifierStack.getModifiedColor(rTextCandidate.getFontColor()));
renderSalLayout(pSalLayout, aRGBFontColor, aFullTextTransform,
getViewInformation2D().getUseAntiAliasing());
diff --git a/drawinglayer/source/processor2d/vclmetafileprocessor2d.cxx b/drawinglayer/source/processor2d/vclmetafileprocessor2d.cxx
index 4435db4bd583..a5f49e9e654b 100644
--- a/drawinglayer/source/processor2d/vclmetafileprocessor2d.cxx
+++ b/drawinglayer/source/processor2d/vclmetafileprocessor2d.cxx
@@ -903,6 +903,13 @@ void VclMetafileProcessor2D::processBasePrimitive2D(const primitive2d::BasePrimi
static_cast<const primitive2d::ObjectInfoPrimitive2D&>(rCandidate));
break;
}
+ case PRIMITIVE2D_ID_TEXTHIERARCHYEMPHASISMARKPRIMITIVE2D:
+ {
+ // EmphasisMarks are traditionally not added to Metafiles, see
+ // OutputDevice::ImplDrawEmphasisMarks which resets GDIMetaFile*
+ // while painting these, so just ignore these
+ break;
+ }
default:
{
// process recursively