[calligra] /: FEATURE: Isolated Mode for Krita nodes

Dmitry Kazakov dimula73 at gmail.com
Thu May 23 16:07:41 UTC 2013


Git commit e1e85dea054bf7718a05a6e63cec83b59ffe0315 by Dmitry Kazakov.
Committed on 23/05/2013 at 18:06.
Pushed by dkazakov into branch 'master'.

FEATURE: Isolated Mode for Krita nodes

Implemented a mode, which allow a user to limit visible layers by
a subtree of a graph. It means you can look at a layer or mask or
a group in isolated environment, without other nodes seen.

To switch on the mode, you can select a context menu item in the
Layers Docker. To switch off, just select another layer or just use
the same context menu.

CCMAIL:kimageshop at kde.org

M  +52   -9    krita/image/kis_image.cc
M  +15   -16   krita/image/kis_image.h
M  +10   -0    krita/image/kis_mask.cc
M  +10   -0    krita/image/kis_mask.h
M  +8    -6    krita/plugins/extensions/dockers/defaultdockers/kis_layer_box.cpp
M  +1    -1    krita/plugins/paintops/libpaintop/kis_custom_brush_widget.cpp
M  +56   -0    krita/ui/kis_node_manager.cpp
M  +4    -0    krita/ui/kis_node_manager.h
M  +45   -2    krita/ui/kis_node_model.cpp
M  +6    -0    krita/ui/kis_node_model.h
M  +2    -2    libs/main/KoDocumentSectionDelegate.cpp
M  +1    -1    libs/main/KoDocumentSectionDelegate.h

http://commits.kde.org/calligra/e1e85dea054bf7718a05a6e63cec83b59ffe0315

diff --git a/krita/image/kis_image.cc b/krita/image/kis_image.cc
index 0c5fc22..ad6cceb 100644
--- a/krita/image/kis_image.cc
+++ b/krita/image/kis_image.cc
@@ -112,6 +112,7 @@ public:
     KisGroupLayerSP rootLayer; // The layers are contained in here
     QList<KisLayer*> dirtyLayers; // for thumbnails
     QList<KisLayerComposition*> compositions;
+    KisNodeSP isolatedRootNode;
 
     KisNameServer *nserver;
 
@@ -177,6 +178,13 @@ KisImage::~KisImage()
     disconnect(); // in case Qt gets confused
 }
 
