Krita Source Code Documentation
Loading...
Searching...
No Matches
kis_tool_transform.cc
Go to the documentation of this file.
1/*
2 * kis_tool_transform.cc -- part of Krita
3 *
4 * SPDX-FileCopyrightText: 2004 Boudewijn Rempt <boud@valdyas.org>
5 * SPDX-FileCopyrightText: 2005 C. Boemann <cbo@boemann.dk>
6 * SPDX-FileCopyrightText: 2010 Marc Pegon <pe.marc@free.fr>
7 * SPDX-FileCopyrightText: 2013 Dmitry Kazakov <dimula73@gmail.com>
8 *
9 * SPDX-License-Identifier: GPL-2.0-or-later
10 */
11
12#include "kis_tool_transform.h"
13
14
15#include <math.h>
16#include <limits>
17
18#include <QPainter>
19#include <QPen>
20#include <QObject>
21#include <QApplication>
22#include <QMatrix4x4>
23#include <QMenu>
24
25#include <kis_debug.h>
26#include <klocalizedstring.h>
27
28#include <KoPointerEvent.h>
29#include <KoID.h>
30#include <KoCanvasBase.h>
31#include <KoViewConverter.h>
32#include <KoSelection.h>
33#include <KoCompositeOp.h>
35
36#include <kis_global.h>
37#include <canvas/kis_canvas2.h>
38#include <KisViewManager.h>
39#include <kis_painter.h>
40#include <kis_cursor.h>
41#include <kis_image.h>
42#include <kis_undo_adapter.h>
43#include <kis_transaction.h>
44#include <kis_selection.h>
45#include <kis_filter_strategy.h>
47#include <kis_statusbar.h>
51#include <kis_pixel_selection.h>
52#include <kis_shape_selection.h>
54#include <krita_utils.h>
57
59#include <KoCanvasController.h>
60
61#include "kis_action_registry.h"
62
64
65#include "kis_transform_utils.h"
72
73#include "kis_transform_mask.h"
75
77#include "kis_layer_utils.h"
79#include "kis_config_notifier.h"
80
83
85 : KisTool(canvas, KisCursor::rotateCursor())
86 , m_warpStrategy(
88 dynamic_cast<KisCanvas2*>(canvas)->coordinatesConverter(),
89 dynamic_cast<KisCanvas2*>(canvas)->snapGuide(),
90 m_currentArgs, m_transaction))
91 , m_cageStrategy(
93 dynamic_cast<KisCanvas2*>(canvas)->coordinatesConverter(),
94 dynamic_cast<KisCanvas2*>(canvas)->snapGuide(),
95 m_currentArgs, m_transaction))
96 , m_liquifyStrategy(
98 dynamic_cast<KisCanvas2*>(canvas)->coordinatesConverter(),
99 m_currentArgs, m_transaction, canvas->resourceManager()))
100 , m_meshStrategy(
102 dynamic_cast<KisCanvas2*>(canvas)->coordinatesConverter(),
103 dynamic_cast<KisCanvas2*>(canvas)->snapGuide(),
104 m_currentArgs, m_transaction))
105 , m_freeStrategy(
107 dynamic_cast<KisCanvas2*>(canvas)->coordinatesConverter(),
108 dynamic_cast<KisCanvas2*>(canvas)->snapGuide(),
109 m_currentArgs, m_transaction))
110 , m_perspectiveStrategy(
112 dynamic_cast<KisCanvas2*>(canvas)->coordinatesConverter(),
113 dynamic_cast<KisCanvas2*>(canvas)->snapGuide(),
114 m_currentArgs, m_transaction))
115 , m_moveShortcutsHelper(this)
116{
117 m_canvas = dynamic_cast<KisCanvas2*>(canvas);
118 Q_ASSERT(m_canvas);
119
120 setObjectName("tool_transform");
121 m_optionsWidget = 0;
122
123 warpAction = new KisAction(i18nc("Warp Transform Tab Label", "Warp"));
124 liquifyAction = new KisAction(i18nc("Liquify Transform Tab Label", "Liquify"));
125 meshAction = new KisAction(i18nc("Mesh Transform Tab Label", "Mesh"));
126 cageAction = new KisAction(i18nc("Cage Transform Tab Label", "Cage"));
127 freeTransformAction = new KisAction(i18nc("Free Transform Tab Label", "Free"));
128 perspectiveAction = new KisAction(i18nc("Perspective Transform Tab Label", "Perspective"));
129
130 // extra actions for free transform that are in the tool options
131 mirrorHorizontalAction = new KisAction(i18n("Mirror Horizontal"));
132 mirrorVerticalAction = new KisAction(i18n("Mirror Vertical"));
133 rotateNinetyCWAction = new KisAction(i18n("Rotate 90 degrees Clockwise"));
134 rotateNinetyCCWAction = new KisAction(i18n("Rotate 90 degrees CounterClockwise"));
135
136 keepAspectRatioAction = new KisAction(i18n("Keep Aspect Ratio"));
137 keepAspectRatioAction->setCheckable(true);
138 keepAspectRatioAction->setChecked(false);
139
140 applyTransformation = new KisAction(i18n("Apply"));
141 resetTransformation = new KisAction(i18n("Reset"));
142
143 m_contextMenu.reset(new QMenu());
144
145 connect(m_warpStrategy.data(), SIGNAL(requestCanvasUpdate()), SLOT(canvasUpdateRequested()));
146 connect(m_warpStrategy.data(), SIGNAL(requestImageRecalculation()), SLOT(requestImageRecalculation()));
147 connect(m_cageStrategy.data(), SIGNAL(requestCanvasUpdate()), SLOT(canvasUpdateRequested()));
148 connect(m_cageStrategy.data(), SIGNAL(requestImageRecalculation()), SLOT(requestImageRecalculation()));
149 connect(m_liquifyStrategy.data(), SIGNAL(requestCanvasUpdate()), SLOT(canvasUpdateRequested()));
150 connect(m_liquifyStrategy.data(), SIGNAL(requestCursorOutlineUpdate(QPointF)), SLOT(cursorOutlineUpdateRequested(QPointF)));
151 connect(m_liquifyStrategy.data(), SIGNAL(requestUpdateOptionWidget()), SLOT(updateOptionWidget()));
153 connect(m_freeStrategy.data(), SIGNAL(requestCanvasUpdate()), SLOT(canvasUpdateRequested()));
154 connect(m_freeStrategy.data(), SIGNAL(requestResetRotationCenterButtons()), SLOT(resetRotationCenterButtonsRequested()));
155 connect(m_freeStrategy.data(), SIGNAL(requestShowImageTooBig(bool)), SLOT(imageTooBigRequested(bool)));
156 connect(m_freeStrategy.data(), SIGNAL(requestImageRecalculation()), SLOT(requestImageRecalculation()));
157 connect(m_freeStrategy.data(), SIGNAL(requestConvexHullCalculation()), SLOT(convexHullCalculationRequested()));
158 connect(m_perspectiveStrategy.data(), SIGNAL(requestCanvasUpdate()), SLOT(canvasUpdateRequested()));
159 connect(m_perspectiveStrategy.data(), SIGNAL(requestShowImageTooBig(bool)), SLOT(imageTooBigRequested(bool)));
161 connect(m_meshStrategy.data(), SIGNAL(requestCanvasUpdate()), SLOT(canvasUpdateRequested()));
162 connect(m_meshStrategy.data(), SIGNAL(requestImageRecalculation()), SLOT(requestImageRecalculation()));
163
164 connect(&m_changesTracker, SIGNAL(sigConfigChanged(KisToolChangesTrackerDataSP)),
166
167 connect(KisConfigNotifier::instance(), SIGNAL(configChanged()), SLOT(slotGlobalConfigChanged()));
168}
169
171{
172 cancelStroke();
173
174 delete warpAction;
175 delete meshAction;
176 delete liquifyAction;
177 delete cageAction;
178 delete freeTransformAction;
179 delete perspectiveAction;
180 delete applyTransformation;
181 delete resetTransformation;
187}
188
190{
191 Q_EMIT freeTransformChanged();
192 m_canvas->updateCanvas();
193}
194
196{
197 m_canvas->updateCanvas();
198}
199
204
206{
207 KConfigGroup group = KSharedConfig::openConfig()->group(toolId());
208 m_preferOverlayPreviewStyle = group.readEntry("useOverlayPreviewStyle", false);
209 m_forceLodMode = group.readEntry("forceLodMode", true);
210}
211
217
223
246void KisToolTransform::slotConvexHullCalculated(QPolygon hull, void *strokeStrategyCookie)
247{
248 if (!m_strokeId || strokeStrategyCookie != m_strokeStrategyCookie) return;
249 QPolygonF hullF = hull;
256 if (hullF.boundingRect() == m_transaction.originalRect()) {
260 } else {
261 warnTools << "WARNING: KisToolTransform: calculated convex hull's bounds "
262 "differ from the bounding rect of the source clip. It shouldn't "
263 "have happened";
264 }
265}
266
268{
270 return m_freeStrategy.data();
272 return m_warpStrategy.data();
274 return m_cageStrategy.data();
276 return m_liquifyStrategy.data();
278 return m_meshStrategy.data();
279 } else /* if (m_currentArgs.mode() == ToolTransformArgs::PERSPECTIVE_4POINT) */ {
280 return m_perspectiveStrategy.data();
281 }
282}
283
284void KisToolTransform::paint(QPainter& gc, const KoViewConverter &converter)
285{
286 Q_UNUSED(converter);
287
288 if (!m_strokeId || m_transaction.rootNodes().isEmpty()) return;
289
290 QRectF newRefRect = KisTransformUtils::imageToFlake(m_canvas->coordinatesConverter(), QRectF(0.0,0.0,1.0,1.0));
291 if (m_refRect != newRefRect) {
292 m_refRect = newRefRect;
294 }
296 currentStrategy()->paint(gc);
297
298
299 if (!m_cursorOutline.isEmpty()) {
300 QPainterPath mappedOutline =
302 m_canvas->coordinatesConverter()).map(m_cursorOutline);
303 paintToolOutline(&gc, mappedOutline);
304 }
305}
306
308{
310 return;
311 }
312
313 if (!m_strokeId) {
315 } else if (m_strokeId && m_transaction.rootNodes().isEmpty()) {
316 // we are in the middle of stroke initialization
318 } else {
319 useCursor(currentStrategy()->getCurrentCursor());
320 }
321}
322
324{
325 QRect canvasUpdateRect;
326
327 if (!m_cursorOutline.isEmpty()) {
328 canvasUpdateRect = m_canvas->coordinatesConverter()->
329 imageToDocument(m_cursorOutline.boundingRect()).toAlignedRect();
330 }
331
333 getCursorOutline().translated(imagePos);
334
335 if (!m_cursorOutline.isEmpty()) {
336 canvasUpdateRect |=
337 m_canvas->coordinatesConverter()->
338 imageToDocument(m_cursorOutline.boundingRect()).toAlignedRect();
339 }
340
341 if (!canvasUpdateRect.isEmpty()) {
342 // grow rect a bit to follow interpolation fuzziness
343 canvasUpdateRect = kisGrowRect(canvasUpdateRect, 2);
344 m_canvas->updateCanvas(canvasUpdateRect);
345 }
346}
347
349{
350 if (!nodeEditable()) {
351 event->ignore();
352 return;
353 }
354
355 if (!m_strokeId) {
357 } else if (!m_transaction.rootNodes().isEmpty()) {
358 bool result = false;
359
360 if (usePrimaryAction) {
361 result = currentStrategy()->beginPrimaryAction(event);
362 } else {
363 result = currentStrategy()->beginAlternateAction(event, action);
364 }
365
366 if (result) {
368 }
369 }
370
372
374}
375
377{
378 if (mode() != KisTool::PAINT_MODE) return;
379 if (m_transaction.rootNodes().isEmpty()) return;
380
382
383 if (usePrimaryAction) {
385 } else {
387 }
388
391}
392
394{
395 if (mode() != KisTool::PAINT_MODE) return;
396
398
400 currentStrategy()->acceptsClicks()) {
401
402 bool result = false;
403
404 if (usePrimaryAction) {
405 result = currentStrategy()->endPrimaryAction(event);
406 } else {
407 result = currentStrategy()->endAlternateAction(event, action);
408 }
409
410 if (result) {
412 }
413
415 }
416
419}
420
422{
423 if (m_contextMenu) {
424 m_contextMenu->clear();
425
426 m_contextMenu->addSection(i18n("Transform Tool Actions"));
427 // add a quick switch to different transform types
430 m_contextMenu->addAction(warpAction);
431 m_contextMenu->addAction(cageAction);
432 m_contextMenu->addAction(liquifyAction);
433 m_contextMenu->addAction(meshAction);
434
435 // extra options if free transform is selected
437 m_contextMenu->addSeparator();
442
443 m_contextMenu->addSeparator();
446 }
447
448 m_contextMenu->addSeparator();
451 }
452
453 return m_contextMenu.data();
454}
455
460
465
470
476
481
487
492
497
502
507
509{
510 // When using touch drawing, we only ever receive move events after the
511 // finger has pressed down. This confuses the strategies greatly, since they
512 // expect to receive a hover to tell which anchor the user wants to
513 // manipulate or similar. So in this case, we send an artificial hover.
514 if (event->isTouchEvent() && this->mode() != KisTool::PAINT_MODE) {
517 }
519}
520
522{
523 QPointF mousePos = m_canvas->coordinatesConverter()->documentToImage(event->point);
524
526
527 if (this->mode() != KisTool::PAINT_MODE) {
531 return;
532 }
533}
534
539
544
549
551{
553
554 switch (m_currentArgs.mode())
555 {
558 break;
561 break;
564 break;
567 break;
570 break;
573 break;
574 default:
575 KIS_ASSERT_RECOVER_NOOP(0 && "unexpected transform mode");
576 }
577
578 return mode;
579}
580
582{
583 return m_currentArgs.transformedCenter().x();
584}
585
587{
588 return m_currentArgs.transformedCenter().y();
589}
590
592{
593 return m_currentArgs.aX();
594}
595
597{
598 return m_currentArgs.aY();
599}
600
602{
603 return m_currentArgs.aZ();
604}
605
607{
608 return m_currentArgs.scaleX();
609}
610
612{
613 return m_currentArgs.scaleY();
614}
615
617{
618 return m_currentArgs.shearX();
619}
620
622{
623 return m_currentArgs.shearY();
624}
625
639
641{
642 return m_currentArgs.alpha();
643}
644
649
679
681{
683
684 if( mode != m_currentArgs.mode() ) {
685 if( newMode == FreeTransformMode ) {
687 } else if( newMode == WarpTransformMode ) {
689 } else if( newMode == CageTransformMode ) {
691 } else if( newMode == LiquifyTransformMode ) {
693 } else if( newMode == PerspectiveTransformMode ) {
695 } else if( newMode == MeshTransformMode ) {
697 }
698
699 Q_EMIT transformModeChanged();
700 }
701}
702
703void KisToolTransform::setRotateX( double rotation )
704{
705 m_currentArgs.setAX( rotation );
706}
707
708void KisToolTransform::setRotateY( double rotation )
709{
710 m_currentArgs.setAY( rotation );
711}
712
713void KisToolTransform::setRotateZ( double rotation )
714{
715 m_currentArgs.setAZ( rotation );
716}
717
734
735void KisToolTransform::setWarpFlexibility( double flexibility )
736{
737 m_currentArgs.setAlpha( flexibility );
738}
739
744
750
759
761{
762 QImage origImg;
763 m_selectedPortionCache = previewDevice;
764
765 QTransform thumbToImageTransform;
766
767 const int maxSize = 2000;
768
769 QRect srcRect(m_transaction.originalRect().toAlignedRect());
770 int x, y, w, h;
771 srcRect.getRect(&x, &y, &w, &h);
772
774 if (w > maxSize || h > maxSize) {
775 qreal scale = qreal(maxSize) / (w > h ? w : h);
776 QTransform scaleTransform = QTransform::fromScale(scale, scale);
777
778 QRect thumbRect = scaleTransform.mapRect(m_transaction.originalRect()).toAlignedRect();
779
780 origImg = m_selectedPortionCache->
781 createThumbnailUncached(thumbRect.width(),
782 thumbRect.height(),
783 srcRect, 1,
786 thumbToImageTransform = scaleTransform.inverted();
787
788 } else {
789 origImg = m_selectedPortionCache->convertToQImage(0, x, y, w, h,
792 thumbToImageTransform = QTransform();
793 }
794 }
795
796 // init both strokes since the thumbnail is initialized only once
797 // during the stroke
798 m_freeStrategy->setThumbnailImage(origImg, thumbToImageTransform);
799 m_perspectiveStrategy->setThumbnailImage(origImg, thumbToImageTransform);
800 m_warpStrategy->setThumbnailImage(origImg, thumbToImageTransform);
801 m_cageStrategy->setThumbnailImage(origImg, thumbToImageTransform);
802 m_liquifyStrategy->setThumbnailImage(origImg, thumbToImageTransform);
803 m_meshStrategy->setThumbnailImage(origImg, thumbToImageTransform);
804}
805
807{
808 m_externalSourceForNextActivation = externalSource;
809 if (isActive()) {
810 QSet<KoShape*> dummy;
811 deactivate();
812 activate(dummy);
813 } else {
814 KoToolManager::instance()->switchToolRequested("KisToolTransform");
815 }
816}
817
818void KisToolTransform::activate(const QSet<KoShape*> &shapes)
819{
820 KisTool::activate(shapes);
821
825
827
828 m_actionConnections.addConnection(action("movetool-move-up"), SIGNAL(triggered(bool)),
829 this, SLOT(slotMoveDiscreteUp()));
830 m_actionConnections.addConnection(action("movetool-move-up-more"), SIGNAL(triggered(bool)),
831 this, SLOT(slotMoveDiscreteUpMore()));
832 m_actionConnections.addConnection(action("movetool-move-down"), SIGNAL(triggered(bool)),
833 this, SLOT(slotMoveDiscreteDown()));
834 m_actionConnections.addConnection(action("movetool-move-down-more"), SIGNAL(triggered(bool)),
835 this, SLOT(slotMoveDiscreteDownMore()));
836 m_actionConnections.addConnection(action("movetool-move-left"), SIGNAL(triggered(bool)),
837 this, SLOT(slotMoveDiscreteLeft()));
838 m_actionConnections.addConnection(action("movetool-move-left-more"), SIGNAL(triggered(bool)),
839 this, SLOT(slotMoveDiscreteLeftMore()));
840 m_actionConnections.addConnection(action("movetool-move-right"), SIGNAL(triggered(bool)),
841 this, SLOT(slotMoveDiscreteRight()));
842 m_actionConnections.addConnection(action("movetool-move-right-more"), SIGNAL(triggered(bool)),
843 this, SLOT(slotMoveDiscreteRightMore()));
844 m_actionConnections.addConnection(action("increase_brush_size"),
845 SIGNAL(triggered()),
846 this,
847 SLOT(slotIncreaseBrushSize()));
848 m_actionConnections.addConnection(action("decrease_brush_size"),
849 SIGNAL(triggered()),
850 this,
851 SLOT(slotDecreaseBrushSize()));
852
853 if (currentNode()) {
855 }
856
859}
860
869
871{
872 if (!m_strokeId || m_transaction.rootNodes().isEmpty() || mode() != HOVER_MODE) return;
873
874 if (!m_changesTracker.canUndo()) {
875 cancelStroke();
876 } else {
878 }
879}
880
882{
883 if (!m_strokeId || m_transaction.rootNodes().isEmpty()) return;
884
887 }
888}
889
894
896{
897 if (m_transaction.rootNodes().isEmpty() || m_currentArgs.isIdentity()) {
898 cancelStroke();
899 } else {
901 }
902}
903
914
916{
917 Q_ASSERT(!m_strokeId);
918
921
922 // set up and null checks before we do anything
923 KisResourcesSnapshotSP resources =
924 new KisResourcesSnapshot(image(), currentNode(), this->canvas()->resourceManager(), 0, selectedNodes(), 0);
925 KisNodeList rootNodes = resources->selectedNodes();
926 //Filter out any nodes that might be children of other selected nodes so they aren't used twice
929
932
933 Q_FOREACH (KisNodeSP currentNode, resources->selectedNodes()) {
934 KisCanvas2 *kisCanvas = dynamic_cast<KisCanvas2*>(canvas());
935 KIS_ASSERT(kisCanvas);
936
937 if (!currentNode || !currentNode->isEditable()) {
939 kisCanvas->viewManager()->
940 showFloatingMessage(
941 i18nc("floating message in transformation tool",
942 "Cannot transform locked layers"),
943 koIcon("object-locked"), 4000, KisFloatingMessage::High);
944 } else if (currentNode && !currentNode->visible()) {
945 kisCanvas->viewManager()->
946 showFloatingMessage(
947 i18nc("floating message in transformation tool",
948 "Cannot transform hidden layers"),
949 koIcon("object-locked"), 4000, KisFloatingMessage::High);
950 } else {
951 kisCanvas->viewManager()->
952 showFloatingMessage(
953 i18nc("floating message in transformation tool",
954 "Cannot use transform tool on this set of layers"),
955 koIcon("object-locked"), 4000, KisFloatingMessage::High);
956 }
957
958 return;
959 }
960
961 // some layer types cannot be transformed. Give a message and return if a user tries it
962 if (currentNode->inherits("KisColorizeMask") ||
963 currentNode->inherits("KisFileLayer") ||
964 currentNode->inherits("KisCloneLayer")) {
965
966 if(currentNode->inherits("KisColorizeMask")){
967 kisCanvas->viewManager()->
968 showFloatingMessage(
969 i18nc("floating message in transformation tool",
970 "Layer type cannot use the transform tool"),
971 koIcon("object-locked"), 4000, KisFloatingMessage::High);
972 }
973 else{
974 kisCanvas->viewManager()->
975 showFloatingMessage(
976 i18nc("floating message in transformation tool",
977 "Layer type cannot use the transform tool. Use transform mask instead."),
978 koIcon("object-locked"), 4000, KisFloatingMessage::High);
979 }
980 return;
981 }
982
983 KisNodeSP impossibleMask =
985 [currentNode] (KisNodeSP node) {
986 // we can process transform masks of the first level
987 if (node == currentNode || node->parent() == currentNode) return false;
988
989 return node->inherits("KisTransformMask") && node->visible(true);
990 });
991
992 if (impossibleMask) {
993 kisCanvas->viewManager()->
994 showFloatingMessage(
995 i18nc("floating message in transformation tool",
996 "Layer has children with transform masks. Please disable them before doing transformation."),
997 koIcon("object-locked"), 8000, KisFloatingMessage::High);
998 return;
999 }
1000
1005 if (selection && dynamic_cast<KisTransformMask*>(currentNode.data())) {
1006 kisCanvas->viewManager()->
1007 showFloatingMessage(
1008 i18nc("floating message in transformation tool",
1009 "Selections are not used when editing transform masks "),
1010 QIcon(), 4000, KisFloatingMessage::Low);
1011
1012 selection = 0;
1013 }
1014 }
1015 // Overlay preview is never used when transforming an externally provided image
1017
1018 KisStrokeStrategy *strategy = 0;
1019
1021 TransformStrokeStrategy *transformStrategy = new TransformStrokeStrategy(mode, m_currentArgs.filterId(), forceReset, rootNodes, selection, image().data(), image().data());
1022 connect(transformStrategy, SIGNAL(sigPreviewDeviceReady(KisPaintDeviceSP)), SLOT(slotPreviewDeviceGenerated(KisPaintDeviceSP)));
1023 connect(transformStrategy, SIGNAL(sigTransactionGenerated(TransformTransactionProperties, ToolTransformArgs, void*)), SLOT(slotTransactionGenerated(TransformTransactionProperties, ToolTransformArgs, void*)));
1024 connect(transformStrategy, SIGNAL(sigConvexHullCalculated(QPolygon, void*)), SLOT(slotConvexHullCalculated(QPolygon, void*)));
1025 strategy = transformStrategy;
1026
1027 // save unique identifier of the stroke so we could
1028 // recognize it when sigTransactionGenerated() is
1029 // received (theoretically, the user can start two
1030 // strokes at the same time, if he is quick enough)
1031 m_strokeStrategyCookie = transformStrategy;
1032
1033 } else {
1034 InplaceTransformStrokeStrategy *transformStrategy = new InplaceTransformStrokeStrategy(mode, m_currentArgs.filterId(), forceReset, rootNodes, selection, externalSource, image().data(), image().data(), image()->root(), m_forceLodMode);
1035 connect(transformStrategy, SIGNAL(sigTransactionGenerated(TransformTransactionProperties, ToolTransformArgs, void*)), SLOT(slotTransactionGenerated(TransformTransactionProperties, ToolTransformArgs, void*)));
1036 connect(transformStrategy, SIGNAL(sigConvexHullCalculated(QPolygon, void*)), SLOT(slotConvexHullCalculated(QPolygon, void*)));
1037 strategy = transformStrategy;
1038
1039 // save unique identifier of the stroke so we could
1040 // recognize it when sigTransactionGenerated() is
1041 // received (theoretically, the user can start two
1042 // strokes at the same time, if he is quick enough)
1043 m_strokeStrategyCookie = transformStrategy;
1044 }
1045
1046 m_strokeId = image()->startStroke(strategy);
1048
1051 }
1052
1054
1056}
1057
1083
1085{
1086 if (!m_strokeId || strokeStrategyCookie != m_strokeStrategyCookie) return;
1087
1088 if (transaction.transformedNodes().isEmpty() ||
1089 transaction.originalRect().isEmpty()) {
1090
1091 KisCanvas2 *kisCanvas = dynamic_cast<KisCanvas2*>(canvas());
1092 KIS_ASSERT(kisCanvas);
1093 kisCanvas->viewManager()->
1094 showFloatingMessage(
1095 i18nc("floating message in transformation tool",
1096 "Cannot transform empty layer "),
1097 QIcon(), 1000, KisFloatingMessage::Medium);
1098
1099 cancelStroke();
1100 return;
1101 }
1102
1103 m_transaction = transaction;
1104 m_currentArgs = args;
1106
1109 }
1110
1112 commitChanges();
1113
1115
1117 KisCanvas2 *kisCanvas = dynamic_cast<KisCanvas2*>(canvas());
1118 KIS_ASSERT(kisCanvas);
1119 kisCanvas->viewManager()->
1120 showFloatingMessage(
1121 i18nc("floating message in transformation tool",
1122 "Invisible sublayers will also be transformed. Lock layers if you do not want them to be transformed "),
1123 QIcon(), 4000, KisFloatingMessage::Low);
1124 }
1125}
1126
1128{
1129 if (device && device->exactBounds().isEmpty()) {
1130 KisCanvas2 *kisCanvas = dynamic_cast<KisCanvas2*>(canvas());
1131 KIS_SAFE_ASSERT_RECOVER(kisCanvas) { cancelStroke(); return; }
1132 kisCanvas->viewManager()->
1133 showFloatingMessage(
1134 i18nc("floating message in transformation tool",
1135 "Cannot transform empty layer "),
1136 QIcon(), 1000, KisFloatingMessage::Medium);
1137
1138 cancelStroke();
1139 } else {
1140 initThumbnailImage(device);
1142 }
1143}
1144
1161
1168
1170{
1171 const ToolTransformArgs *newArgs = dynamic_cast<const ToolTransformArgs*>(status.data());
1173
1174 *m_transaction.currentConfig() = *newArgs;
1175
1176 slotUiChangedConfig(true);
1178}
1179
1181{
1182 if (!m_canvas) return 0;
1183
1185 Q_CHECK_PTR(m_optionsWidget);
1186 m_optionsWidget->setObjectName(toolId() + " option widget");
1187
1188 // See https://bugs.kde.org/show_bug.cgi?id=316896
1189 QWidget *specialSpacer = new QWidget(m_optionsWidget);
1190 specialSpacer->setObjectName("SpecialSpacer");
1191 specialSpacer->setFixedSize(0, 0);
1192 m_optionsWidget->layout()->addWidget(specialSpacer);
1193
1194
1195 connect(m_optionsWidget, SIGNAL(sigConfigChanged(bool)),
1196 this, SLOT(slotUiChangedConfig(bool)));
1197
1198 connect(m_optionsWidget, SIGNAL(sigApplyTransform()),
1199 this, SLOT(slotApplyTransform()));
1200
1201 connect(m_optionsWidget, SIGNAL(sigResetTransform(ToolTransformArgs::TransformMode)),
1203
1204 connect(m_optionsWidget, SIGNAL(sigCancelTransform()),
1205 this, SLOT(slotCancelTransform()));
1206
1207 connect(m_optionsWidget, SIGNAL(sigRestartTransform()),
1208 this, SLOT(slotRestartTransform()));
1209
1210 connect(m_optionsWidget, SIGNAL(sigUpdateGlobalConfig()),
1211 this, SLOT(slotGlobalConfigChanged()));
1212
1213 connect(m_optionsWidget, SIGNAL(sigRestartAndContinueTransform()),
1214 this, SLOT(slotRestartAndContinueTransform()));
1215
1216 connect(m_optionsWidget, SIGNAL(sigEditingFinished()),
1217 this, SLOT(slotEditingFinished()));
1218
1219
1220 connect(mirrorHorizontalAction, SIGNAL(triggered(bool)), m_optionsWidget, SLOT(slotFlipX()));
1221 connect(mirrorVerticalAction, SIGNAL(triggered(bool)), m_optionsWidget, SLOT(slotFlipY()));
1222 connect(rotateNinetyCWAction, SIGNAL(triggered(bool)), m_optionsWidget, SLOT(slotRotateCW()));
1223 connect(rotateNinetyCCWAction, SIGNAL(triggered(bool)), m_optionsWidget, SLOT(slotRotateCCW()));
1224
1225 connect(keepAspectRatioAction, SIGNAL(triggered(bool)), m_optionsWidget, SLOT(slotSetKeepAspectRatio(bool)));
1226
1227
1228 connect(warpAction, SIGNAL(triggered(bool)), this, SLOT(slotUpdateToWarpType()));
1229 connect(perspectiveAction, SIGNAL(triggered(bool)), this, SLOT(slotUpdateToPerspectiveType()));
1230 connect(freeTransformAction, SIGNAL(triggered(bool)), this, SLOT(slotUpdateToFreeTransformType()));
1231 connect(liquifyAction, SIGNAL(triggered(bool)), this, SLOT(slotUpdateToLiquifyType()));
1232 connect(meshAction, SIGNAL(triggered(bool)), this, SLOT(slotUpdateToMeshType()));
1233 connect(cageAction, SIGNAL(triggered(bool)), this, SLOT(slotUpdateToCageType()));
1234
1235 connect(applyTransformation, SIGNAL(triggered(bool)), this, SLOT(slotApplyTransform()));
1236 connect(resetTransformation, SIGNAL(triggered(bool)), this, SLOT(slotCancelTransform()));
1237
1238
1240
1241 return m_optionsWidget;
1242}
1243
1245{
1246 if (!m_optionsWidget) return;
1247
1248 if (!currentNode()) {
1249 m_optionsWidget->setEnabled(false);
1250 return;
1251 }
1252 else {
1253 m_optionsWidget->setEnabled(true);
1255 }
1256}
1257
1264
1265void KisToolTransform::slotUiChangedConfig(bool needsPreviewRecalculation)
1266{
1267 if (mode() == KisTool::PAINT_MODE) return;
1268
1269 if (needsPreviewRecalculation) {
1271 }
1272
1275 }
1276
1279}
1280
1282{
1283 KisCursorOverrideLock cursorLock(KisCursor::waitCursor());
1284 endStroke();
1285}
1286
1288{
1290 const ToolTransformArgs::TransformMode previousMode = config->mode();
1291 config->setMode(mode);
1292
1294 config->setWarpCalculation(KisWarpTransformWorker::WarpCalculation::GRID);
1295 }
1296
1297 if (!m_strokeId || m_transaction.rootNodes().isEmpty()) return;
1298
1301
1311 const bool transformDiffers = !m_currentArgs.continuedTransform()->isSameMode(m_currentArgs);
1312
1313 if (transformDiffers &&
1314 m_currentArgs.continuedTransform()->mode() == savedMode) {
1315
1319
1320 } else {
1321 cancelStroke();
1322 startStroke(savedMode, true);
1323
1325 }
1326 } else {
1332
1333 } else {
1334 cancelStroke();
1336
1337 }
1338 }
1339}
1340
1345
1347{
1348 if (!m_strokeId || m_transaction.rootNodes().isEmpty()) return;
1349
1350 KisNodeSP root = m_transaction.rootNodes()[0];
1351 KIS_ASSERT_RECOVER_RETURN(root); // the stroke is guaranteed to be started by an 'if' above
1352
1354 cancelStroke();
1355 startStroke(savedArgs.mode(), true);
1356}
1357
1359{
1360 if (!m_strokeId || m_transaction.rootNodes().isEmpty()) return;
1361
1362 KisNodeSP root = m_transaction.rootNodes()[0];
1363 KIS_ASSERT_RECOVER_RETURN(root); // the stroke is guaranteed to be started by an 'if' above
1364
1366 endStroke();
1367 startStroke(savedArgs.mode(), false);
1368}
1369
1374
1379
1384
1389
1394
1399
1404
1409
1414
1419
1424
1429
1434
1439
1444
1449
1454
1456{
1458}
1459
1461{
1463}
1464
1466{
1468}
1469
1471{
1473}
1474
1486
1498
1500{
1503
1505
1506 auto makeSubtoolAction = [&actionRegistry, &actions, this](QString actionName, const char *slot) {
1507 QAction *action = actionRegistry->makeQAction(actionName, this);
1508 action->setProperty("always_enabled", true); // To allow this action to be triggered when the transform tool isn't already active
1509 connect(action, SIGNAL(triggered()), slot);
1510 actions << action;
1511 };
1512 makeSubtoolAction("KisToolTransformFree", SLOT(activateSubtoolFree()));
1513 makeSubtoolAction("KisToolTransformPerspective", SLOT(activateSubtoolPerspective()));
1514 makeSubtoolAction("KisToolTransformWarp", SLOT(activateSubtoolWarp()));
1515 makeSubtoolAction("KisToolTransformCage", SLOT(activateSubtoolCage()));
1516 makeSubtoolAction("KisToolTransformLiquify", SLOT(activateSubtoolLiquify()));
1517 makeSubtoolAction("KisToolTransformMesh", SLOT(activateSubtoolMesh()));
1518
1519 return actions;
1520}
1521
1523{
1524 KoToolManager *toolManager = KoToolManager::instance();
1525
1526 KoCanvasController *canvasController = toolManager->activeCanvasController();
1527 if (!canvasController) return;
1528 KoCanvasBase *canvas = canvasController->canvas();
1529 if (!canvas) return;
1530
1531 KoToolBase *tool = toolManager->toolById(canvas, id());
1533 KisToolTransform *transformTool = dynamic_cast<KisToolTransform*>(tool);
1534 KIS_SAFE_ASSERT_RECOVER_RETURN(transformTool);
1535
1536 if (toolManager->activeToolId() == id()) {
1537 // Transform tool is already active, switch the current mode
1538 transformTool->setTransformMode(mode);
1539 } else {
1540 // Works like KoToolFactoryBase::activateTool, but tells the tool beforehand which initial transform mode to use
1541 transformTool->setNextActivationTransformMode(mode);
1542 toolManager->switchToolRequested(id());
1543 }
1544}
float value(const T *src, size_t ch)
QAction * makeQAction(const QString &name, QObject *parent=0)
static KisActionRegistry * instance()
void initUpdateStreamLowLevel(KisStrokesFacade *strokesFacade, KisStrokeId strokeId)
KisViewManager * viewManager() const
static KisConfigNotifier * instance()
static QCursor waitCursor()
Definition kis_cursor.cc:54
static QCursor pointingHandCursor()
bool cancelStroke(KisStrokeId id) override
void addJob(KisStrokeId id, KisStrokeJobData *data) override
KisStrokeId startStroke(KisStrokeStrategy *strokeStrategy) override
void endStroke(KisStrokeId id) override
QRect exactBounds() const
QImage convertToQImage(const KoColorProfile *dstProfile, qint32 x, qint32 y, qint32 w, qint32 h, KoColorConversionTransformation::Intent renderingIntent=KoColorConversionTransformation::internalRenderingIntent(), KoColorConversionTransformation::ConversionFlags conversionFlags=KoColorConversionTransformation::internalConversionFlags()) const
The KisResourcesSnapshot class takes a snapshot of the various resources like colors and settings use...
KisNodeList selectedNodes() const
KisSelectionSP activeSelection() const
void addConnection(Sender sender, Signal signal, Receiver receiver, Method method, Qt::ConnectionType type=Qt::AutoConnection)
void commitConfig(KisToolChangesTrackerDataSP state)
QList< QAction * > createActionsImpl() override
createActionsImpl should be reimplemented if the tool needs any actions. The actions should have a va...
void updateConfig(const ToolTransformArgs &config)
void activateSubtool(KisToolTransform::TransformToolMode mode)
QList< QAction * > createActionsImpl() override
createActionsImpl should be reimplemented if the tool needs any actions. The actions should have a va...
void setRotateZ(double rotation)
void requestStrokeEnd() override
void deactivateAlternateAction(AlternateAction action) override
void setWarpType(WarpType type)
KisAction * perspectiveAction
void paint(QPainter &gc, const KoViewConverter &converter) override
KisAction * resetTransformation
void setRotateY(double rotation)
void setScaleX(double scaleX)
void setTranslateY(double translateY)
static ToolTransformArgs::TransformMode toArgsMode(KisToolTransform::TransformToolMode toolMode)
void mouseReleaseEvent(KoPointerEvent *e) override
KisAction * applyTransformation
KisToolTransformConfigWidget * m_optionsWidget
void mouseMoveEvent(KoPointerEvent *e) override
void slotUiChangedConfig(bool needsPreviewRecalculation)
void beginActionImpl(KoPointerEvent *event, bool usePrimaryAction, KisTool::AlternateAction action)
void endActionImpl(KoPointerEvent *event, bool usePrimaryAction, KisTool::AlternateAction action)
KisToolChangesTracker m_changesTracker
KisAction * keepAspectRatioAction
void initTransformMode(ToolTransformArgs::TransformMode mode)
void setShearY(double shearY)
TransformToolMode transformMode
void deactivate() override
void setWarpFlexibility(double flexibility)
QScopedPointer< KisMeshTransformStrategy > m_meshStrategy
void requestUndoDuringStroke() override
void slotTrackerChangedConfig(KisToolChangesTrackerDataSP status)
QScopedPointer< KisFreeTransformStrategy > m_freeStrategy
void freeTransformChanged()
void initThumbnailImage(KisPaintDeviceSP previewDevice)
void setWarpPointDensity(int density)
void activateAlternateAction(AlternateAction action) override
void cursorOutlineUpdateRequested(const QPointF &imagePos)
void imageTooBigRequested(bool value)
void setRotateX(double rotation)
void slotConvexHullCalculated(QPolygon hull, void *strokeStrategyCookie)
void beginPrimaryAction(KoPointerEvent *event) override
void slotPreviewDeviceGenerated(KisPaintDeviceSP device)
QMenu * popupActionsMenu() override
void newActivationWithExternalSource(KisPaintDeviceSP externalSource) override
newActivationWithExternalSource Makes sure that the tool is active and starts a new stroke,...
void requestStrokeCancellation() override
ToolTransformArgs m_currentArgs
QPointer< KisCanvas2 > m_canvas
void activatePrimaryAction() override
QScopedPointer< KisCageTransformStrategy > m_cageStrategy
QScopedPointer< QMenu > m_contextMenu
void setScaleY(double scaleY)
void mousePressEvent(KoPointerEvent *e) override
TransformToolMode nextActivationTransformMode
KisAction * mirrorVerticalAction
void continueAlternateAction(KoPointerEvent *event, AlternateAction action) override
void beginAlternateAction(KoPointerEvent *event, AlternateAction action) override
void continueActionImpl(KoPointerEvent *event, bool usePrimaryAction, KisTool::AlternateAction action)
void startStroke(ToolTransformArgs::TransformMode mode, bool forceReset)
QScopedPointer< KisPerspectiveTransformStrategy > m_perspectiveStrategy
QPainterPath m_cursorOutline
KisAction * mirrorHorizontalAction
KisPaintDeviceSP m_externalSourceForNextActivation
QWidget * createOptionWidget() override
void setTranslateX(double translateX)
void setNextActivationTransformMode(TransformToolMode mode)
KisSignalAutoConnectionsStore m_actionConnections
KisToolTransform(KoCanvasBase *canvas)
void activate(const QSet< KoShape * > &shapes) override
void endAlternateAction(KoPointerEvent *event, AlternateAction action) override
void setTransformMode(KisToolTransform::TransformToolMode newMode)
KisAction * rotateNinetyCWAction
TransformTransactionProperties m_transaction
KisAction * warpAction
actions for the context click menu
void endPrimaryAction(KoPointerEvent *event) override
void resetRotationCenterButtonsRequested()
void transformModeChanged()
KisToolUtils::MoveShortcutsHelper m_moveShortcutsHelper
void continuePrimaryAction(KoPointerEvent *event) override
void setShearX(double shearX)
KisAction * freeTransformAction
KisAction * rotateNinetyCCWAction
KisPaintDeviceSP m_selectedPortionCache
void slotResetTransform(ToolTransformArgs::TransformMode mode)
void deactivatePrimaryAction() override
void slotTransactionGenerated(TransformTransactionProperties transaction, ToolTransformArgs args, void *strokeStrategyCookie)
void resetCursorStyle() override
QScopedPointer< KisWarpTransformStrategy > m_warpStrategy
KisAsynchronousStrokeUpdateHelper m_asyncUpdateHelper
QScopedPointer< KisLiquifyTransformStrategy > m_liquifyStrategy
KisTransformStrategyBase * currentStrategy() const
void requestRedoDuringStroke() override
static QList< QAction * > createActions()
virtual void activateAlternateAction(KisTool::AlternateAction action)
virtual bool beginAlternateAction(KoPointerEvent *event, KisTool::AlternateAction action)
virtual void continuePrimaryAction(KoPointerEvent *event)=0
virtual bool endPrimaryAction(KoPointerEvent *event)=0
virtual void externalConfigChanged()=0
virtual void hoverActionCommon(KoPointerEvent *event)=0
virtual void decreaseBrushSize(KoCanvasBase *canvas)
virtual void continueAlternateAction(KoPointerEvent *event, KisTool::AlternateAction action)
virtual void paint(QPainter &gc)=0
virtual bool endAlternateAction(KoPointerEvent *event, KisTool::AlternateAction action)
virtual void setDecorationThickness(int thickness)
virtual void increaseBrushSize(KoCanvasBase *canvas)
virtual bool beginPrimaryAction(KoPointerEvent *event)=0
virtual void deactivateAlternateAction(KisTool::AlternateAction action)
static ToolTransformArgs resetArgsForMode(ToolTransformArgs::TransformMode mode, const QString &filterId, const TransformTransactionProperties &transaction, KisPaintDeviceSP externalSource)
static T imageToFlake(const KisCoordinatesConverter *converter, T object)
static QTransform imageToFlakeTransform(const KisCoordinatesConverter *converter)
static bool shouldRestartStrokeOnModeChange(ToolTransformArgs::TransformMode oldMode, ToolTransformArgs::TransformMode newMode, KisNodeList processedNodes)
virtual KoCanvasBase * canvas() const
bool isTouchEvent() const
QPointF point
The point in document coordinates.
Q_INVOKABLE QString toolId() const
virtual KoToolSelection * selection()
void useCursor(const QCursor &cursor)
QAction * action(const QString &name) const
int decorationThickness() const
decorationThickness The minimum thickness for tool decoration lines, this is derived from the screen ...
KoToolBase * toolById(KoCanvasBase *canvas, const QString &id) const
void switchToolRequested(const QString &id)
KoCanvasController * activeCanvasController() const
QString activeToolId() const
Returns the toolId of the currently active tool.
static KoToolManager * instance()
Return the toolmanager singleton.
QPointF transformedCenter() const
void setWarpCalculation(KisWarpTransformWorker::WarpCalculation warpCalc)
KisToolChangesTrackerData * clone() const override
void setAlpha(double alpha)
KisWarpTransformWorker::WarpType warpType() const
void saveLiquifyTransformMode() const
void setTransformedCenter(QPointF transformedCenter)
void setMode(TransformMode mode)
bool isSameMode(const ToolTransformArgs &other) const
QString filterId() const
void setWarpType(KisWarpTransformWorker::WarpType warpType)
TransformMode mode() const
KisPaintDeviceSP externalSource() const
const ToolTransformArgs * continuedTransform() const
void setCurrentConfigLocation(ToolTransformArgs *config)
#define KIS_SAFE_ASSERT_RECOVER(cond)
Definition kis_assert.h:126
#define KIS_SAFE_ASSERT_RECOVER_RETURN(cond)
Definition kis_assert.h:128
#define KIS_ASSERT_RECOVER_RETURN(cond)
Definition kis_assert.h:75
#define KIS_ASSERT_RECOVER_NOOP(cond)
Definition kis_assert.h:97
#define KIS_SAFE_ASSERT_RECOVER_NOOP(cond)
Definition kis_assert.h:130
#define KIS_ASSERT(cond)
Definition kis_assert.h:33
#define warnTools
Definition kis_debug.h:90
T kisGrowRect(const T &rect, U offset)
Definition kis_global.h:186
#define koIcon(name)
Use these macros for icons without any issues.
Definition kis_icon.h:25
QSharedPointer< T > toQShared(T *ptr)
QList< KisNodeSP > KisNodeList
Definition kis_types.h:264
KisNodeSP recursiveFindNode(KisNodeSP node, std::function< bool(KisNodeSP)> func)
void filterMergeableNodes(KisNodeList &nodes, bool allowMasks)
bool isEditable(bool checkVisibility=true) const
bool userLocked() const
virtual bool visible(bool recursive=false) const
KisNodeWSP parent
Definition kis_node.cpp:86
KisNodeList selectedNodes() const
Definition kis_tool.cc:376
virtual ToolMode mode() const
Definition kis_tool.cc:407
void mouseReleaseEvent(KoPointerEvent *event) override
Definition kis_tool.cc:515
bool nodeEditable()
Checks checks if the current node is editable.
Definition kis_tool.cc:651
bool isActive
Definition kis_tool.h:44
KisNodeSP currentNode() const
Definition kis_tool.cc:370
void mousePressEvent(KoPointerEvent *event) override
Definition kis_tool.cc:510
void mouseMoveEvent(KoPointerEvent *event) override
Definition kis_tool.cc:520
bool overrideCursorIfNotEditable()
Override the cursor appropriately if current node is not editable.
Definition kis_tool.cc:618
void activate(const QSet< KoShape * > &shapes) override
Definition kis_tool.cc:93
void paintToolOutline(QPainter *painter, const KisOptimizedBrushOutline &path)
Definition kis_tool.cc:589
void deactivate() override
Definition kis_tool.cc:131
KisImageWSP image() const
Definition kis_tool.cc:332
@ PAINT_MODE
Definition kis_tool.h:300
@ HOVER_MODE
Definition kis_tool.h:299
AlternateAction
Definition kis_tool.h:134
@ ChangeSize
Definition kis_tool.h:135
virtual void setMode(ToolMode mode)
Definition kis_tool.cc:403
KisCanvas2 * canvas