Krita Source Code Documentation
Loading...
Searching...
No Matches
KisPerspectiveTransformStrategy Class Reference

#include <kis_perspective_transform_strategy.h>

+ Inheritance diagram for KisPerspectiveTransformStrategy:

Classes

struct  Private
 

Signals

void requestCanvasUpdate ()
 
void requestImageRecalculation ()
 
void requestShowImageTooBig (bool value)
 

Public Member Functions

bool beginPrimaryAction (const QPointF &pt) override
 
virtual bool beginPrimaryAction (KoPointerEvent *event)=0
 
void continuePrimaryAction (const QPointF &pt, bool shiftModifierActive, bool altModifierActive) override
 
virtual void continuePrimaryAction (KoPointerEvent *event)=0
 
bool endPrimaryAction () override
 
virtual bool endPrimaryAction (KoPointerEvent *event)=0
 
void externalConfigChanged () override
 
QCursor getCurrentCursor () const override
 
 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
 
 ~KisPerspectiveTransformStrategy () override
 
- Public Member Functions inherited from KisSimplifiedActionPolicyStrategy
void activateAlternateAction (KisTool::AlternateAction action) override
 
void activatePrimaryAction () override
 
bool beginAlternateAction (KoPointerEvent *event, KisTool::AlternateAction action) override
 
bool beginPrimaryAction (KoPointerEvent *event) override
 
void continueAlternateAction (KoPointerEvent *event, KisTool::AlternateAction action) override
 
void continuePrimaryAction (KoPointerEvent *event) override
 
void deactivateAlternateAction (KisTool::AlternateAction action) override
 
bool endAlternateAction (KoPointerEvent *event, KisTool::AlternateAction action) override
 
bool endPrimaryAction (KoPointerEvent *event) override
 
virtual QPointF handleSnapPoint (const QPointF &imagePos)
 
void hoverActionCommon (KoPointerEvent *event) override
 
 KisSimplifiedActionPolicyStrategy (const KisCoordinatesConverter *_converter, KoSnapGuide *snapGuide=0)
 
 ~KisSimplifiedActionPolicyStrategy () override
 
- Public Member Functions inherited from KisTransformStrategyBase
virtual bool acceptsClicks () const
 
virtual void deactivatePrimaryAction ()
 
virtual int decorationThickness () const
 
virtual void decreaseBrushSize (KoCanvasBase *canvas)
 
virtual QPainterPath getCursorOutline () const
 
virtual void increaseBrushSize (KoCanvasBase *canvas)
 
 KisTransformStrategyBase ()
 
QImage originalImage () const
 
virtual void setDecorationThickness (int thickness)
 
void setThumbnailImage (const QImage &image, QTransform thumbToImageTransform)
 
QTransform thumbToImageTransform () const
 
 ~KisTransformStrategyBase () override
 

Private Attributes

const QScopedPointer< Privatem_d
 

Additional Inherited Members

- Protected Member Functions inherited from KisSimplifiedActionPolicyStrategy
virtual void hoverActionCommon (const QPointF &pt)
 
virtual bool shiftModifierIsUsed () const
 

Detailed Description

Definition at line 22 of file kis_perspective_transform_strategy.h.

Constructor & Destructor Documentation

◆ KisPerspectiveTransformStrategy()

KisPerspectiveTransformStrategy::KisPerspectiveTransformStrategy ( const KisCoordinatesConverter * converter,
KoSnapGuide * snapGuide,
ToolTransformArgs & currentArgs,
TransformTransactionProperties & transaction )

Definition at line 118 of file kis_perspective_transform_strategy.cpp.

122 : KisSimplifiedActionPolicyStrategy(converter, snapGuide),
123 m_d(new Private(this, converter, currentArgs, transaction))
124{
125}
KisSimplifiedActionPolicyStrategy(const KisCoordinatesConverter *_converter, KoSnapGuide *snapGuide=0)

◆ ~KisPerspectiveTransformStrategy()

KisPerspectiveTransformStrategy::~KisPerspectiveTransformStrategy ( )
override

Definition at line 127 of file kis_perspective_transform_strategy.cpp.

128{
129}

Member Function Documentation

◆ beginPrimaryAction() [1/2]

bool KisPerspectiveTransformStrategy::beginPrimaryAction ( const QPointF & pt)
overridevirtual

Implements KisSimplifiedActionPolicyStrategy.

Definition at line 329 of file kis_perspective_transform_strategy.cpp.

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}
@ NONE

References m_d, and NONE.

◆ beginPrimaryAction() [2/2]

virtual bool KisTransformStrategyBase::beginPrimaryAction ( KoPointerEvent * event)
virtual

