painting/qpainter.cpp

Switch to Source codePreprocessed file
LineSource CodeCoverage
1 -
2 -
3 -
4 -
5 -
6 -
7extern QPixmap qt_pixmapForBrush(int style, bool invert); -
8 -
9void qt_format_text(const QFont &font, -
10 const QRectF &_r, int tf, const QTextOption *option, const QString& str, QRectF *brect, -
11 int tabstops, int* tabarray, int tabarraylen, -
12 QPainter *painter); -
13static void drawTextItemDecoration(QPainter *painter, const QPointF &pos, const QFontEngine *fe, QTextEngine *textEngine, -
14 QTextCharFormat::UnderlineStyle underlineStyle, -
15 QTextItem::RenderFlags flags, qreal width, -
16 const QTextCharFormat &charFormat); -
17 -
18__attribute__((visibility("default"))) void qt_draw_decoration_for_glyphs(QPainter *painter, const glyph_t *glyphArray, -
19 const QFixedPoint *positions, int glyphCount, -
20 QFontEngine *fontEngine, const QFont &font, -
21 const QTextCharFormat &charFormat); -
22 -
23static inline QGradient::CoordinateMode coordinateMode(const QBrush &brush) -
24{ -
25 switch (brush.style()) { -
26 case Qt::LinearGradientPattern: -
27 case Qt::RadialGradientPattern: -
28 case Qt::ConicalGradientPattern: -
29 return brush.gradient()->coordinateMode();
-
30 default: -
31 ; -
32 }
-
33 return QGradient::LogicalMode;
-
34} -
35 -
36 -
37static inline bool check_gradient(const QBrush &brush) -
38{ -
39 return coordinateMode(brush) == QGradient::StretchToDeviceMode;
-
40} -
41 -
42extern bool qHasPixmapTexture(const QBrush &); -
43 -
44static inline bool is_brush_transparent(const QBrush &brush) { -
45 Qt::BrushStyle s = brush.style(); -
46 bool brushBitmap = qHasPixmapTexture(brush)
-
47 ? brush.texture().isQBitmap() -
48 : (brush.textureImage().depth() == 1); -
49 return ((s >= Qt::Dense1Pattern && s <= Qt::DiagCrossPattern) -
50 || (s == Qt::TexturePattern && brushBitmap));
-
51} -
52 -
53static inline bool is_pen_transparent(const QPen &pen) { -
54 return pen.style() > Qt::SolidLine || is_brush_transparent(pen.brush());
-
55} -
56 -
57 -
58 -
59 -
60static inline uint line_emulation(uint emulation) -
61{ -
62 return emulation & (QPaintEngine::PrimitiveTransform -
63 | QPaintEngine::AlphaBlend -
64 | QPaintEngine::Antialiasing -
65 | QPaintEngine::BrushStroke -
66 | QPaintEngine::ConstantOpacity -
67 | 0x10000000 -
68 | QPaintEngine::ObjectBoundingModeGradients -
69 | 0x40000000);
-
70} -
71void QPainterPrivate::checkEmulation() -
72{ -
73 qt_noop(); -
74 if (extended->flags() & QPaintEngineEx::DoNotEmulate)
-
75 return;
-
76 -
77 bool doEmulation = false; -
78 if (state->bgMode == Qt::OpaqueMode)
-
79 doEmulation = true;
-
80 -
81 const QGradient *bg = state->brush.gradient(); -
82 if (bg && bg->coordinateMode() > QGradient::LogicalMode)
-
83 doEmulation = true;
-
84 -
85 const QGradient *pg = qpen_brush(state->pen).gradient(); -
86 if (pg && pg->coordinateMode() > QGradient::LogicalMode)
-
87 doEmulation = true;
-
88 -
89 if (doEmulation) {
-
90 if (extended != emulationEngine) {
-
91 if (!emulationEngine)
-
92 emulationEngine = new QEmulationPaintEngine(extended);
-
93 extended = emulationEngine; -
94 extended->setState(state); -
95 }
-
96 } else if (emulationEngine == extended) {
-
97 extended = emulationEngine->real_engine; -
98 }
-
99} -
100 -
101 -
102QPainterPrivate::~QPainterPrivate() -
103{ -
104 delete emulationEngine; -
105 for (int i=0; i<states.size(); ++i)
-
106 delete states.at(i);
-
107 -
108 if (dummyState)
-
109 delete dummyState;
-
110}
-
111 -
112 -
113QTransform QPainterPrivate::viewTransform() const -
114{ -
115 if (state->VxF) {
-
116 qreal scaleW = qreal(state->vw)/qreal(state->ww); -
117 qreal scaleH = qreal(state->vh)/qreal(state->wh); -
118 return QTransform(scaleW, 0, 0, scaleH, -
119 state->vx - state->wx*scaleW, state->vy - state->wy*scaleH);
-
120 } -
121 return QTransform();
-
122} -
123 -
124QTransform QPainterPrivate::hidpiScaleTransform() const -
125{ -
126 return QTransform();
executed: return QTransform();
Execution Count:57707
57707
127} -
128 -
129 -
130 -
131 -
132 -
133bool QPainterPrivate::attachPainterPrivate(QPainter *q, QPaintDevice *pdev) -
134{ -
135 qt_noop(); -
136 qt_noop(); -
137 -
138 QPainter *sp = pdev->sharedPainter(); -
139 if (!sp)
-
140 return false;
-
141 -
142 -
143 -
144 sp->save(); -
145 if (!sp->d_ptr->d_ptrs) {
-
146 -
147 -
148 -
149 sp->d_ptr->d_ptrs_size = 4; -
150 sp->d_ptr->d_ptrs = (QPainterPrivate **)malloc(4 * sizeof(QPainterPrivate *)); -
151 qt_noop(); -
152 } else if (sp->d_ptr->refcount - 1 == sp->d_ptr->d_ptrs_size) {
-
153 -
154 sp->d_ptr->d_ptrs_size <<= 1; -
155 const int newSize = sp->d_ptr->d_ptrs_size * sizeof(QPainterPrivate *); -
156 sp->d_ptr->d_ptrs = q_check_ptr((QPainterPrivate **)realloc(sp->d_ptr->d_ptrs, newSize)); -
157 }
-
158 sp->d_ptr->d_ptrs[++sp->d_ptr->refcount - 2] = q->d_ptr.data(); -
159 q->d_ptr.take(); -
160 q->d_ptr.reset(sp->d_ptr.data()); -
161 -
162 qt_noop(); -
163 -
164 -
165 q->initFrom(pdev); -
166 QPoint offset; -
167 pdev->redirected(&offset); -
168 offset += q->d_ptr->engine->coordinateOffset(); -
169 -
170 -
171 q->d_ptr->state->ww = q->d_ptr->state->vw = pdev->width(); -
172 q->d_ptr->state->wh = q->d_ptr->state->vh = pdev->height(); -
173 -
174 -
175 if (q->d_ptr->state->WxF) {
-
176 q->d_ptr->state->redirectionMatrix = q->d_ptr->state->matrix; -
177 q->d_ptr->state->redirectionMatrix.translate(-offset.x(), -offset.y()); -
178 q->d_ptr->state->worldMatrix = QTransform(); -
179 q->d_ptr->state->WxF = false; -
180 } else {
-
181 q->d_ptr->state->redirectionMatrix = QTransform::fromTranslate(-offset.x(), -offset.y()); -
182 }
-
183 q->d_ptr->updateMatrix(); -
184 -
185 QPaintEnginePrivate *enginePrivate = q->d_ptr->engine->d_func(); -
186 if (enginePrivate->currentClipDevice == pdev) {
-
187 enginePrivate->systemStateChanged(); -
188 return true;
-
189 } -
190 -
191 -
192 enginePrivate->currentClipDevice = pdev; -
193 enginePrivate->setSystemTransform(q->d_ptr->state->matrix); -
194 return true;
-
195} -
196 -
197void QPainterPrivate::detachPainterPrivate(QPainter *q) -
198{ -
199 qt_noop(); -
200 qt_noop(); -
201 -
202 QPainterPrivate *original = d_ptrs[--refcount - 1]; -
203 if (inDestructor) {
-
204 inDestructor = false; -
205 if (original)
-
206 original->inDestructor = true;
-
207 } else if (!original) {
-
208 original = new QPainterPrivate(q); -
209 }
-
210 -
211 d_ptrs[refcount - 1] = 0; -
212 q->restore(); -
213 q->d_ptr.take(); -
214 q->d_ptr.reset(original); -
215 -
216 if (emulationEngine) {
-
217 extended = emulationEngine->real_engine; -
218 delete emulationEngine; -
219 emulationEngine = 0; -
220 }
-
221}
-
222 -
223 -
224void QPainterPrivate::draw_helper(const QPainterPath &originalPath, DrawOperation op) -
225{ -
226 -
227 -
228 -
229 -
230 -
231 -
232 if (originalPath.isEmpty())
-
233 return;
-
234 -
235 QPaintEngine::PaintEngineFeatures gradientStretch = -
236 QPaintEngine::PaintEngineFeatures(0x10000000 -
237 | QPaintEngine::ObjectBoundingModeGradients); -
238 -
239 const bool mustEmulateObjectBoundingModeGradients = extended
-
240 || ((state->emulationSpecifier & QPaintEngine::ObjectBoundingModeGradients)
-
241 && !engine->hasFeature(QPaintEngine::PatternTransform));
-
242 -
243 if (!(state->emulationSpecifier & ~gradientStretch)
-
244 && !mustEmulateObjectBoundingModeGradients) {
-
245 drawStretchedGradient(originalPath, op); -
246 return;
-
247 } else if (state->emulationSpecifier & 0x40000000) {
-
248 drawOpaqueBackground(originalPath, op); -
249 return;
-
250 } -
251 -
252 QPainter * const q = q_func(); -
253 -
254 qreal strokeOffsetX = 0, strokeOffsetY = 0; -
255 -
256 QPainterPath path = originalPath * state->matrix; -
257 QRectF pathBounds = path.boundingRect(); -
258 QRectF strokeBounds; -
259 bool doStroke = (op & StrokeDraw) && (state->pen.style() != Qt::NoPen);
-
260 if (doStroke) {
-
261 qreal penWidth = state->pen.widthF(); -
262 if (penWidth == 0) {
-
263 strokeOffsetX = 1; -
264 strokeOffsetY = 1; -
265 } else {
-
266 -
267 if (state->matrix.type() > QTransform::TxScale) {
-
268 QPainterPathStroker stroker; -
269 stroker.setWidth(penWidth); -
270 stroker.setJoinStyle(state->pen.joinStyle()); -
271 stroker.setCapStyle(state->pen.capStyle()); -
272 QPainterPath stroke = stroker.createStroke(originalPath); -
273 strokeBounds = (stroke * state->matrix).boundingRect(); -
274 } else {
-
275 strokeOffsetX = qAbs(penWidth * state->matrix.m11() / 2.0); -
276 strokeOffsetY = qAbs(penWidth * state->matrix.m22() / 2.0); -
277 }
-
278 } -
279 } -
280 -
281 QRect absPathRect; -
282 if (!strokeBounds.isEmpty()) {
-
283 absPathRect = strokeBounds.intersected(QRectF(0, 0, device->width(), device->height())).toAlignedRect(); -
284 } else {
-
285 absPathRect = pathBounds.adjusted(-strokeOffsetX, -strokeOffsetY, strokeOffsetX, strokeOffsetY) -
286 .intersected(QRectF(0, 0, device->width(), device->height())).toAlignedRect(); -
287 }
-
288 -
289 if (q->hasClipping()) {
-
290 bool hasPerspectiveTransform = false; -
291 for (int i = 0; i < state->clipInfo.size(); ++i) {
-
292 const QPainterClipInfo &info = state->clipInfo.at(i); -
293 if (info.matrix.type() == QTransform::TxProject) {
-
294 hasPerspectiveTransform = true; -
295 break;
-
296 } -
297 }
-
298 -
299 if (!hasPerspectiveTransform) {
-
300 -
301 -
302 -
303 -
304 -
305 -
306 bool old_txinv = txinv; -
307 QTransform old_invMatrix = invMatrix; -
308 txinv = true; -
309 invMatrix = QTransform(); -
310 QPainterPath clipPath = q->clipPath(); -
311 QRectF r = clipPath.boundingRect().intersected(absPathRect); -
312 absPathRect = r.toAlignedRect(); -
313 txinv = old_txinv; -
314 invMatrix = old_invMatrix; -
315 }
-
316 }
-
317 -
318 -
319 -
320 -
321 -
322 -
323 -
324 if (absPathRect.width() <= 0 || absPathRect.height() <= 0)
-
325 return;
-
326 -
327 QImage image(absPathRect.width(), absPathRect.height(), QImage::Format_ARGB32_Premultiplied); -
328 image.fill(0); -
329 -
330 QPainter p(&image); -
331 -
332 p.d_ptr->helper_device = helper_device; -
333 -
334 p.setOpacity(state->opacity); -
335 p.translate(-absPathRect.x(), -absPathRect.y()); -
336 p.setTransform(state->matrix, true); -
337 p.setPen(doStroke ? state->pen : QPen(Qt::NoPen)); -
338 p.setBrush((op & FillDraw) ? state->brush : QBrush(Qt::NoBrush)); -
339 p.setBackground(state->bgBrush); -
340 p.setBackgroundMode(state->bgMode); -
341 p.setBrushOrigin(state->brushOrigin); -
342 -
343 p.setRenderHint(QPainter::Antialiasing, state->renderHints & QPainter::Antialiasing); -
344 p.setRenderHint(QPainter::SmoothPixmapTransform, -
345 state->renderHints & QPainter::SmoothPixmapTransform); -
346 -
347 p.drawPath(originalPath); -
348 p.end(); -
349 -
350 q->save(); -
351 state->matrix = QTransform(); -
352 if (extended) {
-
353 extended->transformChanged(); -
354 } else {
-
355 state->dirtyFlags |= QPaintEngine::DirtyTransform; -
356 updateState(state); -
357 }
-
358 engine->drawImage(absPathRect, -
359 image, -
360 QRectF(0, 0, absPathRect.width(), absPathRect.height()), -
361 Qt::OrderedDither | Qt::OrderedAlphaDither); -
362 q->restore(); -
363}
-
364 -
365void QPainterPrivate::drawOpaqueBackground(const QPainterPath &path, DrawOperation op) -
366{ -
367 QPainter * const q = q_func(); -
368 -
369 q->setBackgroundMode(Qt::TransparentMode); -
370 -
371 if (op & FillDraw && state->brush.style() != Qt::NoBrush) {
-
372 q->fillPath(path, state->bgBrush.color()); -
373 q->fillPath(path, state->brush); -
374 }
-
375 -
376 if (op & StrokeDraw && state->pen.style() != Qt::NoPen) {
-
377 q->strokePath(path, QPen(state->bgBrush.color(), state->pen.width())); -
378 q->strokePath(path, state->pen); -
379 }
-
380 -
381 q->setBackgroundMode(Qt::OpaqueMode); -
382}
-
383 -
384static inline QBrush stretchGradientToUserSpace(const QBrush &brush, const QRectF &boundingRect) -
385{ -
386 qt_noop(); -
387 -
388 -
389 QTransform gradientToUser(boundingRect.width(), 0, 0, boundingRect.height(), -
390 boundingRect.x(), boundingRect.y()); -
391 -
392 QGradient g = *brush.gradient(); -
393 g.setCoordinateMode(QGradient::LogicalMode); -
394 -
395 QBrush b(g); -
396 b.setTransform(gradientToUser * b.transform()); -
397 return b;
-
398} -
399 -
400void QPainterPrivate::drawStretchedGradient(const QPainterPath &path, DrawOperation op) -
401{ -
402 QPainter * const q = q_func(); -
403 -
404 const qreal sw = helper_device->width(); -
405 const qreal sh = helper_device->height(); -
406 -
407 bool changedPen = false; -
408 bool changedBrush = false; -
409 bool needsFill = false; -
410 -
411 const QPen pen = state->pen; -
412 const QBrush brush = state->brush; -
413 -
414 const QGradient::CoordinateMode penMode = coordinateMode(pen.brush()); -
415 const QGradient::CoordinateMode brushMode = coordinateMode(brush); -
416 -
417 QRectF boundingRect; -
418 -
419 -
420 if ((op & FillDraw) && brush.style() != Qt::NoBrush) {
-
421 if (brushMode == QGradient::StretchToDeviceMode) {
-
422 q->setPen(Qt::NoPen); -
423 changedPen = pen.style() != Qt::NoPen; -
424 q->scale(sw, sh); -
425 updateState(state); -
426 -
427 const qreal isw = 1.0 / sw; -
428 const qreal ish = 1.0 / sh; -
429 QTransform inv(isw, 0, 0, ish, 0, 0); -
430 engine->drawPath(path * inv); -
431 q->scale(isw, ish); -
432 } else {
-
433 needsFill = true; -
434 -
435 if (brushMode == QGradient::ObjectBoundingMode) {
-
436 qt_noop(); -
437 boundingRect = path.boundingRect(); -
438 q->setBrush(stretchGradientToUserSpace(brush, boundingRect)); -
439 changedBrush = true; -
440 }
-
441 }
-
442 } -
443 -
444 if ((op & StrokeDraw) && pen.style() != Qt::NoPen) {
-
445 -
446 if (penMode == QGradient::StretchToDeviceMode) {
-
447 q->setPen(Qt::NoPen); -
448 changedPen = true; -
449 -
450 if (needsFill) {
-
451 updateState(state); -
452 engine->drawPath(path); -
453 }
-
454 -
455 q->scale(sw, sh); -
456 q->setBrush(pen.brush()); -
457 changedBrush = true; -
458 updateState(state); -
459 -
460 QPainterPathStroker stroker; -
461 stroker.setDashPattern(pen.style()); -
462 stroker.setWidth(pen.widthF()); -
463 stroker.setJoinStyle(pen.joinStyle()); -
464 stroker.setCapStyle(pen.capStyle()); -
465 stroker.setMiterLimit(pen.miterLimit()); -
466 QPainterPath stroke = stroker.createStroke(path); -
467 -
468 const qreal isw = 1.0 / sw; -
469 const qreal ish = 1.0 / sh; -
470 QTransform inv(isw, 0, 0, ish, 0, 0); -
471 engine->drawPath(stroke * inv); -
472 q->scale(isw, ish); -
473 } else {
-
474 if (!needsFill && brush.style() != Qt::NoBrush) {
-
475 q->setBrush(Qt::NoBrush); -
476 changedBrush = true; -
477 }
-
478 -
479 if (penMode == QGradient::ObjectBoundingMode) {
-
480 qt_noop(); -
481 -
482 -
483 if (!needsFill || brushMode != QGradient::ObjectBoundingMode)
-
484 boundingRect = path.boundingRect();
-
485 -
486 QPen p = pen; -
487 p.setBrush(stretchGradientToUserSpace(pen.brush(), boundingRect)); -
488 q->setPen(p); -
489 changedPen = true; -
490 } else if (changedPen) {
-
491 q->setPen(pen); -
492 changedPen = false; -
493 }
-
494 -
495 updateState(state); -
496 engine->drawPath(path); -
497 }
-
498 } else if (needsFill) {
-
499 if (pen.style() != Qt::NoPen) {
-
500 q->setPen(Qt::NoPen); -
501 changedPen = true; -
502 }
-
503 -
504 updateState(state); -
505 engine->drawPath(path); -
506 }
-
507 -
508 if (changedPen)
-
509 q->setPen(pen);
-
510 if (changedBrush)
-
511 q->setBrush(brush);
-
512}
-
513 -
514 -
515void QPainterPrivate::updateMatrix() -
516{ -
517 state->matrix = state->WxF ? state->worldMatrix : QTransform();
-
518 if (state->VxF)
-
519 state->matrix *= viewTransform();
-
520 -
521 txinv = false; -
522 state->matrix *= state->redirectionMatrix; -
523 if (extended)
-
524 extended->transformChanged();
-
525 else -
526 state->dirtyFlags |= QPaintEngine::DirtyTransform;
-
527 -
528 state->matrix *= hidpiScaleTransform(); -
529 -
530 -
531 -
532}
-
533 -
534 -
535void QPainterPrivate::updateInvMatrix() -
536{ -
537 qt_noop(); -
538 txinv = true; -
539 invMatrix = state->matrix.inverted(); -
540}
-
541 -
542extern bool qt_isExtendedRadialGradient(const QBrush &brush); -
543 -
544void QPainterPrivate::updateEmulationSpecifier(QPainterState *s) -
545{ -
546 bool alpha = false; -
547 bool linearGradient = false; -
548 bool radialGradient = false; -
549 bool extendedRadialGradient = false; -
550 bool conicalGradient = false; -
551 bool patternBrush = false; -
552 bool xform = false; -
553 bool complexXform = false; -
554 -
555 bool skip = true; -
556 -
557 -
558 -
559 if (s->state() & (QPaintEngine::DirtyPen | QPaintEngine::DirtyBrush | QPaintEngine::DirtyHints)) {
-
560 -
561 if (!s->pen.isSolid() && !engine->hasFeature(QPaintEngine::BrushStroke))
-
562 s->emulationSpecifier |= QPaintEngine::BrushStroke;
-
563 else -
564 s->emulationSpecifier &= ~QPaintEngine::BrushStroke;
-
565 -
566 skip = false; -
567 -
568 QBrush penBrush = (qpen_style(s->pen) == Qt::NoPen) ? QBrush(Qt::NoBrush) : qpen_brush(s->pen);
-
569 Qt::BrushStyle brushStyle = qbrush_style(s->brush); -
570 Qt::BrushStyle penBrushStyle = qbrush_style(penBrush); -
571 alpha = (penBrushStyle != Qt::NoBrush
-
572 && (penBrushStyle < Qt::LinearGradientPattern && penBrush.color().alpha() != 255)
-
573 && !penBrush.isOpaque())
-
574 || (brushStyle != Qt::NoBrush
-
575 && (brushStyle < Qt::LinearGradientPattern && s->brush.color().alpha() != 255)
-
576 && !s->brush.isOpaque());
-
577 linearGradient = ((penBrushStyle == Qt::LinearGradientPattern) ||
-
578 (brushStyle == Qt::LinearGradientPattern));
-
579 radialGradient = ((penBrushStyle == Qt::RadialGradientPattern) ||
-
580 (brushStyle == Qt::RadialGradientPattern));
-
581 extendedRadialGradient = radialGradient && (qt_isExtendedRadialGradient(penBrush) || qt_isExtendedRadialGradient(s->brush));
-
582 conicalGradient = ((penBrushStyle == Qt::ConicalGradientPattern) ||
-
583 (brushStyle == Qt::ConicalGradientPattern));
-
584 patternBrush = (((penBrushStyle > Qt::SolidPattern
-
585 && penBrushStyle < Qt::LinearGradientPattern)
-
586 || penBrushStyle == Qt::TexturePattern) ||
-
587 ((brushStyle > Qt::SolidPattern
-
588 && brushStyle < Qt::LinearGradientPattern)
-
589 || brushStyle == Qt::TexturePattern));
-
590 -
591 bool penTextureAlpha = false; -
592 if (penBrush.style() == Qt::TexturePattern)
-
593 penTextureAlpha = qHasPixmapTexture(penBrush)
-
594 ? (penBrush.texture().depth() > 1) && penBrush.texture().hasAlpha() -
595 : penBrush.textureImage().hasAlphaChannel();
-
596 bool brushTextureAlpha = false; -
597 if (s->brush.style() == Qt::TexturePattern) {
-
598 brushTextureAlpha = qHasPixmapTexture(s->brush)
-
599 ? (s->brush.texture().depth() > 1) && s->brush.texture().hasAlpha() -
600 : s->brush.textureImage().hasAlphaChannel(); -
601 }
-
602 if (((penBrush.style() == Qt::TexturePattern && penTextureAlpha)
-
603 || (s->brush.style() == Qt::TexturePattern && brushTextureAlpha))
-
604 && !engine->hasFeature(QPaintEngine::MaskedBrush))
-
605 s->emulationSpecifier |= QPaintEngine::MaskedBrush;
-
606 else -
607 s->emulationSpecifier &= ~QPaintEngine::MaskedBrush;
-
608 } -
609 -
610 if (s->state() & (QPaintEngine::DirtyHints -
611 | QPaintEngine::DirtyOpacity -
612 | QPaintEngine::DirtyBackgroundMode)) {
-
613 skip = false; -
614 }
-
615 -
616 if (skip)
-
617 return;
-
618 if (s->state() & QPaintEngine::DirtyTransform) {
-
619 xform = !s->matrix.isIdentity(); -
620 complexXform = !s->matrix.isAffine(); -
621 } else if (s->matrix.type() >= QTransform::TxTranslate) {
-
622 xform = true; -
623 complexXform = !s->matrix.isAffine(); -
624 }
-
625 -
626 const bool brushXform = (!s->brush.transform().type() == QTransform::TxNone); -
627 const bool penXform = (!s->pen.brush().transform().type() == QTransform::TxNone); -
628 -
629 const bool patternXform = patternBrush && (xform || brushXform || penXform);
-
630 -
631 -
632 if (alpha && !engine->hasFeature(QPaintEngine::AlphaBlend))
-
633 s->emulationSpecifier |= QPaintEngine::AlphaBlend;
-
634 else -
635 s->emulationSpecifier &= ~QPaintEngine::AlphaBlend;
-
636 -
637 -
638 if (linearGradient && !engine->hasFeature(QPaintEngine::LinearGradientFill))
-
639 s->emulationSpecifier |= QPaintEngine::LinearGradientFill;
-
640 else -
641 s->emulationSpecifier &= ~QPaintEngine::LinearGradientFill;
-
642 -
643 -
644 if (extendedRadialGradient || (radialGradient && !engine->hasFeature(QPaintEngine::RadialGradientFill)))
-
645 s->emulationSpecifier |= QPaintEngine::RadialGradientFill;
-
646 else -
647 s->emulationSpecifier &= ~QPaintEngine::RadialGradientFill;
-
648 -
649 -
650 if (conicalGradient && !engine->hasFeature(QPaintEngine::ConicalGradientFill))
-
651 s->emulationSpecifier |= QPaintEngine::ConicalGradientFill;
-
652 else -
653 s->emulationSpecifier &= ~QPaintEngine::ConicalGradientFill;
-
654 -
655 -
656 if (patternBrush && !engine->hasFeature(QPaintEngine::PatternBrush))
-
657 s->emulationSpecifier |= QPaintEngine::PatternBrush;
-
658 else -
659 s->emulationSpecifier &= ~QPaintEngine::PatternBrush;
-
660 -
661 -
662 if (patternXform && !engine->hasFeature(QPaintEngine::PatternTransform))
-
663 s->emulationSpecifier |= QPaintEngine::PatternTransform;
-
664 else -
665 s->emulationSpecifier &= ~QPaintEngine::PatternTransform;
-
666 -
667 -
668 if (xform && !engine->hasFeature(QPaintEngine::PrimitiveTransform))
-
669 s->emulationSpecifier |= QPaintEngine::PrimitiveTransform;
-
670 else -
671 s->emulationSpecifier &= ~QPaintEngine::PrimitiveTransform;
-
672 -
673 -
674 if (complexXform && !engine->hasFeature(QPaintEngine::PerspectiveTransform))
-
675 s->emulationSpecifier |= QPaintEngine::PerspectiveTransform;
-
676 else -
677 s->emulationSpecifier &= ~QPaintEngine::PerspectiveTransform;
-
678 -
679 -
680 if (state->opacity != 1 && !engine->hasFeature(QPaintEngine::ConstantOpacity))
-
681 s->emulationSpecifier |= QPaintEngine::ConstantOpacity;
-
682 else -
683 s->emulationSpecifier &= ~QPaintEngine::ConstantOpacity;
-
684 -
685 bool gradientStretch = false; -
686 bool objectBoundingMode = false; -
687 if (linearGradient || conicalGradient || radialGradient) {
-
688 QGradient::CoordinateMode brushMode = coordinateMode(s->brush); -
689 QGradient::CoordinateMode penMode = coordinateMode(s->pen.brush()); -
690 -
691 gradientStretch |= (brushMode == QGradient::StretchToDeviceMode); -
692 gradientStretch |= (penMode == QGradient::StretchToDeviceMode); -
693 -
694 objectBoundingMode |= (brushMode == QGradient::ObjectBoundingMode); -
695 objectBoundingMode |= (penMode == QGradient::ObjectBoundingMode); -
696 }
-
697 if (gradientStretch)
-
698 s->emulationSpecifier |= 0x10000000;
-
699 else -
700 s->emulationSpecifier &= ~0x10000000;
-
701 -
702 if (objectBoundingMode && !engine->hasFeature(QPaintEngine::ObjectBoundingModeGradients))
-
703 s->emulationSpecifier |= QPaintEngine::ObjectBoundingModeGradients;
-
704 else -
705 s->emulationSpecifier &= ~QPaintEngine::ObjectBoundingModeGradients;
-
706 -
707 -
708 if (s->bgMode == Qt::OpaqueMode &&
-
709 (is_pen_transparent(s->pen) || is_brush_transparent(s->brush)))
-
710 s->emulationSpecifier |= 0x40000000;
-
711 else -
712 s->emulationSpecifier &= ~0x40000000;
-
713} -
714 -
715void QPainterPrivate::updateStateImpl(QPainterState *newState) -
716{ -
717 -
718 if (!engine->state) {
-
719 engine->state = newState; -
720 engine->setDirty(QPaintEngine::AllDirty); -
721 }
-
722 -
723 if (engine->state->painter() != newState->painter)
-
724 -
725 engine->setDirty(QPaintEngine::AllDirty);
-
726 -
727 -
728 else if (engine->state != newState)
-
729 newState->dirtyFlags |= QPaintEngine::DirtyFlags(static_cast<QPainterState *>(engine->state)->changeFlags);
-
730 -
731 -
732 else -
733 newState->changeFlags |= newState->dirtyFlags;
-
734 -
735 updateEmulationSpecifier(newState); -
736 -
737 -
738 newState->dirtyFlags &= ~(QPaintEngine::DirtyBackgroundMode -
739 | QPaintEngine::DirtyBackground); -
740 -
741 engine->state = newState; -
742 engine->updateState(*newState); -
743 engine->clearDirty(QPaintEngine::AllDirty); -
744 -
745}
-
746 -
747void QPainterPrivate::updateState(QPainterState *newState) -
748{ -
749 -
750 if (!newState) {
-
751 engine->state = newState; -
752 } else if (newState->state() || engine->state!=newState) {
-
753 updateStateImpl(newState); -
754 }
-
755} -
756QPainter::QPainter() -
757 : d_ptr(new QPainterPrivate(this)) -
758{ -
759}
-
760QPainter::QPainter(QPaintDevice *pd) -
761 : d_ptr(0) -
762{ -
763 qt_noop(); -
764 if (!QPainterPrivate::attachPainterPrivate(this, pd)) {
-
765 d_ptr.reset(new QPainterPrivate(this)); -
766 begin(pd); -
767 }
-
768 qt_noop(); -
769}
-
770 -
771 -
772 -
773 -
774QPainter::~QPainter() -
775{ -
776 d_ptr->inDestructor = true; -
777 if (true) {
-
778 if (isActive())
-
779 end();
-
780 else if (d_ptr->refcount > 1)
-
781 d_ptr->detachPainterPrivate(this);
-
782 } else { -
783 -
784 }
-
785 if (d_ptr) {
-
786 -
787 qt_noop(); -
788 d_ptr->inDestructor = false; -
789 qt_noop(); -
790 if (d_ptr->d_ptrs)
-
791 free(d_ptr->d_ptrs);
-
792 }
-
793}
-
794QPaintDevice *QPainter::device() const -
795{ -
796 const QPainterPrivate * const d = d_func(); -
797 if (isActive() && d->engine->d_func()->currentClipDevice)
-
798 return d->engine->d_func()->currentClipDevice;
-
799 return d->original_device;
-
800} -
801bool QPainter::isActive() const -
802{ -
803 const QPainterPrivate * const d = d_func(); -
804 return d->engine;
-
805} -
806void QPainter::initFrom(const QPaintDevice *device) -
807{ -
808 qt_noop(); -
809 QPainterPrivate * const d = d_func(); -
810 if (!d->engine) {
-
811 QMessageLogger("painting/qpainter.cpp", 1540, __PRETTY_FUNCTION__).warning("QPainter::initFrom: Painter not active, aborted"); -
812 return;
-
813 } -
814 -
815 device->initPainter(this); -
816 -
817 if (d->extended) {
-
818 d->extended->penChanged(); -
819 } else if (d->engine) {
-
820 d->engine->setDirty(QPaintEngine::DirtyPen); -
821 d->engine->setDirty(QPaintEngine::DirtyBrush); -
822 d->engine->setDirty(QPaintEngine::DirtyFont); -
823 }
-
824} -
825void QPainter::save() -
826{ -
827 -
828 -
829 -
830 -
831 QPainterPrivate * const d = d_func(); -
832 if (!d->engine) {
-
833 QMessageLogger("painting/qpainter.cpp", 1572, __PRETTY_FUNCTION__).warning("QPainter::save: Painter not active"); -
834 return;
-
835 } -
836 -
837 if (d->extended) {
-
838 d->state = d->extended->createState(d->states.back()); -
839 d->extended->setState(d->state); -
840 } else {
-
841 d->updateState(d->state); -
842 d->state = new QPainterState(d->states.back()); -
843 d->engine->state = d->state; -
844 }
-
845 d->states.push_back(d->state); -
846}
-
847void QPainter::restore() -
848{ -
849 -
850 -
851 -
852 -
853 QPainterPrivate * const d = d_func(); -
854 if (d->states.size()<=1) {
-
855 QMessageLogger("painting/qpainter.cpp", 1602, __PRETTY_FUNCTION__).warning("QPainter::restore: Unbalanced save/restore"); -
856 return;
-
857 } else if (!d->engine) {
-
858 QMessageLogger("painting/qpainter.cpp", 1605, __PRETTY_FUNCTION__).warning("QPainter::restore: Painter not active"); -
859 return;
-
860 } -
861 -
862 QPainterState *tmp = d->state; -
863 d->states.pop_back(); -
864 d->state = d->states.back(); -
865 d->txinv = false; -
866 -
867 if (d->extended) {
-
868 d->checkEmulation(); -
869 d->extended->setState(d->state); -
870 delete tmp; -
871 return;
-
872 } -
873 -
874 -
875 -
876 if (!d->state->clipInfo.isEmpty()
-
877 && (tmp->changeFlags & (QPaintEngine::DirtyClipRegion | QPaintEngine::DirtyClipPath))) {
-
878 -
879 tmp->dirtyFlags = QPaintEngine::DirtyClipPath; -
880 tmp->clipOperation = Qt::NoClip; -
881 tmp->clipPath = QPainterPath(); -
882 d->engine->updateState(*tmp); -
883 -
884 for (int i=0; i<d->state->clipInfo.size(); ++i) {
-
885 const QPainterClipInfo &info = d->state->clipInfo.at(i); -
886 tmp->matrix = info.matrix; -
887 tmp->matrix *= d->state->redirectionMatrix; -
888 tmp->clipOperation = info.operation; -
889 if (info.clipType == QPainterClipInfo::RectClip) {
-
890 tmp->dirtyFlags = QPaintEngine::DirtyClipRegion | QPaintEngine::DirtyTransform; -
891 tmp->clipRegion = info.rect; -
892 } else if (info.clipType == QPainterClipInfo::RegionClip) {
-
893 tmp->dirtyFlags = QPaintEngine::DirtyClipRegion | QPaintEngine::DirtyTransform; -
894 tmp->clipRegion = info.region; -
895 } else {
-
896 tmp->dirtyFlags = QPaintEngine::DirtyClipPath | QPaintEngine::DirtyTransform; -
897 tmp->clipPath = info.path; -
898 }
-
899 d->engine->updateState(*tmp); -
900 }
-
901 -
902 -
903 -
904 d->state->dirtyFlags &= ~(QPaintEngine::DirtyClipPath | QPaintEngine::DirtyClipRegion); -
905 tmp->changeFlags &= ~(QPaintEngine::DirtyClipPath | QPaintEngine::DirtyClipRegion); -
906 tmp->changeFlags |= QPaintEngine::DirtyTransform; -
907 }
-
908 -
909 d->updateState(d->state); -
910 delete tmp; -
911}
-
912static inline void qt_cleanup_painter_state(QPainterPrivate *d) -
913{ -
914 d->states.clear(); -
915 delete d->state; -
916 d->state = 0; -
917 d->engine = 0; -
918 d->device = 0; -
919}
-
920 -
921bool QPainter::begin(QPaintDevice *pd) -
922{ -
923 qt_noop(); -
924 -
925 if (pd->painters > 0) {
evaluated: pd->painters > 0
TRUEFALSE
yes
Evaluation Count:1
yes
Evaluation Count:28784
1-28784
926 QMessageLogger("painting/qpainter.cpp", 1702, __PRETTY_FUNCTION__).warning("QPainter::begin: A paint device can only be painted by one painter at a time."); -
927 return false;
executed: return false;
Execution Count:1
1
928 } -
929 -
930 if (d_ptr->engine) {
evaluated: d_ptr->engine
TRUEFALSE
yes
Evaluation Count:1
yes
Evaluation Count:28783
1-28783
931 QMessageLogger("painting/qpainter.cpp", 1707, __PRETTY_FUNCTION__).warning("QPainter::begin: Painter already active"); -
932 return false;
executed: return false;
Execution Count:1
1
933 } -
934 -
935 if (QPainterPrivate::attachPainterPrivate(this, pd))
evaluated: QPainterPrivate::attachPainterPrivate(this, pd)
TRUEFALSE
yes
Evaluation Count:6
yes
Evaluation Count:28777
6-28777
936 return true;
executed: return true;
Execution Count:6
6
937 -
938 QPainterPrivate * const d = d_func(); -
939 -
940 d->helper_device = pd; -
941 d->original_device = pd; -
942 -
943 QPoint redirectionOffset; -
944 QPaintDevice *rpd = pd->redirected(&redirectionOffset); -
945 if (rpd)
evaluated: rpd
TRUEFALSE
yes
Evaluation Count:23420
yes
Evaluation Count:5357
5357-23420
946 pd = rpd;
executed: pd = rpd;
Execution Count:23420
23420
947 -
948 -
949 -
950 -
951 -
952 -
953 if (pd->devType() == QInternal::Pixmap)
evaluated: pd->devType() == QInternal::Pixmap
TRUEFALSE
yes
Evaluation Count:834
yes
Evaluation Count:27943
834-27943
954 static_cast<QPixmap *>(pd)->detach();
executed: static_cast<QPixmap *>(pd)->detach();
Execution Count:834
834
955 else if (pd->devType() == QInternal::Image)
evaluated: pd->devType() == QInternal::Image
TRUEFALSE
yes
Evaluation Count:27905
yes
Evaluation Count:38
38-27905
956 static_cast<QImage *>(pd)->detach();
executed: static_cast<QImage *>(pd)->detach();
Execution Count:27905
27905
957 -
958 d->engine = pd->paintEngine(); -
959 -
960 if (!d->engine) {
evaluated: !d->engine
TRUEFALSE
yes
Evaluation Count:2
yes
Evaluation Count:28774
2-28774
961 QMessageLogger("painting/qpainter.cpp", 1737, __PRETTY_FUNCTION__).warning("QPainter::begin: Paint device returned engine == 0, type: %d", pd->devType()); -
962 return false;
executed: return false;
Execution Count:2
2
963 } -
964 -
965 d->device = pd; -
966 -
967 d->extended = d->engine->isExtended() ? static_cast<QPaintEngineEx *>(d->engine) : 0;
evaluated: d->engine->isExtended()
TRUEFALSE
yes
Evaluation Count:28736
yes
Evaluation Count:38
38-28736
968 if (d->emulationEngine)
partially evaluated: d->emulationEngine
TRUEFALSE
no
Evaluation Count:0
yes
Evaluation Count:28774
0-28774
969 d->emulationEngine->real_engine = d->extended;
never executed: d->emulationEngine->real_engine = d->extended;
0
970 -
971 -
972 qt_noop(); -
973 d->state = d->extended ? d->extended->createState(0) : new QPainterState;
evaluated: d->extended
TRUEFALSE
yes
Evaluation Count:28736
yes
Evaluation Count:38
38-28736
974 d->state->painter = this; -
975 d->states.push_back(d->state); -
976 -
977 d->state->redirectionMatrix.translate(-redirectionOffset.x(), -redirectionOffset.y()); -
978 d->state->brushOrigin = QPointF(); -
979 -
980 -
981 if (d->extended)
evaluated: d->extended
TRUEFALSE
yes
Evaluation Count:28737
yes
Evaluation Count:38
38-28737
982 d->extended->setState(d->state);
executed: d->extended->setState(d->state);
Execution Count:28737
28737
983 else -
984 d->engine->state = d->state;
executed: d->engine->state = d->state;
Execution Count:38
38
985 -
986 switch (pd->devType()) { -
987 case QInternal::Pixmap: -
988 { -
989 QPixmap *pm = static_cast<QPixmap *>(pd); -
990 qt_noop(); -
991 if (pm->isNull()) {
partially evaluated: pm->isNull()
TRUEFALSE
no
Evaluation Count:0
yes
Evaluation Count:833
0-833
992 QMessageLogger("painting/qpainter.cpp", 1768, __PRETTY_FUNCTION__).warning("QPainter::begin: Cannot paint on a null pixmap"); -
993 qt_cleanup_painter_state(d); -
994 return false;
never executed: return false;
0
995 } -
996 -
997 if (pm->depth() == 1) {
evaluated: pm->depth() == 1
TRUEFALSE
yes
Evaluation Count:5
yes
Evaluation Count:828
5-828
998 d->state->pen = QPen(Qt::color1); -
999 d->state->brush = QBrush(Qt::color0); -
1000 }
executed: }
Execution Count:5
5
1001 break;
executed: break;
Execution Count:833
833
1002 } -
1003 case QInternal::Image: -
1004 { -
1005 QImage *img = static_cast<QImage *>(pd); -
1006 qt_noop(); -
1007 if (img->isNull()) {
partially evaluated: img->isNull()
TRUEFALSE
no
Evaluation Count:0
yes
Evaluation Count:27904
0-27904
1008 QMessageLogger("painting/qpainter.cpp", 1784, __PRETTY_FUNCTION__).warning("QPainter::begin: Cannot paint on a null image"); -
1009 qt_cleanup_painter_state(d); -
1010 return false;
never executed: return false;
0
1011 } else if (img->format() == QImage::Format_Indexed8) {
evaluated: img->format() == QImage::Format_Indexed8
TRUEFALSE
yes
Evaluation Count:1
yes
Evaluation Count:27903
1-27903
1012 -
1013 QMessageLogger("painting/qpainter.cpp", 1789, __PRETTY_FUNCTION__).warning("QPainter::begin: Cannot paint on an image with the QImage::Format_Indexed8 format"); -
1014 qt_cleanup_painter_state(d); -
1015 return false;
executed: return false;
Execution Count:1
1
1016 } -
1017 if (img->depth() == 1) {
evaluated: img->depth() == 1
TRUEFALSE
yes
Evaluation Count:162
yes
Evaluation Count:27741
162-27741
1018 d->state->pen = QPen(Qt::color1); -
1019 d->state->brush = QBrush(Qt::color0); -
1020 }
executed: }
Execution Count:162
162
1021 break;
executed: break;
Execution Count:27903
27903
1022 } -
1023 default: -
1024 break;
executed: break;
Execution Count:38
38
1025 } -
1026 if (d->state->ww == 0)
partially evaluated: d->state->ww == 0
TRUEFALSE
yes
Evaluation Count:28774
no
Evaluation Count:0
0-28774
1027 d->state->ww = d->state->wh = d->state->vw = d->state->vh = 1024;
executed: d->state->ww = d->state->wh = d->state->vw = d->state->vh = 1024;
Execution Count:28774
28774
1028 -
1029 d->engine->setPaintDevice(pd); -
1030 -
1031 bool begun = d->engine->begin(pd); -
1032 if (!begun) {
evaluated: !begun
TRUEFALSE
yes
Evaluation Count:1
yes
Evaluation Count:28773
1-28773
1033 QMessageLogger("painting/qpainter.cpp", 1809, __PRETTY_FUNCTION__).warning("QPainter::begin(): Returned false"); -
1034 if (d->engine->isActive()) {
partially evaluated: d->engine->isActive()
TRUEFALSE
no
Evaluation Count:0
yes
Evaluation Count:1
0-1
1035 end(); -
1036 } else {
never executed: }
0
1037 qt_cleanup_painter_state(d); -
1038 }
executed: }
Execution Count:1
1
1039 return false;
executed: return false;
Execution Count:1
1
1040 } else { -
1041 d->engine->setActive(begun); -
1042 }
executed: }
Execution Count:28773
28773
1043 -
1044 -
1045 -
1046 if (d->original_device->devType() == QInternal::Widget) {
evaluated: d->original_device->devType() == QInternal::Widget
TRUEFALSE
yes
Evaluation Count:23420
yes
Evaluation Count:5353
5353-23420
1047 initFrom(d->original_device); -
1048 } else {
executed: }
Execution Count:23420
23420
1049 d->state->layoutDirection = Qt::LayoutDirectionAuto; -
1050 -
1051 d->state->deviceFont = d->state->font = QFont(d->state->deviceFont, device()); -
1052 }
executed: }
Execution Count:5353
5353
1053 -
1054 QRect systemRect = d->engine->systemRect(); -
1055 if (!systemRect.isEmpty()) {
evaluated: !systemRect.isEmpty()
TRUEFALSE
yes
Evaluation Count:23420
yes
Evaluation Count:5353
5353-23420
1056 d->state->ww = d->state->vw = systemRect.width(); -
1057 d->state->wh = d->state->vh = systemRect.height(); -
1058 } else {
executed: }
Execution Count:23420
23420
1059 d->state->ww = d->state->vw = pd->metric(QPaintDevice::PdmWidth); -
1060 d->state->wh = d->state->vh = pd->metric(QPaintDevice::PdmHeight); -
1061 }
executed: }
Execution Count:5353
5353
1062 -
1063 const QPoint coordinateOffset = d->engine->coordinateOffset(); -
1064 d->state->redirectionMatrix.translate(-coordinateOffset.x(), -coordinateOffset.y()); -
1065 -
1066 qt_noop(); -
1067 -
1068 -
1069 -
1070 -
1071 -
1072 -
1073 const bool isHighDpi = false; -
1074 -
1075 if (!d->state->redirectionMatrix.isIdentity() || isHighDpi)
evaluated: !d->state->redirectionMatrix.isIdentity()
TRUEFALSE
yes
Evaluation Count:15494
yes
Evaluation Count:13279
partially evaluated: isHighDpi
TRUEFALSE
no
Evaluation Count:0
yes
Evaluation Count:13279
0-15494
1076 d->updateMatrix();
executed: d->updateMatrix();
Execution Count:15494
15494
1077 -
1078 qt_noop(); -
1079 d->state->renderHints = QPainter::TextAntialiasing; -
1080 ++d->device->painters; -
1081 -
1082 d->state->emulationSpecifier = 0; -
1083 -
1084 return true;
executed: return true;
Execution Count:28773
28773
1085} -
1086bool QPainter::end() -
1087{ -
1088 -
1089 -
1090 -
1091 -
1092 QPainterPrivate * const d = d_func(); -
1093 -
1094 if (!d->engine) {
-
1095 QMessageLogger("painting/qpainter.cpp", 1882, __PRETTY_FUNCTION__).warning("QPainter::end: Painter not active, aborted"); -
1096 qt_cleanup_painter_state(d); -
1097 return false;
-
1098 } -
1099 -
1100 if (d->refcount > 1) {
-
1101 d->detachPainterPrivate(this); -
1102 return true;
-
1103 } -
1104 -
1105 bool ended = true; -
1106 -
1107 if (d->engine->isActive()) {
-
1108 ended = d->engine->end(); -
1109 d->updateState(0); -
1110 -
1111 --d->device->painters; -
1112 if (d->device->painters == 0) {
-
1113 d->engine->setPaintDevice(0); -
1114 d->engine->setActive(false); -
1115 }
-
1116 }
-
1117 -
1118 if (d->states.size() > 1) {
-
1119 QMessageLogger("painting/qpainter.cpp", 1906, __PRETTY_FUNCTION__).warning("QPainter::end: Painter ended with %d saved states", -
1120 d->states.size()); -
1121 }
-
1122 -
1123 if (d->engine->autoDestruct()) {
-
1124 delete d->engine; -
1125 }
-
1126 -
1127 if (d->emulationEngine) {
-
1128 delete d->emulationEngine; -
1129 d->emulationEngine = 0; -
1130 }
-
1131 -
1132 if (d->extended) {
-
1133 d->extended = 0; -
1134 }
-
1135 -
1136 qt_cleanup_painter_state(d); -
1137 -
1138 return ended;
-
1139} -
1140QPaintEngine *QPainter::paintEngine() const -
1141{ -
1142 const QPainterPrivate * const d = d_func(); -
1143 return d->engine;
-
1144} -
1145void QPainter::beginNativePainting() -
1146{ -
1147 QPainterPrivate * const d = d_func(); -
1148 if (!d->engine) {
-
1149 QMessageLogger("painting/qpainter.cpp", 1977, __PRETTY_FUNCTION__).warning("QPainter::beginNativePainting: Painter not active"); -
1150 return;
-
1151 } -
1152 -
1153 if (d->extended)
-
1154 d->extended->beginNativePainting();
-
1155}
-
1156void QPainter::endNativePainting() -
1157{ -
1158 const QPainterPrivate * const d = d_func(); -
1159 if (!d->engine) {
-
1160 QMessageLogger("painting/qpainter.cpp", 1998, __PRETTY_FUNCTION__).warning("QPainter::beginNativePainting: Painter not active"); -
1161 return;
-
1162 } -
1163 -
1164 if (d->extended)
-
1165 d->extended->endNativePainting();
-
1166 else -
1167 d->engine->syncState();
-
1168} -
1169QFontMetrics QPainter::fontMetrics() const -
1170{ -
1171 const QPainterPrivate * const d = d_func(); -
1172 if (!d->engine) {
-
1173 QMessageLogger("painting/qpainter.cpp", 2019, __PRETTY_FUNCTION__).warning("QPainter::fontMetrics: Painter not active"); -
1174 return QFontMetrics(QFont());
-
1175 } -
1176 return QFontMetrics(d->state->font);
-
1177} -
1178QFontInfo QPainter::fontInfo() const -
1179{ -
1180 const QPainterPrivate * const d = d_func(); -
1181 if (!d->engine) {
-
1182 QMessageLogger("painting/qpainter.cpp", 2037, __PRETTY_FUNCTION__).warning("QPainter::fontInfo: Painter not active"); -
1183 return QFontInfo(QFont());
-
1184 } -
1185 return QFontInfo(d->state->font);
-
1186} -
1187qreal QPainter::opacity() const -
1188{ -
1189 const QPainterPrivate * const d = d_func(); -
1190 if (!d->engine) {
-
1191 QMessageLogger("painting/qpainter.cpp", 2054, __PRETTY_FUNCTION__).warning("QPainter::opacity: Painter not active"); -
1192 return 1.0;
-
1193 } -
1194 return d->state->opacity;
-
1195} -
1196void QPainter::setOpacity(qreal opacity) -
1197{ -
1198 QPainterPrivate * const d = d_func(); -
1199 -
1200 if (!d->engine) {
-
1201 QMessageLogger("painting/qpainter.cpp", 2076, __PRETTY_FUNCTION__).warning("QPainter::setOpacity: Painter not active"); -
1202 return;
-
1203 } -
1204 -
1205 opacity = qMin(qreal(1), qMax(qreal(0), opacity)); -
1206 -
1207 if (opacity == d->state->opacity)
-
1208 return;
-
1209 -
1210 d->state->opacity = opacity; -
1211 -
1212 if (d->extended)
-
1213 d->extended->opacityChanged();
-
1214 else -
1215 d->state->dirtyFlags |= QPaintEngine::DirtyOpacity;
-
1216} -
1217QPoint QPainter::brushOrigin() const -
1218{ -
1219 const QPainterPrivate * const d = d_func(); -
1220 if (!d->engine) {
-
1221 QMessageLogger("painting/qpainter.cpp", 2104, __PRETTY_FUNCTION__).warning("QPainter::brushOrigin: Painter not active"); -
1222 return QPoint();
-
1223 } -
1224 return QPointF(d->state->brushOrigin).toPoint();
-
1225} -
1226void QPainter::setBrushOrigin(const QPointF &p) -
1227{ -
1228 QPainterPrivate * const d = d_func(); -
1229 -
1230 -
1231 -
1232 -
1233 -
1234 if (!d->engine) {
-
1235 QMessageLogger("painting/qpainter.cpp", 2137, __PRETTY_FUNCTION__).warning("QPainter::setBrushOrigin: Painter not active"); -
1236 return;
-
1237 } -
1238 -
1239 d->state->brushOrigin = p; -
1240 -
1241 if (d->extended) {
-
1242 d->extended->brushOriginChanged(); -
1243 return;
-
1244 } -
1245 -
1246 d->state->dirtyFlags |= QPaintEngine::DirtyBrushOrigin; -
1247}
-
1248void QPainter::setCompositionMode(CompositionMode mode) -
1249{ -
1250 QPainterPrivate * const d = d_func(); -
1251 if (!d->engine) {
-
1252 QMessageLogger("painting/qpainter.cpp", 2364, __PRETTY_FUNCTION__).warning("QPainter::setCompositionMode: Painter not active"); -
1253 return;
-
1254 } -
1255 if (d->state->composition_mode == mode)
-
1256 return;
-
1257 if (d->extended) {
-
1258 d->state->composition_mode = mode; -
1259 d->extended->compositionModeChanged(); -
1260 return;
-
1261 } -
1262 -
1263 if (mode >= QPainter::RasterOp_SourceOrDestination) {
-
1264 if (!d->engine->hasFeature(QPaintEngine::RasterOpModes)) {
-
1265 QMessageLogger("painting/qpainter.cpp", 2377, __PRETTY_FUNCTION__).warning("QPainter::setCompositionMode: " -
1266 "Raster operation modes not supported on device"); -
1267 return;
-
1268 } -
1269 } else if (mode >= QPainter::CompositionMode_Plus) {
-
1270 if (!d->engine->hasFeature(QPaintEngine::BlendModes)) {
-
1271 QMessageLogger("painting/qpainter.cpp", 2383, __PRETTY_FUNCTION__).warning("QPainter::setCompositionMode: " -
1272 "Blend modes not supported on device"); -
1273 return;
-
1274 } -
1275 } else if (!d->engine->hasFeature(QPaintEngine::PorterDuff)) {
-
1276 if (mode != CompositionMode_Source && mode != CompositionMode_SourceOver) {
-
1277 QMessageLogger("painting/qpainter.cpp", 2389, __PRETTY_FUNCTION__).warning("QPainter::setCompositionMode: " -
1278 "PorterDuff modes not supported on device"); -
1279 return;
-
1280 } -
1281 }
-
1282 -
1283 d->state->composition_mode = mode; -
1284 d->state->dirtyFlags |= QPaintEngine::DirtyCompositionMode; -
1285}
-
1286 -
1287 -
1288 -
1289 -
1290 -
1291 -
1292QPainter::CompositionMode QPainter::compositionMode() const -
1293{ -
1294 const QPainterPrivate * const d = d_func(); -
1295 if (!d->engine) {
-
1296 QMessageLogger("painting/qpainter.cpp", 2408, __PRETTY_FUNCTION__).warning("QPainter::compositionMode: Painter not active"); -
1297 return QPainter::CompositionMode_SourceOver;
-
1298 } -
1299 return d->state->composition_mode;
-
1300} -
1301 -
1302 -
1303 -
1304 -
1305 -
1306 -
1307 -
1308const QBrush &QPainter::background() const -
1309{ -
1310 const QPainterPrivate * const d = d_func(); -
1311 if (!d->engine) {
-
1312 QMessageLogger("painting/qpainter.cpp", 2424, __PRETTY_FUNCTION__).warning("QPainter::background: Painter not active"); -
1313 return d->fakeState()->brush;
-
1314 } -
1315 return d->state->bgBrush;
-
1316} -
1317bool QPainter::hasClipping() const -
1318{ -
1319 const QPainterPrivate * const d = d_func(); -
1320 if (!d->engine) {
-
1321 QMessageLogger("painting/qpainter.cpp", 2441, __PRETTY_FUNCTION__).warning("QPainter::hasClipping: Painter not active"); -
1322 return false;
-
1323 } -
1324 return d->state->clipEnabled && d->state->clipOperation != Qt::NoClip;
-
1325} -
1326void QPainter::setClipping(bool enable) -
1327{ -
1328 QPainterPrivate * const d = d_func(); -
1329 -
1330 -
1331 -
1332 -
1333 -
1334 -
1335 if (!d->engine) {
-
1336 QMessageLogger("painting/qpainter.cpp", 2465, __PRETTY_FUNCTION__).warning("QPainter::setClipping: Painter not active, state will be reset by begin"); -
1337 return;
-
1338 } -
1339 -
1340 if (hasClipping() == enable)
-
1341 return;
-
1342 -
1343 -
1344 if (enable
-
1345 && (d->state->clipInfo.isEmpty() || d->state->clipInfo.last().operation == Qt::NoClip))
-
1346 return;
-
1347 d->state->clipEnabled = enable; -
1348 -
1349 if (d->extended) {
-
1350 d->extended->clipEnabledChanged(); -
1351 return;
-
1352 } -
1353 -
1354 d->state->dirtyFlags |= QPaintEngine::DirtyClipEnabled; -
1355 d->updateState(d->state); -
1356}
-
1357QRegion QPainter::clipRegion() const -
1358{ -
1359 const QPainterPrivate * const d = d_func(); -
1360 if (!d->engine) {
-
1361 QMessageLogger("painting/qpainter.cpp", 2504, __PRETTY_FUNCTION__).warning("QPainter::clipRegion: Painter not active"); -
1362 return QRegion();
-
1363 } -
1364 -
1365 QRegion region; -
1366 bool lastWasNothing = true; -
1367 -
1368 if (!d->txinv)
-
1369 const_cast<QPainter *>(this)->d_ptr->updateInvMatrix();
-
1370 -
1371 -
1372 for (int i=0; i<d->state->clipInfo.size(); ++i) {
-
1373 const QPainterClipInfo &info = d->state->clipInfo.at(i); -
1374 switch (info.clipType) { -
1375 -
1376 case QPainterClipInfo::RegionClip: { -
1377 QTransform matrix = (info.matrix * d->invMatrix); -
1378 if (lastWasNothing) {
-
1379 region = info.region * matrix; -
1380 lastWasNothing = false; -
1381 continue;
-
1382 } -
1383 if (info.operation == Qt::IntersectClip)
-
1384 region &= info.region * matrix;
-
1385 else if (info.operation == Qt::NoClip) {
-
1386 lastWasNothing = true; -
1387 region = QRegion(); -
1388 } else
-
1389 region = info.region * matrix;
-
1390 break;
-
1391 } -
1392 -
1393 case QPainterClipInfo::PathClip: { -
1394 QTransform matrix = (info.matrix * d->invMatrix); -
1395 if (lastWasNothing) {
-
1396 region = QRegion((info.path * matrix).toFillPolygon().toPolygon(), -
1397 info.path.fillRule()); -
1398 lastWasNothing = false; -
1399 continue;
-
1400 } -
1401 if (info.operation == Qt::IntersectClip) {
-
1402 region &= QRegion((info.path * matrix).toFillPolygon().toPolygon(), -
1403 info.path.fillRule()); -
1404 } else if (info.operation == Qt::NoClip) {
-
1405 lastWasNothing = true; -
1406 region = QRegion(); -
1407 } else {
-
1408 region = QRegion((info.path * matrix).toFillPolygon().toPolygon(), -
1409 info.path.fillRule()); -
1410 }
-
1411 break;
-
1412 } -
1413 -
1414 case QPainterClipInfo::RectClip: { -
1415 QTransform matrix = (info.matrix * d->invMatrix); -
1416 if (lastWasNothing) {
-
1417 region = QRegion(info.rect) * matrix; -
1418 lastWasNothing = false; -
1419 continue;
-
1420 } -
1421 if (info.operation == Qt::IntersectClip) {
-
1422 -
1423 if (matrix.type() <= QTransform::TxScale)
-
1424 region &= matrix.mapRect(info.rect);
-
1425 else -
1426 region &= matrix.map(QRegion(info.rect));
-
1427 } else if (info.operation == Qt::NoClip) {
-
1428 lastWasNothing = true; -
1429 region = QRegion(); -
1430 } else {
-
1431 region = QRegion(info.rect) * matrix; -
1432 }
-
1433 break;
-
1434 } -
1435 -
1436 case QPainterClipInfo::RectFClip: { -
1437 QTransform matrix = (info.matrix * d->invMatrix); -
1438 if (lastWasNothing) {
-
1439 region = QRegion(info.rectf.toRect()) * matrix; -
1440 lastWasNothing = false; -
1441 continue;
-
1442 } -
1443 if (info.operation == Qt::IntersectClip) {
-
1444 -
1445 if (matrix.type() <= QTransform::TxScale)
-
1446 region &= matrix.mapRect(info.rectf.toRect());
-
1447 else -
1448 region &= matrix.map(QRegion(info.rectf.toRect()));
-
1449 } else if (info.operation == Qt::NoClip) {
-
1450 lastWasNothing = true; -
1451 region = QRegion(); -
1452 } else {
-
1453 region = QRegion(info.rectf.toRect()) * matrix; -
1454 }
-
1455 break;
-
1456 } -
1457 } -
1458 }
-
1459 -
1460 return region;
-
1461} -
1462 -
1463extern QPainterPath qt_regionToPath(const QRegion &region); -
1464QPainterPath QPainter::clipPath() const -
1465{ -
1466 const QPainterPrivate * const d = d_func(); -
1467 -
1468 -
1469 -
1470 if (!d->engine) {
-
1471 QMessageLogger("painting/qpainter.cpp", 2626, __PRETTY_FUNCTION__).warning("QPainter::clipPath: Painter not active"); -
1472 return QPainterPath();
-
1473 } -
1474 -
1475 -
1476 if (d->state->clipInfo.size() == 0) {
-
1477 return QPainterPath();
-
1478 } else { -
1479 -
1480 -
1481 if (!d->txinv)
-
1482 const_cast<QPainter *>(this)->d_ptr->updateInvMatrix();
-
1483 -
1484 -
1485 if (d->state->clipInfo.size() == 1
-
1486 && d->state->clipInfo.at(0).clipType == QPainterClipInfo::PathClip) {
-
1487 QTransform matrix = (d->state->clipInfo.at(0).matrix * d->invMatrix); -
1488 return d->state->clipInfo.at(0).path * matrix;
-
1489 -
1490 } else if (d->state->clipInfo.size() == 1
-
1491 && d->state->clipInfo.at(0).clipType == QPainterClipInfo::RectClip) {
-
1492 QTransform matrix = (d->state->clipInfo.at(0).matrix * d->invMatrix); -
1493 QPainterPath path; -
1494 path.addRect(d->state->clipInfo.at(0).rect); -
1495 return path * matrix;
-
1496 } else { -
1497 -
1498 return qt_regionToPath(clipRegion());
-
1499 } -
1500 } -
1501} -
1502QRectF QPainter::clipBoundingRect() const -
1503{ -
1504 const QPainterPrivate * const d = d_func(); -
1505 -
1506 if (!d->engine) {
-
1507 QMessageLogger("painting/qpainter.cpp", 2675, __PRETTY_FUNCTION__).warning("QPainter::clipBoundingRect: Painter not active"); -
1508 return QRectF();
-
1509 } -
1510 -
1511 -
1512 -
1513 -
1514 QRectF bounds; -
1515 for (int i=0; i<d->state->clipInfo.size(); ++i) {
-
1516 QRectF r; -
1517 const QPainterClipInfo &info = d->state->clipInfo.at(i); -
1518 -
1519 if (info.clipType == QPainterClipInfo::RectClip)
-
1520 r = info.rect;
-
1521 else if (info.clipType == QPainterClipInfo::RectFClip)
-
1522 r = info.rectf;
-
1523 else if (info.clipType == QPainterClipInfo::RegionClip)
-
1524 r = info.region.boundingRect();
-
1525 else -
1526 r = info.path.boundingRect();
-
1527 -
1528 r = info.matrix.mapRect(r); -
1529 -
1530 if (i == 0)
-
1531 bounds = r;
-
1532 else if (info.operation == Qt::IntersectClip)
-
1533 bounds &= r;
-
1534 } -
1535 -
1536 -
1537 -
1538 -
1539 if (!d->txinv)
-
1540 const_cast<QPainter *>(this)->d_ptr->updateInvMatrix();
-
1541 -
1542 return d->invMatrix.mapRect(bounds);
-
1543} -
1544void QPainter::setClipRect(const QRectF &rect, Qt::ClipOperation op) -
1545{ -
1546 QPainterPrivate * const d = d_func(); -
1547 -
1548 if (d->extended) {
-
1549 if ((!d->state->clipEnabled && op != Qt::NoClip))
-
1550 op = Qt::ReplaceClip;
-
1551 -
1552 if (!d->engine) {
-
1553 QMessageLogger("painting/qpainter.cpp", 2734, __PRETTY_FUNCTION__).warning("QPainter::setClipRect: Painter not active"); -
1554 return;
-
1555 } -
1556 qreal right = rect.x() + rect.width(); -
1557 qreal bottom = rect.y() + rect.height(); -
1558 qreal pts[] = { rect.x(), rect.y(), -
1559 right, rect.y(), -
1560 right, bottom, -
1561 rect.x(), bottom }; -
1562 QVectorPath vp(pts, 4, 0, QVectorPath::RectangleHint); -
1563 d->state->clipEnabled = true; -
1564 d->extended->clip(vp, op); -
1565 if (op == Qt::ReplaceClip || op == Qt::NoClip)
-
1566 d->state->clipInfo.clear();
-
1567 d->state->clipInfo << QPainterClipInfo(rect, op, d->state->matrix); -
1568 d->state->clipOperation = op; -
1569 return;
-
1570 } -
1571 -
1572 if (qreal(int(rect.top())) == rect.top()
-
1573 && qreal(int(rect.bottom())) == rect.bottom()
-
1574 && qreal(int(rect.left())) == rect.left()
-
1575 && qreal(int(rect.right())) == rect.right())
-
1576 { -
1577 setClipRect(rect.toRect(), op); -
1578 return;
-
1579 } -
1580 -
1581 if (rect.isEmpty()) {
-
1582 setClipRegion(QRegion(), op); -
1583 return;
-
1584 } -
1585 -
1586 QPainterPath path; -
1587 path.addRect(rect); -
1588 setClipPath(path, op); -
1589}
-
1590void QPainter::setClipRect(const QRect &rect, Qt::ClipOperation op) -
1591{ -
1592 QPainterPrivate * const d = d_func(); -
1593 -
1594 if (!d->engine) {
-
1595 QMessageLogger("painting/qpainter.cpp", 2784, __PRETTY_FUNCTION__).warning("QPainter::setClipRect: Painter not active"); -
1596 return;
-
1597 } -
1598 -
1599 if ((!d->state->clipEnabled && op != Qt::NoClip))
-
1600 op = Qt::ReplaceClip;
-
1601 -
1602 if (d->extended) {
-
1603 d->state->clipEnabled = true; -
1604 d->extended->clip(rect, op); -
1605 if (op == Qt::ReplaceClip || op == Qt::NoClip)
-
1606 d->state->clipInfo.clear();
-
1607 d->state->clipInfo << QPainterClipInfo(rect, op, d->state->matrix); -
1608 d->state->clipOperation = op; -
1609 return;
-
1610 } -
1611 -
1612 if (d->state->clipOperation == Qt::NoClip && op == Qt::IntersectClip)
-
1613 op = Qt::ReplaceClip;
-
1614 -
1615 d->state->clipRegion = rect; -
1616 d->state->clipOperation = op; -
1617 if (op == Qt::NoClip || op == Qt::ReplaceClip)
-
1618 d->state->clipInfo.clear();
-
1619 d->state->clipInfo << QPainterClipInfo(rect, op, d->state->matrix); -
1620 d->state->clipEnabled = true; -
1621 d->state->dirtyFlags |= QPaintEngine::DirtyClipRegion | QPaintEngine::DirtyClipEnabled; -
1622 d->updateState(d->state); -
1623}
-
1624void QPainter::setClipRegion(const QRegion &r, Qt::ClipOperation op) -
1625{ -
1626 QPainterPrivate * const d = d_func(); -
1627 -
1628 -
1629 -
1630 -
1631 -
1632 -
1633 if (!d->engine) {
-
1634 QMessageLogger("painting/qpainter.cpp", 2842, __PRETTY_FUNCTION__).warning("QPainter::setClipRegion: Painter not active"); -
1635 return;
-
1636 } -
1637 -
1638 if ((!d->state->clipEnabled && op != Qt::NoClip))
-
1639 op = Qt::ReplaceClip;
-
1640 -
1641 if (d->extended) {
-
1642 d->state->clipEnabled = true; -
1643 d->extended->clip(r, op); -
1644 if (op == Qt::NoClip || op == Qt::ReplaceClip)
-
1645 d->state->clipInfo.clear();
-
1646 d->state->clipInfo << QPainterClipInfo(r, op, d->state->matrix); -
1647 d->state->clipOperation = op; -
1648 return;
-
1649 } -
1650 -
1651 if (d->state->clipOperation == Qt::NoClip && op == Qt::IntersectClip)
-
1652 op = Qt::ReplaceClip;
-
1653 -
1654 d->state->clipRegion = r; -
1655 d->state->clipOperation = op; -
1656 if (op == Qt::NoClip || op == Qt::ReplaceClip)
-
1657 d->state->clipInfo.clear();
-
1658 d->state->clipInfo << QPainterClipInfo(r, op, d->state->matrix); -
1659 d->state->clipEnabled = true; -
1660 d->state->dirtyFlags |= QPaintEngine::DirtyClipRegion | QPaintEngine::DirtyClipEnabled; -
1661 d->updateState(d->state); -
1662}
-
1663void QPainter::setWorldMatrix(const QMatrix &matrix, bool combine) -
1664{ -
1665 setWorldTransform(QTransform(matrix), combine); -
1666}
-
1667const QMatrix &QPainter::worldMatrix() const -
1668{ -
1669 const QPainterPrivate * const d = d_func(); -
1670 if (!d->engine) {
-
1671 QMessageLogger("painting/qpainter.cpp", 2937, __PRETTY_FUNCTION__).warning("QPainter::worldMatrix: Painter not active"); -
1672 return d->fakeState()->transform.toAffine();
-
1673 } -
1674 return d->state->worldMatrix.toAffine();
-
1675} -
1676void QPainter::setMatrix(const QMatrix &matrix, bool combine) -
1677{ -
1678 setWorldTransform(QTransform(matrix), combine); -
1679}
-
1680const QMatrix &QPainter::matrix() const -
1681{ -
1682 return worldMatrix();
-
1683} -
1684QMatrix QPainter::combinedMatrix() const -
1685{ -
1686 return combinedTransform().toAffine();
-
1687} -
1688const QMatrix &QPainter::deviceMatrix() const -
1689{ -
1690 const QPainterPrivate * const d = d_func(); -
1691 if (!d->engine) {
-
1692 QMessageLogger("painting/qpainter.cpp", 3011, __PRETTY_FUNCTION__).warning("QPainter::deviceMatrix: Painter not active"); -
1693 return d->fakeState()->transform.toAffine();
-
1694 } -
1695 return d->state->matrix.toAffine();
-
1696} -
1697void QPainter::resetMatrix() -
1698{ -
1699 resetTransform(); -
1700}
-
1701void QPainter::setWorldMatrixEnabled(bool enable) -
1702{ -
1703 QPainterPrivate * const d = d_func(); -
1704 -
1705 -
1706 -
1707 -
1708 -
1709 if (!d->engine) {
-
1710 QMessageLogger("painting/qpainter.cpp", 3057, __PRETTY_FUNCTION__).warning("QPainter::setMatrixEnabled: Painter not active"); -
1711 return;
-
1712 } -
1713 if (enable == d->state->WxF)
-
1714 return;
-
1715 -
1716 d->state->WxF = enable; -
1717 d->updateMatrix(); -
1718}
-
1719bool QPainter::worldMatrixEnabled() const -
1720{ -
1721 const QPainterPrivate * const d = d_func(); -
1722 if (!d->engine) {
-
1723 QMessageLogger("painting/qpainter.cpp", 3080, __PRETTY_FUNCTION__).warning("QPainter::worldMatrixEnabled: Painter not active"); -
1724 return false;
-
1725 } -
1726 return d->state->WxF;
-
1727} -
1728void QPainter::setMatrixEnabled(bool enable) -
1729{ -
1730 setWorldMatrixEnabled(enable); -
1731}
-
1732bool QPainter::matrixEnabled() const -
1733{ -
1734 return worldMatrixEnabled();
-
1735} -
1736 -
1737 -
1738 -
1739 -
1740 -
1741 -
1742 -
1743void QPainter::scale(qreal sx, qreal sy) -
1744{ -
1745 -
1746 -
1747 -
1748 -
1749 QPainterPrivate * const d = d_func(); -
1750 if (!d->engine) {
-
1751 QMessageLogger("painting/qpainter.cpp", 3126, __PRETTY_FUNCTION__).warning("QPainter::scale: Painter not active"); -
1752 return;
-
1753 } -
1754 -
1755 d->state->worldMatrix.scale(sx,sy); -
1756 d->state->WxF = true; -
1757 d->updateMatrix(); -
1758}
-
1759 -
1760 -
1761 -
1762 -
1763 -
1764 -
1765 -
1766void QPainter::shear(qreal sh, qreal sv) -
1767{ -
1768 -
1769 -
1770 -
1771 -
1772 QPainterPrivate * const d = d_func(); -
1773 if (!d->engine) {
-
1774 QMessageLogger("painting/qpainter.cpp", 3149, __PRETTY_FUNCTION__).warning("QPainter::shear: Painter not active"); -
1775 return;
-
1776 } -
1777 -
1778 d->state->worldMatrix.shear(sh, sv); -
1779 d->state->WxF = true; -
1780 d->updateMatrix(); -
1781}
-
1782void QPainter::rotate(qreal a) -
1783{ -
1784 -
1785 -
1786 -
1787 -
1788 QPainterPrivate * const d = d_func(); -
1789 if (!d->engine) {
-
1790 QMessageLogger("painting/qpainter.cpp", 3174, __PRETTY_FUNCTION__).warning("QPainter::rotate: Painter not active"); -
1791 return;
-
1792 } -
1793 -
1794 d->state->worldMatrix.rotate(a); -
1795 d->state->WxF = true; -
1796 d->updateMatrix(); -
1797}
-
1798 -
1799 -
1800 -
1801 -
1802 -
1803 -
1804 -
1805void QPainter::translate(const QPointF &offset) -
1806{ -
1807 qreal dx = offset.x(); -
1808 qreal dy = offset.y(); -
1809 -
1810 -
1811 -
1812 -
1813 QPainterPrivate * const d = d_func(); -
1814 if (!d->engine) {
-
1815 QMessageLogger("painting/qpainter.cpp", 3199, __PRETTY_FUNCTION__).warning("QPainter::translate: Painter not active"); -
1816 return;
-
1817 } -
1818 -
1819 d->state->worldMatrix.translate(dx, dy); -
1820 d->state->WxF = true; -
1821 d->updateMatrix(); -
1822}
-
1823void QPainter::setClipPath(const QPainterPath &path, Qt::ClipOperation op) -
1824{ -
1825 -
1826 -
1827 -
1828 -
1829 -
1830 -
1831 -
1832 QPainterPrivate * const d = d_func(); -
1833 -
1834 if (!d->engine) {
-
1835 QMessageLogger("painting/qpainter.cpp", 3246, __PRETTY_FUNCTION__).warning("QPainter::setClipPath: Painter not active"); -
1836 return;
-
1837 } -
1838 -
1839 if ((!d->state->clipEnabled && op != Qt::NoClip))
-
1840 op = Qt::ReplaceClip;
-
1841 -
1842 if (d->extended) {
-
1843 d->state->clipEnabled = true; -
1844 d->extended->clip(path, op); -
1845 if (op == Qt::NoClip || op == Qt::ReplaceClip)
-
1846 d->state->clipInfo.clear();
-
1847 d->state->clipInfo << QPainterClipInfo(path, op, d->state->matrix); -
1848 d->state->clipOperation = op; -
1849 return;
-
1850 } -
1851 -
1852 if (d->state->clipOperation == Qt::NoClip && op == Qt::IntersectClip)
-
1853 op = Qt::ReplaceClip;
-
1854 -
1855 d->state->clipPath = path; -
1856 d->state->clipOperation = op; -
1857 if (op == Qt::NoClip || op == Qt::ReplaceClip)
-
1858 d->state->clipInfo.clear();
-
1859 d->state->clipInfo << QPainterClipInfo(path, op, d->state->matrix); -
1860 d->state->clipEnabled = true; -
1861 d->state->dirtyFlags |= QPaintEngine::DirtyClipPath | QPaintEngine::DirtyClipEnabled; -
1862 d->updateState(d->state); -
1863}
-
1864 -
1865 -
1866 -
1867 -
1868 -
1869 -
1870 -
1871void QPainter::strokePath(const QPainterPath &path, const QPen &pen) -
1872{ -
1873 QPainterPrivate * const d = d_func(); -
1874 -
1875 if (!d->engine) {
-
1876 QMessageLogger("painting/qpainter.cpp", 3287, __PRETTY_FUNCTION__).warning("QPainter::strokePath: Painter not active"); -
1877 return;
-
1878 } -
1879 -
1880 if (path.isEmpty())
-
1881 return;
-
1882 -
1883 if (d->extended) {
-
1884 const QGradient *g = qpen_brush(pen).gradient(); -
1885 if (!g || g->coordinateMode() == QGradient::LogicalMode) {
-
1886 d->extended->stroke(qtVectorPathForPath(path), pen); -
1887 return;
-
1888 } -
1889 }
-
1890 -
1891 QBrush oldBrush = d->state->brush; -
1892 QPen oldPen = d->state->pen; -
1893 -
1894 setPen(pen); -
1895 setBrush(Qt::NoBrush); -
1896 -
1897 drawPath(path); -
1898 -
1899 -
1900 setPen(oldPen); -
1901 setBrush(oldBrush); -
1902}
-
1903void QPainter::fillPath(const QPainterPath &path, const QBrush &brush) -
1904{ -
1905 QPainterPrivate * const d = d_func(); -
1906 -
1907 if (!d->engine) {
-
1908 QMessageLogger("painting/qpainter.cpp", 3330, __PRETTY_FUNCTION__).warning("QPainter::fillPath: Painter not active"); -
1909 return;
-
1910 } -
1911 -
1912 if (path.isEmpty())
-
1913 return;
-
1914 -
1915 if (d->extended) {
-
1916 const QGradient *g = brush.gradient(); -
1917 if (!g || g->coordinateMode() == QGradient::LogicalMode) {
-
1918 d->extended->fill(qtVectorPathForPath(path), brush); -
1919 return;
-
1920 } -
1921 }
-
1922 -
1923 QBrush oldBrush = d->state->brush; -
1924 QPen oldPen = d->state->pen; -
1925 -
1926 setPen(Qt::NoPen); -
1927 setBrush(brush); -
1928 -
1929 drawPath(path); -
1930 -
1931 -
1932 setPen(oldPen); -
1933 setBrush(oldBrush); -
1934}
-
1935void QPainter::drawPath(const QPainterPath &path) -
1936{ -
1937 QPainterPrivate * const d = d_func(); -
1938 -
1939 if (!d->engine) {
-
1940 QMessageLogger("painting/qpainter.cpp", 3385, __PRETTY_FUNCTION__).warning("QPainter::drawPath: Painter not active"); -
1941 return;
-
1942 } -
1943 -
1944 if (d->extended) {
-
1945 d->extended->drawPath(path); -
1946 return;
-
1947 } -
1948 d->updateState(d->state); -
1949 -
1950 if (d->engine->hasFeature(QPaintEngine::PainterPaths) && d->state->emulationSpecifier == 0) {
-
1951 d->engine->drawPath(path); -
1952 } else {
-
1953 d->draw_helper(path); -
1954 }
-
1955} -
1956void QPainter::drawRects(const QRectF *rects, int rectCount) -
1957{ -
1958 -
1959 -
1960 -
1961 -
1962 QPainterPrivate * const d = d_func(); -
1963 -
1964 if (!d->engine) {
-
1965 QMessageLogger("painting/qpainter.cpp", 3498, __PRETTY_FUNCTION__).warning("QPainter::drawRects: Painter not active"); -
1966 return;
-
1967 } -
1968 -
1969 if (rectCount <= 0)
-
1970 return;
-
1971 -
1972 if (d->extended) {
-
1973 d->extended->drawRects(rects, rectCount); -
1974 return;
-
1975 } -
1976 -
1977 d->updateState(d->state); -
1978 -
1979 if (!d->state->emulationSpecifier) {
-
1980 d->engine->drawRects(rects, rectCount); -
1981 return;
-
1982 } -
1983 -
1984 if (d->state->emulationSpecifier == QPaintEngine::PrimitiveTransform
-
1985 && d->state->matrix.type() == QTransform::TxTranslate) {
-
1986 for (int i=0; i<rectCount; ++i) {
-
1987 QRectF r(rects[i].x() + d->state->matrix.dx(), -
1988 rects[i].y() + d->state->matrix.dy(), -
1989 rects[i].width(), -
1990 rects[i].height()); -
1991 d->engine->drawRects(&r, 1); -
1992 }
-
1993 } else {
-
1994 if (d->state->brushNeedsResolving() || d->state->penNeedsResolving()) {
-
1995 for (int i=0; i<rectCount; ++i) {
-
1996 QPainterPath rectPath; -
1997 rectPath.addRect(rects[i]); -
1998 d->draw_helper(rectPath, QPainterPrivate::StrokeAndFillDraw); -
1999 }
-
2000 } else {
-
2001 QPainterPath rectPath; -
2002 for (int i=0; i<rectCount; ++i)
-
2003 rectPath.addRect(rects[i]);
-
2004 d->draw_helper(rectPath, QPainterPrivate::StrokeAndFillDraw); -
2005 }
-
2006 } -
2007} -
2008void QPainter::drawRects(const QRect *rects, int rectCount) -
2009{ -
2010 -
2011 -
2012 -
2013 -
2014 QPainterPrivate * const d = d_func(); -
2015 -
2016 if (!d->engine) {
-
2017 QMessageLogger("painting/qpainter.cpp", 3558, __PRETTY_FUNCTION__).warning("QPainter::drawRects: Painter not active"); -
2018 return;
-
2019 } -
2020 -
2021 if (rectCount <= 0)
-
2022 return;
-
2023 -
2024 if (d->extended) {
-
2025 d->extended->drawRects(rects, rectCount); -
2026 return;
-
2027 } -
2028 -
2029 d->updateState(d->state); -
2030 -
2031 if (!d->state->emulationSpecifier) {
-
2032 d->engine->drawRects(rects, rectCount); -
2033 return;
-
2034 } -
2035 -
2036 if (d->state->emulationSpecifier == QPaintEngine::PrimitiveTransform
-
2037 && d->state->matrix.type() == QTransform::TxTranslate) {
-
2038 for (int i=0; i<rectCount; ++i) {
-
2039 QRectF r(rects[i].x() + d->state->matrix.dx(), -
2040 rects[i].y() + d->state->matrix.dy(), -
2041 rects[i].width(), -
2042 rects[i].height()); -
2043 -
2044 d->engine->drawRects(&r, 1); -
2045 }
-
2046 } else {
-
2047 if (d->state->brushNeedsResolving() || d->state->penNeedsResolving()) {
-
2048 for (int i=0; i<rectCount; ++i) {
-
2049 QPainterPath rectPath; -
2050 rectPath.addRect(rects[i]); -
2051 d->draw_helper(rectPath, QPainterPrivate::StrokeAndFillDraw); -
2052 }
-
2053 } else {
-
2054 QPainterPath rectPath; -
2055 for (int i=0; i<rectCount; ++i)
-
2056 rectPath.addRect(rects[i]);
-
2057 -
2058 d->draw_helper(rectPath, QPainterPrivate::StrokeAndFillDraw); -
2059 }
-
2060 } -
2061} -
2062void QPainter::drawPoints(const QPointF *points, int pointCount) -
2063{ -
2064 -
2065 -
2066 -
2067 -
2068 QPainterPrivate * const d = d_func(); -
2069 -
2070 if (!d->engine) {
-
2071 QMessageLogger("painting/qpainter.cpp", 3658, __PRETTY_FUNCTION__).warning("QPainter::drawPoints: Painter not active"); -
2072 return;
-
2073 } -
2074 -
2075 if (pointCount <= 0)
-
2076 return;
-
2077 -
2078 if (d->extended) {
-
2079 d->extended->drawPoints(points, pointCount); -
2080 return;
-
2081 } -
2082 -
2083 d->updateState(d->state); -
2084 -
2085 if (!d->state->emulationSpecifier) {
-
2086 d->engine->drawPoints(points, pointCount); -
2087 return;
-
2088 } -
2089 -
2090 if (d->state->emulationSpecifier == QPaintEngine::PrimitiveTransform
-
2091 && d->state->matrix.type() == QTransform::TxTranslate) {
-
2092 -
2093 for (int i=0; i<pointCount; ++i) {
-
2094 QPointF pt(points[i].x() + d->state->matrix.dx(), -
2095 points[i].y() + d->state->matrix.dy()); -
2096 d->engine->drawPoints(&pt, 1); -
2097 }
-
2098 } else {
-
2099 QPen pen = d->state->pen; -
2100 bool flat_pen = pen.capStyle() == Qt::FlatCap; -
2101 if (flat_pen) {
-
2102 save(); -
2103 pen.setCapStyle(Qt::SquareCap); -
2104 setPen(pen); -
2105 }
-
2106 QPainterPath path; -
2107 for (int i=0; i<pointCount; ++i) {
-
2108 path.moveTo(points[i].x(), points[i].y()); -
2109 path.lineTo(points[i].x() + 0.0001, points[i].y()); -
2110 }
-
2111 d->draw_helper(path, QPainterPrivate::StrokeDraw); -
2112 if (flat_pen)
-
2113 restore();
-
2114 }
-
2115} -
2116void QPainter::drawPoints(const QPoint *points, int pointCount) -
2117{ -
2118 -
2119 -
2120 -
2121 -
2122 QPainterPrivate * const d = d_func(); -
2123 -
2124 if (!d->engine) {
-
2125 QMessageLogger("painting/qpainter.cpp", 3720, __PRETTY_FUNCTION__).warning("QPainter::drawPoints: Painter not active"); -
2126 return;
-
2127 } -
2128 -
2129 if (pointCount <= 0)
-
2130 return;
-
2131 -
2132 if (d->extended) {
-
2133 d->extended->drawPoints(points, pointCount); -
2134 return;
-
2135 } -
2136 -
2137 d->updateState(d->state); -
2138 -
2139 if (!d->state->emulationSpecifier) {
-
2140 d->engine->drawPoints(points, pointCount); -
2141 return;
-
2142 } -
2143 -
2144 if (d->state->emulationSpecifier == QPaintEngine::PrimitiveTransform
-
2145 && d->state->matrix.type() == QTransform::TxTranslate) {
-
2146 -
2147 for (int i=0; i<pointCount; ++i) {
-
2148 QPointF pt(points[i].x() + d->state->matrix.dx(), -
2149 points[i].y() + d->state->matrix.dy()); -
2150 d->engine->drawPoints(&pt, 1); -
2151 }
-
2152 } else {
-
2153 QPen pen = d->state->pen; -
2154 bool flat_pen = (pen.capStyle() == Qt::FlatCap); -
2155 if (flat_pen) {
-
2156 save(); -
2157 pen.setCapStyle(Qt::SquareCap); -
2158 setPen(pen); -
2159 }
-
2160 QPainterPath path; -
2161 for (int i=0; i<pointCount; ++i) {
-
2162 path.moveTo(points[i].x(), points[i].y()); -
2163 path.lineTo(points[i].x() + 0.0001, points[i].y()); -
2164 }
-
2165 d->draw_helper(path, QPainterPrivate::StrokeDraw); -
2166 if (flat_pen)
-
2167 restore();
-
2168 }
-
2169} -
2170void QPainter::setBackgroundMode(Qt::BGMode mode) -
2171{ -
2172 -
2173 -
2174 -
2175 -
2176 -
2177 QPainterPrivate * const d = d_func(); -
2178 if (!d->engine) {
-
2179 QMessageLogger("painting/qpainter.cpp", 3805, __PRETTY_FUNCTION__).warning("QPainter::setBackgroundMode: Painter not active"); -
2180 return;
-
2181 } -
2182 if (d->state->bgMode == mode)
-
2183 return;
-
2184 -
2185 d->state->bgMode = mode; -
2186 if (d->extended) {
-
2187 d->checkEmulation(); -
2188 } else {
-
2189 d->state->dirtyFlags |= QPaintEngine::DirtyBackgroundMode; -
2190 }
-
2191} -
2192 -
2193 -
2194 -
2195 -
2196 -
2197 -
2198Qt::BGMode QPainter::backgroundMode() const -
2199{ -
2200 const QPainterPrivate * const d = d_func(); -
2201 if (!d->engine) {
-
2202 QMessageLogger("painting/qpainter.cpp", 3828, __PRETTY_FUNCTION__).warning("QPainter::backgroundMode: Painter not active"); -
2203 return Qt::TransparentMode;
-
2204 } -
2205 return d->state->bgMode;
-
2206} -
2207void QPainter::setPen(const QColor &color) -
2208{ -
2209 -
2210 -
2211 -
2212 -
2213 QPainterPrivate * const d = d_func(); -
2214 if (!d->engine) {
-
2215 QMessageLogger("painting/qpainter.cpp", 3850, __PRETTY_FUNCTION__).warning("QPainter::setPen: Painter not active"); -
2216 return;
-
2217 } -
2218 -
2219 QPen pen(color.isValid() ? color : QColor(Qt::black)); -
2220 -
2221 if (d->state->pen == pen)
-
2222 return;
-
2223 -
2224 d->state->pen = pen; -
2225 if (d->extended)
-
2226 d->extended->penChanged();
-
2227 else -
2228 d->state->dirtyFlags |= QPaintEngine::DirtyPen;
-
2229} -
2230void QPainter::setPen(const QPen &pen) -
2231{ -
2232 -
2233 -
2234 -
2235 -
2236 -
2237 -
2238 QPainterPrivate * const d = d_func(); -
2239 if (!d->engine) {
-
2240 QMessageLogger("painting/qpainter.cpp", 3885, __PRETTY_FUNCTION__).warning("QPainter::setPen: Painter not active"); -
2241 return;
-
2242 } -
2243 -
2244 if (d->state->pen == pen)
-
2245 return;
-
2246 -
2247 d->state->pen = pen; -
2248 -
2249 if (d->extended) {
-
2250 d->checkEmulation(); -
2251 d->extended->penChanged(); -
2252 return;
-
2253 } -
2254 -
2255 d->state->dirtyFlags |= QPaintEngine::DirtyPen; -
2256}
-
2257void QPainter::setPen(Qt::PenStyle style) -
2258{ -
2259 QPainterPrivate * const d = d_func(); -
2260 if (!d->engine) {
-
2261 QMessageLogger("painting/qpainter.cpp", 3914, __PRETTY_FUNCTION__).warning("QPainter::setPen: Painter not active"); -
2262 return;
-
2263 } -
2264 -
2265 QPen pen = QPen(style); -
2266 -
2267 if (d->state->pen == pen)
-
2268 return;
-
2269 -
2270 d->state->pen = pen; -
2271 -
2272 if (d->extended)
-
2273 d->extended->penChanged();
-
2274 else -
2275 d->state->dirtyFlags |= QPaintEngine::DirtyPen;
-
2276 -
2277} -
2278 -
2279 -
2280 -
2281 -
2282 -
2283 -
2284 -
2285const QPen &QPainter::pen() const -
2286{ -
2287 const QPainterPrivate * const d = d_func(); -
2288 if (!d->engine) {
-
2289 QMessageLogger("painting/qpainter.cpp", 3942, __PRETTY_FUNCTION__).warning("QPainter::pen: Painter not active"); -
2290 return d->fakeState()->pen;
-
2291 } -
2292 return d->state->pen;
-
2293} -
2294void QPainter::setBrush(const QBrush &brush) -
2295{ -
2296 -
2297 -
2298 -
2299 -
2300 QPainterPrivate * const d = d_func(); -
2301 if (!d->engine) {
-
2302 QMessageLogger("painting/qpainter.cpp", 3965, __PRETTY_FUNCTION__).warning("QPainter::setBrush: Painter not active"); -
2303 return;
-
2304 } -
2305 -
2306 if (d->state->brush.d == brush.d)
-
2307 return;
-
2308 -
2309 if (d->extended) {
-
2310 d->state->brush = brush; -
2311 d->checkEmulation(); -
2312 d->extended->brushChanged(); -
2313 return;
-
2314 } -
2315 -
2316 d->state->brush = brush; -
2317 d->state->dirtyFlags |= QPaintEngine::DirtyBrush; -
2318}
-
2319void QPainter::setBrush(Qt::BrushStyle style) -
2320{ -
2321 QPainterPrivate * const d = d_func(); -
2322 if (!d->engine) {
-
2323 QMessageLogger("painting/qpainter.cpp", 3995, __PRETTY_FUNCTION__).warning("QPainter::setBrush: Painter not active"); -
2324 return;
-
2325 } -
2326 if (d->state->brush.style() == style &&
-
2327 (style == Qt::NoBrush
-
2328 || (style == Qt::SolidPattern && d->state->brush.color() == QColor(0, 0, 0))))
-
2329 return;
-
2330 d->state->brush = QBrush(Qt::black, style); -
2331 if (d->extended)
-
2332 d->extended->brushChanged();
-
2333 else -
2334 d->state->dirtyFlags |= QPaintEngine::DirtyBrush;
-
2335} -
2336 -
2337 -
2338 -
2339 -
2340 -
2341 -
2342 -
2343const QBrush &QPainter::brush() const -
2344{ -
2345 const QPainterPrivate * const d = d_func(); -
2346 if (!d->engine) {
-
2347 QMessageLogger("painting/qpainter.cpp", 4019, __PRETTY_FUNCTION__).warning("QPainter::brush: Painter not active"); -
2348 return d->fakeState()->brush;
-
2349 } -
2350 return d->state->brush;
-
2351} -
2352void QPainter::setBackground(const QBrush &bg) -
2353{ -
2354 -
2355 -
2356 -
2357 -
2358 -
2359 QPainterPrivate * const d = d_func(); -
2360 if (!d->engine) {
-
2361 QMessageLogger("painting/qpainter.cpp", 4047, __PRETTY_FUNCTION__).warning("QPainter::setBackground: Painter not active"); -
2362 return;
-
2363 } -
2364 d->state->bgBrush = bg; -
2365 if (!d->extended)
-
2366 d->state->dirtyFlags |= QPaintEngine::DirtyBackground;
-
2367}
-
2368void QPainter::setFont(const QFont &font) -
2369{ -
2370 QPainterPrivate * const d = d_func(); -
2371 -
2372 -
2373 -
2374 -
2375 -
2376 -
2377 if (!d->engine) {
-
2378 QMessageLogger("painting/qpainter.cpp", 4078, __PRETTY_FUNCTION__).warning("QPainter::setFont: Painter not active"); -
2379 return;
-
2380 } -
2381 -
2382 d->state->font = QFont(font.resolve(d->state->deviceFont), device()); -
2383 if (!d->extended)
-
2384 d->state->dirtyFlags |= QPaintEngine::DirtyFont;
-
2385}
-
2386 -
2387 -
2388 -
2389 -
2390 -
2391 -
2392const QFont &QPainter::font() const -
2393{ -
2394 const QPainterPrivate * const d = d_func(); -
2395 if (!d->engine) {
-
2396 QMessageLogger("painting/qpainter.cpp", 4096, __PRETTY_FUNCTION__).warning("QPainter::font: Painter not active"); -
2397 return d->fakeState()->font;
-
2398 } -
2399 return d->state->font;
-
2400} -
2401void QPainter::drawRoundedRect(const QRectF &rect, qreal xRadius, qreal yRadius, Qt::SizeMode mode) -
2402{ -
2403 -
2404 -
2405 -
2406 -
2407 QPainterPrivate * const d = d_func(); -
2408 -
2409 if (!d->engine)
-
2410 return;
-
2411 -
2412 if (xRadius <= 0 || yRadius <= 0) {
-
2413 drawRect(rect); -
2414 return;
-
2415 } -
2416 -
2417 if (d->extended) {
-
2418 d->extended->drawRoundedRect(rect, xRadius, yRadius, mode); -
2419 return;
-
2420 } -
2421 -
2422 QPainterPath path; -
2423 path.addRoundedRect(rect, xRadius, yRadius, mode); -
2424 drawPath(path); -
2425}
-
2426void QPainter::drawRoundRect(const QRectF &r, int xRnd, int yRnd) -
2427{ -
2428 drawRoundedRect(r, xRnd, yRnd, Qt::RelativeSize); -
2429}
-
2430void QPainter::drawEllipse(const QRectF &r) -
2431{ -
2432 -
2433 -
2434 -
2435 -
2436 QPainterPrivate * const d = d_func(); -
2437 -
2438 if (!d->engine)
-
2439 return;
-
2440 -
2441 QRectF rect(r.normalized()); -
2442 -
2443 if (d->extended) {
-
2444 d->extended->drawEllipse(rect); -
2445 return;
-
2446 } -
2447 -
2448 d->updateState(d->state); -
2449 if (d->state->emulationSpecifier) {
-
2450 if (d->state->emulationSpecifier == QPaintEngine::PrimitiveTransform
-
2451 && d->state->matrix.type() == QTransform::TxTranslate) {
-
2452 rect.translate(QPointF(d->state->matrix.dx(), d->state->matrix.dy())); -
2453 } else {
-
2454 QPainterPath path; -
2455 path.addEllipse(rect); -
2456 d->draw_helper(path, QPainterPrivate::StrokeAndFillDraw); -
2457 return;
-
2458 } -
2459 } -
2460 -
2461 d->engine->drawEllipse(rect); -
2462}
-
2463void QPainter::drawEllipse(const QRect &r) -
2464{ -
2465 -
2466 -
2467 -
2468 -
2469 QPainterPrivate * const d = d_func(); -
2470 -
2471 if (!d->engine)
-
2472 return;
-
2473 -
2474 QRect rect(r.normalized()); -
2475 -
2476 if (d->extended) {
-
2477 d->extended->drawEllipse(rect); -
2478 return;
-
2479 } -
2480 -
2481 d->updateState(d->state); -
2482 -
2483 if (d->state->emulationSpecifier) {
-
2484 if (d->state->emulationSpecifier == QPaintEngine::PrimitiveTransform
-
2485 && d->state->matrix.type() == QTransform::TxTranslate) {
-
2486 rect.translate(QPoint(qRound(d->state->matrix.dx()), qRound(d->state->matrix.dy()))); -
2487 } else {
-
2488 QPainterPath path; -
2489 path.addEllipse(rect); -
2490 d->draw_helper(path, QPainterPrivate::StrokeAndFillDraw); -
2491 return;
-
2492 } -
2493 } -
2494 -
2495 d->engine->drawEllipse(rect); -
2496}
-
2497void QPainter::drawArc(const QRectF &r, int a, int alen) -
2498{ -
2499 -
2500 -
2501 -
2502 -
2503 -
2504 QPainterPrivate * const d = d_func(); -
2505 -
2506 if (!d->engine)
-
2507 return;
-
2508 -
2509 QRectF rect = r.normalized(); -
2510 -
2511 QPainterPath path; -
2512 path.arcMoveTo(rect, a/16.0); -
2513 path.arcTo(rect, a/16.0, alen/16.0); -
2514 strokePath(path, d->state->pen); -
2515}
-
2516void QPainter::drawPie(const QRectF &r, int a, int alen) -
2517{ -
2518 -
2519 -
2520 -
2521 -
2522 -
2523 QPainterPrivate * const d = d_func(); -
2524 -
2525 if (!d->engine)
-
2526 return;
-
2527 -
2528 if (a > (360*16)) {
-
2529 a = a % (360*16); -
2530 } else if (a < 0) {
-
2531 a = a % (360*16); -
2532 if (a < 0) a += (360*16);
-
2533 }
-
2534 -
2535 QRectF rect = r.normalized(); -
2536 -
2537 QPainterPath path; -
2538 path.moveTo(rect.center()); -
2539 path.arcTo(rect.x(), rect.y(), rect.width(), rect.height(), a/16.0, alen/16.0); -
2540 path.closeSubpath(); -
2541 drawPath(path); -
2542 -
2543}
-
2544void QPainter::drawChord(const QRectF &r, int a, int alen) -
2545{ -
2546 -
2547 -
2548 -
2549 -
2550 -
2551 QPainterPrivate * const d = d_func(); -
2552 -
2553 if (!d->engine)
-
2554 return;
-
2555 -
2556 QRectF rect = r.normalized(); -
2557 -
2558 QPainterPath path; -
2559 path.arcMoveTo(rect, a/16.0); -
2560 path.arcTo(rect, a/16.0, alen/16.0); -
2561 path.closeSubpath(); -
2562 drawPath(path); -
2563}
-
2564void QPainter::drawLines(const QLineF *lines, int lineCount) -
2565{ -
2566 -
2567 -
2568 -
2569 -
2570 -
2571 QPainterPrivate * const d = d_func(); -
2572 -
2573 if (!d->engine || lineCount < 1)
-
2574 return;
-
2575 -
2576 if (d->extended) {
-
2577 d->extended->drawLines(lines, lineCount); -
2578 return;
-
2579 } -
2580 -
2581 d->updateState(d->state); -
2582 -
2583 uint lineEmulation = line_emulation(d->state->emulationSpecifier); -
2584 -
2585 if (lineEmulation) {
-
2586 if (lineEmulation == QPaintEngine::PrimitiveTransform
-
2587 && d->state->matrix.type() == QTransform::TxTranslate) {
-
2588 for (int i = 0; i < lineCount; ++i) {
-
2589 QLineF line = lines[i]; -
2590 line.translate(d->state->matrix.dx(), d->state->matrix.dy()); -
2591 d->engine->drawLines(&line, 1); -
2592 }
-
2593 } else {
-
2594 QPainterPath linePath; -
2595 for (int i = 0; i < lineCount; ++i) {
-
2596 linePath.moveTo(lines[i].p1()); -
2597 linePath.lineTo(lines[i].p2()); -
2598 }
-
2599 d->draw_helper(linePath, QPainterPrivate::StrokeDraw); -
2600 }
-
2601 return;
-
2602 } -
2603 d->engine->drawLines(lines, lineCount); -
2604}
-
2605void QPainter::drawLines(const QLine *lines, int lineCount) -
2606{ -
2607 -
2608 -
2609 -
2610 -
2611 -
2612 QPainterPrivate * const d = d_func(); -
2613 -
2614 if (!d->engine || lineCount < 1)
-
2615 return;
-
2616 -
2617 if (d->extended) {
-
2618 d->extended->drawLines(lines, lineCount); -
2619 return;
-
2620 } -
2621 -
2622 d->updateState(d->state); -
2623 -
2624 uint lineEmulation = line_emulation(d->state->emulationSpecifier); -
2625 -
2626 if (lineEmulation) {
-
2627 if (lineEmulation == QPaintEngine::PrimitiveTransform
-
2628 && d->state->matrix.type() == QTransform::TxTranslate) {
-
2629 for (int i = 0; i < lineCount; ++i) {
-
2630 QLineF line = lines[i]; -
2631 line.translate(d->state->matrix.dx(), d->state->matrix.dy()); -
2632 d->engine->drawLines(&line, 1); -
2633 }
-
2634 } else {
-
2635 QPainterPath linePath; -
2636 for (int i = 0; i < lineCount; ++i) {
-
2637 linePath.moveTo(lines[i].p1()); -
2638 linePath.lineTo(lines[i].p2()); -
2639 }
-
2640 d->draw_helper(linePath, QPainterPrivate::StrokeDraw); -
2641 }
-
2642 return;
-
2643 } -
2644 d->engine->drawLines(lines, lineCount); -
2645}
-
2646void QPainter::drawLines(const QPointF *pointPairs, int lineCount) -
2647{ -
2648 qt_noop(); -
2649 -
2650 drawLines((QLineF*)pointPairs, lineCount); -
2651}
-
2652 -
2653 -
2654 -
2655 -
2656 -
2657 -
2658 -
2659void QPainter::drawLines(const QPoint *pointPairs, int lineCount) -
2660{ -
2661 qt_noop(); -
2662 -
2663 drawLines((QLine*)pointPairs, lineCount); -
2664}
-
2665void QPainter::drawPolyline(const QPointF *points, int pointCount) -
2666{ -
2667 -
2668 -
2669 -
2670 -
2671 QPainterPrivate * const d = d_func(); -
2672 -
2673 if (!d->engine || pointCount < 2)
-
2674 return;
-
2675 -
2676 if (d->extended) {
-
2677 d->extended->drawPolygon(points, pointCount, QPaintEngine::PolylineMode); -
2678 return;
-
2679 } -
2680 -
2681 d->updateState(d->state); -
2682 -
2683 uint lineEmulation = line_emulation(d->state->emulationSpecifier); -
2684 -
2685 if (lineEmulation) {
-
2686 -
2687 -
2688 -
2689 -
2690 QPainterPath polylinePath(points[0]); -
2691 for (int i=1; i<pointCount; ++i)
-
2692 polylinePath.lineTo(points[i]);
-
2693 d->draw_helper(polylinePath, QPainterPrivate::StrokeDraw); -
2694 -
2695 } else {
-
2696 d->engine->drawPolygon(points, pointCount, QPaintEngine::PolylineMode); -
2697 }
-
2698} -
2699 -
2700 -
2701 -
2702 -
2703 -
2704 -
2705 -
2706void QPainter::drawPolyline(const QPoint *points, int pointCount) -
2707{ -
2708 -
2709 -
2710 -
2711 -
2712 QPainterPrivate * const d = d_func(); -
2713 -
2714 if (!d->engine || pointCount < 2)
-
2715 return;
-
2716 -
2717 if (d->extended) {
-
2718 d->extended->drawPolygon(points, pointCount, QPaintEngine::PolylineMode); -
2719 return;
-
2720 } -
2721 -
2722 d->updateState(d->state); -
2723 -
2724 uint lineEmulation = line_emulation(d->state->emulationSpecifier); -
2725 -
2726 if (lineEmulation) {
-
2727 -
2728 -
2729 -
2730 -
2731 QPainterPath polylinePath(points[0]); -
2732 for (int i=1; i<pointCount; ++i)
-
2733 polylinePath.lineTo(points[i]);
-
2734 d->draw_helper(polylinePath, QPainterPrivate::StrokeDraw); -
2735 -
2736 } else {
-
2737 d->engine->drawPolygon(points, pointCount, QPaintEngine::PolylineMode); -
2738 }
-
2739} -
2740void QPainter::drawPolygon(const QPointF *points, int pointCount, Qt::FillRule fillRule) -
2741{ -
2742 -
2743 -
2744 -
2745 -
2746 -
2747 QPainterPrivate * const d = d_func(); -
2748 -
2749 if (!d->engine || pointCount < 2)
-
2750 return;
-
2751 -
2752 if (d->extended) {
-
2753 d->extended->drawPolygon(points, pointCount, QPaintEngine::PolygonDrawMode(fillRule)); -
2754 return;
-
2755 } -
2756 -
2757 d->updateState(d->state); -
2758 -
2759 uint emulationSpecifier = d->state->emulationSpecifier; -
2760 -
2761 if (emulationSpecifier) {
-
2762 QPainterPath polygonPath(points[0]); -
2763 for (int i=1; i<pointCount; ++i)
-
2764 polygonPath.lineTo(points[i]);
-
2765 polygonPath.closeSubpath(); -
2766 polygonPath.setFillRule(fillRule); -
2767 d->draw_helper(polygonPath); -
2768 return;
-
2769 } -
2770 -
2771 d->engine->drawPolygon(points, pointCount, QPaintEngine::PolygonDrawMode(fillRule)); -
2772}
-
2773 -
2774 -
2775 -
2776 -
2777 -
2778 -
2779void QPainter::drawPolygon(const QPoint *points, int pointCount, Qt::FillRule fillRule) -
2780{ -
2781 -
2782 -
2783 -
2784 -
2785 -
2786 QPainterPrivate * const d = d_func(); -
2787 -
2788 if (!d->engine || pointCount < 2)
-
2789 return;
-
2790 -
2791 if (d->extended) {
-
2792 d->extended->drawPolygon(points, pointCount, QPaintEngine::PolygonDrawMode(fillRule)); -
2793 return;
-
2794 } -
2795 -
2796 d->updateState(d->state); -
2797 -
2798 uint emulationSpecifier = d->state->emulationSpecifier; -
2799 -
2800 if (emulationSpecifier) {
-
2801 QPainterPath polygonPath(points[0]); -
2802 for (int i=1; i<pointCount; ++i)
-
2803 polygonPath.lineTo(points[i]);
-
2804 polygonPath.closeSubpath(); -
2805 polygonPath.setFillRule(fillRule); -
2806 d->draw_helper(polygonPath); -
2807 return;
-
2808 } -
2809 -
2810 d->engine->drawPolygon(points, pointCount, QPaintEngine::PolygonDrawMode(fillRule)); -
2811}
-
2812void QPainter::drawConvexPolygon(const QPoint *points, int pointCount) -
2813{ -
2814 -
2815 -
2816 -
2817 -
2818 -
2819 QPainterPrivate * const d = d_func(); -
2820 -
2821 if (!d->engine || pointCount < 2)
-
2822 return;
-
2823 -
2824 if (d->extended) {
-
2825 d->extended->drawPolygon(points, pointCount, QPaintEngine::ConvexMode); -
2826 return;
-
2827 } -
2828 -
2829 d->updateState(d->state); -
2830 -
2831 uint emulationSpecifier = d->state->emulationSpecifier; -
2832 -
2833 if (emulationSpecifier) {
-
2834 QPainterPath polygonPath(points[0]); -
2835 for (int i=1; i<pointCount; ++i)
-
2836 polygonPath.lineTo(points[i]);
-
2837 polygonPath.closeSubpath(); -
2838 polygonPath.setFillRule(Qt::WindingFill); -
2839 d->draw_helper(polygonPath); -
2840 return;
-
2841 } -
2842 -
2843 d->engine->drawPolygon(points, pointCount, QPaintEngine::ConvexMode); -
2844}
-
2845 -
2846void QPainter::drawConvexPolygon(const QPointF *points, int pointCount) -
2847{ -
2848 -
2849 -
2850 -
2851 -
2852 -
2853 QPainterPrivate * const d = d_func(); -
2854 -
2855 if (!d->engine || pointCount < 2)
-
2856 return;
-
2857 -
2858 if (d->extended) {
-
2859 d->extended->drawPolygon(points, pointCount, QPaintEngine::ConvexMode); -
2860 return;
-
2861 } -
2862 -
2863 d->updateState(d->state); -
2864 -
2865 uint emulationSpecifier = d->state->emulationSpecifier; -
2866 -
2867 if (emulationSpecifier) {
-
2868 QPainterPath polygonPath(points[0]); -
2869 for (int i=1; i<pointCount; ++i)
-
2870 polygonPath.lineTo(points[i]);
-
2871 polygonPath.closeSubpath(); -
2872 polygonPath.setFillRule(Qt::WindingFill); -
2873 d->draw_helper(polygonPath); -
2874 return;
-
2875 } -
2876 -
2877 d->engine->drawPolygon(points, pointCount, QPaintEngine::ConvexMode); -
2878}
-
2879 -
2880static inline QPointF roundInDeviceCoordinates(const QPointF &p, const QTransform &m) -
2881{ -
2882 return m.inverted().map(QPointF(m.map(p).toPoint()));
-
2883} -
2884void QPainter::drawPixmap(const QPointF &p, const QPixmap &pm) -
2885{ -
2886 -
2887 -
2888 -
2889 -
2890 -
2891 -
2892 -
2893 QPainterPrivate * const d = d_func(); -
2894 -
2895 if (!d->engine || pm.isNull())
-
2896 return;
-
2897 -
2898 -
2899 -
2900 -
2901 -
2902 if (d->extended) {
-
2903 d->extended->drawPixmap(p, pm); -
2904 return;
-
2905 } -
2906 -
2907 qreal x = p.x(); -
2908 qreal y = p.y(); -
2909 -
2910 int w = pm.width(); -
2911 int h = pm.height(); -
2912 -
2913 if (w <= 0)
-
2914 return;
-
2915 -
2916 -
2917 if (d->state->bgMode == Qt::OpaqueMode && pm.isQBitmap()) {
-
2918 fillRect(QRectF(x, y, w, h), d->state->bgBrush.color()); -
2919 }
-
2920 -
2921 d->updateState(d->state); -
2922 -
2923 if ((d->state->matrix.type() > QTransform::TxTranslate
-
2924 && !d->engine->hasFeature(QPaintEngine::PixmapTransform))
-
2925 || (!d->state->matrix.isAffine() && !d->engine->hasFeature(QPaintEngine::PerspectiveTransform))
-
2926 || (d->state->opacity != 1.0 && !d->engine->hasFeature(QPaintEngine::ConstantOpacity)))
-
2927 { -
2928 save(); -
2929 -
2930 -
2931 if (d->state->matrix.type() <= QTransform::TxScale) {
-
2932 const QPointF p = roundInDeviceCoordinates(QPointF(x, y), d->state->matrix); -
2933 x = p.x(); -
2934 y = p.y(); -
2935 }
-
2936 translate(x, y); -
2937 setBackgroundMode(Qt::TransparentMode); -
2938 setRenderHint(Antialiasing, renderHints() & SmoothPixmapTransform); -
2939 QBrush brush(d->state->pen.color(), pm); -
2940 setBrush(brush); -
2941 setPen(Qt::NoPen); -
2942 setBrushOrigin(QPointF(0, 0)); -
2943 -
2944 drawRect(pm.rect()); -
2945 restore(); -
2946 } else {
-
2947 if (!d->engine->hasFeature(QPaintEngine::PixmapTransform)) {
-
2948 x += d->state->matrix.dx(); -
2949 y += d->state->matrix.dy(); -
2950 }
-
2951 int scale = pm.devicePixelRatio(); -
2952 d->engine->drawPixmap(QRectF(x, y, w / scale, h / scale), pm, QRectF(0, 0, w, h)); -
2953 }
-
2954} -
2955 -
2956void QPainter::drawPixmap(const QRectF &r, const QPixmap &pm, const QRectF &sr) -
2957{ -
2958 QPainterPrivate * const d = d_func(); -
2959 if (!d->engine || pm.isNull())
-
2960 return;
-
2961 -
2962 -
2963 -
2964 -
2965 qreal x = r.x(); -
2966 qreal y = r.y(); -
2967 qreal w = r.width(); -
2968 qreal h = r.height(); -
2969 qreal sx = sr.x(); -
2970 qreal sy = sr.y(); -
2971 qreal sw = sr.width(); -
2972 qreal sh = sr.height(); -
2973 -
2974 -
2975 -
2976 -
2977 const qreal pmscale = pm.devicePixelRatio(); -
2978 -
2979 -
2980 if (sw <= 0)
-
2981 sw = pm.width() - sx;
-
2982 -
2983 if (sh <= 0)
-
2984 sh = pm.height() - sy;
-
2985 -
2986 if (w < 0)
-
2987 w = sw / pmscale;
-
2988 if (h < 0)
-
2989 h = sh / pmscale;
-
2990 -
2991 if (sx < 0) {
-
2992 qreal w_ratio = sx * w/sw; -
2993 x -= w_ratio; -
2994 w += w_ratio; -
2995 sw += sx; -
2996 sx = 0; -
2997 }
-
2998 -
2999 if (sy < 0) {
-
3000 qreal h_ratio = sy * h/sh; -
3001 y -= h_ratio; -
3002 h += h_ratio; -
3003 sh += sy; -
3004 sy = 0; -
3005 }
-
3006 -
3007 if (sw + sx > pm.width()) {
-
3008 qreal delta = sw - (pm.width() - sx); -
3009 qreal w_ratio = delta * w/sw; -
3010 sw -= delta; -
3011 w -= w_ratio; -
3012 }
-
3013 -
3014 if (sh + sy > pm.height()) {
-
3015 qreal delta = sh - (pm.height() - sy); -
3016 qreal h_ratio = delta * h/sh; -
3017 sh -= delta; -
3018 h -= h_ratio; -
3019 }
-
3020 -
3021 if (w == 0 || h == 0 || sw <= 0 || sh <= 0)
-
3022 return;
-
3023 -
3024 if (d->extended) {
-
3025 d->extended->drawPixmap(QRectF(x, y, w, h), pm, QRectF(sx, sy, sw, sh)); -
3026 return;
-
3027 } -
3028 -
3029 -
3030 if (d->state->bgMode == Qt::OpaqueMode && pm.isQBitmap())
-
3031 fillRect(QRectF(x, y, w, h), d->state->bgBrush.color());
-
3032 -
3033 d->updateState(d->state); -
3034 -
3035 if ((d->state->matrix.type() > QTransform::TxTranslate
-
3036 && !d->engine->hasFeature(QPaintEngine::PixmapTransform))
-
3037 || (!d->state->matrix.isAffine() && !d->engine->hasFeature(QPaintEngine::PerspectiveTransform))
-
3038 || (d->state->opacity != 1.0 && !d->engine->hasFeature(QPaintEngine::ConstantOpacity))
-
3039 || ((sw != w || sh != h) && !d->engine->hasFeature(QPaintEngine::PixmapTransform)))
-
3040 { -
3041 save(); -
3042 -
3043 -
3044 if (d->state->matrix.type() <= QTransform::TxScale) {
-
3045 const QPointF p = roundInDeviceCoordinates(QPointF(x, y), d->state->matrix); -
3046 x = p.x(); -
3047 y = p.y(); -
3048 }
-
3049 -
3050 if (d->state->matrix.type() <= QTransform::TxTranslate && sw == w && sh == h) {
-
3051 sx = qRound(sx); -
3052 sy = qRound(sy); -
3053 sw = qRound(sw); -
3054 sh = qRound(sh); -
3055 }
-
3056 -
3057 translate(x, y); -
3058 scale(w / sw, h / sh); -
3059 setBackgroundMode(Qt::TransparentMode); -
3060 setRenderHint(Antialiasing, renderHints() & SmoothPixmapTransform); -
3061 QBrush brush; -
3062 -
3063 if (sw == pm.width() && sh == pm.height())
-
3064 brush = QBrush(d->state->pen.color(), pm);
-
3065 else -
3066 brush = QBrush(d->state->pen.color(), pm.copy(sx, sy, sw, sh));
-
3067 -
3068 setBrush(brush); -
3069 setPen(Qt::NoPen); -
3070 -
3071 drawRect(QRectF(0, 0, sw, sh)); -
3072 restore(); -
3073 } else {
-
3074 if (!d->engine->hasFeature(QPaintEngine::PixmapTransform)) {
-
3075 x += d->state->matrix.dx(); -
3076 y += d->state->matrix.dy(); -
3077 }
-
3078 d->engine->drawPixmap(QRectF(x, y, w, h), pm, QRectF(sx, sy, sw, sh)); -
3079 }
-
3080} -
3081void QPainter::drawImage(const QPointF &p, const QImage &image) -
3082{ -
3083 QPainterPrivate * const d = d_func(); -
3084 -
3085 if (!d->engine || image.isNull())
-
3086 return;
-
3087 -
3088 if (d->extended) {
-
3089 d->extended->drawImage(p, image); -
3090 return;
-
3091 } -
3092 -
3093 qreal x = p.x(); -
3094 qreal y = p.y(); -
3095 -
3096 int w = image.width(); -
3097 int h = image.height(); -
3098 qreal scale = image.devicePixelRatio(); -
3099 -
3100 d->updateState(d->state); -
3101 -
3102 if (((d->state->matrix.type() > QTransform::TxTranslate)
-
3103 && !d->engine->hasFeature(QPaintEngine::PixmapTransform))
-
3104 || (!d->state->matrix.isAffine() && !d->engine->hasFeature(QPaintEngine::PerspectiveTransform))
-
3105 || (d->state->opacity != 1.0 && !d->engine->hasFeature(QPaintEngine::ConstantOpacity)))
-
3106 { -
3107 save(); -
3108 -
3109 -
3110 if (d->state->matrix.type() <= QTransform::TxScale) {
-
3111 const QPointF p = roundInDeviceCoordinates(QPointF(x, y), d->state->matrix); -
3112 x = p.x(); -
3113 y = p.y(); -
3114 }
-
3115 translate(x, y); -
3116 setBackgroundMode(Qt::TransparentMode); -
3117 setRenderHint(Antialiasing, renderHints() & SmoothPixmapTransform); -
3118 QBrush brush(image); -
3119 setBrush(brush); -
3120 setPen(Qt::NoPen); -
3121 setBrushOrigin(QPointF(0, 0)); -
3122 drawRect(QRect(QPoint(0, 0), image.size() / scale)); -
3123 restore(); -
3124 return;
-
3125 } -
3126 -
3127 if (d->state->matrix.type() == QTransform::TxTranslate
-
3128 && !d->engine->hasFeature(QPaintEngine::PixmapTransform)) {
-
3129 x += d->state->matrix.dx(); -
3130 y += d->state->matrix.dy(); -
3131 }
-
3132 -
3133 d->engine->drawImage(QRectF(x, y, w / scale, h / scale), image, QRectF(0, 0, w, h), Qt::AutoColor); -
3134}
-
3135 -
3136void QPainter::drawImage(const QRectF &targetRect, const QImage &image, const QRectF &sourceRect, -
3137 Qt::ImageConversionFlags flags) -
3138{ -
3139 QPainterPrivate * const d = d_func(); -
3140 -
3141 if (!d->engine || image.isNull())
-
3142 return;
-
3143 -
3144 qreal x = targetRect.x(); -
3145 qreal y = targetRect.y(); -
3146 qreal w = targetRect.width(); -
3147 qreal h = targetRect.height(); -
3148 qreal sx = sourceRect.x(); -
3149 qreal sy = sourceRect.y(); -
3150 qreal sw = sourceRect.width(); -
3151 qreal sh = sourceRect.height(); -
3152 qreal imageScale = image.devicePixelRatio(); -
3153 -
3154 -
3155 if (sw <= 0)
-
3156 sw = image.width() - sx;
-
3157 -
3158 if (sh <= 0)
-
3159 sh = image.height() - sy;
-
3160 -
3161 if (w < 0)
-
3162 w = sw / imageScale;
-
3163 if (h < 0)
-
3164 h = sh / imageScale;
-
3165 -
3166 if (sx < 0) {
-
3167 qreal w_ratio = sx * w/sw; -
3168 x -= w_ratio; -
3169 w += w_ratio; -
3170 sw += sx; -
3171 sx = 0; -
3172 }
-
3173 -
3174 if (sy < 0) {
-
3175 qreal h_ratio = sy * h/sh; -
3176 y -= h_ratio; -
3177 h += h_ratio; -
3178 sh += sy; -
3179 sy = 0; -
3180 }
-
3181 -
3182 if (sw + sx > image.width()) {
-
3183 qreal delta = sw - (image.width() - sx); -
3184 qreal w_ratio = delta * w/sw; -
3185 sw -= delta; -
3186 w -= w_ratio; -
3187 }
-
3188 -
3189 if (sh + sy > image.height()) {
-
3190 qreal delta = sh - (image.height() - sy); -
3191 qreal h_ratio = delta * h/sh; -
3192 sh -= delta; -
3193 h -= h_ratio; -
3194 }
-
3195 -
3196 if (w == 0 || h == 0 || sw <= 0 || sh <= 0)
-
3197 return;
-
3198 -
3199 if (d->extended) {
-
3200 d->extended->drawImage(QRectF(x, y, w, h), image, QRectF(sx, sy, sw, sh), flags); -
3201 return;
-
3202 } -
3203 -
3204 d->updateState(d->state); -
3205 -
3206 if (((d->state->matrix.type() > QTransform::TxTranslate || (sw != w || sh != h))
-
3207 && !d->engine->hasFeature(QPaintEngine::PixmapTransform))
-
3208 || (!d->state->matrix.isAffine() && !d->engine->hasFeature(QPaintEngine::PerspectiveTransform))
-
3209 || (d->state->opacity != 1.0 && !d->engine->hasFeature(QPaintEngine::ConstantOpacity)))
-
3210 { -
3211 save(); -
3212 -
3213 -
3214 if (d->state->matrix.type() <= QTransform::TxScale) {
-
3215 const QPointF p = roundInDeviceCoordinates(QPointF(x, y), d->state->matrix); -
3216 x = p.x(); -
3217 y = p.y(); -
3218 }
-
3219 -
3220 if (d->state->matrix.type() <= QTransform::TxTranslate && sw == w && sh == h) {
-
3221 sx = qRound(sx); -
3222 sy = qRound(sy); -
3223 sw = qRound(sw); -
3224 sh = qRound(sh); -
3225 }
-
3226 translate(x, y); -
3227 scale(w / sw, h / sh); -
3228 setBackgroundMode(Qt::TransparentMode); -
3229 setRenderHint(Antialiasing, renderHints() & SmoothPixmapTransform); -
3230 QBrush brush(image); -
3231 setBrush(brush); -
3232 setPen(Qt::NoPen); -
3233 setBrushOrigin(QPointF(-sx, -sy)); -
3234 -
3235 drawRect(QRectF(0, 0, sw, sh)); -
3236 restore(); -
3237 return;
-
3238 } -
3239 -
3240 if (d->state->matrix.type() == QTransform::TxTranslate
-
3241 && !d->engine->hasFeature(QPaintEngine::PixmapTransform)) {
-
3242 x += d->state->matrix.dx(); -
3243 y += d->state->matrix.dy(); -
3244 }
-
3245 -
3246 d->engine->drawImage(QRectF(x, y, w, h), image, QRectF(sx, sy, sw, sh), flags); -
3247}
-
3248void QPainter::drawGlyphRun(const QPointF &position, const QGlyphRun &glyphRun) -
3249{ -
3250 QPainterPrivate * const d = d_func(); -
3251 -
3252 QRawFont font = glyphRun.rawFont(); -
3253 if (!font.isValid())
-
3254 return;
-
3255 -
3256 QGlyphRunPrivate *glyphRun_d = QGlyphRunPrivate::get(glyphRun); -
3257 -
3258 const quint32 *glyphIndexes = glyphRun_d->glyphIndexData; -
3259 const QPointF *glyphPositions = glyphRun_d->glyphPositionData; -
3260 -
3261 int count = qMin(glyphRun_d->glyphIndexDataSize, glyphRun_d->glyphPositionDataSize); -
3262 QVarLengthArray<QFixedPoint, 128> fixedPointPositions(count); -
3263 -
3264 QRawFontPrivate *fontD = QRawFontPrivate::get(font); -
3265 bool supportsTransformations = d->extended
-
3266 ? d->extended->supportsTransformations(fontD->fontEngine, d->state->matrix) -
3267 : d->engine->type() == QPaintEngine::CoreGraphics || d->state->matrix.isAffine(); -
3268 -
3269 for (int i=0; i<count; ++i) {
-
3270 QPointF processedPosition = position + glyphPositions[i]; -
3271 if (!supportsTransformations)
-
3272 processedPosition = d->state->transform().map(processedPosition);
-
3273 fixedPointPositions[i] = QFixedPoint::fromPointF(processedPosition); -
3274 }
-
3275 -
3276 d->drawGlyphs(glyphIndexes, fixedPointPositions.data(), count, font, glyphRun.overline(), -
3277 glyphRun.underline(), glyphRun.strikeOut()); -
3278}
-
3279 -
3280void QPainterPrivate::drawGlyphs(const quint32 *glyphArray, QFixedPoint *positions, -
3281 int glyphCount, -
3282 const QRawFont &font, bool overline, bool underline, -
3283 bool strikeOut) -
3284{ -
3285 QPainter * const q = q_func(); -
3286 -
3287 updateState(state); -
3288 -
3289 QRawFontPrivate *fontD = QRawFontPrivate::get(font); -
3290 QFontEngine *fontEngine = fontD->fontEngine; -
3291 -
3292 QFixed leftMost; -
3293 QFixed rightMost; -
3294 QFixed baseLine; -
3295 for (int i=0; i<glyphCount; ++i) {
-
3296 glyph_metrics_t gm = fontEngine->boundingBox(glyphArray[i]); -
3297 if (i == 0 || leftMost > positions[i].x)
-
3298 leftMost = positions[i].x;
-
3299 -
3300 -
3301 -
3302 -
3303 if (i == 0 || baseLine < positions[i].y)
-
3304 baseLine = positions[i].y;
-
3305 -
3306 -
3307 if (i == 0 || rightMost < positions[i].x + gm.xoff)
-
3308 rightMost = positions[i].x + gm.xoff;
-
3309 }
-
3310 -
3311 QFixed width = rightMost - leftMost; -
3312 -
3313 if (extended != 0 && state->matrix.isAffine()) {
-
3314 QStaticTextItem staticTextItem; -
3315 staticTextItem.color = state->pen.color(); -
3316 staticTextItem.font = state->font; -
3317 staticTextItem.setFontEngine(fontEngine); -
3318 staticTextItem.numGlyphs = glyphCount; -
3319 staticTextItem.glyphs = reinterpret_cast<glyph_t *>(const_cast<glyph_t *>(glyphArray)); -
3320 staticTextItem.glyphPositions = positions; -
3321 -
3322 extended->drawStaticTextItem(&staticTextItem); -
3323 } else {
-
3324 QTextItemInt textItem; -
3325 textItem.fontEngine = fontEngine; -
3326 -
3327 QVarLengthArray<QFixed, 128> advances(glyphCount); -
3328 QVarLengthArray<QGlyphJustification, 128> glyphJustifications(glyphCount); -
3329 QVarLengthArray<HB_GlyphAttributes, 128> glyphAttributes(glyphCount); -
3330 memset(glyphAttributes.data(), 0, glyphAttributes.size() * sizeof(HB_GlyphAttributes)); -
3331 memset(advances.data(), 0, advances.size() * sizeof(QFixed)); -
3332 memset(glyphJustifications.data(), 0, glyphJustifications.size() * sizeof(QGlyphJustification)); -
3333 -
3334 textItem.glyphs.numGlyphs = glyphCount; -
3335 textItem.glyphs.glyphs = reinterpret_cast<HB_Glyph *>(const_cast<quint32 *>(glyphArray)); -
3336 textItem.glyphs.offsets = positions; -
3337 textItem.glyphs.advances_x = advances.data(); -
3338 textItem.glyphs.advances_y = advances.data(); -
3339 textItem.glyphs.justifications = glyphJustifications.data(); -
3340 textItem.glyphs.attributes = glyphAttributes.data(); -
3341 -
3342 engine->drawTextItem(QPointF(0, 0), textItem); -
3343 }
-
3344 -
3345 QTextItemInt::RenderFlags flags; -
3346 if (underline)
-
3347 flags |= QTextItemInt::Underline;
-
3348 if (overline)
-
3349 flags |= QTextItemInt::Overline;
-
3350 if (strikeOut)
-
3351 flags |= QTextItemInt::StrikeOut;
-
3352 -
3353 drawTextItemDecoration(q, QPointF(leftMost.toReal(), baseLine.toReal()), -
3354 fontEngine, -
3355 0, -
3356 (underline -
3357 ? QTextCharFormat::SingleUnderline -
3358 : QTextCharFormat::NoUnderline), -
3359 flags, width.toReal(), QTextCharFormat()); -
3360}
-
3361void QPainter::drawText(const QPointF &p, const QString &str) -
3362{ -
3363 drawText(p, str, 0, 0); -
3364}
-
3365void QPainter::drawStaticText(const QPointF &topLeftPosition, const QStaticText &staticText) -
3366{ -
3367 QPainterPrivate * const d = d_func(); -
3368 if (!d->engine || staticText.text().isEmpty() || pen().style() == Qt::NoPen)
-
3369 return;
-
3370 -
3371 QStaticTextPrivate *staticText_d = -
3372 const_cast<QStaticTextPrivate *>(QStaticTextPrivate::get(&staticText)); -
3373 -
3374 if (font() != staticText_d->font) {
-
3375 staticText_d->font = font(); -
3376 staticText_d->needsRelayout = true; -
3377 }
-
3378 -
3379 -
3380 -
3381 if (d->extended == 0 || !d->state->matrix.isAffine()) {
-
3382 staticText_d->paintText(topLeftPosition, this); -
3383 return;
-
3384 } -
3385 -
3386 QFontEngine *fe = staticText_d->font.d->engineForScript(QUnicodeTables::Common); -
3387 if (fe->type() == QFontEngine::Multi)
-
3388 fe = static_cast<QFontEngineMulti *>(fe)->engine(0);
-
3389 bool supportsTransformations = d->extended->supportsTransformations(fe, -
3390 d->state->matrix); -
3391 if (supportsTransformations && !staticText_d->untransformedCoordinates) {
-
3392 staticText_d->untransformedCoordinates = true; -
3393 staticText_d->needsRelayout = true; -
3394 } else if (!supportsTransformations && staticText_d->untransformedCoordinates) {
-
3395 staticText_d->untransformedCoordinates = false; -
3396 staticText_d->needsRelayout = true; -
3397 }
-
3398 -
3399 -
3400 -
3401 QPointF transformedPosition = topLeftPosition; -
3402 if (!staticText_d->untransformedCoordinates)
-
3403 transformedPosition = transformedPosition * d->state->matrix;
-
3404 QTransform oldMatrix; -
3405 -
3406 -
3407 -
3408 if (d->state->matrix.isTranslating() && !staticText_d->untransformedCoordinates) {
-
3409 qreal m11 = d->state->matrix.m11(); -
3410 qreal m12 = d->state->matrix.m12(); -
3411 qreal m13 = d->state->matrix.m13(); -
3412 qreal m21 = d->state->matrix.m21(); -
3413 qreal m22 = d->state->matrix.m22(); -
3414 qreal m23 = d->state->matrix.m23(); -
3415 qreal m33 = d->state->matrix.m33(); -
3416 -
3417 oldMatrix = d->state->matrix; -
3418 d->state->matrix.setMatrix(m11, m12, m13, -
3419 m21, m22, m23, -
3420 0.0, 0.0, m33); -
3421 }
-
3422 -
3423 -
3424 -
3425 bool staticTextNeedsReinit = staticText_d->needsRelayout; -
3426 if (!staticText_d->untransformedCoordinates && staticText_d->matrix != d->state->matrix) {
-
3427 staticText_d->matrix = d->state->matrix; -
3428 staticTextNeedsReinit = true; -
3429 }
-
3430 -
3431 -
3432 if (staticTextNeedsReinit)
-
3433 staticText_d->init();
-
3434 -
3435 if (transformedPosition != staticText_d->position) {
-
3436 QFixed fx = QFixed::fromReal(transformedPosition.x()); -
3437 QFixed fy = QFixed::fromReal(transformedPosition.y()); -
3438 QFixed oldX = QFixed::fromReal(staticText_d->position.x()); -
3439 QFixed oldY = QFixed::fromReal(staticText_d->position.y()); -
3440 for (int item=0; item<staticText_d->itemCount;++item) {
-
3441 QStaticTextItem *textItem = staticText_d->items + item; -
3442 for (int i=0; i<textItem->numGlyphs; ++i) {
-
3443 textItem->glyphPositions[i].x += fx - oldX; -
3444 textItem->glyphPositions[i].y += fy - oldY; -
3445 }
-
3446 textItem->userDataNeedsUpdate = true; -
3447 }
-
3448 -
3449 staticText_d->position = transformedPosition; -
3450 }
-
3451 -
3452 QPen oldPen = d->state->pen; -
3453 QColor currentColor = oldPen.color(); -
3454 for (int i=0; i<staticText_d->itemCount; ++i) {
-
3455 QStaticTextItem *item = staticText_d->items + i; -
3456 if (item->color.isValid() && currentColor != item->color) {
-
3457 setPen(item->color); -
3458 currentColor = item->color; -
3459 }
-
3460 d->extended->drawStaticTextItem(item); -
3461 -
3462 qt_draw_decoration_for_glyphs(this, item->glyphs, item->glyphPositions, -
3463 item->numGlyphs, item->fontEngine(), staticText_d->font, -
3464 QTextCharFormat()); -
3465 }
-
3466 if (currentColor != oldPen.color())
-
3467 setPen(oldPen);
-
3468 -
3469 if (!staticText_d->untransformedCoordinates && oldMatrix.isTranslating())
-
3470 d->state->matrix = oldMatrix;
-
3471}
-
3472 -
3473 -
3474 -
3475 -
3476void QPainter::drawText(const QPointF &p, const QString &str, int tf, int justificationPadding) -
3477{ -
3478 -
3479 -
3480 -
3481 -
3482 -
3483 QPainterPrivate * const d = d_func(); -
3484 -
3485 if (!d->engine || str.isEmpty() || pen().style() == Qt::NoPen)
-
3486 return;
-
3487 -
3488 if (tf & Qt::TextBypassShaping) {
-
3489 -
3490 int len = str.length(); -
3491 int numGlyphs = len; -
3492 QVarLengthGlyphLayoutArray glyphs(len); -
3493 QFontEngine *fontEngine = d->state->font.d->engineForScript(QUnicodeTables::Common); -
3494 if (!fontEngine->stringToCMap(str.data(), len, &glyphs, &numGlyphs, 0)) {
-
3495 glyphs.resize(numGlyphs); -
3496 if (!fontEngine->stringToCMap(str.data(), len, &glyphs, &numGlyphs, 0))
-
3497 qt_noop();
-
3498 }
-
3499 -
3500 QTextItemInt gf(glyphs, &d->state->font, str.data(), len, fontEngine); -
3501 drawTextItem(p, gf); -
3502 return;
-
3503 } -
3504 -
3505 QStackTextEngine engine(str, d->state->font); -
3506 engine.option.setTextDirection(d->state->layoutDirection); -
3507 if (tf & (Qt::TextForceLeftToRight|Qt::TextForceRightToLeft)) {
-
3508 engine.ignoreBidi = true; -
3509 engine.option.setTextDirection((tf & Qt::TextForceLeftToRight) ? Qt::LeftToRight : Qt::RightToLeft); -
3510 }
-
3511 engine.itemize(); -
3512 QScriptLine line; -
3513 line.length = str.length(); -
3514 engine.shapeLine(line); -
3515 -
3516 int nItems = engine.layoutData->items.size(); -
3517 QVarLengthArray<int> visualOrder(nItems); -
3518 QVarLengthArray<uchar> levels(nItems); -
3519 for (int i = 0; i < nItems; ++i)
-
3520 levels[i] = engine.layoutData->items[i].analysis.bidiLevel;
-
3521 QTextEngine::bidiReorder(nItems, levels.data(), visualOrder.data()); -
3522 -
3523 if (justificationPadding > 0) {
-
3524 engine.option.setAlignment(Qt::AlignJustify); -
3525 engine.forceJustification = true; -
3526 -
3527 line.width = justificationPadding; -
3528 engine.justify(line); -
3529 }
-
3530 QFixed x = QFixed::fromReal(p.x()); -
3531 -
3532 for (int i = 0; i < nItems; ++i) {
-
3533 int item = visualOrder[i]; -
3534 const QScriptItem &si = engine.layoutData->items.at(item); -
3535 if (si.analysis.flags >= QScriptAnalysis::TabOrObject) {
-
3536 x += si.width; -
3537 continue;
-
3538 } -
3539 QFont f = engine.font(si); -
3540 QTextItemInt gf(si, &f); -
3541 gf.glyphs = engine.shapedGlyphs(&si); -
3542 gf.chars = engine.layoutData->string.unicode() + si.position; -
3543 gf.num_chars = engine.length(item); -
3544 if (engine.forceJustification) {
-
3545 for (int j=0; j<gf.glyphs.numGlyphs; ++j)
-
3546 gf.width += gf.glyphs.effectiveAdvance(j);
-
3547 } else {
-
3548 gf.width = si.width; -
3549 }
-
3550 gf.logClusters = engine.logClusters(&si); -
3551 -
3552 drawTextItem(QPointF(x.toReal(), p.y()), gf); -
3553 -
3554 x += gf.width; -
3555 }
-
3556}
-
3557 -
3558void QPainter::drawText(const QRect &r, int flags, const QString &str, QRect *br) -
3559{ -
3560 -
3561 -
3562 -
3563 -
3564 -
3565 -
3566 QPainterPrivate * const d = d_func(); -
3567 -
3568 if (!d->engine || str.length() == 0 || pen().style() == Qt::NoPen)
-
3569 return;
-
3570 -
3571 if (!d->extended)
-
3572 d->updateState(d->state);
-
3573 -
3574 QRectF bounds; -
3575 qt_format_text(d->state->font, r, flags, 0, str, br ? &bounds : 0, 0, 0, 0, this); -
3576 if (br)
-
3577 *br = bounds.toAlignedRect();
-
3578}
-
3579void QPainter::drawText(const QRectF &r, int flags, const QString &str, QRectF *br) -
3580{ -
3581 -
3582 -
3583 -
3584 -
3585 -
3586 -
3587 QPainterPrivate * const d = d_func(); -
3588 -
3589 if (!d->engine || str.length() == 0 || pen().style() == Qt::NoPen)
-
3590 return;
-
3591 -
3592 if (!d->extended)
-
3593 d->updateState(d->state);
-
3594 -
3595 qt_format_text(d->state->font, r, flags, 0, str, br, 0, 0, 0, this); -
3596}
-
3597void QPainter::drawText(const QRectF &r, const QString &text, const QTextOption &o) -
3598{ -
3599 -
3600 -
3601 -
3602 -
3603 -
3604 -
3605 QPainterPrivate * const d = d_func(); -
3606 -
3607 if (!d->engine || text.length() == 0 || pen().style() == Qt::NoPen)
-
3608 return;
-
3609 -
3610 if (!d->extended)
-
3611 d->updateState(d->state);
-
3612 -
3613 qt_format_text(d->state->font, r, 0, &o, text, 0, 0, 0, 0, this); -
3614}
-
3615static QPixmap generateWavyPixmap(qreal maxRadius, const QPen &pen) -
3616{ -
3617 const qreal radiusBase = qMax(qreal(1), maxRadius); -
3618 -
3619 QString key = QLatin1String("WaveUnderline-") -
3620 % pen.color().name() -
3621 % HexString<qreal>(radiusBase); -
3622 -
3623 QPixmap pixmap; -
3624 if (QPixmapCache::find(key, pixmap))
-
3625 return pixmap;
-
3626 -
3627 const qreal halfPeriod = qMax(qreal(2), qreal(radiusBase * 1.61803399)); -
3628 const int width = qCeil(100 / (2 * halfPeriod)) * (2 * halfPeriod); -
3629 const int radius = qFloor(radiusBase); -
3630 -
3631 QPainterPath path; -
3632 -
3633 qreal xs = 0; -
3634 qreal ys = radius; -
3635 -
3636 while (xs < width) {
-
3637 xs += halfPeriod; -
3638 ys = -ys; -
3639 path.quadTo(xs - halfPeriod / 2, ys, xs, 0); -
3640 }
-
3641 -
3642 pixmap = QPixmap(width, radius * 2); -
3643 pixmap.fill(Qt::transparent); -
3644 { -
3645 QPen wavePen = pen; -
3646 wavePen.setCapStyle(Qt::SquareCap); -
3647 -
3648 -
3649 -
3650 const qreal maxPenWidth = .8 * radius; -
3651 if (wavePen.widthF() > maxPenWidth)
-
3652 wavePen.setWidth(maxPenWidth);
-
3653 -
3654 QPainter imgPainter(&pixmap); -
3655 imgPainter.setPen(wavePen); -
3656 imgPainter.setRenderHint(QPainter::Antialiasing); -
3657 imgPainter.translate(0, radius); -
3658 imgPainter.drawPath(path); -
3659 } -
3660 -
3661 QPixmapCache::insert(key, pixmap); -
3662 -
3663 return pixmap;
-
3664} -
3665 -
3666static void drawTextItemDecoration(QPainter *painter, const QPointF &pos, const QFontEngine *fe, QTextEngine *textEngine, -
3667 QTextCharFormat::UnderlineStyle underlineStyle, -
3668 QTextItem::RenderFlags flags, qreal width, -
3669 const QTextCharFormat &charFormat) -
3670{ -
3671 if (underlineStyle == QTextCharFormat::NoUnderline
-
3672 && !(flags & (QTextItem::StrikeOut | QTextItem::Overline)))
-
3673 return;
-
3674 -
3675 const QPen oldPen = painter->pen(); -
3676 const QBrush oldBrush = painter->brush(); -
3677 painter->setBrush(Qt::NoBrush); -
3678 QPen pen = oldPen; -
3679 pen.setStyle(Qt::SolidLine); -
3680 pen.setWidthF(fe->lineThickness().toReal()); -
3681 pen.setCapStyle(Qt::FlatCap); -
3682 -
3683 QLineF line(pos.x(), pos.y(), pos.x() + qFloor(width), pos.y()); -
3684 -
3685 bool wasCompatiblePainting = painter->renderHints() -
3686 & QPainter::Qt4CompatiblePainting; -
3687 -
3688 if (wasCompatiblePainting)
-
3689 painter->setRenderHint(QPainter::Qt4CompatiblePainting, false);
-
3690 -
3691 const qreal underlineOffset = fe->underlinePosition().toReal(); -
3692 -
3693 -
3694 const qreal underlinePos = pos.y() + qCeil(underlineOffset) + 0.5; -
3695 -
3696 if (underlineStyle == QTextCharFormat::SpellCheckUnderline) {
-
3697 QPlatformTheme *theme = QGuiApplicationPrivate::platformTheme(); -
3698 if (theme)
-
3699 underlineStyle = QTextCharFormat::UnderlineStyle(theme->themeHint(QPlatformTheme::SpellCheckUnderlineStyle).toInt());
-
3700 }
-
3701 -
3702 if (underlineStyle == QTextCharFormat::WaveUnderline) {
-
3703 painter->save(); -
3704 painter->translate(0, pos.y() + 1); -
3705 -
3706 QColor uc = charFormat.underlineColor(); -
3707 if (uc.isValid())
-
3708 pen.setColor(uc);
-
3709 -
3710 -
3711 const QPixmap wave = generateWavyPixmap(qMax(underlineOffset, pen.widthF()), pen); -
3712 const int descent = (int) fe->descent().toReal(); -
3713 -
3714 painter->setBrushOrigin(painter->brushOrigin().x(), 0); -
3715 painter->fillRect(pos.x(), 0, qCeil(width), qMin(wave.height(), descent), wave); -
3716 painter->restore(); -
3717 } else if (underlineStyle != QTextCharFormat::NoUnderline) {
-
3718 QColor uc = charFormat.underlineColor(); -
3719 if (uc.isValid())
-
3720 pen.setColor(uc);
-
3721 -
3722 pen.setStyle((Qt::PenStyle)(underlineStyle)); -
3723 painter->setPen(pen); -
3724 QLineF underline(line.x1(), underlinePos, line.x2(), underlinePos); -
3725 if (textEngine)
-
3726 textEngine->addUnderline(painter, underline);
-
3727 else -
3728 painter->drawLine(underline);
-
3729 } -
3730 -
3731 pen.setStyle(Qt::SolidLine); -
3732 pen.setColor(oldPen.color()); -
3733 -
3734 if (flags & QTextItem::StrikeOut) {
-
3735 QLineF strikeOutLine = line; -
3736 strikeOutLine.translate(0., - fe->ascent().toReal() / 3.); -
3737 painter->setPen(pen); -
3738 if (textEngine)
-
3739 textEngine->addStrikeOut(painter, strikeOutLine);
-
3740 else -
3741 painter->drawLine(strikeOutLine);
-
3742 } -
3743 -
3744 if (flags & QTextItem::Overline) {
-
3745 QLineF overline = line; -
3746 overline.translate(0., - fe->ascent().toReal()); -
3747 painter->setPen(pen); -
3748 if (textEngine)
-
3749 textEngine->addOverline(painter, overline);
-
3750 else -
3751 painter->drawLine(overline);
-
3752 } -
3753 -
3754 painter->setPen(oldPen); -
3755 painter->setBrush(oldBrush); -
3756 -
3757 if (wasCompatiblePainting)
-
3758 painter->setRenderHint(QPainter::Qt4CompatiblePainting);
-
3759}
-
3760 -
3761__attribute__((visibility("default"))) void qt_draw_decoration_for_glyphs(QPainter *painter, const glyph_t *glyphArray, -
3762 const QFixedPoint *positions, int glyphCount, -
3763 QFontEngine *fontEngine, const QFont &font, -
3764 const QTextCharFormat &charFormat) -
3765{ -
3766 if (!(font.underline() || font.strikeOut() || font.overline()))
-
3767 return;
-
3768 -
3769 QFixed leftMost; -
3770 QFixed rightMost; -
3771 QFixed baseLine; -
3772 for (int i=0; i<glyphCount; ++i) {
-
3773 glyph_metrics_t gm = fontEngine->boundingBox(glyphArray[i]); -
3774 if (i == 0 || leftMost > positions[i].x)
-
3775 leftMost = positions[i].x;
-
3776 -
3777 -
3778 -
3779 -
3780 if (i == 0 || baseLine < positions[i].y)
-
3781 baseLine = positions[i].y;
-
3782 -
3783 -
3784 if (i == 0 || rightMost < positions[i].x + gm.xoff)
-
3785 rightMost = positions[i].x + gm.xoff;
-
3786 }
-
3787 -
3788 QFixed width = rightMost - leftMost; -
3789 QTextItem::RenderFlags flags = 0; -
3790 -
3791 if (font.underline())
-
3792 flags |= QTextItem::Underline;
-
3793 if (font.overline())
-
3794 flags |= QTextItem::Overline;
-
3795 if (font.strikeOut())
-
3796 flags |= QTextItem::StrikeOut;
-
3797 -
3798 drawTextItemDecoration(painter, QPointF(leftMost.toReal(), baseLine.toReal()), -
3799 fontEngine, -
3800 0, -
3801 font.underline() ? QTextCharFormat::SingleUnderline -
3802 : QTextCharFormat::NoUnderline, flags, -
3803 width.toReal(), charFormat); -
3804}
-
3805 -
3806void QPainter::drawTextItem(const QPointF &p, const QTextItem &ti) -
3807{ -
3808 QPainterPrivate * const d = d_func(); -
3809 -
3810 d->drawTextItem(p, ti, static_cast<QTextEngine *>(0)); -
3811}
-
3812 -
3813void QPainterPrivate::drawTextItem(const QPointF &p, const QTextItem &_ti, QTextEngine *textEngine) -
3814{ -
3815 -
3816 -
3817 -
3818 -
3819 -
3820 -
3821 QPainter * const q = q_func(); -
3822 -
3823 if (!engine)
-
3824 return;
-
3825 -
3826 -
3827 -
3828 -
3829 -
3830 -
3831 -
3832 QTextItemInt &ti = const_cast<QTextItemInt &>(static_cast<const QTextItemInt &>(_ti)); -
3833 -
3834 if (!extended && state->bgMode == Qt::OpaqueMode) {
-
3835 QRectF rect(p.x(), p.y() - ti.ascent.toReal(), ti.width.toReal(), (ti.ascent + ti.descent + 1).toReal()); -
3836 q->fillRect(rect, state->bgBrush); -
3837 }
-
3838 -
3839 if (q->pen().style() == Qt::NoPen)
-
3840 return;
-
3841 -
3842 const QPainter::RenderHints oldRenderHints = state->renderHints; -
3843 if (!state->renderHints & QPainter::Antialiasing && state->matrix.type() >= QTransform::TxScale) {
-
3844 -
3845 -
3846 -
3847 bool aa = true; -
3848 const QTransform &m = state->matrix; -
3849 if (state->matrix.type() < QTransform::TxShear) {
-
3850 bool isPlain90DegreeRotation = -
3851 (qFuzzyIsNull(m.m11())
-
3852 && qFuzzyIsNull(m.m12() - qreal(1))
-
3853 && qFuzzyIsNull(m.m21() + qreal(1))
-
3854 && qFuzzyIsNull(m.m22())
-
3855 ) -
3856 || -
3857 (qFuzzyIsNull(m.m11() + qreal(1))
-
3858 && qFuzzyIsNull(m.m12())
-
3859 && qFuzzyIsNull(m.m21())
-
3860 && qFuzzyIsNull(m.m22() + qreal(1))
-
3861 ) -
3862 || -
3863 (qFuzzyIsNull(m.m11())
-
3864 && qFuzzyIsNull(m.m12() + qreal(1))
-
3865 && qFuzzyIsNull(m.m21() - qreal(1))
-
3866 && qFuzzyIsNull(m.m22())
-
3867 ) -
3868 ; -
3869 aa = !isPlain90DegreeRotation; -
3870 }
-
3871 if (aa)
-
3872 q->setRenderHint(QPainter::Antialiasing, true);
-
3873 }
-
3874 -
3875 if (!extended)
-
3876 updateState(state);
-
3877 -
3878 if (!ti.glyphs.numGlyphs) {
-
3879 -
3880 } else if (ti.fontEngine->type() == QFontEngine::Multi) {
-
3881 QFontEngineMulti *multi = static_cast<QFontEngineMulti *>(ti.fontEngine); -
3882 -
3883 const QGlyphLayout &glyphs = ti.glyphs; -
3884 int which = glyphs.glyphs[0] >> 24; -
3885 -
3886 qreal x = p.x(); -
3887 qreal y = p.y(); -
3888 -
3889 bool rtl = ti.flags & QTextItem::RightToLeft; -
3890 if (rtl)
-
3891 x += ti.width.toReal();
-
3892 -
3893 int start = 0; -
3894 int end, i; -
3895 for (end = 0; end < ti.glyphs.numGlyphs; ++end) {
-
3896 const int e = glyphs.glyphs[end] >> 24; -
3897 if (e == which)
-
3898 continue;
-
3899 -
3900 -
3901 QTextItemInt ti2 = ti.midItem(multi->engine(which), start, end - start); -
3902 ti2.width = 0; -
3903 -
3904 for (i = start; i < end; ++i) {
-
3905 glyphs.glyphs[i] = glyphs.glyphs[i] & 0xffffff; -
3906 ti2.width += ti.glyphs.effectiveAdvance(i); -
3907 }
-
3908 -
3909 if (rtl)
-
3910 x -= ti2.width.toReal();
-
3911 -
3912 engine->drawTextItem(QPointF(x, y), ti2); -
3913 -
3914 if (!rtl)
-
3915 x += ti2.width.toReal();
-
3916 -
3917 -
3918 const int hi = which << 24; -
3919 for (i = start; i < end; ++i) {
-
3920 glyphs.glyphs[i] = hi | glyphs.glyphs[i]; -
3921 }
-
3922 -
3923 -
3924 start = end; -
3925 which = e; -
3926 }
-
3927 -
3928 QTextItemInt ti2 = ti.midItem(multi->engine(which), start, end - start); -
3929 ti2.width = 0; -
3930 -
3931 for (i = start; i < end; ++i) {
-
3932 glyphs.glyphs[i] = glyphs.glyphs[i] & 0xffffff; -
3933 ti2.width += ti.glyphs.effectiveAdvance(i); -
3934 }
-
3935 -
3936 if (rtl)
-
3937 x -= ti2.width.toReal();
-
3938 -
3939 if (extended)
-
3940 extended->drawTextItem(QPointF(x, y), ti2);
-
3941 else -
3942 engine->drawTextItem(QPointF(x,y), ti2);
-
3943 -
3944 -
3945 const int hi = which << 24; -
3946 for (i = start; i < end; ++i)
-
3947 glyphs.glyphs[i] = hi | glyphs.glyphs[i];
-
3948 -
3949 } else {
-
3950 if (extended)
-
3951 extended->drawTextItem(p, ti);
-
3952 else -
3953 engine->drawTextItem(p, ti);
-
3954 } -
3955 drawTextItemDecoration(q, p, ti.fontEngine, textEngine, ti.underlineStyle, -
3956 ti.flags, ti.width.toReal(), ti.charFormat); -
3957 -
3958 if (state->renderHints != oldRenderHints) {
-
3959 state->renderHints = oldRenderHints; -
3960 if (extended)
-
3961 extended->renderHintsChanged();
-
3962 else -
3963 state->dirtyFlags |= QPaintEngine::DirtyHints;
-
3964 } -
3965}
-
3966QRect QPainter::boundingRect(const QRect &rect, int flags, const QString &str) -
3967{ -
3968 if (str.isEmpty())
-
3969 return QRect(rect.x(),rect.y(), 0,0);
-
3970 QRect brect; -
3971 drawText(rect, flags | Qt::TextDontPrint, str, &brect); -
3972 return brect;
-
3973} -
3974 -
3975 -
3976 -
3977QRectF QPainter::boundingRect(const QRectF &rect, int flags, const QString &str) -
3978{ -
3979 if (str.isEmpty())
-
3980 return QRectF(rect.x(),rect.y(), 0,0);
-
3981 QRectF brect; -
3982 drawText(rect, flags | Qt::TextDontPrint, str, &brect); -
3983 return brect;
-
3984} -
3985QRectF QPainter::boundingRect(const QRectF &r, const QString &text, const QTextOption &o) -
3986{ -
3987 QPainterPrivate * const d = d_func(); -
3988 -
3989 if (!d->engine || text.length() == 0)
-
3990 return QRectF(r.x(),r.y(), 0,0);
-
3991 -
3992 QRectF br; -
3993 qt_format_text(d->state->font, r, Qt::TextDontPrint, &o, text, &br, 0, 0, 0, this); -
3994 return br;
-
3995} -
3996void QPainter::drawTiledPixmap(const QRectF &r, const QPixmap &pixmap, const QPointF &sp) -
3997{ -
3998 QPainterPrivate * const d = d_func(); -
3999 if (!d->engine || pixmap.isNull() || r.isEmpty())
-
4000 return;
-
4001 -
4002 -
4003 -
4004 -
4005 -
4006 qreal sw = pixmap.width(); -
4007 qreal sh = pixmap.height(); -
4008 qreal sx = sp.x(); -
4009 qreal sy = sp.y(); -
4010 if (sx < 0)
-
4011 sx = qRound(sw) - qRound(-sx) % qRound(sw);
-
4012 else -
4013 sx = qRound(sx) % qRound(sw);
-
4014 if (sy < 0)
-
4015 sy = qRound(sh) - -qRound(sy) % qRound(sh);
-
4016 else -
4017 sy = qRound(sy) % qRound(sh);
-
4018 -
4019 -
4020 if (d->extended) {
-
4021 d->extended->drawTiledPixmap(r, pixmap, QPointF(sx, sy)); -
4022 return;
-
4023 } -
4024 -
4025 if (d->state->bgMode == Qt::OpaqueMode && pixmap.isQBitmap())
-
4026 fillRect(r, d->state->bgBrush);
-
4027 -
4028 d->updateState(d->state); -
4029 if ((d->state->matrix.type() > QTransform::TxTranslate
-
4030 && !d->engine->hasFeature(QPaintEngine::PixmapTransform))
-
4031 || (d->state->opacity != 1.0 && !d->engine->hasFeature(QPaintEngine::ConstantOpacity)))
-
4032 { -
4033 save(); -
4034 setBackgroundMode(Qt::TransparentMode); -
4035 setRenderHint(Antialiasing, renderHints() & SmoothPixmapTransform); -
4036 setBrush(QBrush(d->state->pen.color(), pixmap)); -
4037 setPen(Qt::NoPen); -
4038 -
4039 -
4040 -
4041 if (d->state->matrix.type() <= QTransform::TxScale) {
-
4042 const QPointF p = roundInDeviceCoordinates(r.topLeft(), d->state->matrix); -
4043 -
4044 if (d->state->matrix.type() <= QTransform::TxTranslate) {
-
4045 sx = qRound(sx); -
4046 sy = qRound(sy); -
4047 }
-
4048 -
4049 setBrushOrigin(QPointF(r.x()-sx, r.y()-sy)); -
4050 drawRect(QRectF(p, r.size())); -
4051 } else {
-
4052 setBrushOrigin(QPointF(r.x()-sx, r.y()-sy)); -
4053 drawRect(r); -
4054 }
-
4055 restore(); -
4056 return;
-
4057 } -
4058 -
4059 qreal x = r.x(); -
4060 qreal y = r.y(); -
4061 if (d->state->matrix.type() == QTransform::TxTranslate
-
4062 && !d->engine->hasFeature(QPaintEngine::PixmapTransform)) {
-
4063 x += d->state->matrix.dx(); -
4064 y += d->state->matrix.dy(); -
4065 }
-
4066 -
4067 d->engine->drawTiledPixmap(QRectF(x, y, r.width(), r.height()), pixmap, QPointF(sx, sy)); -
4068}
-
4069void QPainter::drawPicture(const QPointF &p, const QPicture &picture) -
4070{ -
4071 QPainterPrivate * const d = d_func(); -
4072 -
4073 if (!d->engine)
-
4074 return;
-
4075 -
4076 if (!d->extended)
-
4077 d->updateState(d->state);
-
4078 -
4079 save(); -
4080 translate(p); -
4081 const_cast<QPicture *>(&picture)->play(this); -
4082 restore(); -
4083}
-
4084void QPainter::eraseRect(const QRectF &r) -
4085{ -
4086 QPainterPrivate * const d = d_func(); -
4087 -
4088 fillRect(r, d->state->bgBrush); -
4089}
-
4090 -
4091static inline bool needsResolving(const QBrush &brush) -
4092{ -
4093 Qt::BrushStyle s = brush.style(); -
4094 return ((s == Qt::LinearGradientPattern || s == Qt::RadialGradientPattern || -
4095 s == Qt::ConicalGradientPattern) && -
4096 brush.gradient()->coordinateMode() == QGradient::ObjectBoundingMode);
-
4097} -
4098void QPainter::fillRect(const QRectF &r, const QBrush &brush) -
4099{ -
4100 QPainterPrivate * const d = d_func(); -
4101 -
4102 if (!d->engine)
-
4103 return;
-
4104 -
4105 if (d->extended) {
-
4106 const QGradient *g = brush.gradient(); -
4107 if (!g || g->coordinateMode() == QGradient::LogicalMode) {
-
4108 d->extended->fillRect(r, brush); -
4109 return;
-
4110 } -
4111 }
-
4112 -
4113 QPen oldPen = pen(); -
4114 QBrush oldBrush = this->brush(); -
4115 setPen(Qt::NoPen); -
4116 if (brush.style() == Qt::SolidPattern) {
-
4117 d->colorBrush.setStyle(Qt::SolidPattern); -
4118 d->colorBrush.setColor(brush.color()); -
4119 setBrush(d->colorBrush); -
4120 } else {
-
4121 setBrush(brush); -
4122 }
-
4123 -
4124 drawRect(r); -
4125 setBrush(oldBrush); -
4126 setPen(oldPen); -
4127}
-
4128void QPainter::fillRect(const QRect &r, const QBrush &brush) -
4129{ -
4130 QPainterPrivate * const d = d_func(); -
4131 -
4132 if (!d->engine)
-
4133 return;
-
4134 -
4135 if (d->extended) {
-
4136 const QGradient *g = brush.gradient(); -
4137 if (!g || g->coordinateMode() == QGradient::LogicalMode) {
-
4138 d->extended->fillRect(r, brush); -
4139 return;
-
4140 } -
4141 }
-
4142 -
4143 QPen oldPen = pen(); -
4144 QBrush oldBrush = this->brush(); -
4145 setPen(Qt::NoPen); -
4146 if (brush.style() == Qt::SolidPattern) {
-
4147 d->colorBrush.setStyle(Qt::SolidPattern); -
4148 d->colorBrush.setColor(brush.color()); -
4149 setBrush(d->colorBrush); -
4150 } else {
-
4151 setBrush(brush); -
4152 }
-
4153 -
4154 drawRect(r); -
4155 setBrush(oldBrush); -
4156 setPen(oldPen); -
4157}
-
4158void QPainter::fillRect(const QRect &r, const QColor &color) -
4159{ -
4160 QPainterPrivate * const d = d_func(); -
4161 -
4162 if (!d->engine)
-
4163 return;
-
4164 -
4165 if (d->extended) {
-
4166 d->extended->fillRect(r, color); -
4167 return;
-
4168 } -
4169 -
4170 fillRect(r, QBrush(color)); -
4171}
-
4172void QPainter::fillRect(const QRectF &r, const QColor &color) -
4173{ -
4174 QPainterPrivate * const d = d_func(); -
4175 -
4176 if (!d->engine)
-
4177 return;
-
4178 -
4179 if (d->extended) {
-
4180 d->extended->fillRect(r, color); -
4181 return;
-
4182 } -
4183 -
4184 fillRect(r, QBrush(color)); -
4185}
-
4186void QPainter::setRenderHint(RenderHint hint, bool on) -
4187{ -
4188 setRenderHints(hint, on); -
4189}
-
4190void QPainter::setRenderHints(RenderHints hints, bool on) -
4191{ -
4192 QPainterPrivate * const d = d_func(); -
4193 -
4194 if (!d->engine) {
-
4195 QMessageLogger("painting/qpainter.cpp", 7054, __PRETTY_FUNCTION__).warning("QPainter::setRenderHint: Painter must be active to set rendering hints"); -
4196 return;
-
4197 } -
4198 -
4199 if (on)
-
4200 d->state->renderHints |= hints;
-
4201 else -
4202 d->state->renderHints &= ~hints;
-
4203 -
4204 if (d->extended)
-
4205 d->extended->renderHintsChanged();
-
4206 else -
4207 d->state->dirtyFlags |= QPaintEngine::DirtyHints;
-
4208} -
4209 -
4210 -
4211 -
4212 -
4213 -
4214 -
4215 -
4216QPainter::RenderHints QPainter::renderHints() const -
4217{ -
4218 const QPainterPrivate * const d = d_func(); -
4219 -
4220 if (!d->engine)
-
4221 return 0;
-
4222 -
4223 return d->state->renderHints;
-
4224} -
4225bool QPainter::viewTransformEnabled() const -
4226{ -
4227 const QPainterPrivate * const d = d_func(); -
4228 if (!d->engine) {
-
4229 QMessageLogger("painting/qpainter.cpp", 7105, __PRETTY_FUNCTION__).warning("QPainter::viewTransformEnabled: Painter not active"); -
4230 return false;
-
4231 } -
4232 return d->state->VxF;
-
4233} -
4234void QPainter::setWindow(const QRect &r) -
4235{ -
4236 -
4237 -
4238 -
4239 -
4240 -
4241 QPainterPrivate * const d = d_func(); -
4242 -
4243 if (!d->engine) {
-
4244 QMessageLogger("painting/qpainter.cpp", 7146, __PRETTY_FUNCTION__).warning("QPainter::setWindow: Painter not active"); -
4245 return;
-
4246 } -
4247 -
4248 d->state->wx = r.x(); -
4249 d->state->wy = r.y(); -
4250 d->state->ww = r.width(); -
4251 d->state->wh = r.height(); -
4252 -
4253 d->state->VxF = true; -
4254 d->updateMatrix(); -
4255}
-
4256 -
4257 -
4258 -
4259 -
4260 -
4261 -
4262 -
4263QRect QPainter::window() const -
4264{ -
4265 const QPainterPrivate * const d = d_func(); -
4266 if (!d->engine) {
-
4267 QMessageLogger("painting/qpainter.cpp", 7169, __PRETTY_FUNCTION__).warning("QPainter::window: Painter not active"); -
4268 return QRect();
-
4269 } -
4270 return QRect(d->state->wx, d->state->wy, d->state->ww, d->state->wh);
-
4271} -
4272void QPainter::setViewport(const QRect &r) -
4273{ -
4274 -
4275 -
4276 -
4277 -
4278 -
4279 QPainterPrivate * const d = d_func(); -
4280 -
4281 if (!d->engine) {
-
4282 QMessageLogger("painting/qpainter.cpp", 7210, __PRETTY_FUNCTION__).warning("QPainter::setViewport: Painter not active"); -
4283 return;
-
4284 } -
4285 -
4286 d->state->vx = r.x(); -
4287 d->state->vy = r.y(); -
4288 d->state->vw = r.width(); -
4289 d->state->vh = r.height(); -
4290 -
4291 d->state->VxF = true; -
4292 d->updateMatrix(); -
4293}
-
4294 -
4295 -
4296 -
4297 -
4298 -
4299 -
4300 -
4301QRect QPainter::viewport() const -
4302{ -
4303 const QPainterPrivate * const d = d_func(); -
4304 if (!d->engine) {
-
4305 QMessageLogger("painting/qpainter.cpp", 7233, __PRETTY_FUNCTION__).warning("QPainter::viewport: Painter not active"); -
4306 return QRect();
-
4307 } -
4308 return QRect(d->state->vx, d->state->vy, d->state->vw, d->state->vh);
-
4309} -
4310void QPainter::setViewTransformEnabled(bool enable) -
4311{ -
4312 -
4313 -
4314 -
4315 -
4316 -
4317 QPainterPrivate * const d = d_func(); -
4318 -
4319 if (!d->engine) {
-
4320 QMessageLogger("painting/qpainter.cpp", 7257, __PRETTY_FUNCTION__).warning("QPainter::setViewTransformEnabled: Painter not active"); -
4321 return;
-
4322 } -
4323 -
4324 if (enable == d->state->VxF)
-
4325 return;
-
4326 -
4327 d->state->VxF = enable; -
4328 d->updateMatrix(); -
4329}
-
4330void QPainter::setRedirected(const QPaintDevice *device, -
4331 QPaintDevice *replacement, -
4332 const QPoint &offset) -
4333{ -
4334 qt_noop(); -
4335 (void)device; -
4336 (void)replacement; -
4337 (void)offset; -
4338 QMessageLogger("painting/qpainter.cpp", 7299, __PRETTY_FUNCTION__).warning("QPainter::setRedirected(): ignoring call to deprecated function, use QWidget::render() instead"); -
4339}
-
4340void QPainter::restoreRedirected(const QPaintDevice *device) -
4341{ -
4342 (void)device; -
4343 QMessageLogger("painting/qpainter.cpp", 7322, __PRETTY_FUNCTION__).warning("QPainter::restoreRedirected(): ignoring call to deprecated function, use QWidget::render() instead"); -
4344}
-
4345QPaintDevice *QPainter::redirected(const QPaintDevice *device, QPoint *offset) -
4346{ -
4347 (void)device; -
4348 (void)offset; -
4349 return 0;
-
4350} -
4351 -
4352void qt_format_text(const QFont &fnt, const QRectF &_r, -
4353 int tf, const QString& str, QRectF *brect, -
4354 int tabstops, int *ta, int tabarraylen, -
4355 QPainter *painter) -
4356{ -
4357 qt_format_text(fnt, _r, -
4358 tf, 0, str, brect, -
4359 tabstops, ta, tabarraylen, -
4360 painter); -
4361}
-
4362void qt_format_text(const QFont &fnt, const QRectF &_r, -
4363 int tf, const QTextOption *option, const QString& str, QRectF *brect, -
4364 int tabstops, int *ta, int tabarraylen, -
4365 QPainter *painter) -
4366{ -
4367 -
4368 qt_noop(); -
4369 -
4370 if (option) {
-
4371 tf |= option->alignment(); -
4372 if (option->wrapMode() != QTextOption::NoWrap)
-
4373 tf |= Qt::TextWordWrap;
-
4374 -
4375 if (option->flags() & QTextOption::IncludeTrailingSpaces)
-
4376 tf |= Qt::TextIncludeTrailingSpaces;
-
4377 -
4378 if (option->tabStop() >= 0 || !option->tabArray().isEmpty())
-
4379 tf |= Qt::TextExpandTabs;
-
4380 }
-
4381 -
4382 -
4383 QRectF r(_r); -
4384 -
4385 bool dontclip = (tf & Qt::TextDontClip); -
4386 bool wordwrap = (tf & Qt::TextWordWrap) || (tf & Qt::TextWrapAnywhere);
-
4387 bool singleline = (tf & Qt::TextSingleLine); -
4388 bool showmnemonic = (tf & Qt::TextShowMnemonic); -
4389 bool hidemnmemonic = (tf & Qt::TextHideMnemonic); -
4390 -
4391 Qt::LayoutDirection layout_direction; -
4392 if (tf & Qt::TextForceLeftToRight)
-
4393 layout_direction = Qt::LeftToRight;
-
4394 else if (tf & Qt::TextForceRightToLeft)
-
4395 layout_direction = Qt::RightToLeft;
-
4396 else if (option)
-
4397 layout_direction = option->textDirection();
-
4398 else if (painter)
-
4399 layout_direction = painter->layoutDirection();
-
4400 else -
4401 layout_direction = Qt::LeftToRight;
-
4402 -
4403 tf = QGuiApplicationPrivate::visualAlignment(layout_direction, QFlag(tf)); -
4404 -
4405 bool isRightToLeft = layout_direction == Qt::RightToLeft; -
4406 bool expandtabs = ((tf & Qt::TextExpandTabs) &&
-
4407 (((tf & Qt::AlignLeft) && !isRightToLeft) ||
-
4408 ((tf & Qt::AlignRight) && isRightToLeft)));
-
4409 -
4410 if (!painter)
-
4411 tf |= Qt::TextDontPrint;
-
4412 -
4413 uint maxUnderlines = 0; -
4414 int numUnderlines = 0; -
4415 QVarLengthArray<int, 32> underlinePositions(1); -
4416 -
4417 QFontMetricsF fm(fnt); -
4418 QString text = str; -
4419 int offset = 0; -
4420start_lengthVariant:
-
4421 bool hasMoreLengthVariants = false; -
4422 -
4423 -
4424 int old_offset = offset; -
4425 for (; offset < text.length(); offset++) {
-
4426 QChar chr = text.at(offset); -
4427 if (chr == QLatin1Char('\r') || (singleline && chr == QLatin1Char('\n'))) {
-
4428 text[offset] = QLatin1Char(' '); -
4429 } else if (chr == QLatin1Char('\n')) {
-
4430 text[offset] = QChar::LineSeparator; -
4431 } else if (chr == QLatin1Char('&')) {
-
4432 ++maxUnderlines; -
4433 } else if (chr == QLatin1Char('\t')) {
-
4434 if (!expandtabs) {
-
4435 text[offset] = QLatin1Char(' '); -
4436 } else if (!tabarraylen && !tabstops) {
-
4437 tabstops = qRound(fm.width(QLatin1Char('x'))*8); -
4438 }
-
4439 } else if (chr == QChar(ushort(0x9c))) {
-
4440 -
4441 hasMoreLengthVariants = true; -
4442 break;
-
4443 } -
4444 } -
4445 -
4446 int length = offset - old_offset; -
4447 if ((hidemnmemonic || showmnemonic) && maxUnderlines > 0) {
-
4448 underlinePositions.resize(maxUnderlines + 1); -
4449 -
4450 QChar *cout = text.data() + old_offset; -
4451 QChar *cin = cout; -
4452 int l = length; -
4453 while (l) {
-
4454 if (*cin == QLatin1Char('&')) {
-
4455 ++cin; -
4456 --length; -
4457 --l; -
4458 if (!l)
-
4459 break;
-
4460 if (*cin != QLatin1Char('&') && !hidemnmemonic)
-
4461 underlinePositions[numUnderlines++] = cout - text.data() - old_offset;
-
4462 }
-
4463 *cout = *cin; -
4464 ++cout; -
4465 ++cin; -
4466 --l; -
4467 }
-
4468 }
-
4469 -
4470 -
4471 if (tf & Qt::TextDontPrint)
-
4472 numUnderlines = 0;
-
4473 -
4474 underlinePositions[numUnderlines] = -1; -
4475 qreal height = 0; -
4476 qreal width = 0; -
4477 -
4478 QString finalText = text.mid(old_offset, length); -
4479 QStackTextEngine engine(finalText, fnt); -
4480 if (option) {
-
4481 engine.option = *option; -
4482 }
-
4483 -
4484 if (engine.option.tabStop() < 0 && tabstops > 0)
-
4485 engine.option.setTabStop(tabstops);
-
4486 -
4487 if (engine.option.tabs().isEmpty() && ta) {
-
4488 QList<qreal> tabs; -
4489 for (int i = 0; i < tabarraylen; i++)
-
4490 tabs.append(qreal(ta[i]));
-
4491 engine.option.setTabArray(tabs); -
4492 }
-
4493 -
4494 engine.option.setTextDirection(layout_direction); -
4495 if (tf & Qt::AlignJustify)
-
4496 engine.option.setAlignment(Qt::AlignJustify);
-
4497 else -
4498 engine.option.setAlignment(Qt::AlignLeft);
-
4499 -
4500 if (!option && (tf & Qt::TextWrapAnywhere))
-
4501 engine.option.setWrapMode(QTextOption::WrapAnywhere);
-
4502 -
4503 if (tf & Qt::TextJustificationForced)
-
4504 engine.forceJustification = true;
-
4505 QTextLayout textLayout(&engine); -
4506 textLayout.setCacheEnabled(true); -
4507 textLayout.engine()->underlinePositions = underlinePositions.data(); -
4508 -
4509 if (finalText.isEmpty()) {
-
4510 height = fm.height(); -
4511 width = 0; -
4512 tf |= Qt::TextDontPrint; -
4513 } else {
-
4514 qreal lineWidth = 0x01000000; -
4515 if (wordwrap || (tf & Qt::TextJustificationForced))
-
4516 lineWidth = qMax<qreal>(0, r.width());
-
4517 if(!wordwrap)
-
4518 tf |= Qt::TextIncludeTrailingSpaces;
-
4519 textLayout.engine()->ignoreBidi = bool(tf & Qt::TextDontPrint); -
4520 textLayout.beginLayout(); -
4521 -
4522 qreal leading = fm.leading(); -
4523 height = -leading; -
4524 -
4525 while (1) {
-
4526 QTextLine l = textLayout.createLine(); -
4527 if (!l.isValid())
-
4528 break;
-
4529 -
4530 l.setLineWidth(lineWidth); -
4531 height += leading; -
4532 -
4533 -
4534 height = qCeil(height); -
4535 l.setPosition(QPointF(0., height)); -
4536 height += textLayout.engine()->lines[l.lineNumber()].height().toReal(); -
4537 width = qMax(width, l.naturalTextWidth()); -
4538 if (!dontclip && !brect && height >= r.height())
-
4539 break;
-
4540 }
-
4541 textLayout.endLayout(); -
4542 }
-
4543 -
4544 qreal yoff = 0; -
4545 qreal xoff = 0; -
4546 if (tf & Qt::AlignBottom)
-
4547 yoff = r.height() - height;
-
4548 else if (tf & Qt::AlignVCenter)
-
4549 yoff = (r.height() - height)/2;
-
4550 -
4551 if (tf & Qt::AlignRight)
-
4552 xoff = r.width() - width;
-
4553 else if (tf & Qt::AlignHCenter)
-
4554 xoff = (r.width() - width)/2;
-
4555 -
4556 QRectF bounds = QRectF(r.x() + xoff, r.y() + yoff, width, height); -
4557 -
4558 if (hasMoreLengthVariants && !(tf & Qt::TextLongestVariant) && !r.contains(bounds)) {
-
4559 offset++; -
4560 goto start_lengthVariant;
-
4561 } -
4562 if (brect)
-
4563 *brect = bounds;
-
4564 -
4565 if (!(tf & Qt::TextDontPrint)) {
-
4566 bool restore = false; -
4567 if (!dontclip && !r.contains(bounds)) {
-
4568 restore = true; -
4569 painter->save(); -
4570 painter->setClipRect(r, Qt::IntersectClip); -
4571 }
-
4572 -
4573 for (int i = 0; i < textLayout.lineCount(); i++) {
-
4574 QTextLine line = textLayout.lineAt(i); -
4575 QTextEngine *eng = textLayout.engine(); -
4576 eng->enableDelayDecorations(); -
4577 -
4578 qreal advance = line.horizontalAdvance(); -
4579 xoff = 0; -
4580 if (tf & Qt::AlignRight) {
-
4581 xoff = r.width() - advance - -
4582 eng->leadingSpaceWidth(eng->lines[line.lineNumber()]).toReal(); -
4583 }
-
4584 else if (tf & Qt::AlignHCenter)
-
4585 xoff = (r.width() - advance) / 2;
-
4586 -
4587 line.draw(painter, QPointF(r.x() + xoff, r.y() + yoff)); -
4588 eng->drawDecorations(painter); -
4589 }
-
4590 -
4591 if (restore) {
-
4592 painter->restore(); -
4593 }
-
4594 }
-
4595}
-
4596void QPainter::setLayoutDirection(Qt::LayoutDirection direction) -
4597{ -
4598 QPainterPrivate * const d = d_func(); -
4599 if (d->state)
-
4600 d->state->layoutDirection = direction;
-
4601}
-
4602 -
4603 -
4604 -
4605 -
4606 -
4607 -
4608Qt::LayoutDirection QPainter::layoutDirection() const -
4609{ -
4610 const QPainterPrivate * const d = d_func(); -
4611 return d->state ? d->state->layoutDirection : Qt::LayoutDirectionAuto;
-
4612} -
4613 -
4614QPainterState::QPainterState(const QPainterState *s) -
4615 : brushOrigin(s->brushOrigin), font(s->font), deviceFont(s->deviceFont), -
4616 pen(s->pen), brush(s->brush), bgBrush(s->bgBrush), -
4617 clipRegion(s->clipRegion), clipPath(s->clipPath), -
4618 clipOperation(s->clipOperation), -
4619 renderHints(s->renderHints), clipInfo(s->clipInfo), -
4620 worldMatrix(s->worldMatrix), matrix(s->matrix), redirectionMatrix(s->redirectionMatrix), -
4621 wx(s->wx), wy(s->wy), ww(s->ww), wh(s->wh), -
4622 vx(s->vx), vy(s->vy), vw(s->vw), vh(s->vh), -
4623 opacity(s->opacity), WxF(s->WxF), VxF(s->VxF), -
4624 clipEnabled(s->clipEnabled), bgMode(s->bgMode), painter(s->painter), -
4625 layoutDirection(s->layoutDirection), -
4626 composition_mode(s->composition_mode), -
4627 emulationSpecifier(s->emulationSpecifier), changeFlags(0) -
4628{ -
4629 dirtyFlags = s->dirtyFlags; -
4630}
-
4631 -
4632QPainterState::QPainterState() -
4633 : brushOrigin(0, 0), bgBrush(Qt::white), clipOperation(Qt::NoClip), -
4634 renderHints(0), -
4635 wx(0), wy(0), ww(0), wh(0), vx(0), vy(0), vw(0), vh(0), -
4636 opacity(1), WxF(false), VxF(false), clipEnabled(true), -
4637 bgMode(Qt::TransparentMode), painter(0), -
4638 layoutDirection(QGuiApplication::layoutDirection()), -
4639 composition_mode(QPainter::CompositionMode_SourceOver), -
4640 emulationSpecifier(0), changeFlags(0) -
4641{ -
4642 dirtyFlags = 0; -
4643}
-
4644 -
4645QPainterState::~QPainterState() -
4646{ -
4647} -
4648 -
4649void QPainterState::init(QPainter *p) { -
4650 bgBrush = Qt::white; -
4651 bgMode = Qt::TransparentMode; -
4652 WxF = false; -
4653 VxF = false; -
4654 clipEnabled = true; -
4655 wx = wy = ww = wh = 0; -
4656 vx = vy = vw = vh = 0; -
4657 painter = p; -
4658 pen = QPen(); -
4659 brushOrigin = QPointF(0, 0); -
4660 brush = QBrush(); -
4661 font = deviceFont = QFont(); -
4662 clipRegion = QRegion(); -
4663 clipPath = QPainterPath(); -
4664 clipOperation = Qt::NoClip; -
4665 clipInfo.clear(); -
4666 worldMatrix.reset(); -
4667 matrix.reset(); -
4668 layoutDirection = QGuiApplication::layoutDirection(); -
4669 composition_mode = QPainter::CompositionMode_SourceOver; -
4670 emulationSpecifier = 0; -
4671 dirtyFlags = 0; -
4672 changeFlags = 0; -
4673 renderHints = 0; -
4674 opacity = 1; -
4675}
-
4676QPen QPaintEngineState::pen() const -
4677{ -
4678 return static_cast<const QPainterState *>(this)->pen;
-
4679} -
4680QBrush QPaintEngineState::brush() const -
4681{ -
4682 return static_cast<const QPainterState *>(this)->brush;
-
4683} -
4684QPointF QPaintEngineState::brushOrigin() const -
4685{ -
4686 return static_cast<const QPainterState *>(this)->brushOrigin;
-
4687} -
4688QBrush QPaintEngineState::backgroundBrush() const -
4689{ -
4690 return static_cast<const QPainterState *>(this)->bgBrush;
-
4691} -
4692Qt::BGMode QPaintEngineState::backgroundMode() const -
4693{ -
4694 return static_cast<const QPainterState *>(this)->bgMode;
-
4695} -
4696QFont QPaintEngineState::font() const -
4697{ -
4698 return static_cast<const QPainterState *>(this)->font;
-
4699} -
4700QMatrix QPaintEngineState::matrix() const -
4701{ -
4702 const QPainterState *st = static_cast<const QPainterState *>(this); -
4703 -
4704 return st->matrix.toAffine();
-
4705} -
4706QTransform QPaintEngineState::transform() const -
4707{ -
4708 const QPainterState *st = static_cast<const QPainterState *>(this); -
4709 -
4710 return st->matrix;
-
4711} -
4712Qt::ClipOperation QPaintEngineState::clipOperation() const -
4713{ -
4714 return static_cast<const QPainterState *>(this)->clipOperation;
-
4715} -
4716bool QPaintEngineState::brushNeedsResolving() const -
4717{ -
4718 const QBrush &brush = static_cast<const QPainterState *>(this)->brush; -
4719 return needsResolving(brush);
-
4720} -
4721bool QPaintEngineState::penNeedsResolving() const -
4722{ -
4723 const QPen &pen = static_cast<const QPainterState *>(this)->pen; -
4724 return needsResolving(pen.brush());
-
4725} -
4726QRegion QPaintEngineState::clipRegion() const -
4727{ -
4728 return static_cast<const QPainterState *>(this)->clipRegion;
-
4729} -
4730QPainterPath QPaintEngineState::clipPath() const -
4731{ -
4732 return static_cast<const QPainterState *>(this)->clipPath;
-
4733} -
4734bool QPaintEngineState::isClipEnabled() const -
4735{ -
4736 return static_cast<const QPainterState *>(this)->clipEnabled;
-
4737} -
4738QPainter::RenderHints QPaintEngineState::renderHints() const -
4739{ -
4740 return static_cast<const QPainterState *>(this)->renderHints;
-
4741} -
4742QPainter::CompositionMode QPaintEngineState::compositionMode() const -
4743{ -
4744 return static_cast<const QPainterState *>(this)->composition_mode;
-
4745} -
4746 -
4747 -
4748 -
4749 -
4750 -
4751 -
4752 -
4753QPainter *QPaintEngineState::painter() const -
4754{ -
4755 return static_cast<const QPainterState *>(this)->painter;
-
4756} -
4757qreal QPaintEngineState::opacity() const -
4758{ -
4759 return static_cast<const QPainterState *>(this)->opacity;
-
4760} -
4761void QPainter::setTransform(const QTransform &transform, bool combine ) -
4762{ -
4763 setWorldTransform(transform, combine); -
4764}
-
4765 -
4766 -
4767 -
4768 -
4769 -
4770 -
4771 -
4772const QTransform & QPainter::transform() const -
4773{ -
4774 return worldTransform();
-
4775} -
4776const QTransform & QPainter::deviceTransform() const -
4777{ -
4778 const QPainterPrivate * const d = d_func(); -
4779 if (!d->engine) {
-
4780 QMessageLogger("painting/qpainter.cpp", 8173, __PRETTY_FUNCTION__).warning("QPainter::deviceTransform: Painter not active"); -
4781 return d->fakeState()->transform;
-
4782 } -
4783 return d->state->matrix;
-
4784} -
4785void QPainter::resetTransform() -
4786{ -
4787 QPainterPrivate * const d = d_func(); -
4788 -
4789 -
4790 -
4791 -
4792 if (!d->engine) {
-
4793 QMessageLogger("painting/qpainter.cpp", 8196, __PRETTY_FUNCTION__).warning("QPainter::resetMatrix: Painter not active"); -
4794 return;
-
4795 } -
4796 -
4797 d->state->wx = d->state->wy = d->state->vx = d->state->vy = 0; -
4798 d->state->ww = d->state->vw = d->device->metric(QPaintDevice::PdmWidth); -
4799 d->state->wh = d->state->vh = d->device->metric(QPaintDevice::PdmHeight); -
4800 d->state->worldMatrix = QTransform(); -
4801 setMatrixEnabled(false); -
4802 setViewTransformEnabled(false); -
4803 if (d->extended)
-
4804 d->extended->transformChanged();
-
4805 else -
4806 d->state->dirtyFlags |= QPaintEngine::DirtyTransform;
-
4807} -
4808void QPainter::setWorldTransform(const QTransform &matrix, bool combine ) -
4809{ -
4810 QPainterPrivate * const d = d_func(); -
4811 -
4812 if (!d->engine) {
-
4813 QMessageLogger("painting/qpainter.cpp", 8225, __PRETTY_FUNCTION__).warning("QPainter::setWorldTransform: Painter not active"); -
4814 return;
-
4815 } -
4816 -
4817 if (combine)
-
4818 d->state->worldMatrix = matrix * d->state->worldMatrix;
-
4819 else -
4820 d->state->worldMatrix = matrix;
-
4821 -
4822 d->state->WxF = true; -
4823 d->updateMatrix(); -
4824}
-
4825 -
4826 -
4827 -
4828 -
4829 -
4830const QTransform & QPainter::worldTransform() const -
4831{ -
4832 const QPainterPrivate * const d = d_func(); -
4833 if (!d->engine) {
-
4834 QMessageLogger("painting/qpainter.cpp", 8246, __PRETTY_FUNCTION__).warning("QPainter::worldTransform: Painter not active"); -
4835 return d->fakeState()->transform;
-
4836 } -
4837 return d->state->worldMatrix;
-
4838} -
4839QTransform QPainter::combinedTransform() const -
4840{ -
4841 const QPainterPrivate * const d = d_func(); -
4842 if (!d->engine) {
-
4843 QMessageLogger("painting/qpainter.cpp", 8263, __PRETTY_FUNCTION__).warning("QPainter::combinedTransform: Painter not active"); -
4844 return QTransform();
-
4845 } -
4846 return d->state->worldMatrix * d->viewTransform() * d->hidpiScaleTransform();
-
4847} -
4848void QPainter::drawPixmapFragments(const PixmapFragment *fragments, int fragmentCount, -
4849 const QPixmap &pixmap, PixmapFragmentHints hints) -
4850{ -
4851 QPainterPrivate * const d = d_func(); -
4852 -
4853 if (!d->engine || pixmap.isNull())
-
4854 return;
-
4855 if (d->engine->isExtended()) {
-
4856 d->extended->drawPixmapFragments(fragments, fragmentCount, pixmap, hints); -
4857 } else {
-
4858 qreal oldOpacity = opacity(); -
4859 QTransform oldTransform = transform(); -
4860 -
4861 for (int i = 0; i < fragmentCount; ++i) {
-
4862 QTransform transform = oldTransform; -
4863 qreal xOffset = 0; -
4864 qreal yOffset = 0; -
4865 if (fragments[i].rotation == 0) {
-
4866 xOffset = fragments[i].x; -
4867 yOffset = fragments[i].y; -
4868 } else {
-
4869 transform.translate(fragments[i].x, fragments[i].y); -
4870 transform.rotate(fragments[i].rotation); -
4871 }
-
4872 setOpacity(oldOpacity * fragments[i].opacity); -
4873 setTransform(transform); -
4874 -
4875 qreal w = fragments[i].scaleX * fragments[i].width; -
4876 qreal h = fragments[i].scaleY * fragments[i].height; -
4877 QRectF sourceRect(fragments[i].sourceLeft, fragments[i].sourceTop, -
4878 fragments[i].width, fragments[i].height); -
4879 drawPixmap(QRectF(-0.5 * w + xOffset, -0.5 * h + yOffset, w, h), pixmap, sourceRect); -
4880 }
-
4881 -
4882 setOpacity(oldOpacity); -
4883 setTransform(oldTransform); -
4884 }
-
4885} -
4886QPainter::PixmapFragment QPainter::PixmapFragment::create(const QPointF &pos, const QRectF &sourceRect, -
4887 qreal scaleX, qreal scaleY, qreal rotation, -
4888 qreal opacity) -
4889{ -
4890 PixmapFragment fragment = {pos.x(), pos.y(), sourceRect.x(), sourceRect.y(), sourceRect.width(), -
4891 sourceRect.height(), scaleX, scaleY, rotation, opacity}; -
4892 return fragment;
-
4893} -
4894void qt_draw_helper(QPainterPrivate *p, const QPainterPath &path, QPainterPrivate::DrawOperation operation) -
4895{ -
4896 p->draw_helper(path, operation); -
4897}
-
4898 -
4899 -
4900 -
Switch to Source codePreprocessed file

Generated by Squish Coco Non-Commercial