5#include "private/qpaintengine_p.h"
6#include "private/qpainter_p.h"
7#include "private/qpicture_p.h"
8#include "private/qfont_p.h"
20#include <private/qtextengine_p.h>
30 Q_DECLARE_PUBLIC(QPicturePaintEngine)
39 : QPaintEngine(*(
new QPicturePaintEnginePrivate), AllFeatures)
41 Q_D(QPicturePaintEngine);
48 Q_D(QPicturePaintEngine);
58 Q_D(QPicturePaintEngine);
59#ifdef QT_PICTURE_DEBUG
60 qDebug(
"QPicturePaintEngine::begin()");
63 QPicture *pic =
static_cast<QPicture *>(pd);
66 d->pic_d = pic->d_func();
69 d->s.setDevice(&d->pic_d->pictb);
70 d->s.setVersion(d->pic_d->formatMajor);
72 d->pic_d->pictb.open(QIODevice::WriteOnly | QIODevice::Truncate);
73 d->sizeLimitExceeded =
false;
74 d->s.writeRawData(qt_mfhdr_tag, 4);
75 d->s << (quint16) 0 << (quint16) d->pic_d->formatMajor << (quint16) d->pic_d->formatMinor;
76 d->s << (quint8) QPicturePrivate::PdcBegin << (quint8)
sizeof(qint32);
77 d->pic_d->brect = QRect();
78 if (d->pic_d->formatMajor >= 4) {
79 QRect r = pic->boundingRect();
80 d->s << (qint32) r.left() << (qint32) r.top() << (qint32) r.width()
81 << (qint32) r.height();
84 d->s << (quint32)d->pic_d->trecs;
85 d->pic_d->formatOk =
false;
92 Q_D(QPicturePaintEngine);
93#ifdef QT_PICTURE_DEBUG
94 qDebug(
"QPicturePaintEngine::end()");
97 d->s << (quint8) QPicturePrivate::PdcEnd << (quint8) 0;
98 int cs_start =
sizeof(quint32);
99 int data_start = cs_start +
sizeof(quint16);
100 int brect_start = data_start + 2*
sizeof(qint16) + 2*
sizeof(quint8);
101 int pos = d->pic_d->pictb.pos();
102 d->pic_d->pictb.seek(brect_start);
103 if (d->pic_d->formatMajor >= 4) {
104 QRect r =
static_cast<QPicture *>(d->pdev)->boundingRect();
105 d->s << (qint32) r.left() << (qint32) r.top() << (qint32) r.width()
106 << (qint32) r.height();
108 d->s << (quint32) d->pic_d->trecs;
109 d->pic_d->pictb.seek(cs_start);
110 const QByteArray buf = d->pic_d->pictb.buffer();
111 quint16 cs = (quint16) qChecksum(QByteArrayView(buf.constData() + data_start, pos - data_start));
113 d->pic_d->pictb.close();
115 return !d->sizeLimitExceeded;
118#define SERIALIZE_CMD(c)
122 pos = d->pic_d->pictb.pos()
126 Q_D(QPicturePaintEngine);
127#ifdef QT_PICTURE_DEBUG
128 qDebug() <<
" -> updatePen(): width:" << pen.width() <<
"style:"
129 << pen.style() <<
"color:" << pen.color();
133 if (d->pic_d->in_memory_only) {
134 int index = d->pic_d->pen_list.size();
135 d->pic_d->pen_list.append(pen);
140 writeCmdLength(pos, QRect(),
false);
145 Q_D(QPicturePaintEngine);
146#ifdef QT_PICTURE_DEBUG
147 qDebug() <<
" -> updateCompositionMode():" << cmode;
151 d->s << (qint32)cmode;
152 writeCmdLength(pos, QRectF(),
false);
157 Q_D(QPicturePaintEngine);
158#ifdef QT_PICTURE_DEBUG
159 qDebug() <<
" -> updateClipEnabled():" << enabled;
164 writeCmdLength(pos, QRectF(),
false);
169 Q_D(QPicturePaintEngine);
170#ifdef QT_PICTURE_DEBUG
171 qDebug() <<
" -> updateOpacity():" << opacity;
175 d->s <<
double(opacity);
176 writeCmdLength(pos, QRectF(),
false);
181 Q_D(QPicturePaintEngine);
182#ifdef QT_PICTURE_DEBUG
183 qDebug() <<
" -> updateBrush(): style:" << brush.style();
187 if (d->pic_d->in_memory_only) {
188 int index = d->pic_d->brush_list.size();
189 d->pic_d->brush_list.append(brush);
194 writeCmdLength(pos, QRect(),
false);
199 Q_D(QPicturePaintEngine);
200#ifdef QT_PICTURE_DEBUG
201 qDebug() <<
" -> updateBrushOrigin(): " << p;
206 writeCmdLength(pos, QRect(),
false);
211 Q_D(QPicturePaintEngine);
212#ifdef QT_PICTURE_DEBUG
213 qDebug() <<
" -> updateFont(): pt sz:" << font.pointSize();
219 writeCmdLength(pos, QRectF(),
false);
224 Q_D(QPicturePaintEngine);
225#ifdef QT_PICTURE_DEBUG
226 qDebug() <<
" -> updateBackground(): mode:" << bgMode <<
"style:" << bgBrush.style();
230 d->s << bgBrush.color();
231 writeCmdLength(pos, QRect(),
false);
234 d->s << (qint8) bgMode;
235 writeCmdLength(pos, QRectF(),
false);
240 Q_D(QPicturePaintEngine);
241#ifdef QT_PICTURE_DEBUG
242 qDebug() <<
" -> updateMatrix():" << matrix;
246 d->s << matrix << (qint8)
false;
247 writeCmdLength(pos, QRectF(),
false);
252 Q_D(QPicturePaintEngine);
253#ifdef QT_PICTURE_DEBUG
254 qDebug() <<
" -> updateClipRegion(): op:" << op
255 <<
"bounding rect:" << region.boundingRect();
259 d->s << region << qint8(op);
260 writeCmdLength(pos, QRectF(),
false);
265 Q_D(QPicturePaintEngine);
266#ifdef QT_PICTURE_DEBUG
267 qDebug() <<
" -> updateClipPath(): op:" << op
268 <<
"bounding rect:" << path.boundingRect();
273 d->s << path << qint8(op);
274 writeCmdLength(pos, QRectF(),
false);
279 Q_D(QPicturePaintEngine);
280#ifdef QT_PICTURE_DEBUG
281 qDebug() <<
" -> updateRenderHints(): " << hints;
285 d->s << (quint32) hints;
286 writeCmdLength(pos, QRect(),
false);
291 Q_D(QPicturePaintEngine);
293 constexpr int sizeLimit =
std::numeric_limits<
int>::max() - 8;
294 if (d->sizeLimitExceeded || d->pic_d->pictb.pos() > sizeLimit) {
296 d->pic_d->pictb.seek(pos - 2);
297 d->pic_d->pictbData.resize(pos - 2);
298 if (!d->sizeLimitExceeded) {
299 d->sizeLimitExceeded =
true;
300 qWarning(
"QPicture: size limit exceeded, will be truncated");
305 int newpos = d->pic_d->pictb.pos();
306 int length = newpos - pos;
310 d->pic_d->pictb.seek(pos - 1);
311 d->s << (quint8)length;
314 d->pic_d->pictb.seek(pos - 1);
316 char *p = d->pic_d->pictb.buffer().data();
317 memmove(p+pos+4, p+pos, length);
318 d->s << (quint32)length;
321 d->pic_d->pictb.seek(newpos);
323 if (br.width() > 0.0 || br.height() > 0.0) {
325 int w2 = painter()->pen().width() / 2;
326 br.setCoords(br.left() - w2, br.top() - w2,
327 br.right() + w2, br.bottom() + w2);
329 br = painter()->transform().mapRect(br);
330 if (painter()->hasClipping()) {
331 QRectF cr = painter()->clipBoundingRect();
335 if (br.width() > 0.0 || br.height() > 0.0) {
336 const auto clampToIntRange = [](qreal v)
338 return qBound(qreal((std::numeric_limits<
int>::min)()),
340 qreal((std::numeric_limits<
int>::max)()));
342 int minx = qFloor(clampToIntRange(br.left()));
343 int miny = qFloor(clampToIntRange(br.top()));
344 int maxx = qCeil(clampToIntRange(br.right()));
345 int maxy = qCeil(clampToIntRange(br.bottom()));
347 if (d->pic_d->brect.width() > 0 || d->pic_d->brect.height() > 0) {
348 minx = qMin(minx, d->pic_d->brect.left());
349 miny = qMin(miny, d->pic_d->brect.top());
350 maxx = qMax(maxx, d->pic_d->brect.x() + d->pic_d->brect.width());
351 maxy = qMax(maxy, d->pic_d->brect.y() + d->pic_d->brect.height());
352 d->pic_d->brect.setCoords(minx, miny, maxx - 1, maxy - 1);
354 d->pic_d->brect.setCoords(minx, miny, maxx - 1, maxy - 1);
362 Q_D(QPicturePaintEngine);
363#ifdef QT_PICTURE_DEBUG
364 qDebug() <<
" -> drawEllipse():" << rect;
369 writeCmdLength(pos, rect,
true);
374 Q_D(QPicturePaintEngine);
375#ifdef QT_PICTURE_DEBUG
376 qDebug() <<
" -> drawPath():" << path.boundingRect();
381 writeCmdLength(pos, path.boundingRect(),
true);
386 Q_D(QPicturePaintEngine);
387#ifdef QT_PICTURE_DEBUG
388 qDebug() <<
" -> drawPolygon(): size=" << numPoints;
393 polygon.reserve(numPoints);
394 for (
int i=0; i<numPoints; ++i)
395 polygon << points[i];
397 if (mode == PolylineMode) {
403 d->s << (qint8)(mode == OddEvenMode ? 0 : 1);
406 writeCmdLength(pos, polygon.boundingRect(),
true);
411 Q_D(QPicturePaintEngine);
412#ifdef QT_PICTURE_DEBUG
413 qDebug() <<
" -> drawPixmap():" << r;
418 if (d->pic_d->in_memory_only) {
419 int index = d->pic_d->pixmap_list.size();
420 d->pic_d->pixmap_list.append(pm);
421 d->s << r << index << sr;
423 d->s << r << pm << sr;
425 writeCmdLength(pos, r,
false);
430 Q_D(QPicturePaintEngine);
431#ifdef QT_PICTURE_DEBUG
432 qDebug() <<
" -> drawTiledPixmap():" << r << s;
436 if (d->pic_d->in_memory_only) {
437 int index = d->pic_d->pixmap_list.size();
438 d->pic_d->pixmap_list.append(pixmap);
439 d->s << r << index << s;
441 d->s << r << pixmap << s;
443 writeCmdLength(pos, r,
false);
447 Qt::ImageConversionFlags flags)
449 Q_D(QPicturePaintEngine);
450#ifdef QT_PICTURE_DEBUG
451 qDebug() <<
" -> drawImage():" << r << sr;
455 if (d->pic_d->in_memory_only) {
456 int index = d->pic_d->image_list.size();
457 d->pic_d->image_list.append(image);
458 d->s << r << index << sr << (quint32) flags;
460 d->s << r << image << sr << (quint32) flags;
462 writeCmdLength(pos, r,
false);
467 Q_D(QPicturePaintEngine);
468#ifdef QT_PICTURE_DEBUG
469 qDebug() <<
" -> drawTextItem():" << p << ti.text();
472 const QTextItemInt &si =
static_cast<
const QTextItemInt &>(ti);
473 if (si.chars ==
nullptr)
474 QPaintEngine::drawTextItem(p, ti);
476 if (d->pic_d->formatMajor >= 9) {
479 QFont fnt = ti.font();
480 fnt.setUnderline(
false);
481 fnt.setStrikeOut(
false);
482 fnt.setOverline(
false);
484 qreal justificationWidth = 0;
486 justificationWidth = si.width.toReal();
488 d->s << p << ti.text() << fnt << ti.renderFlags() <<
double(fnt.d->dpi)/qt_defaultDpi() << justificationWidth;
489 writeCmdLength(pos, QRectF(),
false);
490 }
else if (d->pic_d->formatMajor >= 8) {
494 d->s << QPointF(p.x(), p.y() - ti.ascent()) << ti.text() << ti.font() << ti.renderFlags();
495 writeCmdLength(pos, QRectF(),
false);
500 d->s << p << ti.text();
501 writeCmdLength(pos, QRectF(p, QSizeF(1,1)),
true);
507 QPaintEngine::DirtyFlags flags = state.state();
508 if (flags & DirtyPen) updatePen(state.pen());
509 if (flags & DirtyBrush) updateBrush(state.brush());
510 if (flags & DirtyBrushOrigin) updateBrushOrigin(state.brushOrigin());
511 if (flags & DirtyFont) updateFont(state.font());
512 if (flags & DirtyBackground) updateBackground(state.backgroundMode(), state.backgroundBrush());
513 if (flags & DirtyTransform) updateMatrix(state.transform());
515 if (flags & DirtyClipRegion) updateClipRegion(state.clipRegion(), state.clipOperation());
516 if (flags & DirtyClipPath) updateClipPath(state.clipPath(), state.clipOperation());
517 if (flags & DirtyHints) updateRenderHints(state.renderHints());
518 if (flags & DirtyCompositionMode) updateCompositionMode(state.compositionMode());
519 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.
Combined button and popup list for selecting options.