◆ continuePrimaryAction() [1/2]

void KisPerspectiveTransformStrategy::continuePrimaryAction ( const QPointF & pt,
bool shiftModifierActive,
bool altModifierActive )
overridevirtual

Implements KisSimplifiedActionPolicyStrategy.

Definition at line 479 of file kis_perspective_transform_strategy.cpp.

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}
qreal v
#define KIS_ASSERT_RECOVER_RETURN(cond)
Definition kis_assert.h:75
qreal kisSquareDistance(const QPointF &pt1, const QPointF &pt2)
Definition kis_global.h:194
QVector4D fromQPointF(const QPointF &pt)
Eigen::Matrix3f getTransitionMatrix(const QVector< QPointF > &sp)
QPointF toQPointF(const ExpressionType &expr)
Definition kis_vec.h:29

References A, B, fromQPointF(), getTransitionMatrix(), KIS_ASSERT_RECOVER_RETURN, kisSquareDistance(), m_d, MOVE, NONE, toQPointF(), and v.

◆ continuePrimaryAction() [2/2]

virtual void KisTransformStrategyBase::continuePrimaryAction ( KoPointerEvent * event)
virtual

◆ endPrimaryAction() [1/2]

bool KisPerspectiveTransformStrategy::endPrimaryAction ( )
overridevirtual

Implements KisSimplifiedActionPolicyStrategy.

Definition at line 655 of file kis_perspective_transform_strategy.cpp.

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}

References m_d.

◆ endPrimaryAction() [2/2]

virtual bool KisTransformStrategyBase::endPrimaryAction ( KoPointerEvent * event)
virtual

◆ externalConfigChanged()

void KisPerspectiveTransformStrategy::externalConfigChanged ( )
overridevirtual

Implements KisTransformStrategyBase.

Definition at line 324 of file kis_perspective_transform_strategy.cpp.

325{
326 m_d->recalculateTransformations();
327}

References m_d.

◆ getCurrentCursor()

QCursor KisPerspectiveTransformStrategy::getCurrentCursor ( ) const
overridevirtual

Implements KisTransformStrategyBase.

Definition at line 197 of file kis_perspective_transform_strategy.cpp.

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}
static QCursor moveCursor()
static QCursor arrowCursor()
Definition kis_cursor.cc:24
static QCursor pointingHandCursor()

References KisCursor::arrowCursor(), m_d, MOVE, KisCursor::moveCursor(), NONE, and KisCursor::pointingHandCursor().

◆ paint()

void KisPerspectiveTransformStrategy::paint ( QPainter & gc,
const KoColorDisplayRendererInterface * displayRendererInterface )
overridevirtual

WARNING: we cannot install a transform to paint the handles here!

There is a bug in Qt that prevents painting of cosmetic-pen brushes in openGL mode when a TxProject matrix is active on a QPainter. So just convert it manually.

https://bugreports.qt-project.org/browse/QTBUG-42658

Implements KisTransformStrategyBase.

Definition at line 218 of file kis_perspective_transform_strategy.cpp.

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}
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 KoColorSpaceRegistry * instance()

References KoColorDisplayRendererInterface::convertColorToDisplayColorSpace(), KisTransformStrategyBase::decorationThickness(), KisTransformUtils::handleRect(), KisTransformUtils::handleVisualRadius, KoColorSpaceRegistry::instance(), m_d, and KisTransformStrategyBase::originalImage().

◆ requestCanvasUpdate

void KisPerspectiveTransformStrategy::requestCanvasUpdate ( )
signal

◆ requestImageRecalculation

void KisPerspectiveTransformStrategy::requestImageRecalculation ( )
signal

◆ requestShowImageTooBig

void KisPerspectiveTransformStrategy::requestShowImageTooBig ( bool value)
signal

◆ setTransformFunction()

void KisPerspectiveTransformStrategy::setTransformFunction ( const QPointF & mousePos,
bool perspectiveModifierActive,
bool shiftModifierActive,
bool altModifierActive )
overridevirtual

Implements KisSimplifiedActionPolicyStrategy.

Definition at line 162 of file kis_perspective_transform_strategy.cpp.

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}
static qreal effectiveHandleGrabRadius(const KisCoordinatesConverter *converter)

References KisTransformUtils::HandleChooser< Function >::addFunction(), KisTransformUtils::effectiveHandleGrabRadius(), KisTransformUtils::HandleChooser< Function >::function(), m_d, MOVE, and NONE.

Member Data Documentation

◆ m_d

const QScopedPointer<Private> KisPerspectiveTransformStrategy::m_d
private

Definition at line 53 of file kis_perspective_transform_strategy.h.


The documentation for this class was generated from the following files: