1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
|
// Copyright (C) 2016 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
#include "qsgsoftwareglyphnode_p.h"
#include <QtGui/private/qrawfont_p.h>
QT_BEGIN_NAMESPACE
QSGSoftwareGlyphNode::QSGSoftwareGlyphNode()
: m_geometry(QSGGeometry::defaultAttributes_TexturedPoint2D(), 0)
, m_style(QQuickText::Normal)
{
setMaterial((QSGMaterial*)1);
setGeometry(&m_geometry);
}
namespace {
QRectF calculateBoundingRect(const QPointF &position, const QGlyphRun &glyphs)
{
QFixed minX;
QFixed minY;
QFixed maxX;
QFixed maxY;
QRawFontPrivate *rawFontD = QRawFontPrivate::get(glyphs.rawFont());
QFontEngine *fontEngine = rawFontD->fontEngine;
QFontEngine::GlyphFormat glyphFormat = fontEngine->glyphFormat != QFontEngine::Format_None ? fontEngine->glyphFormat : QFontEngine::Format_A32;
int margin = fontEngine->glyphMargin(glyphFormat);
const QVector<uint> glyphIndexes = glyphs.glyphIndexes();
const QVector<QPointF> glyphPositions = glyphs.positions();
for (int i = 0, n = qMin(glyphIndexes.size(), glyphPositions.size()); i < n; ++i) {
glyph_metrics_t gm = fontEngine->alphaMapBoundingBox(glyphIndexes.at(i), QFixedPoint(), QTransform(), glyphFormat);
gm.x += QFixed::fromReal(glyphPositions.at(i).x()) - margin;
gm.y += QFixed::fromReal(glyphPositions.at(i).y()) - margin;
if (i == 0) {
minX = gm.x;
minY = gm.y;
maxX = gm.x + gm.width;
maxY = gm.y + gm.height;
} else {
minX = qMin(gm.x, minX);
minY = qMin(gm.y, minY);
maxX = qMax(gm.x + gm.width, maxX);
maxY = qMax(gm.y + gm.height, maxY);
}
}
QRectF boundingRect(QPointF(minX.toReal(), minY.toReal()), QPointF(maxX.toReal(), maxY.toReal()));
return boundingRect.translated(position - QPointF(0.0, glyphs.rawFont().ascent()));
}
}
void QSGSoftwareGlyphNode::setGlyphs(const QPointF &position, const QGlyphRun &glyphs)
{
m_position = position;
m_glyphRun = glyphs;
// Decorations handled by text node
m_glyphRun.setOverline(false);
m_glyphRun.setStrikeOut(false);
m_glyphRun.setUnderline(false);
m_bounding_rect = calculateBoundingRect(position, glyphs);
}
void QSGSoftwareGlyphNode::setColor(const QColor &color)
{
m_color = color;
}
void QSGSoftwareGlyphNode::setStyle(QQuickText::TextStyle style)
{
m_style = style;
}
void QSGSoftwareGlyphNode::setStyleColor(const QColor &color)
{
m_styleColor = color;
}
QPointF QSGSoftwareGlyphNode::baseLine() const
{
return QPointF();
}
void QSGSoftwareGlyphNode::setPreferredAntialiasingMode(QSGGlyphNode::AntialiasingMode)
{
}
void QSGSoftwareGlyphNode::update()
{
}
void QSGSoftwareGlyphNode::paint(QPainter *painter)
{
painter->setBrush(QBrush());
QPointF pos = m_position - QPointF(0, m_glyphRun.rawFont().ascent());
qreal offset = 1.0;
if (painter->device()->devicePixelRatio() > 0.0)
offset = 1.0 / painter->device()->devicePixelRatio();
switch (m_style) {
case QQuickText::Normal: break;
case QQuickText::Outline:
painter->setPen(m_styleColor);
painter->drawGlyphRun(pos + QPointF(0, offset), m_glyphRun);
painter->drawGlyphRun(pos + QPointF(0, -offset), m_glyphRun);
painter->drawGlyphRun(pos + QPointF(offset, 0), m_glyphRun);
painter->drawGlyphRun(pos + QPointF(-offset, 0), m_glyphRun);
break;
case QQuickText::Raised:
painter->setPen(m_styleColor);
painter->drawGlyphRun(pos + QPointF(0, offset), m_glyphRun);
break;
case QQuickText::Sunken:
painter->setPen(m_styleColor);
painter->drawGlyphRun(pos + QPointF(0, -offset), m_glyphRun);
break;
}
painter->setPen(m_color);
painter->drawGlyphRun(pos, m_glyphRun);
}
QT_END_NAMESPACE
|