Make text decorations consistent between ways of drawing text
The drawGlyphRun() and drawStaticText() functions would calculate the horizontal origin of the decorations based on the left-most edge of the left-most glyph, in practice including the left bearing of that glyph in the position. But in drawText()/QTextLayout it will always be drawn from the position given as input by the user of the function. The inconsistency was detected in an upgrade of Harfbuzz NG, where the tests on macOS would get a -0.05 left bearing, and as a result the decorations would be painted one pixel to the left in drawGlyphRun()/drawStaticText() compared to QTextLayout. It is not a big deal in practice, but in order to get the Harfbuzz update in, we fix the inconsistency now, by passing the user-input position into the decoration function. This was also an opportunity to consolidate the two code paths, and to unexport the qt_draw_decoration_for_glyphs() symbol, which was exported in Qt 4 only to be usable by Qt Quick 1. Change-Id: I243404b2710ae378e84d7587efae719da3879944 Reviewed-by: Lars Knoll <lars.knoll@qt.io>
This commit is contained in:
parent
31911d387b
commit
78da102810
@ -96,10 +96,15 @@ static void drawTextItemDecoration(QPainter *painter, const QPointF &pos, const
|
|||||||
QTextItem::RenderFlags flags, qreal width,
|
QTextItem::RenderFlags flags, qreal width,
|
||||||
const QTextCharFormat &charFormat);
|
const QTextCharFormat &charFormat);
|
||||||
// Helper function to calculate left most position, width and flags for decoration drawing
|
// Helper function to calculate left most position, width and flags for decoration drawing
|
||||||
Q_GUI_EXPORT void qt_draw_decoration_for_glyphs(QPainter *painter, const glyph_t *glyphArray,
|
static void qt_draw_decoration_for_glyphs(QPainter *painter,
|
||||||
const QFixedPoint *positions, int glyphCount,
|
const QPointF &decorationPosition,
|
||||||
QFontEngine *fontEngine, const QFont &font,
|
const glyph_t *glyphArray,
|
||||||
const QTextCharFormat &charFormat);
|
const QFixedPoint *positions,
|
||||||
|
int glyphCount,
|
||||||
|
QFontEngine *fontEngine,
|
||||||
|
bool underline,
|
||||||
|
bool overline,
|
||||||
|
bool strikeOut);
|
||||||
|
|
||||||
static inline QGradient::CoordinateMode coordinateMode(const QBrush &brush)
|
static inline QGradient::CoordinateMode coordinateMode(const QBrush &brush)
|
||||||
{
|
{
|
||||||
@ -5607,40 +5612,31 @@ void QPainter::drawGlyphRun(const QPointF &position, const QGlyphRun &glyphRun)
|
|||||||
fixedPointPositions[i] = QFixedPoint::fromPointF(processedPosition);
|
fixedPointPositions[i] = QFixedPoint::fromPointF(processedPosition);
|
||||||
}
|
}
|
||||||
|
|
||||||
d->drawGlyphs(glyphIndexes, fixedPointPositions.data(), count, fontD->fontEngine,
|
d->drawGlyphs(engineRequiresPretransformedGlyphPositions
|
||||||
glyphRun.overline(), glyphRun.underline(), glyphRun.strikeOut());
|
? d->state->transform().map(position)
|
||||||
|
: position,
|
||||||
|
glyphIndexes,
|
||||||
|
fixedPointPositions.data(),
|
||||||
|
count,
|
||||||
|
fontD->fontEngine,
|
||||||
|
glyphRun.overline(),
|
||||||
|
glyphRun.underline(),
|
||||||
|
glyphRun.strikeOut());
|
||||||
}
|
}
|
||||||
|
|
||||||
void QPainterPrivate::drawGlyphs(const quint32 *glyphArray, QFixedPoint *positions,
|
void QPainterPrivate::drawGlyphs(const QPointF &decorationPosition,
|
||||||
|
const quint32 *glyphArray,
|
||||||
|
QFixedPoint *positions,
|
||||||
int glyphCount,
|
int glyphCount,
|
||||||
QFontEngine *fontEngine, bool overline, bool underline,
|
QFontEngine *fontEngine,
|
||||||
|
bool overline,
|
||||||
|
bool underline,
|
||||||
bool strikeOut)
|
bool strikeOut)
|
||||||
{
|
{
|
||||||
Q_Q(QPainter);
|
Q_Q(QPainter);
|
||||||
|
|
||||||
updateState(state);
|
updateState(state);
|
||||||
|
|
||||||
QFixed leftMost;
|
|
||||||
QFixed rightMost;
|
|
||||||
QFixed baseLine;
|
|
||||||
for (int i=0; i<glyphCount; ++i) {
|
|
||||||
glyph_metrics_t gm = fontEngine->boundingBox(glyphArray[i]);
|
|
||||||
if (i == 0 || leftMost > positions[i].x)
|
|
||||||
leftMost = positions[i].x;
|
|
||||||
|
|
||||||
// We don't support glyphs that do not share a common baseline. If this turns out to
|
|
||||||
// be a relevant use case, then we need to find clusters of glyphs that share a baseline
|
|
||||||
// and do a drawTextItemDecorations call per cluster.
|
|
||||||
if (i == 0 || baseLine < positions[i].y)
|
|
||||||
baseLine = positions[i].y;
|
|
||||||
|
|
||||||
// We use the advance rather than the actual bounds to match the algorithm in drawText()
|
|
||||||
if (i == 0 || rightMost < positions[i].x + gm.xoff)
|
|
||||||
rightMost = positions[i].x + gm.xoff;
|
|
||||||
}
|
|
||||||
|
|
||||||
QFixed width = rightMost - leftMost;
|
|
||||||
|
|
||||||
if (extended != nullptr && state->matrix.isAffine()) {
|
if (extended != nullptr && state->matrix.isAffine()) {
|
||||||
QStaticTextItem staticTextItem;
|
QStaticTextItem staticTextItem;
|
||||||
staticTextItem.color = state->pen.color();
|
staticTextItem.color = state->pen.color();
|
||||||
@ -5674,21 +5670,15 @@ void QPainterPrivate::drawGlyphs(const quint32 *glyphArray, QFixedPoint *positio
|
|||||||
engine->drawTextItem(QPointF(0, 0), textItem);
|
engine->drawTextItem(QPointF(0, 0), textItem);
|
||||||
}
|
}
|
||||||
|
|
||||||
QTextItemInt::RenderFlags flags;
|
qt_draw_decoration_for_glyphs(q,
|
||||||
if (underline)
|
decorationPosition,
|
||||||
flags |= QTextItemInt::Underline;
|
glyphArray,
|
||||||
if (overline)
|
positions,
|
||||||
flags |= QTextItemInt::Overline;
|
glyphCount,
|
||||||
if (strikeOut)
|
|
||||||
flags |= QTextItemInt::StrikeOut;
|
|
||||||
|
|
||||||
drawTextItemDecoration(q, QPointF(leftMost.toReal(), baseLine.toReal()),
|
|
||||||
fontEngine,
|
fontEngine,
|
||||||
nullptr, // textEngine
|
underline,
|
||||||
(underline
|
overline,
|
||||||
? QTextCharFormat::SingleUnderline
|
strikeOut);
|
||||||
: QTextCharFormat::NoUnderline),
|
|
||||||
flags, width.toReal(), QTextCharFormat());
|
|
||||||
}
|
}
|
||||||
#endif // QT_NO_RAWFONT
|
#endif // QT_NO_RAWFONT
|
||||||
|
|
||||||
@ -5868,9 +5858,15 @@ void QPainter::drawStaticText(const QPointF &topLeftPosition, const QStaticText
|
|||||||
}
|
}
|
||||||
d->extended->drawStaticTextItem(item);
|
d->extended->drawStaticTextItem(item);
|
||||||
|
|
||||||
qt_draw_decoration_for_glyphs(this, item->glyphs, item->glyphPositions,
|
qt_draw_decoration_for_glyphs(this,
|
||||||
item->numGlyphs, item->fontEngine(), staticText_d->font,
|
topLeftPosition,
|
||||||
QTextCharFormat());
|
item->glyphs,
|
||||||
|
item->glyphPositions,
|
||||||
|
item->numGlyphs,
|
||||||
|
item->fontEngine(),
|
||||||
|
staticText_d->font.underline(),
|
||||||
|
staticText_d->font.overline(),
|
||||||
|
staticText_d->font.strikeOut());
|
||||||
}
|
}
|
||||||
if (currentColor != oldPen.color())
|
if (currentColor != oldPen.color())
|
||||||
setPen(oldPen);
|
setPen(oldPen);
|
||||||
@ -6375,49 +6371,44 @@ static void drawTextItemDecoration(QPainter *painter, const QPointF &pos, const
|
|||||||
painter->setRenderHint(QPainter::Qt4CompatiblePainting);
|
painter->setRenderHint(QPainter::Qt4CompatiblePainting);
|
||||||
}
|
}
|
||||||
|
|
||||||
Q_GUI_EXPORT void qt_draw_decoration_for_glyphs(QPainter *painter, const glyph_t *glyphArray,
|
static void qt_draw_decoration_for_glyphs(QPainter *painter,
|
||||||
const QFixedPoint *positions, int glyphCount,
|
const QPointF &decorationPosition,
|
||||||
QFontEngine *fontEngine, const QFont &font,
|
const glyph_t *glyphArray,
|
||||||
const QTextCharFormat &charFormat)
|
const QFixedPoint *positions,
|
||||||
|
int glyphCount,
|
||||||
|
QFontEngine *fontEngine,
|
||||||
|
bool underline,
|
||||||
|
bool overline,
|
||||||
|
bool strikeOut)
|
||||||
{
|
{
|
||||||
if (!(font.underline() || font.strikeOut() || font.overline()))
|
if (!underline && !overline && !strikeOut)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
QFixed leftMost;
|
|
||||||
QFixed rightMost;
|
|
||||||
QFixed baseLine;
|
|
||||||
for (int i=0; i<glyphCount; ++i) {
|
|
||||||
glyph_metrics_t gm = fontEngine->boundingBox(glyphArray[i]);
|
|
||||||
if (i == 0 || leftMost > positions[i].x)
|
|
||||||
leftMost = positions[i].x;
|
|
||||||
|
|
||||||
// We don't support glyphs that do not share a common baseline. If this turns out to
|
|
||||||
// be a relevant use case, then we need to find clusters of glyphs that share a baseline
|
|
||||||
// and do a drawTextItemDecoration call per cluster.
|
|
||||||
if (i == 0 || baseLine < positions[i].y)
|
|
||||||
baseLine = positions[i].y;
|
|
||||||
|
|
||||||
// We use the advance rather than the actual bounds to match the algorithm in drawText()
|
|
||||||
if (i == 0 || rightMost < positions[i].x + gm.xoff)
|
|
||||||
rightMost = positions[i].x + gm.xoff;
|
|
||||||
}
|
|
||||||
|
|
||||||
QFixed width = rightMost - leftMost;
|
|
||||||
QTextItem::RenderFlags flags;
|
QTextItem::RenderFlags flags;
|
||||||
|
if (underline)
|
||||||
if (font.underline())
|
|
||||||
flags |= QTextItem::Underline;
|
flags |= QTextItem::Underline;
|
||||||
if (font.overline())
|
if (overline)
|
||||||
flags |= QTextItem::Overline;
|
flags |= QTextItem::Overline;
|
||||||
if (font.strikeOut())
|
if (strikeOut)
|
||||||
flags |= QTextItem::StrikeOut;
|
flags |= QTextItem::StrikeOut;
|
||||||
|
|
||||||
drawTextItemDecoration(painter, QPointF(leftMost.toReal(), baseLine.toReal()),
|
bool rtl = positions[glyphCount - 1].x < positions[0].x;
|
||||||
|
QFixed baseline = positions[0].y;
|
||||||
|
glyph_metrics_t gm = fontEngine->boundingBox(glyphArray[rtl ? 0 : glyphCount - 1]);
|
||||||
|
|
||||||
|
qreal width = rtl
|
||||||
|
? (positions[0].x + gm.xoff - positions[glyphCount - 1].x).toReal()
|
||||||
|
: (positions[glyphCount - 1].x + gm.xoff - positions[0].x).toReal();
|
||||||
|
|
||||||
|
drawTextItemDecoration(painter,
|
||||||
|
QPointF(decorationPosition.x(), baseline.toReal()),
|
||||||
fontEngine,
|
fontEngine,
|
||||||
nullptr, // textEngine
|
nullptr, // textEngine
|
||||||
font.underline() ? QTextCharFormat::SingleUnderline
|
underline ? QTextCharFormat::SingleUnderline
|
||||||
: QTextCharFormat::NoUnderline, flags,
|
: QTextCharFormat::NoUnderline,
|
||||||
width.toReal(), charFormat);
|
flags,
|
||||||
|
width,
|
||||||
|
QTextCharFormat());
|
||||||
}
|
}
|
||||||
|
|
||||||
void QPainter::drawTextItem(const QPointF &p, const QTextItem &ti)
|
void QPainter::drawTextItem(const QPointF &p, const QTextItem &ti)
|
||||||
|
@ -235,7 +235,7 @@ public:
|
|||||||
void drawTextItem(const QPointF &p, const QTextItem &_ti, QTextEngine *textEngine);
|
void drawTextItem(const QPointF &p, const QTextItem &_ti, QTextEngine *textEngine);
|
||||||
|
|
||||||
#if !defined(QT_NO_RAWFONT)
|
#if !defined(QT_NO_RAWFONT)
|
||||||
void drawGlyphs(const quint32 *glyphArray, QFixedPoint *positionArray, int glyphCount,
|
void drawGlyphs(const QPointF &decorationPosition, const quint32 *glyphArray, QFixedPoint *positionArray, int glyphCount,
|
||||||
QFontEngine *fontEngine, bool overline = false, bool underline = false,
|
QFontEngine *fontEngine, bool overline = false, bool underline = false,
|
||||||
bool strikeOut = false);
|
bool strikeOut = false);
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
Reference in New Issue
Block a user