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( node->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
829{
830 // the node must still be in the graph, some asynchronous
831 // signals may easily break this requirement
832 if (node && !node->graphListener()) {
833 node = 0;
834 }
835
836 if (node) {
837 QStringList vectorTools = QStringList()
838 << "InteractionTool"
839 << "KarbonGradientTool"
840 << "KarbonCalligraphyTool"
841 << "PathTool";
842
843 QStringList pixelTools = QStringList()
844 << "KritaShape/KisToolBrush"
845 << "KritaShape/KisToolDyna"
846 << "KritaShape/KisToolMultiBrush"
847 << "KritaFill/KisToolFill"
848 << "KritaFill/KisToolGradient";
849
850 KisSelectionMask *selectionMask = dynamic_cast<KisSelectionMask*>(node.data());
851 const bool nodeHasVectorAbilities = node->inherits("KisShapeLayer") ||
852 (selectionMask && selectionMask->selection()->hasShapeSelection());
853
854 if (nodeHasVectorAbilities) {
855 if (pixelTools.contains(KoToolManager::instance()->activeToolId())) {
856 KoToolManager::instance()->switchToolRequested("InteractionTool");
857 }
858 }
859 else {
860 if (vectorTools.contains(KoToolManager::instance()->activeToolId())) {
861 KoToolManager::instance()->switchToolRequested("KritaShape/KisToolBrush");
862 }
863 }
864 }
865
866 if (node == activeNode()) return;
867
869}
870
872{
873 KisNodeSP node = activeNode();
874 if (!node) return;
875
878
879 m_d->view->updateGUI();
881
882 {
884 m_d->pinToTimeline->setChecked(node->isPinnedToTimeline());
885 }
886}
887
894
896{
897 if ((selectedNodes().size() > 1 && node->inherits("KisLayer")) || node->inherits("KisLayer")) {
899 }
900 else if (node->inherits("KisMask")) {
902 }
903}
904
906{
907 Q_ASSERT(node);
908
909 // Change the current node temporarily
910 KisNodeSP originalNode = m_d->imageView->currentNode();
911 m_d->imageView->setCurrentNode(node);
912
913 if (node->inherits("KisLayer")) {
915 }
916 else if (node->inherits("KisMask")) {
918 }
919
920 m_d->imageView->setCurrentNode(originalNode);
921}
922
927
929{
935 return qMin(255, int(opacity * 2.55 + 0.5));
936}
937
938void KisNodeManager::setNodeName(KisNodeSP node, const QString &name)
939{
940 if (!node) return;
941 if (node->name() == name) return;
942
943 m_d->commandsAdapter.setNodeName(node, name);
944
945}
946
948{
949 if (!node) return;
950 if (node->opacity() == opacity) return;
951
952 m_d->commandsAdapter.setOpacity(node, opacity);
953}
954
956 const KoCompositeOp* compositeOp)
957{
958 if (!node) return;
959 if (node->compositeOp() == compositeOp) return;
960
961 m_d->commandsAdapter.setCompositeOp(node, compositeOp);
962}
963
965{
966 if (activeNode) {
968 }
969 if (!selectedNodes.isEmpty()) {
971 }
972}
973
975{
976 m_d->selectedNodes = nodes;
977 Q_EMIT sigUiNeedChangeSelectedNodes(nodes);
978}
979
984
989
994
999
1000bool KisNodeManager::isNodeHidden(KisNodeSP node, bool isGlobalSelectionHidden)
1001{
1002 if (node && node->isFakeNode()) {
1003 return true;
1004 }
1005
1006 if (isGlobalSelectionHidden && dynamic_cast<KisSelectionMask *>(node.data()) &&
1007 (!node->parent() || !node->parent()->parent())) {
1008 return true;
1009 }
1010
1011 return false;
1012}
1013
1015{
1016 const KisPaintLayer *paintLayer = dynamic_cast<KisPaintLayer*>(node.data());
1017 if (paintLayer) {
1019
1020 if (properties.contains(onionSkinOn)) {
1021 const KisPaintDeviceSP &paintDevice = paintLayer->paintDevice();
1022 if (paintDevice && paintDevice->defaultPixel().opacityU8() == 255) {
1023 m_d->view->showFloatingMessage(i18n("Onion skins require a layer with transparent background."), QIcon());
1024 return false;
1025 }
1026 }
1027 }
1028
1030
1031 return true;
1032}
1033
1035{
1036 KisNodeSP node = activeNode();
1037
1038 setNodeOpacity(node, convertOpacityToInt(opacity));
1039}
1040
1042{
1043 KisNodeSP node = activeNode();
1044
1045 setNodeCompositeOp(node, op);
1046}
1047
1049{
1050 KUndo2MagicString actionName = kundo2_i18n("Duplicate Nodes");
1051 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1052 juggler->duplicateNode(selectedNodes());
1053}
1054
1056{
1057 KisImageWSP image = view->image();
1058
1059 if (!nodeJuggler ||
1060 (nodeJuggler &&
1061 (nodeJuggler->isEnded() ||
1062 !nodeJuggler->canMergeAction(actionName)))) {
1063
1064 nodeJuggler = new KisNodeJugglerCompressed(actionName, image, q, 750);
1065 nodeJuggler->setAutoDelete(true);
1066 }
1067
1068 return nodeJuggler;
1069}
1070
1072{
1073 if (!canMoveLayers(selectedNodes())) return;
1074
1075 KUndo2MagicString actionName = kundo2_i18n("Raise Nodes");
1076 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1077 juggler->raiseNode(selectedNodes());
1078}
1079
1081{
1082 if (!canMoveLayers(selectedNodes())) return;
1083
1084 KUndo2MagicString actionName = kundo2_i18n("Lower Nodes");
1085 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1086 juggler->lowerNode(selectedNodes());
1087}
1088
1090{
1091 if (!node || !node->parent()) {
1092 return;
1093 }
1094
1095 KisNodeList nodes;
1096 nodes << node;
1097 removeSelectedNodes(nodes);
1098}
1099
1101{
1102 if (!canModifyLayers(nodes)) return;
1103
1104 KUndo2MagicString actionName = kundo2_i18n("Remove Nodes");
1105 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1106 juggler->removeNode(nodes);
1107}
1108
1113
1115{
1116 KisNodeList nodes = selectedNodes();
1117
1118 KUndo2MagicString commandName;
1119 if (nodes.size() == 1 && nodes[0]->inherits("KisMask")) {
1120 commandName = kundo2_i18n("Mirror Mask Horizontally");
1121 }
1122 else {
1123 commandName = kundo2_i18np("Mirror Layer Horizontally", "Mirror %1 Layers Horizontally", nodes.size());
1124 }
1125 mirrorNodes(nodes, commandName, Qt::Horizontal, m_d->view->selection());
1126}
1127
1129{
1130 KisNodeList nodes = selectedNodes();
1131
1132 KUndo2MagicString commandName;
1133 if (nodes.size() == 1 && nodes[0]->inherits("KisMask")) {
1134 commandName = kundo2_i18n("Mirror Mask Vertically");
1135 }
1136 else {
1137 commandName = kundo2_i18np("Mirror Layer Vertically", "Mirror %1 Layers Vertically", nodes.size());
1138 }
1139 mirrorNodes(nodes, commandName, Qt::Vertical, m_d->view->selection());
1140}
1141
1143{
1144 KisNodeSP node = m_d->view->image()->root();
1145 mirrorNode(node, kundo2_i18n("Mirror All Layers Horizontally"),
1146 Qt::Horizontal, m_d->view->selection());
1147}
1148
1150{
1151 KisNodeSP node = m_d->view->image()->root();
1152 mirrorNode(node, kundo2_i18n("Mirror All Layers Vertically"),
1153 Qt::Vertical, m_d->view->selection());
1154}
1155
1157{
1159 if (!activeNode) return;
1160
1161 KisNodeSP nextNode = activeNode->nextSibling();
1162
1163 if (!siblingsOnly) {
1164 // Recurse groups...
1165 while (nextNode && nextNode->childCount() > 0) {
1166 nextNode = nextNode->firstChild();
1167 }
1168
1169 // Out of nodes? Back out of group...
1170 if (!nextNode && activeNode->parent()) {
1171 nextNode = activeNode->parent();
1172 }
1173 }
1174
1175 // Skip nodes hidden from tree view..
1176 while (nextNode && isNodeHidden(nextNode, m_d->nodeDisplayModeAdapter->showGlobalSelectionMask())) {
1177 nextNode = nextNode->nextSibling();
1178 }
1179
1180 // Select node, unless root..
1181 if (nextNode && nextNode->parent()) {
1182 slotNonUiActivatedNode(nextNode);
1183 }
1184}
1185
1190
1192{
1194 if (!activeNode) return;
1195
1196 KisNodeSP nextNode = activeNode->prevSibling();
1197
1198 if (!siblingsOnly) {
1199 // Enter groups..
1200 if (activeNode->childCount() > 0) {
1201 nextNode = activeNode->lastChild();
1202 }
1203
1204 // Out of nodes? Back out of group...
1205 if (!nextNode && activeNode->parent()) {
1206 nextNode = activeNode->parent()->prevSibling();
1207 }
1208 }
1209
1210 // Skip nodes hidden from tree view..
1211 while (nextNode && isNodeHidden(nextNode, m_d->nodeDisplayModeAdapter->showGlobalSelectionMask())) {
1212 nextNode = nextNode->prevSibling();
1213 }
1214
1215 // Select node, unless root..
1216 if (nextNode && nextNode->parent()) {
1217 slotNonUiActivatedNode(nextNode);
1218 }
1219}
1220
1225
1232
1234 const KUndo2MagicString& actionName,
1235 Qt::Orientation orientation,
1236 KisSelectionSP selection)
1237{
1238 KisNodeList nodes = {node};
1239 mirrorNodes(nodes, actionName, orientation, selection);
1240}
1241
1243 const KUndo2MagicString& actionName,
1244 Qt::Orientation orientation,
1245 KisSelectionSP selection)
1246{
1247 Q_FOREACH(KisNodeSP node, nodes) {
1248 if (!canModifyLayer(node)) return;
1249 }
1250
1251 KisImageSignalVector emitSignals;
1252
1253 KisProcessingApplicator applicator(m_d->view->image(), nodes,
1255 emitSignals, actionName);
1256
1257 KisProcessingVisitorSP visitor;
1258
1259 if (selection) {
1260 visitor = new KisMirrorProcessingVisitor(selection, orientation);
1261 } else {
1262 visitor = new KisMirrorProcessingVisitor(m_d->view->image()->bounds(), orientation);
1263 }
1264
1265 if (!selection) {
1267 } else {
1268 applicator.applyVisitor(visitor, KisStrokeJobData::CONCURRENT);
1269 }
1270
1271 applicator.end();
1272
1273 nodesUpdated();
1274}
1275
1277 const QString &defaultName,
1278 const QRect &bounds,
1279 qreal xRes,
1280 qreal yRes,
1281 quint8 opacity)
1282{
1283 KoFileDialog dialog(view->mainWindowAsQWidget(), KoFileDialog::SaveFile, "savenodeasimage");
1284 dialog.setCaption(i18n("Export \"%1\"", defaultName));
1285 dialog.setDefaultDir(QStandardPaths::writableLocation(QStandardPaths::PicturesLocation));
1287 QString filename = dialog.filename();
1288
1289 if (filename.isEmpty()) return;
1290
1291 QString mimefilter = KisMimeDatabase::mimeTypeForFile(filename, false);
1292
1293 QScopedPointer<KisDocument> doc(KisPart::instance()->createDocument());
1294
1295 KisImageSP dst = new KisImage(doc->createUndoStore(),
1296 bounds.width(),
1297 bounds.height(),
1299 defaultName);
1300 dst->setResolution(xRes, yRes);
1301 doc->setCurrentImage(dst);
1302 KisPaintLayer* paintLayer = new KisPaintLayer(dst, "paint device", opacity);
1303 paintLayer->paintDevice()->makeCloneFrom(device, bounds);
1304 dst->addNode(paintLayer, dst->rootLayer(), KisLayerSP(0));
1305
1306 dst->initialRefreshGraph();
1307
1308 if (!doc->exportDocumentSync(filename, mimefilter.toLatin1())) {
1309 QMessageBox::warning(qApp->activeWindow(),
1310 i18nc("@title:window", "Krita"),
1311 i18n("Could not save the layer. %1", doc->errorMessage().toUtf8().data()),
1312 QMessageBox::Ok);
1313
1314 }
1315}
1316
1318{
1319 KisNodeSP node = activeNode();
1320
1321 if (!node) {
1322 warnKrita << "BUG: Save Node As Image was called without any node selected";
1323 return;
1324 }
1325
1326 KisPaintDeviceSP saveDevice = node->projection();
1327
1328 if (!saveDevice) {
1329 m_d->view->showFloatingMessage(i18nc("warning message when trying to export a transform mask", "Layer has no pixel data"), QIcon());
1330 return;
1331 }
1332
1333 KisImageSP image = m_d->view->image();
1334 QRect saveRect = image->bounds() | node->exactBounds();
1335
1336 m_d->saveDeviceAsImage(saveDevice,
1337 node->name(),
1338 saveRect,
1339 image->xRes(), image->yRes(),
1340 node->opacity());
1341}
1342
1343#include "SvgWriter.h"
1344
1346{
1347 KisShapeLayerSP shapeLayer = qobject_cast<KisShapeLayer*>(activeNode().data());
1348 if (!shapeLayer) {
1349 return;
1350 }
1351
1352 KoFileDialog dialog(m_d->view->mainWindowAsQWidget(), KoFileDialog::SaveFile, "savenodeasimage");
1353 dialog.setCaption(i18nc("@title:window", "Export to SVG"));
1354 dialog.setDefaultDir(QStandardPaths::writableLocation(QStandardPaths::PicturesLocation));
1355 dialog.setMimeTypeFilters(QStringList() << "image/svg+xml", "image/svg+xml");
1356 QString filename = dialog.filename();
1357
1358 if (filename.isEmpty()) return;
1359
1360 QUrl url = QUrl::fromLocalFile(filename);
1361
1362 if (url.isEmpty()) return;
1363
1364 const QSizeF sizeInPx = m_d->view->image()->bounds().size();
1365 const QSizeF sizeInPt(sizeInPx.width() / m_d->view->image()->xRes(),
1366 sizeInPx.height() / m_d->view->image()->yRes());
1367
1368 QList<KoShape*> shapes = shapeLayer->shapes();
1369 std::sort(shapes.begin(), shapes.end(), KoShape::compareShapeZIndex);
1370
1371 SvgWriter writer(shapes);
1372 if (!writer.save(filename, sizeInPt, true)) {
1373 QMessageBox::warning(qApp->activeWindow(), i18nc("@title:window", "Krita"), i18n("Could not save to svg: %1", filename));
1374 }
1375}
1376
1378{
1379 KisNodeSP node = activeNode();
1380 if (!canModifyLayer(node)) return;
1381
1382 // guaranteed by KisActionManager
1384
1385 KisLayerUtils::splitAlphaToMask(node->image(), node, m_d->maskManager.createMaskNameCommon(node, "KisTransparencyMask", i18n("Transparency Mask")));
1386}
1387
1389{
1390 KisNodeSP node = q->activeNode();
1391 KisNodeSP parentNode = node->parent();
1392
1393 // guaranteed by KisActionManager
1394 KIS_ASSERT_RECOVER_RETURN(node->inherits("KisTransparencyMask"));
1395
1396 if (writeToLayers && (!parentNode->hasEditablePaintDevice() || !node->isEditable(false))) {
1397 QMessageBox::information(view->mainWindowAsQWidget(),
1398 i18nc("@title:window", "Layer %1 is not editable", parentNode->name()),
1399 i18n("Cannot write alpha channel of "
1400 "the parent layer \"%1\".\n"
1401 "The operation will be cancelled.", parentNode->name()));
1402 return;
1403 }
1404
1405 KisPaintDeviceSP dstDevice;
1406 if (writeToLayers) {
1408 dstDevice = parentNode->paintDevice();
1409 } else {
1410 KisPaintDeviceSP copyDevice = parentNode->paintDevice();
1411 if (!copyDevice) {
1412 copyDevice = parentNode->original();
1413 }
1414 dstDevice = new KisPaintDevice(*copyDevice);
1415 }
1416
1417 const KoColorSpace *dstCS = dstDevice->colorSpace();
1418
1419 KisPaintDeviceSP selectionDevice = node->paintDevice();
1420 KIS_ASSERT_RECOVER_RETURN(selectionDevice->colorSpace()->pixelSize() == 1);
1421
1422 const QRect processRect =
1423 selectionDevice->exactBounds() |
1424 dstDevice->exactBounds() |
1425 selectionDevice->defaultBounds()->bounds();
1426
1427 QScopedPointer<KisTransaction> transaction;
1428
1429 if (writeToLayers) {
1430 commandsAdapter.beginMacro(kundo2_i18n("Write Alpha into a Layer"));
1431 transaction.reset(new KisTransaction(kundo2_noi18n("__write_alpha_channel__"), dstDevice));
1432 }
1433
1434 KisSequentialIterator srcIt(selectionDevice, processRect);
1435 KisSequentialIterator dstIt(dstDevice, processRect);
1436
1437 while (srcIt.nextPixel() && dstIt.nextPixel()) {
1438 quint8 *alpha8Ptr = srcIt.rawData();
1439 quint8 *dstPtr = dstIt.rawData();
1440
1441 dstCS->setOpacity(dstPtr, *alpha8Ptr, 1);
1442 }
1443
1444 if (writeToLayers) {
1445 commandsAdapter.addExtraCommand(transaction->endAndTake());
1446 commandsAdapter.removeNode(node);
1447 commandsAdapter.endMacro();
1448 } else {
1449 KisImageWSP image = view->image();
1450 QRect saveRect = image->bounds();
1451
1452 saveDeviceAsImage(dstDevice, parentNode->name(),
1453 saveRect,
1454 image->xRes(), image->yRes(),
1456 }
1457}
1458
1459
1464
1469
1471{
1472 KisNodeList nodes = this->selectedNodes();
1473 KisNodeSP active = activeNode();
1474 if (nodes.isEmpty() || !active) return;
1475
1476 bool isLocked = active->userLocked();
1477
1478 for (auto &node : nodes) {
1480 }
1481}
1482
1484{
1485 KisNodeList nodes = this->selectedNodes();
1486 KisNodeSP active = activeNode();
1487 if (nodes.isEmpty() || !active) return;
1488
1489 bool isVisible = active->visible();
1490
1491 for (auto &node : nodes) {
1493 }
1494}
1495
1497{
1498 KisNodeList nodes = this->selectedNodes();
1499 KisNodeSP active = activeNode();
1500 if (nodes.isEmpty() || !active) return;
1501
1502 auto layer = qobject_cast<KisPaintLayer*>(active.data());
1503 if (!layer) {
1504 return;
1505 }
1506
1507 bool isAlphaLocked = layer->alphaLocked();
1508 for (auto &node : nodes) {
1509 auto layer = qobject_cast<KisPaintLayer*>(node.data());
1510 if (layer) {
1512 }
1513 }
1514}
1515
1517{
1518 KisNodeList nodes = this->selectedNodes();
1519 KisNodeSP active = activeNode();
1520 if (nodes.isEmpty() || !active) return;
1521
1522 auto layer = qobject_cast<KisLayer*>(active.data());
1523 if (!layer) {
1524 return;
1525 }
1526
1527 bool isAlphaDisabled = layer->alphaChannelDisabled();
1528 for (auto &node : nodes) {
1529 auto layer = qobject_cast<KisLayer*>(node.data());
1530 if (layer) {
1532 }
1533 }
1534}
1535
1537{
1538 Q_ASSERT(node);
1539 KisLayerSP layer = qobject_cast<KisLayer*>(node.data());
1540 if (!layer) {
1541 return;
1542 }
1543
1544 KisFilterMaskSP mask = layer->colorOverlayMask();
1545 if (!mask) {
1546 // This layer does not use fast color overlay mask.
1547 return;
1548 }
1549
1551}
1552
1554{
1555 KisNodeList nodes = this->selectedNodes();
1556 if (nodes.isEmpty()) return;
1557
1558 KisNodeList::Iterator it = nodes.begin();
1559 while (it != nodes.end()) {
1560 // make sure the deleted nodes aren't referenced here again
1561 if (!it->data()->parent()) {
1562 nodes.erase(it);
1563 }
1564 it++;
1565 }
1566
1567 KisClipboard::instance()->setLayers(nodes, m_d->view->image(), false);
1568
1569 if (canModifyLayers(nodes)) {
1570 KUndo2MagicString actionName = kundo2_i18n("Cut Nodes");
1571 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1572 juggler->removeNode(nodes);
1573 }
1574}
1575
1577{
1578 KisNodeList nodes = this->selectedNodes();
1579 KisClipboard::instance()->setLayers(nodes, m_d->view->image(), true);
1580}
1581
1582void KisNodeManager::pasteLayersFromClipboard(bool changeOffset, QPointF offset, KisProcessingApplicator *applicator)
1583{
1584 const QMimeData *data = KisClipboard::instance()->layersMimeData();
1585 if (!data) return;
1586
1588
1589 KisShapeController *shapeController = dynamic_cast<KisShapeController*>(m_d->imageView->document()->shapeController());
1590 Q_ASSERT(shapeController);
1591
1592 KisDummiesFacadeBase *dummiesFacade = dynamic_cast<KisDummiesFacadeBase*>(m_d->imageView->document()->shapeController());
1593 Q_ASSERT(dummiesFacade);
1594
1595 const bool copyNode = false;
1596 KisImageSP image = m_d->view->image();
1597 KisNodeDummy *parentDummy = dummiesFacade->dummyForNode(activeNode ? activeNode : image->root());
1598 KisNodeDummy *aboveThisDummy = parentDummy ? parentDummy->lastChild() : 0;
1599
1601 image,
1602 shapeController,
1603 parentDummy,
1604 aboveThisDummy,
1605 copyNode,
1607 changeOffset,
1608 offset,
1609 applicator);
1610}
1611
1613 const QString &overrideGroupName,
1614 KisNodeSP *newGroup,
1615 KisNodeSP *newLastChild)
1616{
1617 KisNodeSP active = activeNode();
1618 if (!active) return false;
1619
1620 if (!canMoveLayer(active)) return false;
1621
1622 KisImageSP image = m_d->view->image();
1623 QString groupName = !overrideGroupName.isEmpty() ? overrideGroupName : image->nextLayerName(i18nc("A group of layers", "Group"));
1624 KisGroupLayerSP group = new KisGroupLayer(image.data(), groupName, OPACITY_OPAQUE_U8);
1625
1626 KisNodeList nodes1;
1627 nodes1 << group;
1628
1629 KisNodeList nodes2;
1632
1633 if (nodes2.size() == 0) return false;
1634
1635 if (KisLayerUtils::checkIsChildOf(active, nodes2)) {
1636 active = nodes2.first();
1637 }
1638
1639 KisNodeSP parent = active->parent();
1640 KisNodeSP aboveThis = active;
1641
1642 juggler->addNode(nodes1, parent, aboveThis);
1643 juggler->moveNode(nodes2, group, 0);
1644
1645 *newGroup = group;
1646 *newLastChild = nodes2.last();
1647
1648 return true;
1649}
1650
1652{
1653 KUndo2MagicString actionName = kundo2_i18n("Quick Group");
1654 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1655
1656 KisNodeSP parent;
1657 KisNodeSP above;
1658
1659 createQuickGroupImpl(juggler, "", &parent, &above);
1660}
1661
1663{
1664 KUndo2MagicString actionName = kundo2_i18n("Quick Clipping Group");
1665 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1666
1667 KisNodeSP parent;
1668 KisNodeSP above;
1669
1670 KisImageSP image = m_d->view->image();
1671 if (createQuickGroupImpl(juggler, image->nextLayerName(i18nc("default name for a clipping group layer", "Clipping Group")), &parent, &above)) {
1672 KisPaintLayerSP maskLayer = new KisPaintLayer(image.data(), i18nc("default name for quick clip group mask layer", "Mask Layer"), OPACITY_OPAQUE_U8, image->colorSpace());
1673 maskLayer->disableAlphaChannel(true);
1674
1675 juggler->addNode(KisNodeList() << maskLayer, parent, above);
1676 }
1677}
1678
1680{
1681 KisNodeSP active = activeNode();
1682 if (!active) return;
1683
1684 if (!canModifyLayer(active)) return;
1685
1686 KisNodeSP parent = active->parent();
1687 KisNodeSP aboveThis = active;
1688
1689 auto checkCanMoveLayers = [this] (KisNodeList nodes, KisNodeSP newParent) -> bool {
1690 auto incompatibleNode =
1691 std::find_if(nodes.begin(), nodes.end(),
1692 [newParent] (KisNodeSP node) {
1693 return !newParent->allowAsChild(node);
1694 });
1695
1696 if (incompatibleNode != nodes.end()) {
1697 const QString message =
1698 newParent->parent() ?
1699 i18n("Cannot move layer \"%1\" into new parent \"%2\"",
1700 (*incompatibleNode)->name(),
1701 newParent->name()) :
1702 i18n("Cannot move layer \"%1\" into the root layer",
1703 (*incompatibleNode)->name());
1704 m_d->view->showFloatingMessage(message, QIcon());
1705 return false;
1706 }
1707 return true;
1708 };
1709
1710 KUndo2MagicString actionName = kundo2_i18n("Quick Ungroup");
1711
1712 if (parent && dynamic_cast<KisGroupLayer*>(active.data())) {
1713 KisNodeList nodes = active->childNodes(QStringList(), KoProperties());
1714
1715 if (checkCanMoveLayers(nodes, parent)) {
1716 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1717 juggler->moveNode(nodes, parent, active);
1718 juggler->removeNode(KisNodeList() << active);
1719 }
1720 } else if (parent && parent->parent()) {
1721 KisNodeSP grandParent = parent->parent();
1722
1723 KisNodeList allChildNodes = parent->childNodes(QStringList(), KoProperties());
1724 KisNodeList allSelectedNodes = selectedNodes();
1725
1726 const bool removeParent = KritaUtils::compareListsUnordered(allChildNodes, allSelectedNodes);
1727
1728 if (checkCanMoveLayers(allSelectedNodes, parent)) {
1729 KisNodeJugglerCompressed *juggler = m_d->lazyGetJuggler(actionName);
1730 juggler->moveNode(allSelectedNodes, grandParent, parent);
1731 if (removeParent) {
1732 juggler->removeNode(KisNodeList() << parent);
1733 }
1734 }
1735 }
1736}
1737
1738void KisNodeManager::selectLayersImpl(const KoProperties &props, const KoProperties &invertedProps)
1739{
1740 KisImageSP image = m_d->view->image();
1741 KisNodeList nodes = KisLayerUtils::findNodesWithProps(image->root(), props, true);
1742
1744
1746 nodes = KisLayerUtils::findNodesWithProps(image->root(), invertedProps, true);
1747 }
1748
1749 if (!nodes.isEmpty()) {
1750 slotImageRequestNodeReselection(nodes.last(), nodes);
1751 }
1752}
1753
1755{
1756 KoProperties props;
1757 selectLayersImpl(props, props);
1758}
1759
1761{
1762 KoProperties props;
1763 props.setProperty("visible", true);
1764
1765 KoProperties invertedProps;
1766 invertedProps.setProperty("visible", false);
1767
1768 selectLayersImpl(props, invertedProps);
1769}
1770
1772{
1773 KoProperties props;
1774 props.setProperty("locked", true);
1775
1776 KoProperties invertedProps;
1777 invertedProps.setProperty("locked", false);
1778
1779 selectLayersImpl(props, invertedProps);
1780}
1781
1783{
1784 KoProperties props;
1785 props.setProperty("visible", false);
1786
1787 KoProperties invertedProps;
1788 invertedProps.setProperty("visible", true);
1789
1790 selectLayersImpl(props, invertedProps);
1791}
1792
1794{
1795 KoProperties props;
1796 props.setProperty("locked", false);
1797
1798 KoProperties invertedProps;
1799 invertedProps.setProperty("locked", true);
1800
1801 selectLayersImpl(props, invertedProps);
1802}
1803
1805{
1806 if (!sender()->property("node").isNull()) {
1807 QString name = sender()->property("node").toString();
1808 KisNodeSP node = KisLayerUtils::findNodeByName(m_d->imageView->image()->rootLayer(),name);
1809 if (node) {
1810 slotUiActivatedNode(node);
1811 }
1812 }
1813}
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:388
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:826
void disableAlphaChannel(bool disable)
Definition kis_layer.cc:319
virtual KisSelectionMaskSP selectionMask() const
Definition kis_layer.cc:504
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:573
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