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)
38 : QPaintEngine(*(
new QPicturePaintEnginePrivate), AllFeatures)
40 Q_D(QPicturePaintEngine);
47 Q_D(QPicturePaintEngine);
57 Q_D(QPicturePaintEngine);
58#ifdef QT_PICTURE_DEBUG
59 qDebug(
"QPicturePaintEngine::begin()");
62 QPicture *pic =
static_cast<QPicture *>(pd);
65 d->pic_d = pic->d_func();
68 d->s.setDevice(&d->pic_d->pictb);
69 d->s.setVersion(d->pic_d->formatMajor);
71 d->pic_d->pictb.open(QIODevice::WriteOnly | QIODevice::Truncate);
72 d->sizeLimitExceeded =
false;
73 d->s.writeRawData(qt_mfhdr_tag, 4);
74 d->s << (quint16) 0 << (quint16) d->pic_d->formatMajor << (quint16) d->pic_d->formatMinor;
75 d->s << (quint8) QPicturePrivate::PdcBegin << (quint8)
sizeof(qint32);
76 d->pic_d->brect = QRect();
77 if (d->pic_d->formatMajor >= 4) {
78 QRect r = pic->boundingRect();
79 d->s << (qint32) r.left() << (qint32) r.top() << (qint32) r.width()
80 << (qint32) r.height();
83 d->s << (quint32)d->pic_d->trecs;
84 d->pic_d->formatOk =
false;
91 Q_D(QPicturePaintEngine);
92#ifdef QT_PICTURE_DEBUG
93 qDebug(
"QPicturePaintEngine::end()");
96 d->s << (quint8) QPicturePrivate::PdcEnd << (quint8) 0;
97 int cs_start =
sizeof(quint32);
98 int data_start = cs_start +
sizeof(quint16);
99 int brect_start = data_start + 2*
sizeof(qint16) + 2*
sizeof(quint8);
100 int pos = d->pic_d->pictb.pos();
101 d->pic_d->pictb.seek(brect_start);
102 if (d->pic_d->formatMajor >= 4) {
103 QRect r =
static_cast<QPicture *>(d->pdev)->boundingRect();
104 d->s << (qint32) r.left() << (qint32) r.top() << (qint32) r.width()
105 << (qint32) r.height();
107 d->s << (quint32) d->pic_d->trecs;
108 d->pic_d->pictb.seek(cs_start);
109 const QByteArray buf = d->pic_d->pictb.buffer();
110 quint16 cs = (quint16) qChecksum(QByteArrayView(buf.constData() + data_start, pos - data_start));
112 d->pic_d->pictb.close();
114 return !d->sizeLimitExceeded;
117#define SERIALIZE_CMD(c)
121 pos = d->pic_d->pictb.pos()
125 Q_D(QPicturePaintEngine);
126#ifdef QT_PICTURE_DEBUG
127 qDebug() <<
" -> updatePen(): width:" << pen.width() <<
"style:"
128 << pen.style() <<
"color:" << pen.color();
132 if (d->pic_d->in_memory_only) {
133 int index = d->pic_d->pen_list.size();
134 d->pic_d->pen_list.append(pen);
139 writeCmdLength(pos, QRect(),
false);
144 Q_D(QPicturePaintEngine);
145#ifdef QT_PICTURE_DEBUG
146 qDebug() <<
" -> updateCompositionMode():" << cmode;
150 d->s << (qint32)cmode;
151 writeCmdLength(pos, QRectF(),
false);
156 Q_D(QPicturePaintEngine);
157#ifdef QT_PICTURE_DEBUG
158 qDebug() <<
" -> updateClipEnabled():" << enabled;
163 writeCmdLength(pos, QRectF(),
false);
168 Q_D(QPicturePaintEngine);
169#ifdef QT_PICTURE_DEBUG
170 qDebug() <<
" -> updateOpacity():" << opacity;
174 d->s <<
double(opacity);
175 writeCmdLength(pos, QRectF(),
false);
180 Q_D(QPicturePaintEngine);
181#ifdef QT_PICTURE_DEBUG
182 qDebug() <<
" -> updateBrush(): style:" << brush.style();
186 if (d->pic_d->in_memory_only) {
187 int index = d->pic_d->brush_list.size();
188 d->pic_d->brush_list.append(brush);
193 writeCmdLength(pos, QRect(),
false);
198 Q_D(QPicturePaintEngine);
199#ifdef QT_PICTURE_DEBUG
200 qDebug() <<
" -> updateBrushOrigin(): " << p;
205 writeCmdLength(pos, QRect(),
false);
210 Q_D(QPicturePaintEngine);
211#ifdef QT_PICTURE_DEBUG
212 qDebug() <<
" -> updateFont(): pt sz:" << font.pointSize();
218 writeCmdLength(pos, QRectF(),
false);
223 Q_D(QPicturePaintEngine);
224#ifdef QT_PICTURE_DEBUG
225 qDebug() <<
" -> updateBackground(): mode:" << bgMode <<
"style:" << bgBrush.style();
229 d->s << bgBrush.color();
230 writeCmdLength(pos, QRect(),
false);
233 d->s << (qint8) bgMode;
234 writeCmdLength(pos, QRectF(),
false);
239 Q_D(QPicturePaintEngine);
240#ifdef QT_PICTURE_DEBUG
241 qDebug() <<
" -> updateMatrix():" << matrix;
245 d->s << matrix << (qint8)
false;
246 writeCmdLength(pos, QRectF(),
false);
251 Q_D(QPicturePaintEngine);
252#ifdef QT_PICTURE_DEBUG
253 qDebug() <<
" -> updateClipRegion(): op:" << op
254 <<
"bounding rect:" << region.boundingRect();
258 d->s << region << qint8(op);
259 writeCmdLength(pos, QRectF(),
false);
264 Q_D(QPicturePaintEngine);
265#ifdef QT_PICTURE_DEBUG
266 qDebug() <<
" -> updateClipPath(): op:" << op
267 <<
"bounding rect:" << path.boundingRect();
272 d->s << path << qint8(op);
273 writeCmdLength(pos, QRectF(),
false);
278 Q_D(QPicturePaintEngine);
279#ifdef QT_PICTURE_DEBUG
280 qDebug() <<
" -> updateRenderHints(): " << hints;
284 d->s << (quint32) hints;
285 writeCmdLength(pos, QRect(),
false);
290 Q_D(QPicturePaintEngine);
292 constexpr int sizeLimit =
std::numeric_limits<
int>::max() - 8;
293 if (d->sizeLimitExceeded || d->pic_d->pictb.pos() > sizeLimit) {
295 d->pic_d->pictb.seek(pos - 2);
296 d->pic_d->pictbData.resize(pos - 2);
297 if (!d->sizeLimitExceeded) {
298 d->sizeLimitExceeded =
true;
299 qWarning(
"QPicture: size limit exceeded, will be truncated");
304 int newpos = d->pic_d->pictb.pos();
305 int length = newpos - pos;
309 d->pic_d->pictb.seek(pos - 1);
310 d->s << (quint8)length;
313 d->pic_d->pictb.seek(pos - 1);
315 char *p = d->pic_d->pictb.buffer().data();
316 memmove(p+pos+4, p+pos, length);
317 d->s << (quint32)length;
320 d->pic_d->pictb.seek(newpos);
322 if (br.width() > 0.0 || br.height() > 0.0) {
324 int w2 = painter()->pen().width() / 2;
325 br.setCoords(br.left() - w2, br.top() - w2,
326 br.right() + w2, br.bottom() + w2);
328 br = painter()->transform().mapRect(br);
329 if (painter()->hasClipping()) {
330 QRectF cr = painter()->clipBoundingRect();
334 if (br.width() > 0.0 || br.height() > 0.0) {
335 const auto clampToIntRange = [](qreal v)
337 return qBound(qreal((std::numeric_limits<
int>::min)()),
339 qreal((std::numeric_limits<
int>::max)()));
341 int minx = qFloor(clampToIntRange(br.left()));
342 int miny = qFloor(clampToIntRange(br.top()));
343 int maxx = qCeil(clampToIntRange(br.right()));
344 int maxy = qCeil(clampToIntRange(br.bottom()));
346 if (d->pic_d->brect.width() > 0 || d->pic_d->brect.height() > 0) {
347 minx = qMin(minx, d->pic_d->brect.left());
348 miny = qMin(miny, d->pic_d->brect.top());
349 maxx = qMax(maxx, d->pic_d->brect.x() + d->pic_d->brect.width());
350 maxy = qMax(maxy, d->pic_d->brect.y() + d->pic_d->brect.height());
351 d->pic_d->brect.setCoords(minx, miny, maxx - 1, maxy - 1);
353 d->pic_d->brect.setCoords(minx, miny, maxx - 1, maxy - 1);
361 Q_D(QPicturePaintEngine);
362#ifdef QT_PICTURE_DEBUG
363 qDebug() <<
" -> drawEllipse():" << rect;
368 writeCmdLength(pos, rect,
true);
373 Q_D(QPicturePaintEngine);
374#ifdef QT_PICTURE_DEBUG
375 qDebug() <<
" -> drawPath():" << path.boundingRect();
380 writeCmdLength(pos, path.boundingRect(),
true);
385 Q_D(QPicturePaintEngine);
386#ifdef QT_PICTURE_DEBUG
387 qDebug() <<
" -> drawPolygon(): size=" << numPoints;
392 polygon.reserve(numPoints);
393 for (
int i=0; i<numPoints; ++i)
394 polygon << points[i];
396 if (mode == PolylineMode) {
402 d->s << (qint8)(mode == OddEvenMode ? 0 : 1);
405 writeCmdLength(pos, polygon.boundingRect(),
true);
410 Q_D(QPicturePaintEngine);
411#ifdef QT_PICTURE_DEBUG
412 qDebug() <<
" -> drawPixmap():" << r;
417 if (d->pic_d->in_memory_only) {
418 int index = d->pic_d->pixmap_list.size();
419 d->pic_d->pixmap_list.append(pm);
420 d->s << r << index << sr;
422 d->s << r << pm << sr;
424 writeCmdLength(pos, r,
false);
429 Q_D(QPicturePaintEngine);
430#ifdef QT_PICTURE_DEBUG
431 qDebug() <<
" -> drawTiledPixmap():" << r << s;
435 if (d->pic_d->in_memory_only) {
436 int index = d->pic_d->pixmap_list.size();
437 d->pic_d->pixmap_list.append(pixmap);
438 d->s << r << index << s;
440 d->s << r << pixmap << s;
442 writeCmdLength(pos, r,
false);
446 Qt::ImageConversionFlags flags)
448 Q_D(QPicturePaintEngine);
449#ifdef QT_PICTURE_DEBUG
450 qDebug() <<
" -> drawImage():" << r << sr;
454 if (d->pic_d->in_memory_only) {
455 int index = d->pic_d->image_list.size();
456 d->pic_d->image_list.append(image);
457 d->s << r << index << sr << (quint32) flags;
459 d->s << r << image << sr << (quint32) flags;
461 writeCmdLength(pos, r,
false);
466 Q_D(QPicturePaintEngine);
467#ifdef QT_PICTURE_DEBUG
468 qDebug() <<
" -> drawTextItem():" << p << ti.text();
471 const QTextItemInt &si =
static_cast<
const QTextItemInt &>(ti);
472 if (si.chars ==
nullptr)
473 QPaintEngine::drawTextItem(p, ti);
475 if (d->pic_d->formatMajor >= 9) {
478 QFont fnt = ti.font();
479 fnt.setUnderline(
false);
480 fnt.setStrikeOut(
false);
481 fnt.setOverline(
false);
483 qreal justificationWidth = 0;
485 justificationWidth = si.width.toReal();
487 d->s << p << ti.text() << fnt << ti.renderFlags() <<
double(fnt.d->dpi)/qt_defaultDpi() << justificationWidth;
488 writeCmdLength(pos, QRectF(),
false);
489 }
else if (d->pic_d->formatMajor >= 8) {
493 d->s << QPointF(p.x(), p.y() - ti.ascent()) << ti.text() << ti.font() << ti.renderFlags();
494 writeCmdLength(pos, QRectF(),
false);
499 d->s << p << ti.text();
500 writeCmdLength(pos, QRectF(p, QSizeF(1,1)),
true);
506 QPaintEngine::DirtyFlags flags = state.state();
507 if (flags & DirtyPen) updatePen(state.pen());
508 if (flags & DirtyBrush) updateBrush(state.brush());
509 if (flags & DirtyBrushOrigin) updateBrushOrigin(state.brushOrigin());
510 if (flags & DirtyFont) updateFont(state.font());
511 if (flags & DirtyBackground) updateBackground(state.backgroundMode(), state.backgroundBrush());
512 if (flags & DirtyTransform) updateMatrix(state.transform());
514 if (flags & DirtyClipRegion) updateClipRegion(state.clipRegion(), state.clipOperation());
515 if (flags & DirtyClipPath) updateClipPath(state.clipPath(), state.clipOperation());
516 if (flags & DirtyHints) updateRenderHints(state.renderHints());
517 if (flags & DirtyCompositionMode) updateCompositionMode(state.compositionMode());
518 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.