Krita Source Code Documentation
Loading...
Searching...
No Matches
kis_node_manager.cpp
Go to the documentation of this file.
1/*
2 * SPDX-FileCopyrightText: 2007 Boudewijn Rempt <boud@valdyas.org>
3 *
4 * SPDX-License-Identifier: GPL-2.0-or-later
5 */
6
7#include "kis_node_manager.h"
8
9#include <QStandardPaths>
10#include <QMessageBox>
11#include <KisSignalMapper.h>
12#include <QApplication>
13
14#include <kactioncollection.h>
15
16#include <QKeySequence>
17
18#include <kis_icon.h>
19#include <KoSelection.h>
20#include <KoShapeManager.h>
21#include <KoShape.h>
22#include <KoShapeLayer.h>
24#include <KoFileDialog.h>
25#include <KoToolManager.h>
26#include <KoProperties.h>
27
28#include <KoColorSpace.h>
31
32#include <kis_types.h>
33#include <kis_node.h>
34#include <kis_selection.h>
35#include <kis_selection_mask.h>
36#include <kis_layer.h>
37#include <kis_mask.h>
38#include <kis_image.h>
39#include <kis_painter.h>
40#include <kis_paint_layer.h>
41#include <KisMimeDatabase.h>
43
44#include "KisPart.h"
45#include "canvas/kis_canvas2.h"
48#include "KisViewManager.h"
49#include "KisDocument.h"
50#include "kis_mask_manager.h"
51#include "kis_group_layer.h"
52#include "kis_layer_manager.h"
55#include "kis_action.h"
56#include "kis_action_manager.h"
59#include "kis_transaction.h"
64#include "kis_clipboard.h"
66#include "kis_mimedata.h"
67#include "kis_layer_utils.h"
68#include "krita_utils.h"
69#include "kis_shape_layer.h"
73#include "kis_layer_utils.h"
74#include "kis_filter_mask.h"
75
77#include "KisView.h"
78
79#include <kis_signals_blocker.h>
83
98
99 KisNodeManager * q {nullptr};
100 KisViewManager * view {nullptr};
105 QScopedPointer<KisNodeSelectionAdapter> nodeSelectionAdapter;
106 QScopedPointer<KisNodeInsertionAdapter> nodeInsertionAdapter;
107 QScopedPointer<KisNodeDisplayModeAdapter> nodeDisplayModeAdapter;
108
110
113
115
116 bool activateNodeImpl(KisNodeSP node);
117
120
123
125 const QString &defaultName,
126 const QRect &bounds,
127 qreal xRes,
128 qreal yRes,
129 quint8 opacity);
130
131 void mergeTransparencyMaskAsAlpha(bool writeToLayers);
133};
134
136{
137 Q_ASSERT(view);
138 Q_ASSERT(view->canvasBase());
139 Q_ASSERT(view->canvasBase()->globalShapeManager());
140 Q_ASSERT(imageView);
141 if (node && node == q->activeNode()) {
142 return false;
143 }
144
145 // Set the selection on the shape manager to the active layer
146 // and set call KoSelection::setActiveLayer( KoShapeLayer* layer )
147 // with the parent of the active layer.
149 Q_ASSERT(selection);
150 selection->deselectAll();
151
152 // Disable all enter-group modes that were active
154
155 if (!node) {
156 selection->setActiveLayer(0);
157 imageView->setCurrentNode(0);
161 } else {
163
164 KoShape * shape = view->document()->shapeForNode(node);
165
167
168 selection->select(shape);
169 KoShapeLayer * shapeLayer = dynamic_cast<KoShapeLayer*>(shape);
170
171 KIS_SAFE_ASSERT_RECOVER_RETURN_VALUE(shapeLayer, false);
172
173 // shapeLayer->setGeometryProtected(node->userLocked());
174 // shapeLayer->setVisible(node->visible());
175 selection->setActiveLayer(shapeLayer);
176
177 imageView->setCurrentNode(node);
178 if (KisLayerSP layer = qobject_cast<KisLayer*>(node.data())) {
181 } else if (KisMaskSP mask = dynamic_cast<KisMask*>(node.data())) {
183 // XXX_NODE: for now, masks cannot be nested.
184 layerManager.activateLayer(static_cast<KisLayer*>(node->parent().data()));
185 }
186 }
187 return true;
188}
189
190//=====================================================================================
191
197
199{
200 delete m_d;
201}
202
204{
205 m_d->maskManager.setView(imageView);
206 m_d->layerManager.setView(imageView);
207
208 if (m_d->imageView) {
209 KisShapeController *shapeController = dynamic_cast<KisShapeController*>(m_d->imageView->document()->shapeController());
210 Q_ASSERT(shapeController);
211 shapeController->disconnect(SIGNAL(sigActivateNode(KisNodeSP)), this);
212 m_d->imageView->image()->disconnect(this);
213 m_d->imageView->image()->disconnect(&m_d->activateNodeConnection);
214 }
215
216 m_d->imageView = imageView;
217
218 if (m_d->imageView) {
219 KisShapeController *shapeController = dynamic_cast<KisShapeController*>(m_d->imageView->document()->shapeController());
220 Q_ASSERT(shapeController);
221 connect(shapeController, SIGNAL(sigActivateNode(KisNodeSP)), SLOT(slotNonUiActivatedNode(KisNodeSP)));
222
223 if (!m_d->imageView->currentNode()) {
228 if (shapeController->lastActivatedNode() && !m_d->imageView->currentNode()) {
229 slotNonUiActivatedNode(shapeController->lastActivatedNode());
230 } else {
231 // if last activated node is null, most probably, it means that the shape controller
232 // is going to Q_EMIT the activation signal very soon
233 }
234 } else {
241 if (!m_d->imageView->currentNode()->graphListener()) {
242 slotNonUiActivatedNode(m_d->imageView->image()->root()->lastChild());
243 }
244 }
245
247 m_d->imageView->resourceProvider()->slotNodeActivated(m_d->imageView->currentNode());
248 connect(m_d->imageView->image(), SIGNAL(sigIsolatedModeChanged()), this, SLOT(handleExternalIsolationChange()));
249 }
250
251}
252
253#define NEW_LAYER_ACTION(id, layerType) \
254{ \
255 action = actionManager->createAction(id); \
256 m_d->nodeCreationSignalMapper.setMapping(action, layerType); \
257 connect(action, SIGNAL(triggered()), \
258 &m_d->nodeCreationSignalMapper, SLOT(map())); \
259 }
260
261#define CONVERT_NODE_ACTION_2(id, layerType, exclude) \
262{ \
263 action = actionManager->createAction(id); \
264 action->setExcludedNodeTypes(QStringList(exclude)); \
265 actionManager->addAction(id, action); \
266 m_d->nodeConversionSignalMapper.setMapping(action, layerType); \
267 connect(action, SIGNAL(triggered()), \
268 &m_d->nodeConversionSignalMapper, SLOT(map())); \
269 }
270
271#define CONVERT_NODE_ACTION(id, layerType) \
272 CONVERT_NODE_ACTION_2(id, layerType, layerType)
273
274void KisNodeManager::setup(KisKActionCollection * actionCollection, KisActionManager* actionManager)
275{
276 m_d->layerManager.setup(actionManager);
277 m_d->maskManager.setup(actionCollection, actionManager);
278
279 KisAction * action = 0;
280
281 action = actionManager->createAction("mirrorNodeX");
282 connect(action, SIGNAL(triggered()), this, SLOT(mirrorNodeX()));
283
284 action = actionManager->createAction("mirrorNodeY");
285 connect(action, SIGNAL(triggered()), this, SLOT(mirrorNodeY()));
286
287 action = actionManager->createAction("mirrorAllNodesX");
288 connect(action, SIGNAL(triggered()), this, SLOT(mirrorAllNodesX()));
289
290 action = actionManager->createAction("mirrorAllNodesY");
291 connect(action, SIGNAL(triggered()), this, SLOT(mirrorAllNodesY()));
292
293 action = actionManager->createAction("activateNextLayer");
294 connect(action, SIGNAL(triggered()), this, SLOT(activateNextNode()));
295
296 action = actionManager->createAction("activateNextSiblingLayer");
297 connect(action, SIGNAL(triggered()), this, SLOT(activateNextSiblingNode()));
298
299 action = actionManager->createAction("activatePreviousLayer");
300 connect(action, SIGNAL(triggered()), this, SLOT(activatePreviousNode()));
301
302 action = actionManager->createAction("activatePreviousSiblingLayer");
303 connect(action, SIGNAL(triggered()), this, SLOT(activatePreviousSiblingNode()));
304
305 action = actionManager->createAction("switchToPreviouslyActiveNode");
306 connect(action, SIGNAL(triggered()), this, SLOT(switchToPreviouslyActiveNode()));
307
308 action = actionManager->createAction("save_node_as_image");
309 connect(action, SIGNAL(triggered()), this, SLOT(saveNodeAsImage()));
310
311 action = actionManager->createAction("save_vector_node_to_svg");
312 connect(action, SIGNAL(triggered()), this, SLOT(saveVectorLayerAsImage()));
314
315 action = actionManager->createAction("duplicatelayer");
316 connect(action, SIGNAL(triggered()), this, SLOT(duplicateActiveNode()));
317
318 action = actionManager->createAction("copy_layer_clipboard");
319 connect(action, SIGNAL(triggered()), this, SLOT(copyLayersToClipboard()));
320
321 action = actionManager->createAction("cut_layer_clipboard");
322 connect(action, SIGNAL(triggered()), this, SLOT(cutLayersToClipboard()));
323
324 action = actionManager->createAction("paste_layer_from_clipboard");
325 connect(action, SIGNAL(triggered()), this, SLOT(pasteLayersFromClipboard()));
326
327 action = actionManager->createAction("create_quick_group");
328 connect(action, SIGNAL(triggered()), this, SLOT(createQuickGroup()));
329
330 action = actionManager->createAction("create_quick_clipping_group");
331 connect(action, SIGNAL(triggered()), this, SLOT(createQuickClippingGroup()));
332
333 action = actionManager->createAction("quick_ungroup");
334 connect(action, SIGNAL(triggered()), this, SLOT(quickUngroup()));
335
336 action = actionManager->createAction("select_all_layers");
337 connect(action, SIGNAL(triggered()), this, SLOT(selectAllNodes()));
338
339 action = actionManager->createAction("select_visible_layers");
340 connect(action, SIGNAL(triggered()), this, SLOT(selectVisibleNodes()));
341
342 action = actionManager->createAction("select_locked_layers");
343 connect(action, SIGNAL(triggered()), this, SLOT(selectLockedNodes()));
344
345 action = actionManager->createAction("select_invisible_layers");
346 connect(action, SIGNAL(triggered()), this, SLOT(selectInvisibleNodes()));
347
348 action = actionManager->createAction("select_unlocked_layers");
349 connect(action, SIGNAL(triggered()), this, SLOT(selectUnlockedNodes()));
350
351 action = actionManager->createAction("new_from_visible");
352 connect(action, SIGNAL(triggered()), this, SLOT(createFromVisible()));
353
354 action = actionManager->createAction("create_reference_image_from_active_layer");
355 connect(action, SIGNAL(triggered()), this, SLOT(createReferenceImageFromLayer()));
356
357 action = actionManager->createAction("create_reference_image_from_visible_canvas");
358 connect(action, SIGNAL(triggered()), this, SLOT(createReferenceImageFromVisible()));
359
360 action = actionManager->createAction("pin_to_timeline");
361 action->setCheckable(true);
362 connect(action, SIGNAL(toggled(bool)), this, SLOT(slotPinToTimeline(bool)));
363 m_d->pinToTimeline = action;
364
365 NEW_LAYER_ACTION("add_new_paint_layer", "KisPaintLayer");
366
367 NEW_LAYER_ACTION("add_new_group_layer", "KisGroupLayer");
368
369 NEW_LAYER_ACTION("add_new_clone_layer", "KisCloneLayer");
370
371 NEW_LAYER_ACTION("add_new_shape_layer", "KisShapeLayer");
372
373 NEW_LAYER_ACTION("add_new_adjustment_layer", "KisAdjustmentLayer");
374
375 NEW_LAYER_ACTION("add_new_fill_layer", "KisGeneratorLayer");
376
377 NEW_LAYER_ACTION("add_new_file_layer", "KisFileLayer");
378
379 NEW_LAYER_ACTION("add_new_transparency_mask", "KisTransparencyMask");
380
381 NEW_LAYER_ACTION("add_new_filter_mask", "KisFilterMask");
382
383 // NOTE: FastColorOverlayFilterMask is just an identifier, not an actual class name
384 NEW_LAYER_ACTION("add_new_fast_color_overlay_mask", "FastColorOverlayFilterMask");
385
386 NEW_LAYER_ACTION("add_new_colorize_mask", "KisColorizeMask");
387
388 NEW_LAYER_ACTION("add_new_transform_mask", "KisTransformMask");
389
390 NEW_LAYER_ACTION("add_new_selection_mask", "KisSelectionMask");
391
392 connect(&m_d->nodeCreationSignalMapper, SIGNAL(mapped(QString)),
393 this, SLOT(createNode(QString)));
394
395 CONVERT_NODE_ACTION("convert_to_paint_layer", "KisPaintLayer");
396
397 CONVERT_NODE_ACTION_2("convert_to_selection_mask", "KisSelectionMask", QStringList() << "KisSelectionMask" << "KisColorizeMask");
398
399 CONVERT_NODE_ACTION_2("convert_to_filter_mask", "KisFilterMask", QStringList() << "KisFilterMask" << "KisColorizeMask");
400
401 CONVERT_NODE_ACTION_2("convert_to_transparency_mask", "KisTransparencyMask", QStringList() << "KisTransparencyMask" << "KisColorizeMask");
402
403 CONVERT_NODE_ACTION("convert_to_animated", "animated");
404
405 CONVERT_NODE_ACTION_2("convert_to_file_layer", "KisFileLayer", QStringList() << "KisFileLayer" << "KisCloneLayer");
406
407 connect(&m_d->nodeConversionSignalMapper, SIGNAL(mapped(QString)),
408 this, SLOT(convertNode(QString)));
409
410 // Isolation Modes...
411 // Post Qt5.14 this can be replaced with QActionGroup + ExclusionPolicy::ExclusiveOptional.
412 action = actionManager->createAction("isolate_active_layer");
413 connect(action, SIGNAL(toggled(bool)), this, SLOT(setIsolateActiveLayerMode(bool)));
414 action = actionManager->createAction("isolate_active_group");
415 connect(action, SIGNAL(triggered(bool)), this, SLOT(setIsolateActiveGroupMode(bool)));
416 connect(this, SIGNAL(sigNodeActivated(KisNodeSP)), SLOT(changeIsolationRoot(KisNodeSP)));
417
418 action = actionManager->createAction("toggle_layer_visibility");
419 connect(action, SIGNAL(triggered()), this, SLOT(toggleVisibility()));
420
421 action = actionManager->createAction("toggle_layer_lock");
422 connect(action, SIGNAL(triggered()), this, SLOT(toggleLock()));
423
424 action = actionManager->createAction("toggle_layer_inherit_alpha");
425 connect(action, SIGNAL(triggered()), this, SLOT(toggleInheritAlpha()));
426
427 action = actionManager->createAction("toggle_layer_alpha_lock");
428 connect(action, SIGNAL(triggered()), this, SLOT(toggleAlphaLock()));
429
430 action = actionManager->createAction("split_alpha_into_mask");
431 connect(action, SIGNAL(triggered()), this, SLOT(slotSplitAlphaIntoMask()));
432
433 action = actionManager->createAction("split_alpha_write");
434 connect(action, SIGNAL(triggered()), this, SLOT(slotSplitAlphaWrite()));
435
436 // HINT: we can save even when the nodes are not editable
437 action = actionManager->createAction("split_alpha_save_merged");
438 connect(action, SIGNAL(triggered()), this, SLOT(slotSplitAlphaSaveMerged()));
439}
440
442{
443 // enable/disable all relevant actions
446}
447
449{
450 if (m_d->imageView) {
451 return m_d->imageView->currentNode();
452 }
453 return 0;
454}
455
460
462{
465 } else {
466 Q_ASSERT(m_d->layerManager.activeLayer());
469 else
470 return m_d->view->image()->colorSpace();
471 }
472}
473
474bool KisNodeManager::canModifyLayers(KisNodeList nodes, bool showWarning)
475{
476 KisNodeSP lockedNode;
477 Q_FOREACH (KisNodeSP node, nodes) {
478 if (!node->isEditable(false)) {
479 lockedNode = node;
480 break;
481 }
482 }
483
484 if (lockedNode && showWarning) {
485 QString errorMessage;
486
487 if (nodes.size() <= 1) {
488 errorMessage = i18n("Layer is locked");
489 } else {
490 errorMessage = i18n("Layer \"%1\" is locked", lockedNode->name());
491 }
492
493 m_d->view->showFloatingMessage(errorMessage, QIcon());
494 }
495
496 return !lockedNode;
497}
498
499bool KisNodeManager::canModifyLayer(KisNodeSP node, bool showWarning)
500{
501 return canModifyLayers({node}, showWarning);
502}
503
504bool KisNodeManager::canMoveLayers(KisNodeList nodes, bool showWarning)
505{
506 KisNodeSP lockedNode;
507 Q_FOREACH (KisNodeSP node, nodes) {
508 if (node->parent() && !node->parent()->isEditable(false)) {
509 lockedNode = node->parent();
510 break;
511 }
512 }
513
514 if (lockedNode && showWarning) {
515 QString errorMessage = i18n("Layer \"%1\" is locked", lockedNode->name());
516 m_d->view->showFloatingMessage(errorMessage, QIcon());
517 }
518
519 return !lockedNode;
520}
521
522bool KisNodeManager::canMoveLayer(KisNodeSP node, bool showWarning)
523{
524 return canMoveLayers({node}, showWarning);
525}
526
527void KisNodeManager::moveNodeAt(KisNodeSP node, KisNodeSP parent, int index)
528{
529 if (parent->allowAsChild(node)) {
530 if (node->inherits("KisSelectionMask") && parent->inherits("KisLayer")) {
531 KisSelectionMask *m = dynamic_cast<KisSelectionMask*>(node.data());
532 KisLayer *l = qobject_cast<KisLayer*>(parent.data());
533 if (m && m->active() && l && l->selectionMask()) {
534 l->selectionMask()->setActive(false);
535 }
536 }
537 m_d->commandsAdapter.moveNode(node, parent, index);
538 }
539}
540
542{
543 KUndo2MagicString actionName = kundo2_i18n("Move Nodes");
544 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
545 juggler->moveNode(nodes, parent, aboveThis);
546}
547
549{
550 KUndo2MagicString actionName = kundo2_i18n("Copy Nodes");
551 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
552 juggler->copyNode(nodes, parent, aboveThis);
553}
554
556{
557 KUndo2MagicString actionName = kundo2_i18n("Add Nodes");
558 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
559 juggler->addNode(nodes, parent, aboveThis);
560}
561
563{
564 QAction* action = m_d->view->actionManager()->actionByName("isolate_active_layer");
565 action->toggle();
566}
567
569{
570 KisImageWSP image = m_d->view->image();
572
573 const bool groupIsolationState = image->isIsolatingGroup();
574 changeIsolationMode(checked, groupIsolationState);
575}
576
578{
579 KisImageWSP image = m_d->view->image();
581
582 const bool layerIsolationState = image->isIsolatingLayer();
583 changeIsolationMode(layerIsolationState, checked);
584}
585
586void KisNodeManager::changeIsolationMode(bool isolateActiveLayer, bool isolateActiveGroup)
587{
588 KisImageWSP image = m_d->view->image();
591
592 if (isolateActiveLayer || isolateActiveGroup) {
593 if (image->startIsolatedMode(activeNode, isolateActiveLayer, isolateActiveGroup) == false) {
595 }
596 } else {
597 image->stopIsolatedMode();
598 }
599}
600
602{
603 KisImageWSP image = m_d->view->image();
604 if (!image || !isolationRoot) return;
605
606 const bool isIsolatingLayer = image->isIsolatingLayer();
607 const bool isIsolatingGroup = image->isIsolatingGroup();
608
609 // Restart isolation with a new root node and the same settings.
610 if (image->startIsolatedMode(isolationRoot, isIsolatingLayer, isIsolatingGroup) == false) {
612 }
613}
614
616{
617 // It might be that we have multiple Krita windows open. In such a case
618 // only the currently active one should restart isolated mode
619 if (!m_d->view->mainWindowAsQWidget()->isActiveWindow()) return;
620
621 KisImageWSP image = m_d->view->image();
623
624 const bool isIsolatingLayer = image->isIsolatingLayer();
625 const bool isIsolatingGroup = image->isIsolatingGroup();
626
627 m_d->view->actionManager()->actionByName("isolate_active_layer")->setChecked(isIsolatingLayer);
628 m_d->view->actionManager()->actionByName("isolate_active_group")->setChecked(isIsolatingGroup);
629}
630
632{
633 m_d->view->actionManager()->actionByName("isolate_active_layer")->setChecked(false);
634 m_d->view->actionManager()->actionByName("isolate_active_group")->setChecked(false);
635}
636
637KisNodeSP KisNodeManager::createNode(const QString & nodeType, bool quiet, KisPaintDeviceSP copyFrom)
638{
640 return 0;
641 }
642
644 if (!activeNode) {
645 activeNode = m_d->view->image()->root();
646 }
647
649
653
654 // XXX: make factories for this kind of stuff,
655 // with a registry
656
657 if (nodeType == "KisPaintLayer") {
659 } else if (nodeType == "KisGroupLayer") {
661 } else if (nodeType == "KisAdjustmentLayer") {
663 } else if (nodeType == "KisGeneratorLayer") {
665 } else if (nodeType == "KisShapeLayer") {
667 } else if (nodeType == "KisCloneLayer") {
668 KisNodeList nodes = selectedNodes();
669 if (nodes.isEmpty()) {
670 nodes.append(activeNode);
671 }
672 return m_d->layerManager.addCloneLayer(nodes);
673 } else if (nodeType == "KisTransparencyMask") {
674 return m_d->maskManager.createTransparencyMask(activeNode, copyFrom, false);
675 } else if (nodeType == "KisFilterMask") {
676 return m_d->maskManager.createFilterMask(activeNode, copyFrom, quiet, false);
677 } else if (nodeType == "FastColorOverlayFilterMask") {
679 } else if (nodeType == "KisColorizeMask") {
681 } else if (nodeType == "KisTransformMask") {
683 } else if (nodeType == "KisSelectionMask") {
684 return m_d->maskManager.createSelectionMask(activeNode, copyFrom, false);
685 } else if (nodeType == "KisFileLayer") {
687 }
688 return 0;
689}
690
695
697{
698 Q_FOREACH (KisNodeSP node, selectedNodes()) {
700 }
701}
702
704{
705 KisNodeSP node = createNode("KisPaintLayer");
706 return dynamic_cast<KisLayer*>(node.data());
707}
708
710{
712 return;
713 }
714
716 if (!activeNode) return;
717
718 if (!canModifyLayer(activeNode)) return;
719
720 if (nodeType == "KisPaintLayer") {
722 } else if (nodeType == "KisSelectionMask" ||
723 nodeType == "KisFilterMask" ||
724 nodeType == "KisTransparencyMask") {
725
728
729 m_d->commandsAdapter.beginMacro(kundo2_i18n("Convert to a Selection Mask"));
730
731 bool result = false;
732
733 if (nodeType == "KisSelectionMask") {
734 result = !m_d->maskManager.createSelectionMask(activeNode, copyFrom, true).isNull();
735 } else if (nodeType == "KisFilterMask") {
736 result = !m_d->maskManager.createFilterMask(activeNode, copyFrom, false, true).isNull();
737 } else if (nodeType == "KisTransparencyMask") {
738 result = !m_d->maskManager.createTransparencyMask(activeNode, copyFrom, true).isNull();
739 }
740
742
743 if (!result) {
746 }
747 } else if (nodeType == "KisFileLayer") {
749 } else {
750 warnKrita << "Unsupported node conversion type:" << nodeType;
751 }
752}
753
755 KisViewManager* m_view = m_d->view;
756 KisDocument *document = m_view->document();
757 KisCanvas2 *canvas = m_view->canvasBase();
758
759 const KisPaintDeviceSP paintDevice = fromLayer ? m_view->activeLayer()->projection()
760 : canvas->currentImage()->projection();
763 QScopedPointer<KisReferenceImage> reference(KisReferenceImage::fromQImage(*canvas->coordinatesConverter(), image));
765 if (reference) {
766 if (document->referenceImagesLayer()) {
767 reference->setZIndex(document->referenceImagesLayer()->shapes().size());
768 }
769 canvas->addCommand(KisReferenceImagesLayer::addReferenceImages(document, {reference.take()}));
770
771 KoToolManager::instance()->switchToolRequested("ToolReferenceImages");
772
773 } else {
774 if (canvas->canvasWidget()) {
775 QString strMessage = fromLayer ? i18nc("error dialog from the reference tool", "Could not create a reference image from the active layer.")
776 : i18nc("error dialog from the reference tool", "Could not create a reference image from the visible canvas.");
777
778 m_d->view->showFloatingMessage(strMessage, QIcon(), 5000, KisFloatingMessage::High, Qt::TextSingleLine);
779 }
780 }
781}
782
786
790
792{
793 KisDummiesFacadeBase *dummiesFacade = dynamic_cast<KisDummiesFacadeBase*>(m_d->imageView->document()->shapeController());
794 KIS_SAFE_ASSERT_RECOVER_RETURN(dummiesFacade);
795
796 const bool nodeVisible = !isNodeHidden(node, !m_d->nodeDisplayModeAdapter->showGlobalSelectionMask());
797 if (!nodeVisible) {
798 return;
799 }
800
802 if (m_d->activateNodeImpl(node)) {
803 Q_EMIT sigUiNeedChangeActiveNode(node);
804 Q_EMIT sigNodeActivated(node);
805 nodesUpdated();
806 if (node) {
807 bool toggled = m_d->view->actionCollection()->action("view_show_canvas_only")->isChecked();
808 if (toggled) {
809 m_d->view->showFloatingMessage( activeLayer()->name(), QIcon(), 1600, KisFloatingMessage::Medium, Qt::TextSingleLine);
810 }
811 }
812 }
813}
814
816{
817 // the node must still be in the graph, some asynchronous
818 // signals may easily break this requirement
819 if (node && !node->graphListener()) {
820 node = 0;
821 }
822
823 if (node == activeNode()) return;
824
826
827 if (node) {
828 bool toggled = m_d->view->actionCollection()->action("view_show_canvas_only")->isChecked();
829 if (toggled) {
830 m_d->view->showFloatingMessage( activeLayer()->name(), QIcon(), 1600, KisFloatingMessage::Medium, Qt::TextSingleLine);
831 }
832 }
833}
834
836{
837 // the node must still be in the graph, some asynchronous
838 // signals may easily break this requirement
839 if (node && !node->graphListener()) {
840 node = 0;
841 }
842
843 if (node) {
844 QStringList vectorTools = QStringList()
845 << "InteractionTool"
846 << "KarbonGradientTool"
847 << "KarbonCalligraphyTool"
848 << "PathTool";
849
850 QStringList pixelTools = QStringList()
851 << "KritaShape/KisToolBrush"
852 << "KritaShape/KisToolDyna"
853 << "KritaShape/KisToolMultiBrush"
854 << "KritaFill/KisToolFill"
855 << "KritaFill/KisToolGradient";
856
857 KisSelectionMask *selectionMask = dynamic_cast<KisSelectionMask*>(node.data());
858 const bool nodeHasVectorAbilities = node->inherits("KisShapeLayer") ||
859 (selectionMask && selectionMask->selection()->hasShapeSelection());
860
861 if (nodeHasVectorAbilities) {
862 if (pixelTools.contains(KoToolManager::instance()->activeToolId())) {
863 KoToolManager::instance()->switchToolRequested("InteractionTool");
864 }
865 }
866 else {
867 if (vectorTools.contains(KoToolManager::instance()->activeToolId())) {
868 KoToolManager::instance()->switchToolRequested("KritaShape/KisToolBrush");
869 }
870 }
871 }
872
873 if (node == activeNode()) return;
874
876}
877
879{
880 KisNodeSP node = activeNode();
881 if (!node) return;
882
885
886 m_d->view->updateGUI();
888
889 {
891 m_d->pinToTimeline->setChecked(node->isPinnedToTimeline());
892 }
893}
894
901
903{
904 if ((selectedNodes().size() > 1 && node->inherits("KisLayer")) || node->inherits("KisLayer")) {
906 }
907 else if (node->inherits("KisMask")) {
909 }
910}
911
913{
914 Q_ASSERT(node);
915
916 // Change the current node temporarily
917 KisNodeSP originalNode = m_d->imageView->currentNode();
918 m_d->imageView->setCurrentNode(node);
919
920 if (node->inherits("KisLayer")) {
922 }
923 else if (node->inherits("KisMask")) {
925 }
926
927 m_d->imageView->setCurrentNode(originalNode);
928}
929
934
936{
942 return qMin(255, int(opacity * 2.55 + 0.5));
943}
944
945void KisNodeManager::setNodeName(KisNodeSP node, const QString &name)
946{
947 if (!node) return;
948 if (node->name() == name) return;
949
950 m_d->commandsAdapter.setNodeName(node, name);
951
952}
953
955{
956 if (!node) return;
957 if (node->opacity() == opacity) return;
958
959 m_d->commandsAdapter.setOpacity(node, opacity);
960}
961
963 const KoCompositeOp* compositeOp)
964{
965 if (!node) return;
966 if (node->compositeOp() == compositeOp) return;
967
968 m_d->commandsAdapter.setCompositeOp(node, compositeOp);
969}
970
972{
973 if (activeNode) {
975 }
976 if (!selectedNodes.isEmpty()) {
978 }
979}
980
982{
983 m_d->selectedNodes = nodes;
984 Q_EMIT sigUiNeedChangeSelectedNodes(nodes);
985}
986
991
996
1001
1006
1007bool KisNodeManager::isNodeHidden(KisNodeSP node, bool isGlobalSelectionHidden)
1008{
1009 if (node && node->isFakeNode()) {
1010 return true;
1011 }
1012
1013 if (isGlobalSelectionHidden && dynamic_cast<KisSelectionMask *>(node.data()) &&
1014 (!node->parent() || !node->parent()->parent())) {
1015 return true;
1016 }
1017
1018 return false;
1019}
1020
1022{
1023 const KisPaintLayer *paintLayer = dynamic_cast<KisPaintLayer*>(node.data());
1024 if (paintLayer) {
1026
1027 if (properties.contains(onionSkinOn)) {
1028 const KisPaintDeviceSP &paintDevice = paintLayer->paintDevice();
1029 if (paintDevice && paintDevice->defaultPixel().opacityU8() == 255) {
1030 m_d->view->showFloatingMessage(i18n("Onion skins require a layer with transparent background."), QIcon());
1031 return false;
1032 }
1033 }
1034 }
1035
1037
1038 return true;
1039}
1040
1042{
1043 KisNodeSP node = activeNode();
1044
1045 setNodeOpacity(node, convertOpacityToInt(opacity));
1046}
1047
1049{
1050 KisNodeSP node = activeNode();
1051
1052 setNodeCompositeOp(node, op);
1053}
1054
1056{
1057 KUndo2MagicString actionName = kundo2_i18n("Duplicate Nodes");
1058 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1059 juggler->duplicateNode(selectedNodes());
1060}
1061
1063{
1064 KisImageWSP image = view->image();
1065
1066 if (!nodeJuggler ||
1067 (nodeJuggler &&
1068 (nodeJuggler->isEnded() ||
1069 !nodeJuggler->canMergeAction(actionName)))) {
1070
1071 nodeJuggler = new KisNodeJugglerCompressed(actionName, image, q, 750);
1072 nodeJuggler->setAutoDelete(true);
1073 }
1074
1075 return nodeJuggler;
1076}
1077
1079{
1080 if (!canMoveLayers(selectedNodes())) return;
1081
1082 KUndo2MagicString actionName = kundo2_i18n("Raise Nodes");
1083 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1084 juggler->raiseNode(selectedNodes());
1085}
1086
1088{
1089 if (!canMoveLayers(selectedNodes())) return;
1090
1091 KUndo2MagicString actionName = kundo2_i18n("Lower Nodes");
1092 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1093 juggler->lowerNode(selectedNodes());
1094}
1095
1097{
1098 if (!node || !node->parent()) {
1099 return;
1100 }
1101
1102 KisNodeList nodes;
1103 nodes << node;
1104 removeSelectedNodes(nodes);
1105}
1106
1108{
1109 if (!canModifyLayers(nodes)) return;
1110
1111 KUndo2MagicString actionName = kundo2_i18n("Remove Nodes");
1112 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1113 juggler->removeNode(nodes);
1114}
1115
1120
1122{
1123 KisNodeList nodes = selectedNodes();
1124
1125 KUndo2MagicString commandName;
1126 if (nodes.size() == 1 && nodes[0]->inherits("KisMask")) {
1127 commandName = kundo2_i18n("Mirror Mask Horizontally");
1128 }
1129 else {
1130 commandName = kundo2_i18np("Mirror Layer Horizontally", "Mirror %1 Layers Horizontally", nodes.size());
1131 }
1132 mirrorNodes(nodes, commandName, Qt::Horizontal, m_d->view->selection());
1133}
1134
1136{
1137 KisNodeList nodes = selectedNodes();
1138
1139 KUndo2MagicString commandName;
1140 if (nodes.size() == 1 && nodes[0]->inherits("KisMask")) {
1141 commandName = kundo2_i18n("Mirror Mask Vertically");
1142 }
1143 else {
1144 commandName = kundo2_i18np("Mirror Layer Vertically", "Mirror %1 Layers Vertically", nodes.size());
1145 }
1146 mirrorNodes(nodes, commandName, Qt::Vertical, m_d->view->selection());
1147}
1148
1150{
1151 KisNodeSP node = m_d->view->image()->root();
1152 mirrorNode(node, kundo2_i18n("Mirror All Layers Horizontally"),
1153 Qt::Horizontal, m_d->view->selection());
1154}
1155
1157{
1158 KisNodeSP node = m_d->view->image()->root();
1159 mirrorNode(node, kundo2_i18n("Mirror All Layers Vertically"),
1160 Qt::Vertical, m_d->view->selection());
1161}
1162
1164{
1166 if (!activeNode) return;
1167
1168 KisNodeSP nextNode = activeNode->nextSibling();
1169
1170 if (!siblingsOnly) {
1171 // Recurse groups...
1172 while (nextNode && nextNode->childCount() > 0) {
1173 nextNode = nextNode->firstChild();
1174 }
1175
1176 // Out of nodes? Back out of group...
1177 if (!nextNode && activeNode->parent()) {
1178 nextNode = activeNode->parent();
1179 }
1180 }
1181
1182 // Skip nodes hidden from tree view..
1183 while (nextNode && isNodeHidden(nextNode, m_d->nodeDisplayModeAdapter->showGlobalSelectionMask())) {
1184 nextNode = nextNode->nextSibling();
1185 }
1186
1187 // Select node, unless root..
1188 if (nextNode && nextNode->parent()) {
1189 slotNonUiActivatedNode(nextNode);
1190 }
1191}
1192
1197
1199{
1201 if (!activeNode) return;
1202
1203 KisNodeSP nextNode = activeNode->prevSibling();
1204
1205 if (!siblingsOnly) {
1206 // Enter groups..
1207 if (activeNode->childCount() > 0) {
1208 nextNode = activeNode->lastChild();
1209 }
1210
1211 // Out of nodes? Back out of group...
1212 if (!nextNode && activeNode->parent()) {
1213 nextNode = activeNode->parent()->prevSibling();
1214 }
1215 }
1216
1217 // Skip nodes hidden from tree view..
1218 while (nextNode && isNodeHidden(nextNode, m_d->nodeDisplayModeAdapter->showGlobalSelectionMask())) {
1219 nextNode = nextNode->prevSibling();
1220 }
1221
1222 // Select node, unless root..
1223 if (nextNode && nextNode->parent()) {
1224 slotNonUiActivatedNode(nextNode);
1225 }
1226}
1227
1232
1239
1241 const KUndo2MagicString& actionName,
1242 Qt::Orientation orientation,
1243 KisSelectionSP selection)
1244{
1245 KisNodeList nodes = {node};
1246 mirrorNodes(nodes, actionName, orientation, selection);
1247}
1248
1250 const KUndo2MagicString& actionName,
1251 Qt::Orientation orientation,
1252 KisSelectionSP selection)
1253{
1254 Q_FOREACH(KisNodeSP node, nodes) {
1255 if (!canModifyLayer(node)) return;
1256 }
1257
1258 KisImageSignalVector emitSignals;
1259
1260 KisProcessingApplicator applicator(m_d->view->image(), nodes,
1262 emitSignals, actionName);
1263
1264 KisProcessingVisitorSP visitor;
1265
1266 if (selection) {
1267 visitor = new KisMirrorProcessingVisitor(selection, orientation);
1268 } else {
1269 visitor = new KisMirrorProcessingVisitor(m_d->view->image()->bounds(), orientation);
1270 }
1271
1272 if (!selection) {
1274 } else {
1275 applicator.applyVisitor(visitor, KisStrokeJobData::CONCURRENT);
1276 }
1277
1278 applicator.end();
1279
1280 nodesUpdated();
1281}
1282
1284 const QString &defaultName,
1285 const QRect &bounds,
1286 qreal xRes,
1287 qreal yRes,
1288 quint8 opacity)
1289{
1290 KoFileDialog dialog(view->mainWindowAsQWidget(), KoFileDialog::SaveFile, "savenodeasimage");
1291 dialog.setCaption(i18n("Export \"%1\"", defaultName));
1292 dialog.setDefaultDir(QStandardPaths::writableLocation(QStandardPaths::PicturesLocation));
1294 QString filename = dialog.filename();
1295
1296 if (filename.isEmpty()) return;
1297
1298 QString mimefilter = KisMimeDatabase::mimeTypeForFile(filename, false);
1299
1300 QScopedPointer<KisDocument> doc(KisPart::instance()->createDocument());
1301
1302 KisImageSP dst = new KisImage(doc->createUndoStore(),
1303 bounds.width(),
1304 bounds.height(),
1306 defaultName);
1307 dst->setResolution(xRes, yRes);
1308 doc->setCurrentImage(dst);
1309 KisPaintLayer* paintLayer = new KisPaintLayer(dst, "paint device", opacity);
1310 paintLayer->paintDevice()->makeCloneFrom(device, bounds);
1311 dst->addNode(paintLayer, dst->rootLayer(), KisLayerSP(0));
1312
1313 dst->initialRefreshGraph();
1314
1315 if (!doc->exportDocumentSync(filename, mimefilter.toLatin1())) {
1316 QMessageBox::warning(qApp->activeWindow(),
1317 i18nc("@title:window", "Krita"),
1318 i18n("Could not save the layer. %1", doc->errorMessage().toUtf8().data()),
1319 QMessageBox::Ok);
1320
1321 }
1322}
1323
1325{
1326 KisNodeSP node = activeNode();
1327
1328 if (!node) {
1329 warnKrita << "BUG: Save Node As Image was called without any node selected";
1330 return;
1331 }
1332
1333 KisPaintDeviceSP saveDevice = node->projection();
1334
1335 if (!saveDevice) {
1336 m_d->view->showFloatingMessage(i18nc("warning message when trying to export a transform mask", "Layer has no pixel data"), QIcon());
1337 return;
1338 }
1339
1340 KisImageSP image = m_d->view->image();
1341 QRect saveRect = image->bounds() | node->exactBounds();
1342
1343 m_d->saveDeviceAsImage(saveDevice,
1344 node->name(),
1345 saveRect,
1346 image->xRes(), image->yRes(),
1347 node->opacity());
1348}
1349
1350#include "SvgWriter.h"
1351
1353{
1354 KisShapeLayerSP shapeLayer = qobject_cast<KisShapeLayer*>(activeNode().data());
1355 if (!shapeLayer) {
1356 return;
1357 }
1358
1359 KoFileDialog dialog(m_d->view->mainWindowAsQWidget(), KoFileDialog::SaveFile, "savenodeasimage");
1360 dialog.setCaption(i18nc("@title:window", "Export to SVG"));
1361 dialog.setDefaultDir(QStandardPaths::writableLocation(QStandardPaths::PicturesLocation));
1362 dialog.setMimeTypeFilters(QStringList() << "image/svg+xml", "image/svg+xml");
1363 QString filename = dialog.filename();
1364
1365 if (filename.isEmpty()) return;
1366
1367 QUrl url = QUrl::fromLocalFile(filename);
1368
1369 if (url.isEmpty()) return;
1370
1371 const QSizeF sizeInPx = m_d->view->image()->bounds().size();
1372 const QSizeF sizeInPt(sizeInPx.width() / m_d->view->image()->xRes(),
1373 sizeInPx.height() / m_d->view->image()->yRes());
1374
1375 QList<KoShape*> shapes = shapeLayer->shapes();
1376 std::sort(shapes.begin(), shapes.end(), KoShape::compareShapeZIndex);
1377
1378 SvgWriter writer(shapes);
1379 if (!writer.save(filename, sizeInPt, true)) {
1380 QMessageBox::warning(qApp->activeWindow(), i18nc("@title:window", "Krita"), i18n("Could not save to svg: %1", filename));
1381 }
1382}
1383
1385{
1386 KisNodeSP node = activeNode();
1387 if (!canModifyLayer(node)) return;
1388
1389 // guaranteed by KisActionManager
1391
1392 KisLayerUtils::splitAlphaToMask(node->image(), node, m_d->maskManager.createMaskNameCommon(node, "KisTransparencyMask", i18n("Transparency Mask")));
1393}
1394
1396{
1397 KisNodeSP node = q->activeNode();
1398 KisNodeSP parentNode = node->parent();
1399
1400 // guaranteed by KisActionManager
1401 KIS_ASSERT_RECOVER_RETURN(node->inherits("KisTransparencyMask"));
1402
1403 if (writeToLayers && (!parentNode->hasEditablePaintDevice() || !node->isEditable(false))) {
1404 QMessageBox::information(view->mainWindowAsQWidget(),
1405 i18nc("@title:window", "Layer %1 is not editable", parentNode->name()),
1406 i18n("Cannot write alpha channel of "
1407 "the parent layer \"%1\".\n"
1408 "The operation will be cancelled.", parentNode->name()));
1409 return;
1410 }
1411
1412 KisPaintDeviceSP dstDevice;
1413 if (writeToLayers) {
1415 dstDevice = parentNode->paintDevice();
1416 } else {
1417 KisPaintDeviceSP copyDevice = parentNode->paintDevice();
1418 if (!copyDevice) {
1419 copyDevice = parentNode->original();
1420 }
1421 dstDevice = new KisPaintDevice(*copyDevice);
1422 }
1423
1424 const KoColorSpace *dstCS = dstDevice->colorSpace();
1425
1426 KisPaintDeviceSP selectionDevice = node->paintDevice();
1427 KIS_ASSERT_RECOVER_RETURN(selectionDevice->colorSpace()->pixelSize() == 1);
1428
1429 const QRect processRect =
1430 selectionDevice->exactBounds() |
1431 dstDevice->exactBounds() |
1432 selectionDevice->defaultBounds()->bounds();
1433
1434 QScopedPointer<KisTransaction> transaction;
1435
1436 if (writeToLayers) {
1437 commandsAdapter.beginMacro(kundo2_i18n("Write Alpha into a Layer"));
1438 transaction.reset(new KisTransaction(kundo2_noi18n("__write_alpha_channel__"), dstDevice));
1439 }
1440
1441 KisSequentialIterator srcIt(selectionDevice, processRect);
1442 KisSequentialIterator dstIt(dstDevice, processRect);
1443
1444 while (srcIt.nextPixel() && dstIt.nextPixel()) {
1445 quint8 *alpha8Ptr = srcIt.rawData();
1446 quint8 *dstPtr = dstIt.rawData();
1447
1448 dstCS->setOpacity(dstPtr, *alpha8Ptr, 1);
1449 }
1450
1451 if (writeToLayers) {
1452 commandsAdapter.addExtraCommand(transaction->endAndTake());
1453 commandsAdapter.removeNode(node);
1454 commandsAdapter.endMacro();
1455 } else {
1456 KisImageWSP image = view->image();
1457 QRect saveRect = image->bounds();
1458
1459 saveDeviceAsImage(dstDevice, parentNode->name(),
1460 saveRect,
1461 image->xRes(), image->yRes(),
1463 }
1464}
1465
1466
1471
1476
1478{
1479 KisNodeList nodes = this->selectedNodes();
1480 KisNodeSP active = activeNode();
1481 if (nodes.isEmpty() || !active) return;
1482
1483 bool isLocked = active->userLocked();
1484
1485 for (auto &node : nodes) {
1487 }
1488}
1489
1491{
1492 KisNodeList nodes = this->selectedNodes();
1493 KisNodeSP active = activeNode();
1494 if (nodes.isEmpty() || !active) return;
1495
1496 bool isVisible = active->visible();
1497
1498 for (auto &node : nodes) {
1500 }
1501}
1502
1504{
1505 KisNodeList nodes = this->selectedNodes();
1506 KisNodeSP active = activeNode();
1507 if (nodes.isEmpty() || !active) return;
1508
1509 auto layer = qobject_cast<KisPaintLayer*>(active.data());
1510 if (!layer) {
1511 return;
1512 }
1513
1514 bool isAlphaLocked = layer->alphaLocked();
1515 for (auto &node : nodes) {
1516 auto layer = qobject_cast<KisPaintLayer*>(node.data());
1517 if (layer) {
1519 }
1520 }
1521}
1522
1524{
1525 KisNodeList nodes = this->selectedNodes();
1526 KisNodeSP active = activeNode();
1527 if (nodes.isEmpty() || !active) return;
1528
1529 auto layer = qobject_cast<KisLayer*>(active.data());
1530 if (!layer) {
1531 return;
1532 }
1533
1534 bool isAlphaDisabled = layer->alphaChannelDisabled();
1535 for (auto &node : nodes) {
1536 auto layer = qobject_cast<KisLayer*>(node.data());
1537 if (layer) {
1539 }
1540 }
1541}
1542
1544{
1545 Q_ASSERT(node);
1546 KisLayerSP layer = qobject_cast<KisLayer*>(node.data());
1547 if (!layer) {
1548 return;
1549 }
1550
1551 KisFilterMaskSP mask = layer->colorOverlayMask();
1552 if (!mask) {
1553 // This layer does not use fast color overlay mask.
1554 return;
1555 }
1556
1558}
1559
1561{
1562 KisNodeList nodes = this->selectedNodes();
1563 if (nodes.isEmpty()) return;
1564
1565 KisNodeList::Iterator it = nodes.begin();
1566 while (it != nodes.end()) {
1567 // make sure the deleted nodes aren't referenced here again
1568 if (!it->data()->parent()) {
1569 nodes.erase(it);
1570 }
1571 it++;
1572 }
1573
1574 KisClipboard::instance()->setLayers(nodes, m_d->view->image(), false);
1575
1576 if (canModifyLayers(nodes)) {
1577 KUndo2MagicString actionName = kundo2_i18n("Cut Nodes");
1578 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1579 juggler->removeNode(nodes);
1580 }
1581}
1582
1584{
1585 KisNodeList nodes = this->selectedNodes();
1586 KisClipboard::instance()->setLayers(nodes, m_d->view->image(), true);
1587}
1588
1589void KisNodeManager::pasteLayersFromClipboard(bool changeOffset, QPointF offset, KisProcessingApplicator *applicator)
1590{
1591 const QMimeData *data = KisClipboard::instance()->layersMimeData();
1592 if (!data) return;
1593
1595
1596 KisShapeController *shapeController = dynamic_cast<KisShapeController*>(m_d->imageView->document()->shapeController());
1597 Q_ASSERT(shapeController);
1598
1599 KisDummiesFacadeBase *dummiesFacade = dynamic_cast<KisDummiesFacadeBase*>(m_d->imageView->document()->shapeController());
1600 Q_ASSERT(dummiesFacade);
1601
1602 const bool copyNode = false;
1603 KisImageSP image = m_d->view->image();
1604 KisNodeDummy *parentDummy = dummiesFacade->dummyForNode(activeNode ? activeNode : image->root());
1605 KisNodeDummy *aboveThisDummy = parentDummy ? parentDummy->lastChild() : 0;
1606
1608 image,
1609 shapeController,
1610 parentDummy,
1611 aboveThisDummy,
1612 copyNode,
1614 changeOffset,
1615 offset,
1616 applicator);
1617}
1618
1620 const QString &overrideGroupName,
1621 KisNodeSP *newGroup,
1622 KisNodeSP *newLastChild)
1623{
1624 KisNodeSP active = activeNode();
1625 if (!active) return false;
1626
1627 if (!canMoveLayer(active)) return false;
1628
1629 KisImageSP image = m_d->view->image();
1630 QString groupName = !overrideGroupName.isEmpty() ? overrideGroupName : image->nextLayerName(i18nc("A group of layers", "Group"));
1631 KisGroupLayerSP group = new KisGroupLayer(image.data(), groupName, OPACITY_OPAQUE_U8);
1632
1633 KisNodeList nodes1;
1634 nodes1 << group;
1635
1636 KisNodeList nodes2;
1639
1640 if (nodes2.size() == 0) return false;
1641
1642 if (KisLayerUtils::checkIsChildOf(active, nodes2)) {
1643 active = nodes2.first();
1644 }
1645
1646 KisNodeSP parent = active->parent();
1647 KisNodeSP aboveThis = active;
1648
1649 juggler->addNode(nodes1, parent, aboveThis);
1650 juggler->moveNode(nodes2, group, 0);
1651
1652 *newGroup = group;
1653 *newLastChild = nodes2.last();
1654
1655 return true;
1656}
1657
1659{
1660 KUndo2MagicString actionName = kundo2_i18n("Quick Group");
1661 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1662
1663 KisNodeSP parent;
1664 KisNodeSP above;
1665
1666 createQuickGroupImpl(juggler, "", &parent, &above);
1667}
1668
1670{
1671 KUndo2MagicString actionName = kundo2_i18n("Quick Clipping Group");
1672 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1673
1674 KisNodeSP parent;
1675 KisNodeSP above;
1676
1677 KisImageSP image = m_d->view->image();
1678 if (createQuickGroupImpl(juggler, image->nextLayerName(i18nc("default name for a clipping group layer", "Clipping Group")), &parent, &above)) {
1679 KisPaintLayerSP maskLayer = new KisPaintLayer(image.data(), i18nc("default name for quick clip group mask layer", "Mask Layer"), OPACITY_OPAQUE_U8, image->colorSpace());
1680 maskLayer->disableAlphaChannel(true);
1681
1682 juggler->addNode(KisNodeList() << maskLayer, parent, above);
1683 }
1684}
1685
1687{
1688 KisNodeSP active = activeNode();
1689 if (!active) return;
1690
1691 if (!canModifyLayer(active)) return;
1692
1693 KisNodeSP parent = active->parent();
1694 KisNodeSP aboveThis = active;
1695
1696 auto checkCanMoveLayers = [this] (KisNodeList nodes, KisNodeSP newParent) -> bool {
1697 auto incompatibleNode =
1698 std::find_if(nodes.begin(), nodes.end(),
1699 [newParent] (KisNodeSP node) {
1700 return !newParent->allowAsChild(node);
1701 });
1702
1703 if (incompatibleNode != nodes.end()) {
1704 const QString message =
1705 newParent->parent() ?
1706 i18n("Cannot move layer \"%1\" into new parent \"%2\"",
1707 (*incompatibleNode)->name(),
1708 newParent->name()) :
1709 i18n("Cannot move layer \"%1\" into the root layer",
1710 (*incompatibleNode)->name());
1711 m_d->view->showFloatingMessage(message, QIcon());
1712 return false;
1713 }
1714 return true;
1715 };
1716
1717 KUndo2MagicString actionName = kundo2_i18n("Quick Ungroup");
1718
1719 if (parent && dynamic_cast<KisGroupLayer*>(active.data())) {
1720 KisNodeList nodes = active->childNodes(QStringList(), KoProperties());
1721
1722 if (checkCanMoveLayers(nodes, parent)) {
1723 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1724 juggler->moveNode(nodes, parent, active);
1725 juggler->removeNode(KisNodeList() << active);
1726 }
1727 } else if (parent && parent->parent()) {
1728 KisNodeSP grandParent = parent->parent();
1729
1730 KisNodeList allChildNodes = parent->childNodes(QStringList(), KoProperties());
1731 KisNodeList allSelectedNodes = selectedNodes();
1732
1733 const bool removeParent = KritaUtils::compareListsUnordered(allChildNodes, allSelectedNodes);
1734
1735 if (checkCanMoveLayers(allSelectedNodes, parent)) {
1736 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1737 juggler->moveNode(allSelectedNodes, grandParent, parent);
1738 if (removeParent) {
1739 juggler->removeNode(KisNodeList() << parent);
1740 }
1741 }
1742 }
1743}
1744
1745void KisNodeManager::selectLayersImpl(const KoProperties &props, const KoProperties &invertedProps)
1746{
1747 KisImageSP image = m_d->view->image();
1748 KisNodeList nodes = KisLayerUtils::findNodesWithProps(image->root(), props, true);
1749
1751
1753 nodes = KisLayerUtils::findNodesWithProps(image->root(), invertedProps, true);
1754 }
1755
1756 if (!nodes.isEmpty()) {
1757 slotImageRequestNodeReselection(nodes.last(), nodes);
1758 }
1759}
1760
1762{
1763 KoProperties props;
1764 selectLayersImpl(props, props);
1765}
1766
1768{
1769 KoProperties props;
1770 props.setProperty("visible", true);
1771
1772 KoProperties invertedProps;
1773 invertedProps.setProperty("visible", false);
1774
1775 selectLayersImpl(props, invertedProps);
1776}
1777
1779{
1780 KoProperties props;
1781 props.setProperty("locked", true);
1782
1783 KoProperties invertedProps;
1784 invertedProps.setProperty("locked", false);
1785
1786 selectLayersImpl(props, invertedProps);
1787}
1788
1790{
1791 KoProperties props;
1792 props.setProperty("visible", false);
1793
1794 KoProperties invertedProps;
1795 invertedProps.setProperty("visible", true);
1796
1797 selectLayersImpl(props, invertedProps);
1798}
1799
1801{
1802 KoProperties props;
1803 props.setProperty("locked", false);
1804
1805 KoProperties invertedProps;
1806 invertedProps.setProperty("locked", true);
1807
1808 selectLayersImpl(props, invertedProps);
1809}
1810
1812{
1813 if (!sender()->property("node").isNull()) {
1814 QString name = sender()->property("node").toString();
1815 KisNodeSP node = KisLayerUtils::findNodeByName(m_d->imageView->image()->rootLayer(),name);
1816 if (node) {
1817 slotUiActivatedNode(node);
1818 }
1819 }
1820}
float value(const T *src, size_t ch)
qreal v
QList< QString > QStringList
const quint8 OPACITY_OPAQUE_U8
char nodeType(const KoPathPoint *point)
A KisActionManager class keeps track of KisActions. These actions are always associated with the GUI....
KisAction * createAction(const QString &name)
KisAction * actionByName(const QString &name) const
void setActivationFlags(ActivationFlags flags)
@ ACTIVE_SHAPE_LAYER
Activate if the current node is a vector layer.
Definition kis_action.h:50
KisImageWSP currentImage() const
KisCoordinatesConverter * coordinatesConverter
void setCurrentShapeManagerOwnerShape(KoShape *source) override
sets the group shape that is supposed to be "entered"
void addCommand(KUndo2Command *command) override
KisAbstractCanvasWidget * canvasWidget
KoShapeManager * globalShapeManager() const
void setLayers(KisNodeList nodes, KisImageSP image, bool forceCopy=false)
static KisClipboard * instance()
const QMimeData * layersMimeData() const
virtual QRect bounds() const =0
KoShapeLayer * shapeForNode(KisNodeSP layer) const
virtual KisNodeDummy * dummyForNode(KisNodeSP node) const =0
bool isIsolatingLayer() const
bool startIsolatedMode(KisNodeSP node, bool isolateLayer, bool isolateGroup)
KisGroupLayerSP rootLayer() const
void sigRequestNodeReselection(KisNodeSP activeNode, const KisNodeList &selectedNodes)
const KoColorSpace * colorSpace() const
QString nextLayerName(const QString &baseName="") const
Definition kis_image.cc:715
KisPaintDeviceSP projection() const
void initialRefreshGraph()
void stopIsolatedMode()
bool isIsolatingGroup() const
double xRes() const
double yRes() const
QRect bounds() const override
void setResolution(double xres, double yres)
static QStringList supportedMimeTypes(Direction direction)
A container for a set of QAction objects.
QAction * action(int index) const
KisNodeSP addShapeLayer(KisNodeSP activeNode)
KisNodeSP addGeneratorLayer(KisNodeSP activeNode)
void setView(QPointer< KisView >view)
KisNodeSP addFileLayer(KisNodeSP activeNode)
void activateLayer(KisLayerSP layer)
KisPaintDeviceSP activeDevice()
KisLayerSP activeLayer()
void setup(KisActionManager *actionManager)
void convertLayerToFileLayer(KisNodeSP source)
KisLayerSP addPaintLayer(KisNodeSP activeNode)
KisNodeSP addGroupLayer(KisNodeSP activeNode)
KisNodeSP addCloneLayer(KisNodeList nodes)
void convertNodeToPaintLayer(KisNodeSP source)
KisNodeSP addAdjustmentLayer(KisNodeSP activeNode)
static void setNodePropertyAutoUndo(KisNodeSP node, const KoID &id, const QVariant &value, KisImageSP image)
static KisBaseNode::Property getProperty(const KoID &id, bool state)
KisNodeSP createTransformMask(KisNodeSP activeNode)
void setup(KisKActionCollection *actionCollection, KisActionManager *actionManager)
void activateMask(KisMaskSP mask)
KisNodeSP createSelectionMask(KisNodeSP activeNode, KisPaintDeviceSP copyFrom, bool convertActiveNode)
KisPaintDeviceSP activeDevice()
KisNodeSP createColorizeMask(KisNodeSP activeNode)
KisMaskSP activeMask()
void setView(QPointer< KisView >view)
KisNodeSP createFastColorOverlayMask(KisNodeSP activeNode)
KisNodeSP createTransparencyMask(KisNodeSP activeNode, KisPaintDeviceSP copyFrom, bool convertActiveNode)
KisNodeSP createFilterMask(KisNodeSP activeNode, KisPaintDeviceSP copyFrom, bool quiet, bool convertActiveNode)
QString createMaskNameCommon(KisNodeSP targetNode, const QString &nodeType, const QString &desiredName)
static bool insertMimeLayers(const QMimeData *data, KisImageSP image, KisShapeController *shapeController, KisNodeDummy *parentDummy, KisNodeDummy *aboveThisDummy, bool copyNode, KisNodeInsertionAdapter *nodeInsertionAdapter, bool changeOffset=false, QPointF offset=QPointF(), KisProcessingApplicator *applicator=nullptr)
static QString mimeTypeForFile(const QString &file, bool checkExistingFiles=true)
Find the mimetype for the given filename. The filename must include a suffix.
void moveNode(KisNodeSP node, KisNodeSP parent, KisNodeSP aboveThis)
void setNodeName(KisNodeSP node, const QString &name)
void setOpacity(KisNodeSP node, qint32 opacity)
void setCompositeOp(KisNodeSP node, const KoCompositeOp *compositeOp)
void beginMacro(const KUndo2MagicString &macroName)
KisNodeDummy * lastChild() const
void removeNode(const KisNodeList &nodes)
void raiseNode(const KisNodeList &nodes)
void addNode(const KisNodeList &nodes, KisNodeSP dstParent, KisNodeSP dstAbove)
void duplicateNode(const KisNodeList &nodes)
void copyNode(const KisNodeList &nodes, KisNodeSP dstParent, KisNodeSP dstAbove)
void lowerNode(const KisNodeList &nodes)
void moveNode(KisNodeSP node, KisNodeSP parent, KisNodeSP above)
void activateNextNode(bool siblingsOnly=false)
static bool isNodeHidden(KisNodeSP node, bool isGlobalSelectionHidden)
void selectLayersImpl(const KoProperties &props, const KoProperties &invertedProps)
KisLayerSP activeLayer()
void setIsolateActiveGroupMode(bool checked)
KisNodeDisplayModeAdapter * nodeDisplayModeAdapter() const
void createReferenceImageFromVisible()
KisNodeInsertionAdapter * nodeInsertionAdapter() const
void setView(QPointer< KisView >imageView)
bool canMoveLayer(KisNodeSP node, bool showWarning=true)
void copyNodesDirect(KisNodeList nodes, KisNodeSP parent, KisNodeSP aboveThis)
void colorOverlayMaskProperties(KisNodeSP node)
void nodeOpacityChanged(qreal opacity)
void sigUiNeedChangeActiveNode(KisNodeSP node)
void slotPinToTimeline(bool value)
void setNodeOpacity(KisNodeSP node, qint32 opacity)
void slotUiActivateNode()
slotUiActivateNode inspects the sender to see which node needs to be activated.
void changeCloneSource()
pop up a window for changing the source of the selected Clone Layers
void moveNodesDirect(KisNodeList nodes, KisNodeSP parent, KisNodeSP aboveThis)
void switchToPreviouslyActiveNode()
void setIsolateActiveLayerMode(bool checked)
void setup(KisKActionCollection *collection, KisActionManager *actionManager)
void handleExternalIsolationChange()
KisNodeSP activeNode()
Convenience function to get the active layer or mask.
KisNodeList selectedNodes()
void nodePropertiesIgnoreSelection(KisNodeSP node)
void slotImageRequestNodeReselection(KisNodeSP activeNode, const KisNodeList &selectedNodes)
void mirrorNodes(KisNodeList nodes, const KUndo2MagicString &commandName, Qt::Orientation orientation, KisSelectionSP selection)
void sigUiNeedChangeSelectedNodes(const QList< KisNodeSP > &nodes)
void activatePreviousNode(bool siblingsOnly=false)
Private *const m_d
const KoColorSpace * activeColorSpace()
void setNodeCompositeOp(KisNodeSP node, const KoCompositeOp *compositeOp)
void changeIsolationRoot(KisNodeSP isolationRoot)
void addNodesDirect(KisNodeList nodes, KisNodeSP parent, KisNodeSP aboveThis)
KisPaintDeviceSP activePaintDevice()
Get the paint device the user wants to paint on now.
void setNodeName(KisNodeSP node, const QString &name)
qint32 convertOpacityToInt(qreal opacity)
void mirrorNode(KisNodeSP node, const KUndo2MagicString &commandName, Qt::Orientation orientation, KisSelectionSP selection)
bool createQuickGroupImpl(KisNodeJugglerCompressed *juggler, const QString &overrideGroupName, KisNodeSP *newGroup, KisNodeSP *newLastChild)
void slotSetSelectedNodes(const KisNodeList &nodes)
slotSetSelectedNodes set the list of nodes selected in the layerbox. Selected nodes are not necessari...
void pasteLayersFromClipboard(bool changeOffset=false, QPointF offset=QPointF(), KisProcessingApplicator *applicator=nullptr)
bool canModifyLayers(KisNodeList nodes, bool showWarning=true)
KisNodeSelectionAdapter * nodeSelectionAdapter() const
bool trySetNodeProperties(KisNodeSP node, KisImageSP image, KisBaseNode::PropertyList properties) const
KisNodeSP createNode(const QString &nodeType, bool quiet=false, KisPaintDeviceSP copyFrom=0)
bool canModifyLayer(KisNodeSP node, bool showWarning=true)
void slotNonUiActivatedNode(KisNodeSP node)
void nodeProperties(KisNodeSP node)
void sigNodeActivated(KisNodeSP node)
emitted whenever a node is selected.
void createReferenceImage(bool fromLayer)
KisNodeManager(KisViewManager *view)
void moveNodeAt(KisNodeSP node, KisNodeSP parent, int index)
~KisNodeManager() override
void removeSingleNode(KisNodeSP node)
void createReferenceImageFromLayer()
void reinitializeIsolationActionGroup()
void slotSomethingActivatedNodeImpl(KisNodeSP node)
KisLayerSP createPaintLayer()
bool canMoveLayers(KisNodeList nodes, bool showWarning=true)
void nodeCompositeOpChanged(const KoCompositeOp *op)
void removeSelectedNodes(KisNodeList selectedNodes)
void changeIsolationMode(bool isolateActiveLayer, bool isolateActiveGroup)
void convertNode(const QString &nodeType)
void slotUiActivatedNode(KisNodeSP node)
static void setNodePropertiesAutoUndo(KisNodeSP node, KisImageSP image, PropertyList proplist)
virtual const KoColorSpace * compositionSourceColorSpace() const
QRect exactBounds() const
const KoColorSpace * colorSpace() const
KoColor defaultPixel() 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
KisDefaultBoundsBaseSP defaultBounds() const
void makeCloneFrom(KisPaintDeviceSP src, const QRect &rect)
static KisPart * instance()
Definition KisPart.cpp:131
void applyVisitor(KisProcessingVisitorSP visitor, KisStrokeJobData::Sequentiality sequentiality=KisStrokeJobData::SEQUENTIAL, KisStrokeJobData::Exclusivity exclusivity=KisStrokeJobData::NORMAL)
void applyVisitorAllFrames(KisProcessingVisitorSP visitor, KisStrokeJobData::Sequentiality sequentiality=KisStrokeJobData::SEQUENTIAL, KisStrokeJobData::Exclusivity exclusivity=KisStrokeJobData::NORMAL)
static KisReferenceImage * fromQImage(const KisCoordinatesConverter &converter, const QImage &img)
static KUndo2Command * addReferenceImages(KisDocument *document, QList< KoShape * > referenceImages)
ALWAYS_INLINE quint8 * rawData()
bool isNull() const
The KisSignalMapper class bundles signals from identifiable senders.
bool blockUntilOperationsFinished(KisImageSP image)
blockUntilOperationsFinished blocks the GUI of the application until execution of actions on image is...
KisDocument * document() const
KisActionManager * actionManager() const
KisCanvas2 * canvasBase() const
Return the canvas base class.
KisSelectionSP selection()
KisLayerSP activeLayer()
Convenience method to get at the active layer.
void blockUntilOperationsFinishedForced(KisImageSP image)
blockUntilOperationsFinished blocks the GUI of the application until execution of actions on image is...
virtual KisKActionCollection * actionCollection() const
KisSelectionManager * selectionManager()
QWidget * mainWindowAsQWidget() const
KisImageWSP image() const
Return the image this view is displaying.
void showFloatingMessage(const QString &message, const QIcon &icon, int timeout=4500, KisFloatingMessage::Priority priority=KisFloatingMessage::Medium, int alignment=Qt::AlignCenter|Qt::TextWordWrap)
shows a floating message in the top right corner of the canvas
virtual quint32 pixelSize() const =0
virtual void setOpacity(quint8 *pixels, quint8 alpha, qint32 nPixels) const =0
quint8 opacityU8() const
Definition KoColor.cpp:341
void setProperty(const QString &name, const QVariant &value)
void deselectAll()
clear the selections list
void setActiveLayer(KoShapeLayer *layer)
void select(KoShape *shape)
QList< KoShape * > shapes() const
KoSelection * selection
static bool compareShapeZIndex(KoShape *s1, KoShape *s2)
Definition KoShape.cpp:393
void switchToolRequested(const QString &id)
QString activeToolId() const
Returns the toolId of the currently active tool.
static KoToolManager * instance()
Return the toolmanager singleton.
Implements exporting shapes to SVG.
Definition SvgWriter.h:33
bool save(QIODevice &outputDevice, const QSizeF &pageSize)
Writes svg to specified output device.
Definition SvgWriter.cpp:82
#define KIS_ASSERT_RECOVER_RETURN_VALUE(cond, val)
Definition kis_assert.h:85
#define KIS_SAFE_ASSERT_RECOVER_RETURN_VALUE(cond, val)
Definition kis_assert.h:129
#define KIS_SAFE_ASSERT_RECOVER_RETURN(cond)
Definition kis_assert.h:128
#define KIS_ASSERT_RECOVER_RETURN(cond)
Definition kis_assert.h:75
#define bounds(x, a, b)
#define warnKrita
Definition kis_debug.h:87
KisDocument * createDocument(QList< KisNodeSP > nodes, KisImageSP srcImage, const QRect &copiedBounds)
#define CONVERT_NODE_ACTION_2(id, layerType, exclude)
#define CONVERT_NODE_ACTION(id, layerType)
#define NEW_LAYER_ACTION(id, layerType)
QList< KisNodeSP > KisNodeList
Definition kis_types.h:264
KUndo2MagicString kundo2_i18n(const char *text)
KUndo2MagicString kundo2_noi18n(const QString &text)
KUndo2MagicString kundo2_i18np(const char *sing, const char *plur, const A1 &a1)
void splitAlphaToMask(KisImageSP image, KisNodeSP node, const QString &maskName)
void sortMergeableNodes(KisNodeSP root, KisNodeList &inputNodes, KisNodeList &outputNodes)
KisNodeSP findNodeByName(KisNodeSP root, const QString &name)
KisNodeList findNodesWithProps(KisNodeSP root, const KoProperties &props, bool excludeRoot)
void newLayerFromVisible(KisImageSP image, KisNodeSP putAfter, MergeFlags flags)
void filterMergeableNodes(KisNodeList &nodes, bool allowMasks)
bool checkIsChildOf(KisNodeSP node, const KisNodeList &parents)
bool compareListsUnordered(const QList< T > &a, const QList< T > &b)
void setPinnedToTimeline(bool pinned)
virtual KisPaintDeviceSP projection() const =0
bool isEditable(bool checkVisibility=true) const
bool isPinnedToTimeline() const
virtual QRect exactBounds() const
virtual KisPaintDeviceSP original() const =0
QString compositeOp
KisImageWSP image
bool userLocked() const
virtual KisPaintDeviceSP paintDevice() const =0
QString name() const
virtual bool isFakeNode() const
quint8 opacity() const
virtual bool visible(bool recursive=false) const
bool hasEditablePaintDevice() const
KisPaintDeviceSP projection() const override
Definition kis_layer.cc:820
void disableAlphaChannel(bool disable)
Definition kis_layer.cc:319
virtual KisSelectionMaskSP selectionMask() const
Definition kis_layer.cc:498
const KoColorSpace * colorSpace() const override
returns the image's colorSpace or null, if there is no image
Definition kis_layer.cc:225
KisFilterMaskSP colorOverlayMask() const
Definition kis_layer.cc:567
KisLayerSP parentLayer() const
KisSelectionSP selection
Definition kis_mask.cc:44
bool addNode(KisNodeSP node, KisNodeSP parent=KisNodeSP(), KisNodeAdditionFlags flags=KisNodeAdditionFlag::None)
void mergeTransparencyMaskAsAlpha(bool writeToLayers)
QScopedPointer< KisNodeSelectionAdapter > nodeSelectionAdapter
bool activateNodeImpl(KisNodeSP node)
void saveDeviceAsImage(KisPaintDeviceSP device, const QString &defaultName, const QRect &bounds, qreal xRes, qreal yRes, quint8 opacity)
KisNodeCommandsAdapter commandsAdapter
KisSynchronizedConnection< KisNodeSP, KisNodeList > activateNodeConnection
QScopedPointer< KisNodeInsertionAdapter > nodeInsertionAdapter
KisSignalMapper nodeCreationSignalMapper
QPointer< KisNodeJugglerCompressed > nodeJuggler
KisNodeJugglerCompressed * lazyGetJuggler(const KUndo2MagicString &actionName)
Private(KisNodeManager *_q, KisViewManager *v)
QPointer< KisView > imageView
KisSignalMapper nodeConversionSignalMapper
QScopedPointer< KisNodeDisplayModeAdapter > nodeDisplayModeAdapter
KisNodeSP prevSibling() const
Definition kis_node.cpp:402
KisNodeSP firstChild() const
Definition kis_node.cpp:361
QList< KisNodeSP > childNodes(const QStringList &nodeTypes, const KoProperties &properties) const
Definition kis_node.cpp:439
quint32 childCount() const
Definition kis_node.cpp:414
KisNodeWSP parent
Definition kis_node.cpp:86
KisNodeSP lastChild() const
Definition kis_node.cpp:367
KisNodeSP nextSibling() const
Definition kis_node.cpp:408
KisNodeGraphListener * graphListener
Definition kis_node.cpp:87
KisPaintDeviceSP paintDevice
void setActive(bool active)
bool hasShapeSelection() const