IDEA-121775 XDebugger: merge Watches and Variables in one view - allow to switch...
[idea/community.git] / platform / xdebugger-impl / src / com / intellij / xdebugger / impl / ui / tree / nodes / WatchesRootNode.java
1 /*
2  * Copyright 2000-2016 JetBrains s.r.o.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 package com.intellij.xdebugger.impl.ui.tree.nodes;
17
18 import com.intellij.util.ArrayUtil;
19 import com.intellij.util.containers.ContainerUtil;
20 import com.intellij.util.ui.tree.TreeUtil;
21 import com.intellij.xdebugger.XDebugSession;
22 import com.intellij.xdebugger.XExpression;
23 import com.intellij.xdebugger.evaluation.XDebuggerEvaluator;
24 import com.intellij.xdebugger.frame.XCompositeNode;
25 import com.intellij.xdebugger.frame.XStackFrame;
26 import com.intellij.xdebugger.frame.XValueChildrenList;
27 import com.intellij.xdebugger.frame.XValueContainer;
28 import com.intellij.xdebugger.impl.breakpoints.XExpressionImpl;
29 import com.intellij.xdebugger.impl.frame.WatchInplaceEditor;
30 import com.intellij.xdebugger.impl.frame.XWatchesView;
31 import com.intellij.xdebugger.impl.ui.tree.XDebuggerTree;
32 import org.jetbrains.annotations.NotNull;
33 import org.jetbrains.annotations.Nullable;
34
35 import javax.swing.tree.TreeNode;
36 import java.util.Collection;
37 import java.util.List;
38
39 import static com.intellij.xdebugger.impl.frame.XDebugView.getSession;
40
41 /**
42  * @author nik
43  */
44 public class WatchesRootNode extends XValueContainerNode<XValueContainer> {
45   private final XWatchesView myWatchesView;
46   private final List<WatchNodeImpl> myChildren;
47
48   public WatchesRootNode(@NotNull XDebuggerTree tree,
49                          @NotNull XWatchesView watchesView,
50                          @NotNull XExpression[] expressions) {
51     this(tree, watchesView, expressions, null, false);
52   }
53
54   public WatchesRootNode(@NotNull XDebuggerTree tree,
55                          @NotNull XWatchesView watchesView,
56                          @NotNull XExpression[] expressions,
57                          @Nullable XStackFrame stackFrame,
58                          boolean watchesInVariables) {
59     super(tree, null, new XValueContainer() {
60       @Override
61       public void computeChildren(@NotNull XCompositeNode node) {
62         if (stackFrame != null && watchesInVariables) {
63           stackFrame.computeChildren(node);
64         }
65         else {
66           node.addChildren(XValueChildrenList.EMPTY, true);
67         }
68       }
69     });
70     setLeaf(false);
71     myWatchesView = watchesView;
72     myChildren = ContainerUtil.newArrayList();
73     for (XExpression watchExpression : expressions) {
74       myChildren.add(new WatchNodeImpl(myTree, this, watchExpression, stackFrame));
75     }
76   }
77
78   @NotNull
79   @Override
80   public List<? extends XValueContainerNode<?>> getLoadedChildren() {
81     return ContainerUtil.concat(myChildren, super.getLoadedChildren());
82   }
83
84   @NotNull
85   @Override
86   public List<? extends TreeNode> getChildren() {
87     List<? extends TreeNode> children = super.getChildren();
88     return ContainerUtil.concat(myChildren, children);
89   }
90
91   @NotNull
92   public List<? extends WatchNode> getWatchChildren() {
93     return myChildren;
94   }
95
96   @Override
97   public void clearChildren() {
98     super.clearChildren();
99     myChildren.clear();
100   }
101
102   /**
103    * @deprecated Use {@link #addWatchExpression(XStackFrame, XExpression, int, boolean)}
104    */
105   @Deprecated
106   public void addWatchExpression(@Nullable XDebuggerEvaluator evaluator,
107                                  @NotNull XExpression expression,
108                                  int index,
109                                  boolean navigateToWatchNode) {
110     addWatchExpression((XStackFrame)null, expression, index, navigateToWatchNode);
111   }
112
113   public void addWatchExpression(@Nullable XStackFrame stackFrame,
114                                  @NotNull XExpression expression,
115                                  int index,
116                                  boolean navigateToWatchNode) {
117     WatchNodeImpl message = new WatchNodeImpl(myTree, this, expression, stackFrame);
118     if (index == -1) {
119       myChildren.add(message);
120       index = myChildren.size() - 1;
121     }
122     else {
123       myChildren.add(index, message);
124     }
125     fireNodeInserted(index);
126     TreeUtil.selectNode(myTree, message);
127     if (navigateToWatchNode) {
128       myTree.scrollPathToVisible(message.getPath());
129     }
130   }
131
132   private void fireNodeInserted(int index) {
133     myTree.getTreeModel().nodesWereInserted(this, new int[]{index});
134   }
135
136   @SuppressWarnings("SuspiciousMethodCalls")
137   public int removeChildNode(XDebuggerTreeNode node) {
138     int index = myChildren.indexOf(node);
139     if (index != -1) {
140       myChildren.remove(node);
141       fireNodesRemoved(new int[]{index}, new TreeNode[]{node});
142     }
143     return index;
144   }
145
146   public void removeChildren(Collection<? extends XDebuggerTreeNode> nodes) {
147     int[] indices = getNodesIndices(nodes);
148     TreeNode[] removed = getChildNodes(indices);
149     myChildren.removeAll(nodes);
150     fireNodesRemoved(indices, removed);
151   }
152
153   public void removeAllChildren() {
154     myChildren.clear();
155     fireNodeStructureChanged();
156   }
157
158   public void moveUp(WatchNode node) {
159     int index = getIndex(node);
160     if (index > 0) {
161       ContainerUtil.swapElements(myChildren, index, index - 1);
162     }
163     fireNodeStructureChanged();
164     getTree().setSelectionRow(index - 1);
165   }
166
167   public void moveDown(WatchNode node) {
168     int index = getIndex(node);
169     if (index < myChildren.size() - 1) {
170       ContainerUtil.swapElements(myChildren, index, index + 1);
171     }
172     fireNodeStructureChanged();
173     getTree().setSelectionRow(index + 1);
174   }
175
176   public void addNewWatch() {
177     editWatch(null);
178   }
179
180   public void editWatch(@Nullable WatchNodeImpl node) {
181     WatchNodeImpl messageNode;
182     int index = node != null ? myChildren.indexOf(node) : -1;
183     if (index == -1) {
184       int selectedIndex = myChildren.indexOf(ArrayUtil.getFirstElement(myTree.getSelectedNodes(WatchNodeImpl.class, null)));
185       int targetIndex = selectedIndex == - 1 ? myChildren.size() : selectedIndex + 1;
186       messageNode = new WatchNodeImpl(myTree, this, XExpressionImpl.EMPTY_EXPRESSION, null);
187       myChildren.add(targetIndex, messageNode);
188       fireNodeInserted(targetIndex);
189       getTree().setSelectionRows(ArrayUtil.EMPTY_INT_ARRAY);
190     }
191     else {
192       messageNode = node;
193     }
194     XDebugSession session = getSession(myTree);
195     new WatchInplaceEditor(this, session, myWatchesView, messageNode, "watch", node).show();
196   }
197 }