12#include <QPainterPath>
30 DRAG_X_VANISHING_POINT,
31 DRAG_Y_VANISHING_POINT,
123 m_d(new
Private(this, converter, currentArgs, transaction))
151 v = QVector4D(1, 0, 0, 0);
156 v = QVector4D(0, 1, 0, 0);
164 Q_UNUSED(perspectiveModifierActive);
165 Q_UNUSED(shiftModifierActive);
166 Q_UNUSED(altModifierActive);
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);
175 if (!
m_d->transformedHandles.xVanishing.isNull()) {
177 handleRadius, DRAG_X_VANISHING_POINT);
180 if (!
m_d->transformedHandles.yVanishing.isNull()) {
182 handleRadius, DRAG_Y_VANISHING_POINT);
185 m_d->currentDraggingHandlePoint = -1;
186 for (
int i = 0; i <
m_d->dstHandlePoints.size(); i++) {
188 handleRadius, DRAG_HANDLE)) {
190 m_d->currentDraggingHandlePoint = i;
201 switch (
m_d->function) {
209 case DRAG_X_VANISHING_POINT:
210 case DRAG_Y_VANISHING_POINT:
222 gc.setOpacity(
m_d->transaction.basePreviewOpacity());
223 gc.setTransform(
m_d->paintingTransform,
true);
229 QPainterPath handles;
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());
238 auto addHandleRectFunc =
239 [&](
const QPointF &pt) {
242 m_d->handlesTransform,
243 m_d->transaction.originalRect(), pt)
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());
258 if (
m_d->isTransforming) {
274 QPainterPath mappedHandles =
m_d->handlesTransform.map(handles);
278 pen[0].setCosmetic(
true);
280 pen[1].setCosmetic(
true);
283 for (
int i = 1; i >= 0; --i) {
285 gc.drawPath(mappedHandles);
291 QPainterPath perspectiveHandles;
296 m_d->transaction.originalRect(), 0, 0);
298 if (
m_d->transformedHandles.xVanishingExists) {
299 QRectF rc = handleRect.translated(
m_d->transformedHandles.xVanishing);
300 perspectiveHandles.addEllipse(rc);
303 if (
m_d->transformedHandles.yVanishingExists) {
304 QRectF rc = handleRect.translated(
m_d->transformedHandles.yVanishing);
305 perspectiveHandles.addEllipse(rc);
308 if (!perspectiveHandles.isEmpty()) {
310 gc.setTransform(
m_d->converter->imageToWidgetTransform());
312 gc.setBrush(Qt::red);
314 for (
int i = 1; i >= 0; --i) {
316 gc.drawPath(perspectiveHandles);
326 m_d->recalculateTransformations();
333 if (
m_d->function ==
NONE)
return false;
336 m_d->clickArgs =
m_d->currentArgs;
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()
350 v3 << sp[HANDLE_BOTTOM_RIGHT].x() , sp[HANDLE_BOTTOM_RIGHT].y() , 1;
352 Eigen::Vector3f coeffs =
A.colPivHouseholderQr().solve(v3);
354 A.col(0) *= coeffs(0);
355 A.col(1) *= coeffs(1);
356 A.col(2) *= coeffs(2);
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));
372 m << t.m11() , t.m21() , t.m31()
373 ,t.m12() , t.m22() , t.m32()
374 ,t.m13() , t.m23() , t.m33();
414 Eigen::Matrix3f TS =
fromTranslate(-currentArgs.originalCenter());
416 Eigen::Matrix3f m = t * TS.inverse();
418 qreal tX = m(0,2) / m(2,2);
419 qreal tY = m(1,2) / m(2,2);
443 currentArgs.setScaleX(dm.
scaleX);
444 currentArgs.setScaleY(dm.
scaleY);
446 currentArgs.setShearX(dm.
shearXY);
447 currentArgs.setShearY(0.0);
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);
461 currentArgs.setTransformedCenter(QPointF(tX, tY));
462 currentArgs.setFlattenedPerspectiveTransform(
toQTransform(m));
472 return QVector4D(pt.x(), pt.y(), 0, 1.0);
476 return v.toVector2DAffine().toPointF();
481 Q_UNUSED(shiftModifierActive);
482 Q_UNUSED(altModifierActive);
484 m_d->isTransforming =
true;
486 switch (
m_d->function) {
490 QPointF diff = mousePos -
m_d->clickPos;
491 m_d->currentArgs.setTransformedCenter(
492 m_d->clickArgs.transformedCenter() + diff);
498 if (
m_d->currentDraggingHandlePoint < HANDLE_MIDDLE_TOP) {
500 m_d->dstHandlePoints[
m_d->currentDraggingHandlePoint] = mousePos;
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;
509 case HANDLE_MIDDLE_BOTTOM:
510 m_d->dstHandlePoints[HANDLE_BOTTOM_LEFT] += delta;
511 m_d->dstHandlePoints[HANDLE_BOTTOM_RIGHT] += delta;
513 case HANDLE_MIDDLE_LEFT:
514 m_d->dstHandlePoints[HANDLE_TOP_LEFT] += delta;
515 m_d->dstHandlePoints[HANDLE_BOTTOM_LEFT] += delta;
517 case HANDLE_MIDDLE_RIGHT:
518 m_d->dstHandlePoints[HANDLE_TOP_RIGHT] += delta;
519 m_d->dstHandlePoints[HANDLE_BOTTOM_RIGHT] += delta;
526 Eigen::Matrix3f result =
B *
A.inverse();
528 m_d->transformIntoArgs(result);
532 case DRAG_X_VANISHING_POINT:
533 case DRAG_Y_VANISHING_POINT: {
535 QMatrix4x4 m(
m_d->transform);
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();
542 QVector4D
v(1,0,0,0);
543 QVector4D otherV(0,1,0,0);
545 if (
m_d->function == DRAG_X_VANISHING_POINT) {
546 v = QVector4D(1,0,0,0);
547 otherV = QVector4D(0,1,0,0);
549 v = QVector4D(0,1,0,0);
550 otherV = QVector4D(1,0,0,0);
558 QPointF otherV_dst =
toQPointF(m * otherV);
573 if (
m_d->function == DRAG_X_VANISHING_POINT) {
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);
633 srcPoints << far1_src;
634 srcPoints << far2_src;
635 srcPoints << near1_src;
636 srcPoints << near2_src;
638 dstPoints << far1_dst;
639 dstPoints << far2_dst;
640 dstPoints << near1_dst;
641 dstPoints << near2_dst;
645 Eigen::Matrix3f result =
B *
A.inverse();
647 m_d->transformIntoArgs(result);
652 m_d->recalculateTransformations();
657 bool shouldSave = !
m_d->imageTooBig;
658 m_d->isTransforming =
false;
660 if (
m_d->imageTooBig) {
661 m_d->currentArgs =
m_d->clickArgs;
662 m_d->recalculateTransformations();
670 transform = transformFromArgs();
672 QTransform viewScaleTransform = converter->imageToDocumentTransform() * converter->documentToFlakeTransform();
673 handlesTransform = transform * viewScaleTransform;
675 QTransform tl = QTransform::fromTranslate(transaction.originalTopLeft().x(), transaction.originalTopLeft().y());
676 paintingTransform = tl.inverted() * q->thumbToImageTransform() * tl * transform * viewScaleTransform;
677 paintingOffset = transaction.originalTopLeft();
680 const qreal maxScale = 20.0;
684 if (qAbs(currentArgs.scaleX()) > maxScale ||
685 qAbs(currentArgs.scaleY()) > maxScale) {
691 points << transaction.originalRect().topLeft();
692 points << transaction.originalRect().topRight();
693 points << transaction.originalRect().bottomRight();
694 points << transaction.originalRect().bottomLeft();
696 for (
int i = 0; i < points.size(); i++) {
697 points[i] = transform.map(points[i]);
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];
706 QLineF l1(pt, other);
707 QLineF l2(prev, next);
709 QPointF intersection;
710 l1.intersects(l2, &intersection);
721 const qreal thresholdDistance = 0.02 * l2.length();
731 recalculateTransformedHandles();
733 Q_EMIT q->requestShowImageTooBig(imageTooBig);
734 Q_EMIT q->requestImageRecalculation();
static QCursor moveCursor()
static QCursor arrowCursor()
static QCursor pointingHandCursor()
virtual QColor convertColorToDisplayColorSpace(const KoColor color) const =0
convertColorToDisplayColorSpace
static bool qFuzzyCompare(half p1, half p2)
#define KIS_ASSERT_RECOVER_RETURN(cond)
T kisDegreesToRadians(T degrees)
qreal kisDistanceToLine(const QPointF &m, const QLineF &line)
qreal kisSquareDistance(const QPointF &pt1, const QPointF &pt2)
QPointF toQPointF(const ExpressionType &expr)
QTransform shearTransform() const
QTransform rotateTransform() const
QTransform scaleTransform() const
static KoColorSpaceRegistry * instance()