4#include "private/qpaintengine_p.h"
5#include "private/qpainter_p.h"
6#include "private/qpicture_p.h"
7#include "private/qfont_p.h"
19#include <private/qtextengine_p.h>
29 Q_DECLARE_PUBLIC(QPicturePaintEngine)
37 : QPaintEngine(*(
new QPicturePaintEnginePrivate), AllFeatures)
39 Q_D(QPicturePaintEngine);
46 Q_D(QPicturePaintEngine);
56 Q_D(QPicturePaintEngine);
57#ifdef QT_PICTURE_DEBUG
58 qDebug(
"QPicturePaintEngine::begin()");
61 QPicture *pic =
static_cast<QPicture *>(pd);
64 d->pic_d = pic->d_func();
67 d->s.setDevice(&d->pic_d->pictb);
68 d->s.setVersion(d->pic_d->formatMajor);
70 d->pic_d->pictb.open(QIODevice::WriteOnly | QIODevice::Truncate);
71 d->s.writeRawData(qt_mfhdr_tag, 4);
72 d->s << (quint16) 0 << (quint16) d->pic_d->formatMajor << (quint16) d->pic_d->formatMinor;
73 d->s << (quint8) QPicturePrivate::PdcBegin << (quint8)
sizeof(qint32);
74 d->pic_d->brect = QRect();
75 if (d->pic_d->formatMajor >= 4) {
76 QRect r = pic->boundingRect();
77 d->s << (qint32) r.left() << (qint32) r.top() << (qint32) r.width()
78 << (qint32) r.height();
81 d->s << (quint32)d->pic_d->trecs;
82 d->pic_d->formatOk =
false;
89 Q_D(QPicturePaintEngine);
90#ifdef QT_PICTURE_DEBUG
91 qDebug(
"QPicturePaintEngine::end()");
94 d->s << (quint8) QPicturePrivate::PdcEnd << (quint8) 0;
95 int cs_start =
sizeof(quint32);
96 int data_start = cs_start +
sizeof(quint16);
97 int brect_start = data_start + 2*
sizeof(qint16) + 2*
sizeof(quint8);
98 int pos = d->pic_d->pictb.pos();
99 d->pic_d->pictb.seek(brect_start);
100 if (d->pic_d->formatMajor >= 4) {
101 QRect r =
static_cast<QPicture *>(d->pdev)->boundingRect();
102 d->s << (qint32) r.left() << (qint32) r.top() << (qint32) r.width()
103 << (qint32) r.height();
105 d->s << (quint32) d->pic_d->trecs;
106 d->pic_d->pictb.seek(cs_start);
107 const QByteArray buf = d->pic_d->pictb.buffer();
108 quint16 cs = (quint16) qChecksum(QByteArrayView(buf.constData() + data_start, pos - data_start));
110 d->pic_d->pictb.close();
115#define SERIALIZE_CMD(c)
119 pos = d->pic_d->pictb.pos()
123 Q_D(QPicturePaintEngine);
124#ifdef QT_PICTURE_DEBUG
125 qDebug() <<
" -> updatePen(): width:" << pen.width() <<
"style:"
126 << pen.style() <<
"color:" << pen.color();
130 if (d->pic_d->in_memory_only) {
131 int index = d->pic_d->pen_list.size();
132 d->pic_d->pen_list.append(pen);
137 writeCmdLength(pos, QRect(),
false);
142 Q_D(QPicturePaintEngine);
143#ifdef QT_PICTURE_DEBUG
144 qDebug() <<
" -> updateCompositionMode():" << cmode;
148 d->s << (qint32)cmode;
149 writeCmdLength(pos, QRectF(),
false);
154 Q_D(QPicturePaintEngine);
155#ifdef QT_PICTURE_DEBUG
156 qDebug() <<
" -> updateClipEnabled():" << enabled;
161 writeCmdLength(pos, QRectF(),
false);
166 Q_D(QPicturePaintEngine);
167#ifdef QT_PICTURE_DEBUG
168 qDebug() <<
" -> updateOpacity():" << opacity;
172 d->s <<
double(opacity);
173 writeCmdLength(pos, QRectF(),
false);
178 Q_D(QPicturePaintEngine);
179#ifdef QT_PICTURE_DEBUG
180 qDebug() <<
" -> updateBrush(): style:" << brush.style();
184 if (d->pic_d->in_memory_only) {
185 int index = d->pic_d->brush_list.size();
186 d->pic_d->brush_list.append(brush);
191 writeCmdLength(pos, QRect(),
false);
196 Q_D(QPicturePaintEngine);
197#ifdef QT_PICTURE_DEBUG
198 qDebug() <<
" -> updateBrushOrigin(): " << p;
203 writeCmdLength(pos, QRect(),
false);
208 Q_D(QPicturePaintEngine);
209#ifdef QT_PICTURE_DEBUG
210 qDebug() <<
" -> updateFont(): pt sz:" << font.pointSize();
216 writeCmdLength(pos, QRectF(),
false);
221 Q_D(QPicturePaintEngine);
222#ifdef QT_PICTURE_DEBUG
223 qDebug() <<
" -> updateBackground(): mode:" << bgMode <<
"style:" << bgBrush.style();
227 d->s << bgBrush.color();
228 writeCmdLength(pos, QRect(),
false);
231 d->s << (qint8) bgMode;
232 writeCmdLength(pos, QRectF(),
false);
237 Q_D(QPicturePaintEngine);
238#ifdef QT_PICTURE_DEBUG
239 qDebug() <<
" -> updateMatrix():" << matrix;
243 d->s << matrix << (qint8)
false;
244 writeCmdLength(pos, QRectF(),
false);
249 Q_D(QPicturePaintEngine);
250#ifdef QT_PICTURE_DEBUG
251 qDebug() <<
" -> updateClipRegion(): op:" << op
252 <<
"bounding rect:" << region.boundingRect();
256 d->s << region << qint8(op);
257 writeCmdLength(pos, QRectF(),
false);
262 Q_D(QPicturePaintEngine);
263#ifdef QT_PICTURE_DEBUG
264 qDebug() <<
" -> updateClipPath(): op:" << op
265 <<
"bounding rect:" << path.boundingRect();
270 d->s << path << qint8(op);
271 writeCmdLength(pos, QRectF(),
false);
276 Q_D(QPicturePaintEngine);
277#ifdef QT_PICTURE_DEBUG
278 qDebug() <<
" -> updateRenderHints(): " << hints;
282 d->s << (quint32) hints;
283 writeCmdLength(pos, QRect(),
false);
288 Q_D(QPicturePaintEngine);
289 int newpos = d->pic_d->pictb.pos();
290 int length = newpos - pos;
294 d->pic_d->pictb.seek(pos - 1);
295 d->s << (quint8)length;
298 d->pic_d->pictb.seek(pos - 1);
300 char *p = d->pic_d->pictb.buffer().data();
301 memmove(p+pos+4, p+pos, length);
302 d->s << (quint32)length;
305 d->pic_d->pictb.seek(newpos);
307 if (br.width() > 0.0 || br.height() > 0.0) {
309 int w2 = painter()->pen().width() / 2;
310 br.setCoords(br.left() - w2, br.top() - w2,
311 br.right() + w2, br.bottom() + w2);
313 br = painter()->transform().mapRect(br);
314 if (painter()->hasClipping()) {
315 QRectF cr = painter()->clipBoundingRect();
319 if (br.width() > 0.0 || br.height() > 0.0) {
320 const auto clampToIntRange = [](qreal v)
322 return qBound(qreal((std::numeric_limits<
int>::min)()),
324 qreal((std::numeric_limits<
int>::max)()));
326 int minx = qFloor(clampToIntRange(br.left()));
327 int miny = qFloor(clampToIntRange(br.top()));
328 int maxx = qCeil(clampToIntRange(br.right()));
329 int maxy = qCeil(clampToIntRange(br.bottom()));
331 if (d->pic_d->brect.width() > 0 || d->pic_d->brect.height() > 0) {
332 minx = qMin(minx, d->pic_d->brect.left());
333 miny = qMin(miny, d->pic_d->brect.top());
334 maxx = qMax(maxx, d->pic_d->brect.x() + d->pic_d->brect.width());
335 maxy = qMax(maxy, d->pic_d->brect.y() + d->pic_d->brect.height());
336 d->pic_d->brect.setCoords(minx, miny, maxx - 1, maxy - 1);
338 d->pic_d->brect.setCoords(minx, miny, maxx - 1, maxy - 1);
346 Q_D(QPicturePaintEngine);
347#ifdef QT_PICTURE_DEBUG
348 qDebug() <<
" -> drawEllipse():" << rect;
353 writeCmdLength(pos, rect,
true);
358 Q_D(QPicturePaintEngine);
359#ifdef QT_PICTURE_DEBUG
360 qDebug() <<
" -> drawPath():" << path.boundingRect();
365 writeCmdLength(pos, path.boundingRect(),
true);
370 Q_D(QPicturePaintEngine);
371#ifdef QT_PICTURE_DEBUG
372 qDebug() <<
" -> drawPolygon(): size=" << numPoints;
377 polygon.reserve(numPoints);
378 for (
int i=0; i<numPoints; ++i)
379 polygon << points[i];
381 if (mode == PolylineMode) {
387 d->s << (qint8)(mode == OddEvenMode ? 0 : 1);
390 writeCmdLength(pos, polygon.boundingRect(),
true);
395 Q_D(QPicturePaintEngine);
396#ifdef QT_PICTURE_DEBUG
397 qDebug() <<
" -> drawPixmap():" << r;
402 if (d->pic_d->in_memory_only) {
403 int index = d->pic_d->pixmap_list.size();
404 d->pic_d->pixmap_list.append(pm);
405 d->s << r << index << sr;
407 d->s << r << pm << sr;
409 writeCmdLength(pos, r,
false);
414 Q_D(QPicturePaintEngine);
415#ifdef QT_PICTURE_DEBUG
416 qDebug() <<
" -> drawTiledPixmap():" << r << s;
420 if (d->pic_d->in_memory_only) {
421 int index = d->pic_d->pixmap_list.size();
422 d->pic_d->pixmap_list.append(pixmap);
423 d->s << r << index << s;
425 d->s << r << pixmap << s;
427 writeCmdLength(pos, r,
false);
431 Qt::ImageConversionFlags flags)
433 Q_D(QPicturePaintEngine);
434#ifdef QT_PICTURE_DEBUG
435 qDebug() <<
" -> drawImage():" << r << sr;
439 if (d->pic_d->in_memory_only) {
440 int index = d->pic_d->image_list.size();
441 d->pic_d->image_list.append(image);
442 d->s << r << index << sr << (quint32) flags;
444 d->s << r << image << sr << (quint32) flags;
446 writeCmdLength(pos, r,
false);
451 Q_D(QPicturePaintEngine);
452#ifdef QT_PICTURE_DEBUG
453 qDebug() <<
" -> drawTextItem():" << p << ti.text();
456 const QTextItemInt &si =
static_cast<
const QTextItemInt &>(ti);
457 if (si.chars ==
nullptr)
458 QPaintEngine::drawTextItem(p, ti);
460 if (d->pic_d->formatMajor >= 9) {
463 QFont fnt = ti.font();
464 fnt.setUnderline(
false);
465 fnt.setStrikeOut(
false);
466 fnt.setOverline(
false);
468 qreal justificationWidth = 0;
470 justificationWidth = si.width.toReal();
472 d->s << p << ti.text() << fnt << ti.renderFlags() <<
double(fnt.d->dpi)/qt_defaultDpi() << justificationWidth;
473 writeCmdLength(pos, QRectF(),
false);
474 }
else if (d->pic_d->formatMajor >= 8) {
478 d->s << QPointF(p.x(), p.y() - ti.ascent()) << ti.text() << ti.font() << ti.renderFlags();
479 writeCmdLength(pos, QRectF(),
false);
484 d->s << p << ti.text();
485 writeCmdLength(pos, QRectF(p, QSizeF(1,1)),
true);
491 QPaintEngine::DirtyFlags flags = state.state();
492 if (flags & DirtyPen) updatePen(state.pen());
493 if (flags & DirtyBrush) updateBrush(state.brush());
494 if (flags & DirtyBrushOrigin) updateBrushOrigin(state.brushOrigin());
495 if (flags & DirtyFont) updateFont(state.font());
496 if (flags & DirtyBackground) updateBackground(state.backgroundMode(), state.backgroundBrush());
497 if (flags & DirtyTransform) updateMatrix(state.transform());
499 if (flags & DirtyClipRegion) updateClipRegion(state.clipRegion(), state.clipOperation());
500 if (flags & DirtyClipPath) updateClipPath(state.clipPath(), state.clipOperation());
501 if (flags & DirtyHints) updateRenderHints(state.renderHints());
502 if (flags & DirtyCompositionMode) updateCompositionMode(state.compositionMode());
503 if (flags & DirtyOpacity) updateOpacity(state.opacity());
void updateOpacity(qreal opacity)
void updatePen(const QPen &pen)
void drawPolygon(const QPointF *points, int numPoints, PolygonDrawMode mode) override
Reimplement this virtual function to draw the polygon defined by the pointCount first points in point...
void updateClipRegion(const QRegion ®ion, Qt::ClipOperation op)
void drawPath(const QPainterPath &path) override
The default implementation ignores the path and does nothing.
void drawTextItem(const QPointF &p, const QTextItem &ti) override
This function draws the text item textItem at position p.
QPicturePaintEngine(QPaintEnginePrivate &dptr)
void updateBrushOrigin(const QPointF &origin)
void updateFont(const QFont &font)
void updateCompositionMode(QPainter::CompositionMode cmode)
void updateClipEnabled(bool enabled)
void updateState(const QPaintEngineState &state) override
Reimplement this function to update the state of a paint engine.
bool end() override
Reimplement this function to finish painting on the current paint device.
void updateBrush(const QBrush &brush)
void updateClipPath(const QPainterPath &path, Qt::ClipOperation op)
void updateRenderHints(QPainter::RenderHints hints)
void updateBackground(Qt::BGMode bgmode, const QBrush &bgBrush)
bool begin(QPaintDevice *pdev) override
Reimplement this function to initialise your paint engine when painting is to start on the paint devi...
The QPolygonF class provides a list of points using floating point precision.