+void KisImage::aboutToAddANode(KisNode *parent, int index)
+{
+    KisNodeGraphListener::aboutToAddANode(parent, index);
+    SANITY_CHECK_LOCKED("aboutToAddANode");
+    stopIsolatedMode();
+}
+
 void KisImage::nodeHasBeenAdded(KisNode *parent, int index)
 {
     KisNodeGraphListener::nodeHasBeenAdded(parent, index);
@@ -187,6 +195,7 @@ void KisImage::nodeHasBeenAdded(KisNode *parent, int index)
 
 void KisImage::aboutToRemoveANode(KisNode *parent, int index)
 {
+    stopIsolatedMode();
     KisNodeGraphListener::aboutToRemoveANode(parent, index);
 
     SANITY_CHECK_LOCKED("aboutToRemoveANode");
@@ -807,6 +816,11 @@ KisGroupLayerSP KisImage::rootLayer() const
 
 KisPaintDeviceSP KisImage::projection()
 {
+    if (m_d->isolatedRootNode) {
+        return m_d->isolatedRootNode->projection();
+    }
+
+
     Q_ASSERT(m_d->rootLayer);
     KisPaintDeviceSP projection = m_d->rootLayer->projection();
     Q_ASSERT(projection);
@@ -1059,7 +1073,7 @@ QImage KisImage::convertToQImage(qint32 x,
                                  qint32 h,
                                  const KoColorProfile * profile)
 {
-    KisPaintDeviceSP dev = m_d->rootLayer->projection();
+    KisPaintDeviceSP dev = projection();
     if (!dev) return QImage();
     QImage image = dev->convertToQImage(const_cast<KoColorProfile*>(profile), x, y, w, h,
                                         KoColorConversionTransformation::InternalRenderingIntent,
@@ -1114,7 +1128,7 @@ QImage KisImage::convertToQImage(const QRect& scaledRect, const QSize& scaledIma
     srcRect.setTop(static_cast<int>(scaledRect.top() * yScale));
     srcRect.setBottom(static_cast<int>(ceil((scaledRect.bottom() + 1) * yScale)) - 1);
 
-    KisPaintDeviceSP mergedImage = m_d->rootLayer->projection();
+    KisPaintDeviceSP mergedImage = projection();
     quint8 *scaledImageData = new quint8[scaledRect.width() * scaledRect.height() * pixelSize];
 
     quint8 *imageRow = new quint8[srcRect.width() * pixelSize];
@@ -1173,13 +1187,6 @@ QImage KisImage::convertToQImage(const QRect& scaledRect, const QSize& scaledIma
     return image;
 }
 
-
-KisPaintDeviceSP KisImage::mergedImage()
-{
-    refreshGraph();
-    return m_d->rootLayer->projection();
-}
-
 void KisImage::notifyLayersChanged()
 {
     m_d->signalRouter->emitNotification(LayersChangedSignal);
@@ -1221,6 +1228,8 @@ KisActionRecorder* KisImage::actionRecorder() const
 
 void KisImage::setRootLayer(KisGroupLayerSP rootLayer)
 {
+    stopIsolatedMode();
+
     if (m_d->rootLayer) {
         m_d->rootLayer->setGraphListener(0);
         m_d->rootLayer->disconnect();
@@ -1348,6 +1357,40 @@ KisStrokeId KisImage::startStroke(KisStrokeStrategy *strokeStrategy)
     return id;
 }
 
+void KisImage::startIsolatedMode(KisNodeSP node)
+{
+    barrierLock();
+    unlock();
+
+    m_d->isolatedRootNode = node;
+    emit sigIsolatedModeChanged();
+
+    notifyProjectionUpdated(bounds());
+}
+
+void KisImage::stopIsolatedMode()
+{
+    if (!m_d->isolatedRootNode)  return;
+
+    KisNodeSP oldRootNode = m_d->isolatedRootNode;
+    m_d->isolatedRootNode = 0;
+
+    emit sigIsolatedModeChanged();
+
+    notifyProjectionUpdated(bounds());
+
+    // TODO: Substitute notifyProjectionUpdated() with this code
+    // when update optimization is implemented
+    // 
+    // QRect updateRect = bounds() | oldRootNode->extent();
+    // oldRootNode->setDirty(updateRect);
+}
+
+KisNodeSP KisImage::isolatedModeRoot() const
+{
+    return m_d->isolatedRootNode;
+}
+
 void KisImage::addJob(KisStrokeId id, KisStrokeJobData *data)
 {
     if (m_d->scheduler) {
diff --git a/krita/image/kis_image.h b/krita/image/kis_image.h
index 11e17ca..7988df5 100644
--- a/krita/image/kis_image.h
+++ b/krita/image/kis_image.h
@@ -84,6 +84,7 @@ public:
 
 public: // KisNodeGraphListener implementation
 
+    void aboutToAddANode(KisNode *parent, int index);
     void nodeHasBeenAdded(KisNode *parent, int index);
     void aboutToRemoveANode(KisNode *parent, int index);
     void nodeChanged(KisNode * node);
@@ -387,22 +388,6 @@ public:
     }
 
     /**
-     * Starting form 2.3 mergedImage() is declared deprecated.
-     * If you want to get a projection of the image, please use
-     * something like:
-     *
-     * image->lock();
-     * read_something_from_the_image(image->projection());
-     * image->unlock();
-     *
-     * or if you want to get a full refresh of the image graph
-     * performed beforehand (do you really want it?) (sure?) then
-     * you can add a call to image->refreshGraph() before locking
-     * the image.
-     */
-    KDE_DEPRECATED KisPaintDeviceSP mergedImage();
-
-    /**
      * @return the root node of the image node graph
      */
     KisGroupLayerSP rootLayer() const;
@@ -503,6 +488,11 @@ public:
      */
     void removeComposition(KisLayerComposition* composition);
 
+public:
+    void startIsolatedMode(KisNodeSP node);
+    void stopIsolatedMode();
+    KisNodeSP isolatedModeRoot() const;
+
 signals:
 
     /**
@@ -622,6 +612,15 @@ signals:
      */
     void sigStrokeEndRequested();
 
+    /**
+     * Emitted when the isolated mode status has changed.
+     *
+     * Can be used by the recievers to catch a fact of forcefully
+     * stopping the isolated mode by the image when some complex
+     * action was requested
+     */
+    void sigIsolatedModeChanged();
+
 public slots:
     KisCompositeProgressProxy* compositeProgressProxy();
 
diff --git a/krita/image/kis_mask.cc b/krita/image/kis_mask.cc
index d4ba755..c3953a6 100644
--- a/krita/image/kis_mask.cc
+++ b/krita/image/kis_mask.cc
@@ -179,6 +179,16 @@ KisPaintDeviceSP KisMask::paintDevice() const
     return selection()->getOrCreatePixelSelection();
 }
 
+KisPaintDeviceSP KisMask::original() const
+{
+    return paintDevice();
+}
+
+KisPaintDeviceSP KisMask::projection() const
+{
+    return paintDevice();
+}
+
 void KisMask::setSelection(KisSelectionSP selection)
 {
     m_d->selection = selection;
diff --git a/krita/image/kis_mask.h b/krita/image/kis_mask.h
index 056ec68..92c2eaf 100644
--- a/krita/image/kis_mask.h
+++ b/krita/image/kis_mask.h
@@ -119,6 +119,16 @@ public:
     KisPaintDeviceSP paintDevice() const;
 
     /**
+     * @return the same as paintDevice()
+     */
+    KisPaintDeviceSP original() const;
+
+    /**
+     * @return the same as paintDevice()
+     */
+    KisPaintDeviceSP projection() const;
+
+    /**
      * Change the selection to the specified selection object. The
      * selection is deep copied.
      */
diff --git a/krita/plugins/extensions/dockers/defaultdockers/kis_layer_box.cpp b/krita/plugins/extensions/dockers/defaultdockers/kis_layer_box.cpp
index 00755a3..81e1b78 100644
--- a/krita/plugins/extensions/dockers/defaultdockers/kis_layer_box.cpp
+++ b/krita/plugins/extensions/dockers/defaultdockers/kis_layer_box.cpp
@@ -454,14 +454,16 @@ void KisLayerBox::slotContextMenuRequested(const QPoint &pos, const QModelIndex
         QAction* mergeLayerDown = menu.addAction(i18n("&Merge with Layer Below"), this, SLOT(slotMergeLayer()));
         if (!index.sibling(index.row() + 1, 0).isValid()) mergeLayerDown->setEnabled(false);
         menu.addSeparator();
+
+        QMenu *convertToMenu = menu.addMenu(i18n("&Convert"));
+        addActionToMenu(convertToMenu, "convert_to_paint_layer");
+        addActionToMenu(convertToMenu, "convert_to_transparency_mask");
+        addActionToMenu(convertToMenu, "convert_to_filter_mask");
+        addActionToMenu(convertToMenu, "convert_to_selection_mask");
+
+        addActionToMenu(&menu, "isolate_layer");
     }
     menu.addSeparator();
-    QMenu *convertToMenu = menu.addMenu(i18n("&Convert"));
-    addActionToMenu(convertToMenu, "convert_to_paint_layer");
-    addActionToMenu(convertToMenu, "convert_to_transparency_mask");
-    addActionToMenu(convertToMenu, "convert_to_filter_mask");
-    addActionToMenu(convertToMenu, "convert_to_selection_mask");
-    menu.addSeparator();
     addActionToMenu(&menu, "add_new_transparency_mask");
     addActionToMenu(&menu, "add_new_filter_mask");
     addActionToMenu(&menu, "add_new_selection_mask");
diff --git a/krita/plugins/paintops/libpaintop/kis_custom_brush_widget.cpp b/krita/plugins/paintops/libpaintop/kis_custom_brush_widget.cpp
index ac1601c..0d77ccd 100644
--- a/krita/plugins/paintops/libpaintop/kis_custom_brush_widget.cpp
+++ b/krita/plugins/paintops/libpaintop/kis_custom_brush_widget.cpp
@@ -189,7 +189,7 @@ void KisCustomBrushWidget::createBrush()
         KisSelectionSP selection = m_image->globalSelection();
         // create copy of the data
         m_image->lock();
-        KisPaintDeviceSP dev = new KisPaintDevice(*m_image->mergedImage());
+        KisPaintDeviceSP dev = new KisPaintDevice(*m_image->projection());
         m_image->unlock();
 
         if (!selection){
diff --git a/krita/ui/kis_node_manager.cpp b/krita/ui/kis_node_manager.cpp
index 1a7b650..1604ea5 100644
--- a/krita/ui/kis_node_manager.cpp
+++ b/krita/ui/kis_node_manager.cpp
@@ -270,6 +270,16 @@ void KisNodeManager::setup(KActionCollection * actionCollection, KisActionManage
     connect(&m_d->nodeConversionSignalMapper, SIGNAL(mapped(const QString &)),
             this, SLOT(convertNode(const QString &)));
 
+    action = new KisAction(koIcon("view-filter"), i18n("&Isolate Layer"), this);
+    action->setCheckable(true);
+    action->setActivationFlags(KisAction::ACTIVE_NODE);
+    actionManager->addAction("isolate_layer", action, actionCollection);
+    connect(action, SIGNAL(triggered(bool)), this, SLOT(toggleIsolateMode(bool)));
+
+    connect(m_d->view->image(), SIGNAL(sigIsolatedModeChanged()),
+            this, SLOT(slotUpdateIsolateModeAction()));
+    connect(this, SIGNAL(sigNodeActivated(KisNodeSP)), SLOT(slotUpdateIsolateModeAction()));
+    connect(this, SIGNAL(sigNodeActivated(KisNodeSP)), SLOT(slotTryFinishIsolatedMode()));
 }
 
 void KisNodeManager::updateGUI()
@@ -333,6 +343,52 @@ void KisNodeManager::moveNodeDirect(KisNodeSP node, KisNodeSP parent, KisNodeSP
     m_d->commandsAdapter->moveNode(node, parent, aboveThis);
 }
 
+void KisNodeManager::toggleIsolateMode(bool checked)
+{
+    KisImageWSP image = m_d->view->image();
+
+    if (checked) {
+        KisNodeSP activeNode = this->activeNode();
+        Q_ASSERT(activeNode);
+
+        image->startIsolatedMode(activeNode);
+    } else {
+        image->stopIsolatedMode();
+    }
+}
+
+void KisNodeManager::slotUpdateIsolateModeAction()
+{
+    KisAction *action = m_d->view->actionManager()->actionByName("isolate_layer");
+    Q_ASSERT(action);
+
+    KisNodeSP activeNode = this->activeNode();
+    KisNodeSP isolatedRootNode = m_d->view->image()->isolatedModeRoot();
+
+    action->setChecked(isolatedRootNode && isolatedRootNode == activeNode);
+}
+
+void KisNodeManager::slotTryFinishIsolatedMode()
+{
+    KisNodeSP isolatedRootNode = m_d->view->image()->isolatedModeRoot();
+    if (!isolatedRootNode) return;
+
+    bool belongsToIsolatedGroup = false;
+
+    KisNodeSP node = this->activeNode();
+    while(node) {
+        if (node == isolatedRootNode) {
+            belongsToIsolatedGroup = true;
+            break;
+        }
+        node = node->parent();
+    }
+
+    if (!belongsToIsolatedGroup) {
+        m_d->view->image()->stopIsolatedMode();
+    }
+}
+
 void KisNodeManager::createNode(const QString & nodeType)
 {
     KisNodeSP activeNode = this->activeNode();
diff --git a/krita/ui/kis_node_manager.h b/krita/ui/kis_node_manager.h
index 9e8596b..1f81239 100644
--- a/krita/ui/kis_node_manager.h
+++ b/krita/ui/kis_node_manager.h
@@ -138,6 +138,10 @@ public slots:
      */
     void moveNodeDirect(KisNodeSP node, KisNodeSP parent, KisNodeSP aboveThis);
 
+    void toggleIsolateMode(bool checked);
+    void slotUpdateIsolateModeAction();
+    void slotTryFinishIsolatedMode();
+
     void moveNodeAt(KisNodeSP node, KisNodeSP parent, int index);
     void createNode(const QString &node);
     void convertNode(const QString &nodeType);
diff --git a/krita/ui/kis_node_model.cpp b/krita/ui/kis_node_model.cpp
index 8d64888..1b354c3 100644
--- a/krita/ui/kis_node_model.cpp
+++ b/krita/ui/kis_node_model.cpp
@@ -99,6 +99,26 @@ QModelIndex KisNodeModel::indexFromNode(KisNodeSP node) const
     return m_d->indexConverter->indexFromDummy(dummy);
 }
 
+bool KisNodeModel::belongsToIsolatedGroup(KisNodeSP node) const
+{
+    KisNodeSP isolatedRoot = m_d->image->isolatedModeRoot();
+    if (!isolatedRoot) return true;
+
+    KisNodeDummy *isolatedRootDummy =
+        m_d->dummiesFacade->dummyForNode(isolatedRoot);
+    KisNodeDummy *dummy =
+        m_d->dummiesFacade->dummyForNode(node);
+
+    while (dummy) {
+        if (dummy == isolatedRootDummy) {
+            return true;
+        }
+        dummy = dummy->parent();
+    }
+
+    return false;
+}
+
 void KisNodeModel::resetIndexConverter()
 {
     delete m_d->indexConverter;
@@ -116,6 +136,23 @@ void KisNodeModel::resetIndexConverter()
     }
 }
 
+void KisNodeModel::regenerateItems(KisNodeDummy *dummy)
+{
+    const QModelIndex &index = m_d->indexConverter->indexFromDummy(dummy);
+    emit dataChanged(index, index);
+
+    dummy = dummy->firstChild();
+    while (dummy) {
+        regenerateItems(dummy);
+        dummy = dummy->nextSibling();
+    }
+}
+
+void KisNodeModel::slotIsolatedModeChanged()
+{
+    regenerateItems(m_d->dummiesFacade->rootDummy());
+}
+
 void KisNodeModel::updateSettings()
 {
     KisConfig cfg;
@@ -163,14 +200,16 @@ void KisNodeModel::connectDummies(KisNodeDummy *dummy, bool needConnect)
 
 void KisNodeModel::setDummiesFacade(KisDummiesFacadeBase *dummiesFacade, KisImageWSP image, KisShapeController *shapeController)
 {
-    m_d->image = image;
+
     m_d->shapeController = shapeController;
 
     if(m_d->dummiesFacade) {
+        m_d->image->disconnect(this);
         m_d->dummiesFacade->disconnect(this);
         connectDummies(m_d->dummiesFacade->rootDummy(), false);
     }
 
+    m_d->image = image;
     m_d->dummiesFacade = dummiesFacade;
     resetIndexConverter();
 
@@ -191,6 +230,8 @@ void KisNodeModel::setDummiesFacade(KisDummiesFacadeBase *dummiesFacade, KisImag
 
         connect(m_d->dummiesFacade, SIGNAL(sigDummyChanged(KisNodeDummy*)),
                 SLOT(slotDummyChanged(KisNodeDummy*)));
+
+        connect(m_d->image, SIGNAL(sigIsolatedModeChanged()), SLOT(slotIsolatedModeChanged()));
     }
 
     reset();
@@ -320,6 +361,8 @@ QVariant KisNodeModel::data(const QModelIndex &index, int role) const
     case Qt::DecorationRole: return node->icon();
     case Qt::EditRole: return node->name();
     case Qt::SizeHintRole: return m_d->image->size(); // FIXME
+    case Qt::TextColorRole:
+        return belongsToIsolatedGroup(node) ? QVariant() : Qt::gray;
     case PropertiesRole: return QVariant::fromValue(node->sectionModelProperties());
     case AspectRatioRole: return double(m_d->image->width()) / m_d->image->height();
     case ProgressRole: {
@@ -327,7 +370,7 @@ QVariant KisNodeModel::data(const QModelIndex &index, int role) const
         return proxy ? proxy->percentage() : -1;
     }
     default:
-        if (role >= int(BeginThumbnailRole))
+        if (role >= int(BeginThumbnailRole) && belongsToIsolatedGroup(node))
             return node->createThumbnail(role - int(BeginThumbnailRole), role - int(BeginThumbnailRole));
         else
             return QVariant();
diff --git a/krita/ui/kis_node_model.h b/krita/ui/kis_node_model.h
index 6352853..76d04eb 100644
--- a/krita/ui/kis_node_model.h
+++ b/krita/ui/kis_node_model.h
@@ -79,6 +79,8 @@ private slots:
     void slotEndRemoveDummy();
     void slotDummyChanged(KisNodeDummy *dummy);
 
+    void slotIsolatedModeChanged();
+
     void updateSettings();
     void processUpdateQueue();
     void progressPercentageChanged(int, const KisNodeSP);
@@ -95,6 +97,10 @@ private:
     bool correctNewNodeLocation(KisNodeSP node,
                                 KisNodeDummy* &parentDummy,
                                 KisNodeDummy* &aboveThisDummy);
+
+    void regenerateItems(KisNodeDummy *dummy);
+    bool belongsToIsolatedGroup(KisNodeSP node) const;
+
 private:
 
     struct Private;
diff --git a/libs/main/KoDocumentSectionDelegate.cpp b/libs/main/KoDocumentSectionDelegate.cpp
index 13d6100..81ae7b1 100644
--- a/libs/main/KoDocumentSectionDelegate.cpp
+++ b/libs/main/KoDocumentSectionDelegate.cpp
@@ -264,9 +264,9 @@ bool KoDocumentSectionDelegate::eventFilter(QObject *object, QEvent *event)
 // PRIVATE
 
 
-QStyleOptionViewItem KoDocumentSectionDelegate::getOptions(const QStyleOptionViewItem &o, const QModelIndex &index)
+QStyleOptionViewItemV4 KoDocumentSectionDelegate::getOptions(const QStyleOptionViewItem &o, const QModelIndex &index)
 {
-    QStyleOptionViewItem option = o;
+    QStyleOptionViewItemV4 option = o;
     QVariant v = index.data(Qt::FontRole);
     if (v.isValid()) {
         option.font = v.value<QFont>();
diff --git a/libs/main/KoDocumentSectionDelegate.h b/libs/main/KoDocumentSectionDelegate.h
index e804f5d..ad4b9f1 100644
--- a/libs/main/KoDocumentSectionDelegate.h
+++ b/libs/main/KoDocumentSectionDelegate.h
@@ -57,7 +57,7 @@ private:
     class Private;
     Private* const d;
 
-    static QStyleOptionViewItem getOptions(const QStyleOptionViewItem &option, const QModelIndex &index);
+    static QStyleOptionViewItemV4 getOptions(const QStyleOptionViewItem &option, const QModelIndex &index);
     int thumbnailHeight(const QStyleOptionViewItem &option, const QModelIndex &index) const;
     int availableWidth() const;
     int textBoxHeight(const QStyleOptionViewItem &option) const;


More information about the kimageshop mailing list