Krita Source Code Documentation
Loading...
Searching...
No Matches
kis_perspective_transform_strategy.cpp
Go to the documentation of this file.
1/*
2 * SPDX-FileCopyrightText: 2014 Dmitry Kazakov <dimula73@gmail.com>
3 * SPDX-FileCopyrightText: 2022 Carsten Hartenfels <carsten.hartenfels@pm.me>
4 *
5 * SPDX-License-Identifier: GPL-2.0-or-later
6 */
7
9
10#include <QPointF>
11#include <QPainter>
12#include <QPainterPath>
13#include <QMatrix4x4>
14#include <QVector2D>
15
16#include <Eigen/Dense>
17
20#include "tool_transform_args.h"
22#include "krita_utils.h"
23#include "kis_cursor.h"
24#include "kis_transform_utils.h"
26
27namespace {
28enum StrokeFunction {
29 DRAG_HANDLE = 0,
30 DRAG_X_VANISHING_POINT,
31 DRAG_Y_VANISHING_POINT,
32 MOVE,
33 NONE
34};
35
36enum HandleIndexes {
37 HANDLE_TOP_LEFT = 0,
38 HANDLE_TOP_RIGHT,
39 HANDLE_BOTTOM_LEFT,
40 HANDLE_BOTTOM_RIGHT,
41 HANDLE_MIDDLE_TOP,
42 HANDLE_MIDDLE_BOTTOM,
43 HANDLE_MIDDLE_LEFT,
44 HANDLE_MIDDLE_RIGHT,
45 HANDLE_COUNT,
46};
47}
48
50{
52 const KisCoordinatesConverter *_converter,
53 ToolTransformArgs &_currentArgs,
55 : q(_q),
56 converter(_converter),
57 currentArgs(_currentArgs),
58 transaction(_transaction),
59 imageTooBig(false),
60 isTransforming(false)
61 {
62 }
63
65
67
69
74
75
78
81
82 QTransform handlesTransform;
83
85
86 StrokeFunction function {NONE};
87
88 struct HandlePoints {
89 bool xVanishingExists {false};
90 bool yVanishingExists {false};
91
92 QPointF xVanishing;
93 QPointF yVanishing;
94 };
96
97 QTransform transform;
98
102
103 bool imageTooBig {false};
104
105 QPointF clickPos;
107 bool isTransforming {false};
108
109 QCursor getScaleCursor(const QPointF &handlePt);
110 QCursor getShearCursor(const QPointF &start, const QPointF &end);
113
114 void transformIntoArgs(const Eigen::Matrix3f &t);
115 QTransform transformFromArgs();
116};
117
119 KoSnapGuide *snapGuide,
120 ToolTransformArgs &currentArgs,
122 : KisSimplifiedActionPolicyStrategy(converter, snapGuide),
123 m_d(new Private(this, converter, currentArgs, transaction))
124{
125}
126
130
132{
133 srcHandlePoints.resize(HANDLE_COUNT);
134 srcHandlePoints[HANDLE_TOP_LEFT] = transaction.originalTopLeft();
135 srcHandlePoints[HANDLE_TOP_RIGHT] = transaction.originalTopRight();
136 srcHandlePoints[HANDLE_BOTTOM_LEFT] = transaction.originalBottomLeft();
137 srcHandlePoints[HANDLE_BOTTOM_RIGHT] = transaction.originalBottomRight();
138 srcHandlePoints[HANDLE_MIDDLE_TOP] = transaction.originalMiddleTop();
139 srcHandlePoints[HANDLE_MIDDLE_BOTTOM] = transaction.originalMiddleBottom();
140 srcHandlePoints[HANDLE_MIDDLE_LEFT] = transaction.originalMiddleLeft();
141 srcHandlePoints[HANDLE_MIDDLE_RIGHT] = transaction.originalMiddleRight();
142
143 dstHandlePoints.clear();
144 Q_FOREACH (const QPointF &pt, srcHandlePoints) {
145 dstHandlePoints << transform.map(pt);
146 }
147
148 QMatrix4x4 realMatrix(transform);
149 QVector4D v;
150
151 v = QVector4D(1, 0, 0, 0);
152 v = realMatrix * v;
154 transformedHandles.xVanishing = v.toVector2DAffine().toPointF();
155
156 v = QVector4D(0, 1, 0, 0);
157 v = realMatrix * v;
159 transformedHandles.yVanishing = v.toVector2DAffine().toPointF();
160}
161
162void KisPerspectiveTransformStrategy::setTransformFunction(const QPointF &mousePos, bool perspectiveModifierActive, bool shiftModifierActive, bool altModifierActive)
163{
164 Q_UNUSED(perspectiveModifierActive);
165 Q_UNUSED(shiftModifierActive);
166 Q_UNUSED(altModifierActive);
167
168 QPolygonF transformedPolygon = m_d->transform.map(QPolygonF(m_d->transaction.originalRect()));
169 StrokeFunction defaultFunction = transformedPolygon.containsPoint(mousePos, Qt::OddEvenFill) ? MOVE : NONE;
171 handleChooser(mousePos, defaultFunction);
172
173 qreal handleRadius = KisTransformUtils::effectiveHandleGrabRadius(m_d->converter);
174
175 if (!m_d->transformedHandles.xVanishing.isNull()) {
176 handleChooser.addFunction(m_d->transformedHandles.xVanishing,
177 handleRadius, DRAG_X_VANISHING_POINT);
178 }
179
180 if (!m_d->transformedHandles.yVanishing.isNull()) {
181 handleChooser.addFunction(m_d->transformedHandles.yVanishing,
182 handleRadius, DRAG_Y_VANISHING_POINT);
183 }
184
185 m_d->currentDraggingHandlePoint = -1;
186 for (int i = 0; i < m_d->dstHandlePoints.size(); i++) {
187 if (handleChooser.addFunction(m_d->dstHandlePoints[i],
188 handleRadius, DRAG_HANDLE)) {
189
190 m_d->currentDraggingHandlePoint = i;
191 }
192 }
193
194 m_d->function = handleChooser.function();
195}
196
198{
199 QCursor cursor;
200
201 switch (m_d->function) {
202 case NONE:
203 cursor = KisCursor::arrowCursor();
204 break;
205 case MOVE:
206 cursor = KisCursor::moveCursor();
207 break;
208 case DRAG_HANDLE:
209 case DRAG_X_VANISHING_POINT:
210 case DRAG_Y_VANISHING_POINT:
212 break;
213 }
214
215 return cursor;
216}
217
218void KisPerspectiveTransformStrategy::paint(QPainter &gc, const KoColorDisplayRendererInterface *displayRendererInterface)
219{
220 gc.save();
221
222 gc.setOpacity(m_d->transaction.basePreviewOpacity());
223 gc.setTransform(m_d->paintingTransform, true);
224 gc.drawImage(m_d->paintingOffset, originalImage());
225
226 gc.restore();
227
228 // Draw Handles
229 QPainterPath handles;
230
231 handles.moveTo(m_d->transaction.originalTopLeft());
232 handles.lineTo(m_d->transaction.originalTopRight());
233 handles.lineTo(m_d->transaction.originalBottomRight());
234 handles.lineTo(m_d->transaction.originalBottomLeft());
235 handles.lineTo(m_d->transaction.originalTopLeft());
236
237
238 auto addHandleRectFunc =
239 [&](const QPointF &pt) {
240 handles.addRect(
242 m_d->handlesTransform,
243 m_d->transaction.originalRect(), pt)
244 .translated(pt));
245 };
246
247 addHandleRectFunc(m_d->transaction.originalTopLeft());
248 addHandleRectFunc(m_d->transaction.originalTopRight());
249 addHandleRectFunc(m_d->transaction.originalBottomLeft());
250 addHandleRectFunc(m_d->transaction.originalBottomRight());
251 addHandleRectFunc(m_d->transaction.originalMiddleTop());
252 addHandleRectFunc(m_d->transaction.originalMiddleBottom());
253 addHandleRectFunc(m_d->transaction.originalMiddleLeft());
254 addHandleRectFunc(m_d->transaction.originalMiddleRight());
255
256 gc.save();
257
258 if (m_d->isTransforming) {
259 gc.setOpacity(0.1);
260 }
261
272 //gc.setTransform(m_d->handlesTransform, true); <-- don't do like this!
273
274 QPainterPath mappedHandles = m_d->handlesTransform.map(handles);
275
276 QPen pen[2];
277 pen[0].setWidth(decorationThickness());
278 pen[0].setCosmetic(true);
279 pen[1].setWidth(decorationThickness() * 2);
280 pen[1].setCosmetic(true);
281 pen[1].setColor(displayRendererInterface->convertColorToDisplayColorSpace(KoColor(Qt::lightGray, KoColorSpaceRegistry::instance()->rgb8())));
282
283 for (int i = 1; i >= 0; --i) {
284 gc.setPen(pen[i]);
285 gc.drawPath(mappedHandles);
286 }
287
288 gc.restore();
289
290 { // painting perspective handles
291 QPainterPath perspectiveHandles;
292
293 QRectF handleRect =
295 QTransform(),
296 m_d->transaction.originalRect(), 0, 0);
297
298 if (m_d->transformedHandles.xVanishingExists) {
299 QRectF rc = handleRect.translated(m_d->transformedHandles.xVanishing);
300 perspectiveHandles.addEllipse(rc);
301 }
302
303 if (m_d->transformedHandles.yVanishingExists) {
304 QRectF rc = handleRect.translated(m_d->transformedHandles.yVanishing);
305 perspectiveHandles.addEllipse(rc);
306 }
307
308 if (!perspectiveHandles.isEmpty()) {
309 gc.save();
310 gc.setTransform(m_d->converter->imageToWidgetTransform());
311
312 gc.setBrush(Qt::red);
313
314 for (int i = 1; i >= 0; --i) {
315 gc.setPen(pen[i]);
316 gc.drawPath(perspectiveHandles);
317 }
318
319 gc.restore();
320 }
321 }
322}
323
325{
326 m_d->recalculateTransformations();
327}
328
330{
331 Q_UNUSED(pt);
332
333 if (m_d->function == NONE) return false;
334
335 m_d->clickPos = pt;
336 m_d->clickArgs = m_d->currentArgs;
337
338 return true;
339}
340
341Eigen::Matrix3f getTransitionMatrix(const QVector<QPointF> &sp)
342{
343 Eigen::Matrix3f A;
344 Eigen::Vector3f v3;
345
346 A << sp[HANDLE_TOP_LEFT].x() , sp[HANDLE_TOP_RIGHT].x() , sp[HANDLE_BOTTOM_LEFT].x()
347 ,sp[HANDLE_TOP_LEFT].y() , sp[HANDLE_TOP_RIGHT].y() , sp[HANDLE_BOTTOM_LEFT].y()
348 , 1 , 1 , 1;
349
350 v3 << sp[HANDLE_BOTTOM_RIGHT].x() , sp[HANDLE_BOTTOM_RIGHT].y() , 1;
351
352 Eigen::Vector3f coeffs = A.colPivHouseholderQr().solve(v3);
353
354 A.col(0) *= coeffs(0);
355 A.col(1) *= coeffs(1);
356 A.col(2) *= coeffs(2);
357
358 return A;
359}
360
361QTransform toQTransform(const Eigen::Matrix3f &m)
362{
363 return QTransform(m(0,0), m(1,0), m(2,0),
364 m(0,1), m(1,1), m(2,1),
365 m(0,2), m(1,2), m(2,2));
366}
367
368Eigen::Matrix3f fromQTransform(const QTransform &t)
369{
370 Eigen::Matrix3f m;
371
372 m << t.m11() , t.m21() , t.m31()
373 ,t.m12() , t.m22() , t.m32()
374 ,t.m13() , t.m23() , t.m33();
375
376 return m;
377}
378
379Eigen::Matrix3f fromTranslate(const QPointF &pt)
380{
381 Eigen::Matrix3f m;
382
383 m << 1 , 0 , pt.x()
384 ,0 , 1 , pt.y()
385 ,0 , 0 , 1;
386
387 return m;
388}
389
390Eigen::Matrix3f fromScale(qreal sx, qreal sy)
391{
392 Eigen::Matrix3f m;
393
394 m << sx , 0 , 0
395 ,0 , sy , 0
396 ,0 , 0 , 1;
397
398 return m;
399}
400
401Eigen::Matrix3f fromShear(qreal sx, qreal sy)
402{
403 Eigen::Matrix3f m;
404
405 m << 1 , sx , 0
406 ,sy , sx*sy + 1, 0
407 ,0 , 0 , 1;
408
409 return m;
410}
411
413{
414 Eigen::Matrix3f TS = fromTranslate(-currentArgs.originalCenter());
415
416 Eigen::Matrix3f m = t * TS.inverse();
417
418 qreal tX = m(0,2) / m(2,2);
419 qreal tY = m(1,2) / m(2,2);
420
421 Eigen::Matrix3f T = fromTranslate(QPointF(tX, tY));
422
423 m = T.inverse() * m;
424
438#if 0
439 // Decomposition according to:
440 // https://www.w3.org/TR/css-transforms-1/#decomposing-a-3d-matrix
442
443 currentArgs.setScaleX(dm.scaleX);
444 currentArgs.setScaleY(dm.scaleY);
445
446 currentArgs.setShearX(dm.shearXY);
447 currentArgs.setShearY(0.0);
448
449 currentArgs.setAZ(kisDegreesToRadians(dm.angle));
450
451 QTransform pre = dm.scaleTransform() * dm.shearTransform() * dm.rotateTransform();
452 m = m * fromQTransform(pre.inverted());
453#else
454 currentArgs.setScaleX(1.0);
455 currentArgs.setScaleY(1.0);
456 currentArgs.setShearX(0.0);
457 currentArgs.setShearY(0.0);
458 currentArgs.setAZ(0.0);
459#endif
460
461 currentArgs.setTransformedCenter(QPointF(tX, tY));
462 currentArgs.setFlattenedPerspectiveTransform(toQTransform(m));
463}
464
470
471QVector4D fromQPointF(const QPointF &pt) {
472 return QVector4D(pt.x(), pt.y(), 0, 1.0);
473}
474
475QPointF toQPointF(const QVector4D &v) {
476 return v.toVector2DAffine().toPointF();
477}
478
479void KisPerspectiveTransformStrategy::continuePrimaryAction(const QPointF &mousePos, bool shiftModifierActive, bool altModifierActive)
480{
481 Q_UNUSED(shiftModifierActive);
482 Q_UNUSED(altModifierActive);
483
484 m_d->isTransforming = true;
485
486 switch (m_d->function) {
487 case NONE:
488 break;
489 case MOVE: {
490 QPointF diff = mousePos - m_d->clickPos;
491 m_d->currentArgs.setTransformedCenter(
492 m_d->clickArgs.transformedCenter() + diff);
493 break;
494 }
495 case DRAG_HANDLE: {
496 KIS_ASSERT_RECOVER_RETURN(m_d->currentDraggingHandlePoint >= 0);
497 KIS_ASSERT_RECOVER_RETURN(m_d->currentDraggingHandlePoint < HANDLE_COUNT);
498 if (m_d->currentDraggingHandlePoint < HANDLE_MIDDLE_TOP) {
499 // Corner point, transform directly.
500 m_d->dstHandlePoints[m_d->currentDraggingHandlePoint] = mousePos;
501 } else {
502 // Middle point, move adjacent corners.
503 QPointF delta = mousePos - m_d->dstHandlePoints[m_d->currentDraggingHandlePoint];
504 switch(m_d->currentDraggingHandlePoint) {
505 case HANDLE_MIDDLE_TOP:
506 m_d->dstHandlePoints[HANDLE_TOP_LEFT] += delta;
507 m_d->dstHandlePoints[HANDLE_TOP_RIGHT] += delta;
508 break;
509 case HANDLE_MIDDLE_BOTTOM:
510 m_d->dstHandlePoints[HANDLE_BOTTOM_LEFT] += delta;
511 m_d->dstHandlePoints[HANDLE_BOTTOM_RIGHT] += delta;
512 break;
513 case HANDLE_MIDDLE_LEFT:
514 m_d->dstHandlePoints[HANDLE_TOP_LEFT] += delta;
515 m_d->dstHandlePoints[HANDLE_BOTTOM_LEFT] += delta;
516 break;
517 case HANDLE_MIDDLE_RIGHT:
518 m_d->dstHandlePoints[HANDLE_TOP_RIGHT] += delta;
519 m_d->dstHandlePoints[HANDLE_BOTTOM_RIGHT] += delta;
520 break;
521 }
522 }
523
524 Eigen::Matrix3f A = getTransitionMatrix(m_d->srcHandlePoints);
525 Eigen::Matrix3f B = getTransitionMatrix(m_d->dstHandlePoints);
526 Eigen::Matrix3f result = B * A.inverse();
527
528 m_d->transformIntoArgs(result);
529
530 break;
531 }
532 case DRAG_X_VANISHING_POINT:
533 case DRAG_Y_VANISHING_POINT: {
534
535 QMatrix4x4 m(m_d->transform);
536
537 QPointF tl = m_d->transaction.originalTopLeft();
538 QPointF tr = m_d->transaction.originalTopRight();
539 QPointF bl = m_d->transaction.originalBottomLeft();
540 QPointF br = m_d->transaction.originalBottomRight();
541
542 QVector4D v(1,0,0,0);
543 QVector4D otherV(0,1,0,0);
544
545 if (m_d->function == DRAG_X_VANISHING_POINT) {
546 v = QVector4D(1,0,0,0);
547 otherV = QVector4D(0,1,0,0);
548 } else {
549 v = QVector4D(0,1,0,0);
550 otherV = QVector4D(1,0,0,0);
551 }
552
553 QPointF tl_dst = toQPointF(m * fromQPointF(tl));
554 QPointF tr_dst = toQPointF(m * fromQPointF(tr));
555 QPointF bl_dst = toQPointF(m * fromQPointF(bl));
556 QPointF br_dst = toQPointF(m * fromQPointF(br));
557 QPointF v_dst = toQPointF(m * v);
558 QPointF otherV_dst = toQPointF(m * otherV);
559
560 QVector<QPointF> srcPoints;
561 QVector<QPointF> dstPoints;
562
563 QPointF far1_src;
564 QPointF far2_src;
565 QPointF near1_src;
566 QPointF near2_src;
567
568 QPointF far1_dst;
569 QPointF far2_dst;
570 QPointF near1_dst;
571 QPointF near2_dst;
572
573 if (m_d->function == DRAG_X_VANISHING_POINT) {
574
575 // topLeft (far) --- topRight (near) --- vanishing
576 if (kisSquareDistance(v_dst, tl_dst) > kisSquareDistance(v_dst, tr_dst)) {
577 far1_src = tl;
578 far2_src = bl;
579 near1_src = tr;
580 near2_src = br;
581
582 far1_dst = tl_dst;
583 far2_dst = bl_dst;
584 near1_dst = tr_dst;
585 near2_dst = br_dst;
586
587 // topRight (far) --- topLeft (near) --- vanishing
588 } else {
589 far1_src = tr;
590 far2_src = br;
591 near1_src = tl;
592 near2_src = bl;
593
594 far1_dst = tr_dst;
595 far2_dst = br_dst;
596 near1_dst = tl_dst;
597 near2_dst = bl_dst;
598 }
599
600 } else /* if (m_d->function == DRAG_Y_VANISHING_POINT) */{
601 // topLeft (far) --- bottomLeft (near) --- vanishing
602 if (kisSquareDistance(v_dst, tl_dst) > kisSquareDistance(v_dst, bl_dst)) {
603 far1_src = tl;
604 far2_src = tr;
605 near1_src = bl;
606 near2_src = br;
607
608 far1_dst = tl_dst;
609 far2_dst = tr_dst;
610 near1_dst = bl_dst;
611 near2_dst = br_dst;
612
613 // bottomLeft (far) --- topLeft (near) --- vanishing
614 } else {
615 far1_src = bl;
616 far2_src = br;
617 near1_src = tl;
618 near2_src = tr;
619
620 far1_dst = bl_dst;
621 far2_dst = br_dst;
622 near1_dst = tl_dst;
623 near2_dst = tr_dst;
624 }
625 }
626
627 QLineF l0(far1_dst, mousePos);
628 QLineF l1(far2_dst, mousePos);
629 QLineF l2(otherV_dst, near1_dst);
630 l0.intersects(l2, &near1_dst);
631 l1.intersects(l2, &near2_dst);
632
633 srcPoints << far1_src;
634 srcPoints << far2_src;
635 srcPoints << near1_src;
636 srcPoints << near2_src;
637
638 dstPoints << far1_dst;
639 dstPoints << far2_dst;
640 dstPoints << near1_dst;
641 dstPoints << near2_dst;
642
643 Eigen::Matrix3f A = getTransitionMatrix(srcPoints);
644 Eigen::Matrix3f B = getTransitionMatrix(dstPoints);
645 Eigen::Matrix3f result = B * A.inverse();
646
647 m_d->transformIntoArgs(result);
648 break;
649 }
650 }
651
652 m_d->recalculateTransformations();
653}
654
656{
657 bool shouldSave = !m_d->imageTooBig;
658 m_d->isTransforming = false;
659
660 if (m_d->imageTooBig) {
661 m_d->currentArgs = m_d->clickArgs;
662 m_d->recalculateTransformations();
663 }
664
665 return shouldSave;
666}
667
669{
670 transform = transformFromArgs();
671
672 QTransform viewScaleTransform = converter->imageToDocumentTransform() * converter->documentToFlakeTransform();
673 handlesTransform = transform * viewScaleTransform;
674
675 QTransform tl = QTransform::fromTranslate(transaction.originalTopLeft().x(), transaction.originalTopLeft().y());
676 paintingTransform = tl.inverted() * q->thumbToImageTransform() * tl * transform * viewScaleTransform;
677 paintingOffset = transaction.originalTopLeft();
678
679 // check whether image is too big to be displayed or not
680 const qreal maxScale = 20.0;
681
682 imageTooBig = false;
683
684 if (qAbs(currentArgs.scaleX()) > maxScale ||
685 qAbs(currentArgs.scaleY()) > maxScale) {
686
687 imageTooBig = true;
688
689 } else {
690 QVector<QPointF> points;
691 points << transaction.originalRect().topLeft();
692 points << transaction.originalRect().topRight();
693 points << transaction.originalRect().bottomRight();
694 points << transaction.originalRect().bottomLeft();
695
696 for (int i = 0; i < points.size(); i++) {
697 points[i] = transform.map(points[i]);
698 }
699
700 for (int i = 0; i < points.size(); i++) {
701 const QPointF &pt = points[i];
702 const QPointF &prev = points[(i - 1 + 4) % 4];
703 const QPointF &next = points[(i + 1) % 4];
704 const QPointF &other = points[(i + 2) % 4];
705
706 QLineF l1(pt, other);
707 QLineF l2(prev, next);
708
709 QPointF intersection;
710 l1.intersects(l2, &intersection);
711
712 qreal maxDistance = kisSquareDistance(pt, other);
713
714 if (kisSquareDistance(pt, intersection) > maxDistance ||
715 kisSquareDistance(other, intersection) > maxDistance) {
716
717 imageTooBig = true;
718 break;
719 }
720
721 const qreal thresholdDistance = 0.02 * l2.length();
722
723 if (kisDistanceToLine(pt, l2) < thresholdDistance) {
724 imageTooBig = true;
725 break;
726 }
727 }
728 }
729
730 // recalculate cached handles position
731 recalculateTransformedHandles();
732
733 Q_EMIT q->requestShowImageTooBig(imageTooBig);
734 Q_EMIT q->requestImageRecalculation();
735}
qreal v
@ NONE
static QCursor moveCursor()
static QCursor arrowCursor()
Definition kis_cursor.cc:24
static QCursor pointingHandCursor()
KisPerspectiveTransformStrategy(const KisCoordinatesConverter *converter, KoSnapGuide *snapGuide, ToolTransformArgs &currentArgs, TransformTransactionProperties &transaction)
void paint(QPainter &gc, const KoColorDisplayRendererInterface *displayRendererInterface) override
void setTransformFunction(const QPointF &mousePos, bool perspectiveModifierActive, bool shiftModifierActive, bool altModifierActive) override
void continuePrimaryAction(const QPointF &pt, bool shiftModifierActive, bool altModifierActive) override
bool beginPrimaryAction(const QPointF &pt) override
bool addFunction(const QPointF &pt, qreal radius, Function function)
static qreal effectiveHandleGrabRadius(const KisCoordinatesConverter *converter)
static const int handleVisualRadius
static QRectF handleRect(qreal radius, const QTransform &t, const QRectF &limitingRect, qreal *dOutX, qreal *dOutY)
virtual QColor convertColorToDisplayColorSpace(const KoColor color) const =0
convertColorToDisplayColorSpace
static bool qFuzzyCompare(half p1, half p2)
#define KIS_ASSERT_RECOVER_RETURN(cond)
Definition kis_assert.h:75
T kisDegreesToRadians(T degrees)
Definition kis_global.h:176
qreal kisDistanceToLine(const QPointF &m, const QLineF &line)
Definition kis_global.h:234
qreal kisSquareDistance(const QPointF &pt1, const QPointF &pt2)
Definition kis_global.h:194
Eigen::Matrix3f fromShear(qreal sx, qreal sy)
Eigen::Matrix3f fromScale(qreal sx, qreal sy)
Eigen::Matrix3f fromTranslate(const QPointF &pt)
QVector4D fromQPointF(const QPointF &pt)
Eigen::Matrix3f fromQTransform(const QTransform &t)
QTransform toQTransform(const Eigen::Matrix3f &m)
Eigen::Matrix3f getTransitionMatrix(const QVector< QPointF > &sp)
QPointF toQPointF(const ExpressionType &expr)
Definition kis_vec.h:29
QTransform shearTransform() const
QTransform rotateTransform() const
QTransform scaleTransform() const
QCursor getScaleCursor(const QPointF &handlePt)
Private(KisPerspectiveTransformStrategy *_q, const KisCoordinatesConverter *_converter, ToolTransformArgs &_currentArgs, TransformTransactionProperties &_transaction)
QCursor getShearCursor(const QPointF &start, const QPointF &end)
const KisCoordinatesConverter * converter
standard members ///
static KoColorSpaceRegistry * instance()