]> git.argeo.org Git - gpl/argeo-slc.git/blobdiff - plugins/org.argeo.slc.client.ui/src/main/java/org/argeo/slc/client/ui/views/JcrResultTreeView.java
Migrate SLC modules
[gpl/argeo-slc.git] / plugins / org.argeo.slc.client.ui / src / main / java / org / argeo / slc / client / ui / views / JcrResultTreeView.java
index 9d5bb8cd97a87b09e4f016e8a87bed5528a0b73b..f0ec30ed91406448220fd1386ab87222aa3ee18d 100644 (file)
@@ -1,52 +1,69 @@
+/*\r
+ * Copyright (C) 2007-2012 Argeo GmbH\r
+ *\r
+ * Licensed under the Apache License, Version 2.0 (the "License");\r
+ * you may not use this file except in compliance with the License.\r
+ * You may obtain a copy of the License at\r
+ *\r
+ *         http://www.apache.org/licenses/LICENSE-2.0\r
+ *\r
+ * Unless required by applicable law or agreed to in writing, software\r
+ * distributed under the License is distributed on an "AS IS" BASIS,\r
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r
+ * See the License for the specific language governing permissions and\r
+ * limitations under the License.\r
+ */\r
 package org.argeo.slc.client.ui.views;\r
 \r
+import java.text.DateFormat;\r
+import java.text.SimpleDateFormat;\r
 import java.util.ArrayList;\r
 import java.util.Calendar;\r
+import java.util.Iterator;\r
 import java.util.List;\r
 \r
 import javax.jcr.Node;\r
 import javax.jcr.NodeIterator;\r
 import javax.jcr.Property;\r
-import javax.jcr.PropertyIterator;\r
-import javax.jcr.PropertyType;\r
+import javax.jcr.Repository;\r
 import javax.jcr.RepositoryException;\r
 import javax.jcr.Session;\r
-import javax.jcr.Value;\r
 import javax.jcr.nodetype.NodeType;\r
 import javax.jcr.observation.Event;\r
 import javax.jcr.observation.EventListener;\r
 import javax.jcr.observation.ObservationManager;\r
 \r
-import org.apache.commons.logging.Log;\r
-import org.apache.commons.logging.LogFactory;\r
 import org.argeo.ArgeoException;\r
 import org.argeo.eclipse.ui.jcr.AsyncUiEventListener;\r
 import org.argeo.eclipse.ui.utils.CommandUtils;\r
 import org.argeo.jcr.JcrUtils;\r
-import org.argeo.jcr.UserJcrUtils;\r
 import org.argeo.slc.SlcException;\r
 import org.argeo.slc.client.ui.ClientUiPlugin;\r
 import org.argeo.slc.client.ui.SlcUiConstants;\r
 import org.argeo.slc.client.ui.commands.AddResultFolder;\r
+import org.argeo.slc.client.ui.commands.DeleteItems;\r
+import org.argeo.slc.client.ui.commands.RefreshJcrResultTreeView;\r
 import org.argeo.slc.client.ui.commands.RenameResultFolder;\r
 import org.argeo.slc.client.ui.commands.RenameResultNode;\r
 import org.argeo.slc.client.ui.editors.ProcessEditor;\r
 import org.argeo.slc.client.ui.editors.ProcessEditorInput;\r
 import org.argeo.slc.client.ui.model.ParentNodeFolder;\r
 import org.argeo.slc.client.ui.model.ResultFolder;\r
+import org.argeo.slc.client.ui.model.ResultItemsComparator;\r
+import org.argeo.slc.client.ui.model.ResultItemsComparer;\r
 import org.argeo.slc.client.ui.model.ResultParent;\r
 import org.argeo.slc.client.ui.model.ResultParentUtils;\r
 import org.argeo.slc.client.ui.model.SingleResultNode;\r
 import org.argeo.slc.client.ui.model.VirtualFolder;\r
 import org.argeo.slc.client.ui.providers.ResultTreeContentProvider;\r
 import org.argeo.slc.client.ui.providers.ResultTreeLabelProvider;\r
-import org.argeo.slc.client.ui.wizards.ConfirmOverwriteWizard;\r
 import org.argeo.slc.jcr.SlcJcrResultUtils;\r
 import org.argeo.slc.jcr.SlcNames;\r
 import org.argeo.slc.jcr.SlcTypes;\r
 import org.eclipse.jface.action.IMenuListener;\r
 import org.eclipse.jface.action.IMenuManager;\r
 import org.eclipse.jface.action.MenuManager;\r
+import org.eclipse.jface.dialogs.MessageDialog;\r
 import org.eclipse.jface.viewers.ColumnLabelProvider;\r
 import org.eclipse.jface.viewers.DecoratingLabelProvider;\r
 import org.eclipse.jface.viewers.DoubleClickEvent;\r
@@ -62,7 +79,6 @@ import org.eclipse.jface.viewers.TreePath;
 import org.eclipse.jface.viewers.TreeViewer;\r
 import org.eclipse.jface.viewers.Viewer;\r
 import org.eclipse.jface.viewers.ViewerDropAdapter;\r
-import org.eclipse.jface.wizard.WizardDialog;\r
 import org.eclipse.swt.SWT;\r
 import org.eclipse.swt.custom.SashForm;\r
 import org.eclipse.swt.dnd.DND;\r
@@ -87,7 +103,11 @@ import org.eclipse.ui.part.ViewPart;
 public class JcrResultTreeView extends ViewPart {\r
        public final static String ID = ClientUiPlugin.ID + ".jcrResultTreeView";\r
 \r
-       private final static Log log = LogFactory.getLog(JcrResultTreeView.class);\r
+       private final static DateFormat dateFormat = new SimpleDateFormat(\r
+                       SlcUiConstants.DEFAULT_DISPLAY_DATE_TIME_FORMAT);\r
+\r
+       // private final static Log log =\r
+       // LogFactory.getLog(JcrResultTreeView.class);\r
 \r
        /* DEPENDENCY INJECTION */\r
        private Session session;\r
@@ -96,22 +116,18 @@ public class JcrResultTreeView extends ViewPart {
        private TreeViewer resultTreeViewer;\r
        private TableViewer propertiesViewer;\r
 \r
-       private EventListener resultsObserver = null;\r
+       private EventListener myResultsObserver = null;\r
+       private EventListener allResultsObserver = null;\r
 \r
-       private final static String[] observedNodeTypes = {\r
+       // under My Results\r
+       private final static String[] observedNodeTypesUnderMyResult = {\r
                        SlcTypes.SLC_TEST_RESULT, SlcTypes.SLC_RESULT_FOLDER,\r
-                       NodeType.NT_UNSTRUCTURED };\r
+                       SlcTypes.SLC_MY_RESULT_ROOT_FOLDER };\r
 \r
-       // FIXME cache to ease D&D\r
-       private boolean isActionUnderMyResult = false;\r
-       private ResultParent lastSelectedTargetElement;\r
-       private ResultParent lastSelectedSourceElement;\r
-       private ResultParent lastSelectedSourceElementParent;\r
-       private boolean isResultFolder = false;\r
+       private final static String[] observedNodeTypesUnderAllResults = {\r
+                       SlcTypes.SLC_TEST_RESULT, NodeType.NT_UNSTRUCTURED };\r
 \r
-       // FIXME we cache the fact that we are moving a node to avoid exception\r
-       // triggered by the "Add Node" event while moving\r
-       // boolean isMoveInProgress = false;\r
+       private boolean isResultFolder = false;\r
 \r
        /**\r
         * To be overridden to adapt size of form and result frames.\r
@@ -141,22 +157,44 @@ public class JcrResultTreeView extends ViewPart {
 \r
                sashForm.setWeights(getWeights());\r
 \r
-               resultTreeViewer.setInput(initializeResultTree());\r
+               setOrderedInput(resultTreeViewer);\r
+\r
                // Initialize observer\r
                try {\r
                        ObservationManager observationManager = session.getWorkspace()\r
                                        .getObservationManager();\r
-                       resultsObserver = new ResultObserver(resultTreeViewer.getTree()\r
-                                       .getDisplay());\r
-                       observationManager.addEventListener(resultsObserver,\r
-                                       Event.NODE_MOVED | Event.NODE_ADDED | Event.NODE_REMOVED,\r
-                                       UserJcrUtils.getUserHome(session).getPath(), true, null,\r
-                                       observedNodeTypes, false);\r
+                       myResultsObserver = new MyResultsObserver(resultTreeViewer\r
+                                       .getTree().getDisplay());\r
+                       allResultsObserver = new AllResultsObserver(resultTreeViewer\r
+                                       .getTree().getDisplay());\r
+\r
+                       // observe tree changes under MyResults\r
+                       observationManager.addEventListener(myResultsObserver,\r
+                                       Event.NODE_ADDED | Event.NODE_REMOVED,\r
+                                       SlcJcrResultUtils.getMyResultsBasePath(session), true,\r
+                                       null, observedNodeTypesUnderMyResult, false);\r
+                       // observe tree changes under All results\r
+                       observationManager.addEventListener(allResultsObserver,\r
+                                       Event.NODE_ADDED | Event.NODE_REMOVED,\r
+                                       SlcJcrResultUtils.getSlcResultsBasePath(session), true,\r
+                                       null, observedNodeTypesUnderAllResults, false);\r
                } catch (RepositoryException e) {\r
                        throw new SlcException("Cannot register listeners", e);\r
                }\r
        }\r
 \r
+       /**\r
+        * Override default behaviour so that default defined order remains\r
+        * unchanged on first level of the tree\r
+        */\r
+       private void setOrderedInput(TreeViewer viewer) {\r
+               // Add specific ordering\r
+               viewer.setInput(null);\r
+               viewer.setComparator(null);\r
+               viewer.setInput(initializeResultTree());\r
+               viewer.setComparator(new ResultItemsComparator());\r
+       }\r
+\r
        // The main tree viewer\r
        protected TreeViewer createResultsTreeViewer(Composite parent) {\r
                int style = SWT.BORDER | SWT.MULTI;\r
@@ -175,6 +213,11 @@ public class JcrResultTreeView extends ViewPart {
                                decorator));\r
                viewer.addDoubleClickListener(new ViewDoubleClickListener());\r
 \r
+               // Override default behaviour to insure that 2 distincts results that\r
+               // have the same name will be correctly and distincly returned by\r
+               // corresponding TreeViewer.getSelection() method.\r
+               viewer.setComparer(new ResultItemsComparer());\r
+\r
                // viewer.setLabelProvider(rtLblProvider);\r
                getSite().setSelectionProvider(viewer);\r
 \r
@@ -193,6 +236,8 @@ public class JcrResultTreeView extends ViewPart {
                        }\r
                });\r
                viewer.getTree().setMenu(menu);\r
+               menuManager.setRemoveAllWhenShown(true);\r
+\r
                getSite().registerContextMenu(menuManager, viewer);\r
 \r
                // add change listener to display TestResult information in the property\r
@@ -211,11 +256,23 @@ public class JcrResultTreeView extends ViewPart {
                TableViewerColumn col = new TableViewerColumn(propertiesViewer,\r
                                SWT.NONE);\r
                col.getColumn().setText("Name");\r
-               col.getColumn().setWidth(200);\r
+               col.getColumn().setWidth(100);\r
                col.setLabelProvider(new ColumnLabelProvider() {\r
                        public String getText(Object element) {\r
                                try {\r
-                                       return ((Property) element).getName();\r
+                                       String name = ((Property) element).getName();\r
+                                       String value = null;\r
+                                       if (SlcNames.SLC_TEST_CASE.equals(name))\r
+                                               value = "Test case";\r
+                                       else if (SlcNames.SLC_COMPLETED.equals(name))\r
+                                               value = "Completed on";\r
+                                       else if (SlcNames.SLC_SUCCESS.equals(name))\r
+                                               value = "Status";\r
+                                       else if (SlcNames.SLC_MESSAGE.equals(name))\r
+                                               value = "Message";\r
+                                       else if (SlcNames.SLC_ERROR_MESSAGE.equals(name))\r
+                                               value = "Error";\r
+                                       return value;\r
                                } catch (RepositoryException e) {\r
                                        throw new ArgeoException(\r
                                                        "Unexpected exception in label provider", e);\r
@@ -224,39 +281,33 @@ public class JcrResultTreeView extends ViewPart {
                });\r
                col = new TableViewerColumn(propertiesViewer, SWT.NONE);\r
                col.getColumn().setText("Value");\r
-               col.getColumn().setWidth(400);\r
+               col.getColumn().setWidth(200);\r
                col.setLabelProvider(new ColumnLabelProvider() {\r
                        public String getText(Object element) {\r
                                try {\r
                                        Property property = (Property) element;\r
-                                       if (property.getType() == PropertyType.BINARY)\r
-                                               return "<binary>";\r
-                                       else if (property.isMultiple()) {\r
-                                               StringBuffer buf = new StringBuffer("[");\r
-                                               Value[] values = property.getValues();\r
-                                               for (int i = 0; i < values.length; i++) {\r
-                                                       if (i != 0)\r
-                                                               buf.append(", ");\r
-                                                       buf.append(values[i].getString());\r
+                                       String name = property.getName();\r
+                                       String value = null;\r
+\r
+                                       if (SlcNames.SLC_TEST_CASE.equals(name)\r
+                                                       || SlcNames.SLC_ERROR_MESSAGE.equals(name)\r
+                                                       || SlcNames.SLC_MESSAGE.equals(name))\r
+                                               value = property.getValue().getString();\r
+                                       else if (SlcNames.SLC_COMPLETED.equals(name)) {\r
+                                               Calendar date = property.getValue().getDate();\r
+                                               value = dateFormat.format(date.getTime());\r
+                                       } else if (SlcNames.SLC_SUCCESS.equals(name)) {\r
+                                               if (property.getValue().getBoolean())\r
+                                                       value = "PASSED";\r
+                                               else {\r
+                                                       if (property.getParent().hasProperty(\r
+                                                                       SlcNames.SLC_ERROR_MESSAGE))\r
+                                                               value = "ERROR";\r
+                                                       else\r
+                                                               value = "FAILED";\r
                                                }\r
-                                               buf.append(']');\r
-                                               return buf.toString();\r
-                                       } else\r
-                                               return property.getValue().getString();\r
-                               } catch (RepositoryException e) {\r
-                                       throw new ArgeoException(\r
-                                                       "Unexpected exception in label provider", e);\r
-                               }\r
-                       }\r
-               });\r
-               col = new TableViewerColumn(propertiesViewer, SWT.NONE);\r
-               col.getColumn().setText("Type");\r
-               col.getColumn().setWidth(200);\r
-               col.setLabelProvider(new ColumnLabelProvider() {\r
-                       public String getText(Object element) {\r
-                               try {\r
-                                       return PropertyType.nameFromValue(((Property) element)\r
-                                                       .getType());\r
+                                       }\r
+                                       return value;\r
                                } catch (RepositoryException e) {\r
                                        throw new ArgeoException(\r
                                                        "Unexpected exception in label provider", e);\r
@@ -306,11 +357,13 @@ public class JcrResultTreeView extends ViewPart {
         * \r
         */\r
        public void refresh(ResultParent resultParent) {\r
-               // if (log.isDebugEnabled())\r
-               // log.debug("Refreshing '" + resultParent + "'...");\r
-               // Thread.dumpStack();\r
                if (resultParent == null) {\r
-                       resultTreeViewer.setInput(initializeResultTree());\r
+                       if (!resultTreeViewer.getTree().isDisposed()) {\r
+                               TreePath[] tps = resultTreeViewer.getExpandedTreePaths();\r
+                               setOrderedInput(resultTreeViewer);\r
+                               resultTreeViewer.setExpandedTreePaths(tps);\r
+                       } else\r
+                               setOrderedInput(resultTreeViewer);\r
                } else {\r
                        if (resultParent instanceof ParentNodeFolder) {\r
                                ParentNodeFolder currFolder = (ParentNodeFolder) resultParent;\r
@@ -318,10 +371,8 @@ public class JcrResultTreeView extends ViewPart {
                                currFolder.forceFullRefresh();\r
                        }\r
                        // FIXME: specific refresh does not work\r
-                       // resultTreeViewer.refresh(currFolder, true);\r
-                       TreePath[] tps = resultTreeViewer.getExpandedTreePaths();\r
-                       resultTreeViewer.setInput(initializeResultTree());\r
-                       resultTreeViewer.setExpandedTreePaths(tps);\r
+                       // resultTreeViewer.refresh(resultParent, true);\r
+                       refresh(null);\r
                }\r
        }\r
 \r
@@ -339,20 +390,17 @@ public class JcrResultTreeView extends ViewPart {
                boolean isPassed = true;\r
                try {\r
                        if (node.isNodeType(SlcTypes.SLC_TEST_RESULT)) {\r
-                               isPassed = node.getNode(SlcNames.SLC_STATUS)\r
+                               isPassed = node.getNode(SlcNames.SLC_AGGREGATED_STATUS)\r
                                                .getProperty(SlcNames.SLC_SUCCESS).getBoolean();\r
                        } else if (node.isNodeType(SlcTypes.SLC_RESULT_FOLDER)) {\r
                                NodeIterator ni = node.getNodes();\r
-                               // quicker but wrong : refresh will stop as soon as a failed\r
-                               // test is found and the whole tree won't be refreshed\r
-                               // while (isPassed && ni.hasNext()){\r
                                while (ni.hasNext()) {\r
                                        Node currChild = ni.nextNode();\r
                                        isPassed = isPassed & jcrRefresh(currChild);\r
                                }\r
-                               if (isPassed != node.getNode(SlcNames.SLC_STATUS)\r
+                               if (isPassed != node.getNode(SlcNames.SLC_AGGREGATED_STATUS)\r
                                                .getProperty(SlcNames.SLC_SUCCESS).getBoolean()) {\r
-                                       node.getNode(SlcNames.SLC_STATUS).setProperty(\r
+                                       node.getNode(SlcNames.SLC_AGGREGATED_STATUS).setProperty(\r
                                                        SlcNames.SLC_SUCCESS, isPassed);\r
                                        node.getSession().save();\r
                                        return isPassed;\r
@@ -438,10 +486,13 @@ public class JcrResultTreeView extends ViewPart {
                boolean canAddSubfolder = false;\r
                boolean canRenamefolder = false;\r
                boolean isSingleResultNode = false;\r
-               // Building conditions\r
-               if (selection.size() == 1) {\r
-                       Object obj = selection.getFirstElement();\r
-                       try {\r
+               boolean isUnderMyResult = false;\r
+               boolean validMultipleDelete = false;\r
+               try {\r
+\r
+                       // Building conditions\r
+                       if (selection.size() == 1) {\r
+                               Object obj = selection.getFirstElement();\r
                                if (obj instanceof SingleResultNode)\r
                                        isSingleResultNode = true;\r
                                else if (obj instanceof ParentNodeFolder) {\r
@@ -449,18 +500,50 @@ public class JcrResultTreeView extends ViewPart {
                                        if (cNode.isNodeType(SlcTypes.SLC_RESULT_FOLDER)) {\r
                                                canAddSubfolder = true;\r
                                                canRenamefolder = true;\r
+                                               isUnderMyResult = true;\r
                                        } else if (cNode\r
                                                        .isNodeType(SlcTypes.SLC_MY_RESULT_ROOT_FOLDER)) {\r
                                                canAddSubfolder = true;\r
                                        }\r
                                }\r
-                       } catch (RepositoryException re) {\r
-                               throw new SlcException(\r
-                                               "unexpected error while building condition for context menu",\r
-                                               re);\r
+                       } else {\r
+                               @SuppressWarnings("rawtypes")\r
+                               Iterator it = selection.iterator();\r
+                               multicheck: while (it.hasNext()) {\r
+                                       validMultipleDelete = true;\r
+                                       Object obj = it.next();\r
+                                       if (obj instanceof SingleResultNode)\r
+                                               continue multicheck;\r
+                                       else if (obj instanceof ParentNodeFolder) {\r
+                                               Node cNode = ((ParentNodeFolder) obj).getNode();\r
+                                               if (cNode.isNodeType(SlcTypes.SLC_RESULT_FOLDER))\r
+                                                       continue multicheck;\r
+                                               else {\r
+                                                       validMultipleDelete = false;\r
+                                                       break multicheck;\r
+                                               }\r
+                                       } else {\r
+                                               validMultipleDelete = false;\r
+                                               break multicheck;\r
+                                       }\r
+                               }\r
                        }\r
+               } catch (RepositoryException re) {\r
+                       throw new SlcException(\r
+                                       "unexpected error while building condition for context menu",\r
+                                       re);\r
                }\r
+\r
                // Effective Refresh\r
+               CommandUtils.refreshCommand(menuManager, window,\r
+                               RefreshJcrResultTreeView.ID,\r
+                               RefreshJcrResultTreeView.DEFAULT_LABEL,\r
+                               RefreshJcrResultTreeView.DEFAULT_IMG_DESCRIPTOR, true);\r
+\r
+               CommandUtils.refreshCommand(menuManager, window, DeleteItems.ID,\r
+                               DeleteItems.DEFAULT_LABEL, DeleteItems.DEFAULT_IMG_DESCRIPTOR,\r
+                               isUnderMyResult || isSingleResultNode || validMultipleDelete);\r
+\r
                CommandUtils.refreshCommand(menuManager, window, AddResultFolder.ID,\r
                                AddResultFolder.DEFAULT_LABEL,\r
                                ClientUiPlugin.getDefault().getWorkbench().getSharedImages()\r
@@ -471,9 +554,21 @@ public class JcrResultTreeView extends ViewPart {
                                RenameResultFolder.DEFAULT_LABEL,\r
                                RenameResultFolder.DEFAULT_IMG_DESCRIPTOR, canRenamefolder);\r
 \r
+               // Command removed for the time being.\r
                CommandUtils.refreshCommand(menuManager, window, RenameResultNode.ID,\r
                                RenameResultNode.DEFAULT_LABEL,\r
-                               RenameResultNode.DEFAULT_IMG_DESCRIPTOR, isSingleResultNode);\r
+                               RenameResultNode.DEFAULT_IMG_DESCRIPTOR, false);\r
+\r
+               // Test to be removed\r
+               // If you use this pattern, do not forget to call\r
+               // menuManager.setRemoveAllWhenShown(true);\r
+               // when creating the menuManager\r
+\r
+               // menuManager.add(new Action("Test") {\r
+               // public void run() {\r
+               // log.debug("do something");\r
+               // }\r
+               // });\r
        }\r
 \r
        /* INNER CLASSES */\r
@@ -484,29 +579,41 @@ public class JcrResultTreeView extends ViewPart {
                        IStructuredSelection selection = (IStructuredSelection) resultTreeViewer\r
                                        .getSelection();\r
                        boolean doIt = false;\r
-                       // only one node at a time for the time being.\r
-                       if (selection.size() == 1) {\r
-                               try {\r
+\r
+                       // FIXME clean this code.\r
+                       try {\r
+                               if (selection.size() == 1) {\r
                                        Object obj = selection.getFirstElement();\r
-                                       if (obj instanceof SingleResultNode) {\r
-                                               Node tNode = ((SingleResultNode) obj).getNode();\r
-                                               if (tNode.getPrimaryNodeType().isNodeType(\r
-                                                               SlcTypes.SLC_TEST_RESULT)) {\r
-                                                       doIt = true;\r
-                                                       isResultFolder = false;\r
-                                               }\r
-                                       } else if (obj instanceof ResultFolder) {\r
+                                       if (obj instanceof ResultFolder) {\r
                                                Node tNode = ((ResultFolder) obj).getNode();\r
                                                if (tNode.getPrimaryNodeType().isNodeType(\r
                                                                SlcTypes.SLC_RESULT_FOLDER)) {\r
                                                        doIt = true;\r
                                                        isResultFolder = true;\r
                                                }\r
+                                       } else\r
+                                               isResultFolder = false;\r
+                               } else\r
+                                       isResultFolder = false;\r
+\r
+                               if (!isResultFolder) {\r
+                                       @SuppressWarnings("rawtypes")\r
+                                       Iterator it = selection.iterator();\r
+                                       while (it.hasNext()) {\r
+                                               Object obj = it.next();\r
+                                               if (obj instanceof SingleResultNode) {\r
+                                                       Node tNode = ((SingleResultNode) obj).getNode();\r
+                                                       if (tNode.getPrimaryNodeType().isNodeType(\r
+                                                                       SlcTypes.SLC_TEST_RESULT)) {\r
+                                                               doIt = true;\r
+                                                       }\r
+                                               }\r
                                        }\r
-                               } catch (RepositoryException re) {\r
-                                       throw new SlcException(\r
-                                                       "unexpected error while validating drag source", re);\r
                                }\r
+\r
+                       } catch (RepositoryException re) {\r
+                               throw new SlcException(\r
+                                               "unexpected error while validating drag source", re);\r
                        }\r
                        event.doit = doIt;\r
                }\r
@@ -514,15 +621,27 @@ public class JcrResultTreeView extends ViewPart {
                public void dragSetData(DragSourceEvent event) {\r
                        IStructuredSelection selection = (IStructuredSelection) resultTreeViewer\r
                                        .getSelection();\r
-                       Object obj = selection.getFirstElement();\r
+\r
                        try {\r
-                               Node first;\r
-                               if (obj instanceof SingleResultNode) {\r
-                                       first = ((SingleResultNode) obj).getNode();\r
-                                       event.data = first.getIdentifier();\r
-                               } else if (obj instanceof ResultFolder) {\r
-                                       first = ((ResultFolder) obj).getNode();\r
-                                       event.data = first.getIdentifier();\r
+                               // specific case of a result folder\r
+                               if (isResultFolder) {\r
+                                       Object obj = selection.getFirstElement();\r
+                                       event.data = ((ResultFolder) obj).getNode().getIdentifier();\r
+                               } else {\r
+                                       @SuppressWarnings("rawtypes")\r
+                                       Iterator it = selection.iterator();\r
+                                       StringBuilder nodes = new StringBuilder();\r
+                                       while (it.hasNext()) {\r
+                                               Object obj = it.next();\r
+                                               if (obj instanceof SingleResultNode) {\r
+                                                       Node tNode = ((SingleResultNode) obj).getNode();\r
+                                                       if (tNode.getPrimaryNodeType().isNodeType(\r
+                                                                       SlcTypes.SLC_TEST_RESULT)) {\r
+                                                               nodes.append(tNode.getIdentifier()).append(";");\r
+                                                       }\r
+                                               }\r
+                                       }\r
+                                       event.data = nodes.toString();\r
                                }\r
                        } catch (RepositoryException re) {\r
                                throw new SlcException("unexpected error while setting data",\r
@@ -565,29 +684,8 @@ public class JcrResultTreeView extends ViewPart {
                                }\r
 \r
                                if (tpNode != null) {\r
-                                       // Sanity check : we cannot move a folder to one of its sub\r
-                                       // folder or neither move an object in the same parent\r
-                                       // folder\r
-                                       boolean doit = true;\r
-                                       Node source = null;\r
-                                       if (isResultFolder) {\r
-                                               source = ((ParentNodeFolder) lastSelectedSourceElement)\r
-                                                               .getNode();\r
-                                               if (tpNode.getPath().startsWith(source.getPath()))\r
-                                                       doit = false;\r
-                                       } else if (lastSelectedSourceElement instanceof SingleResultNode) {\r
-                                               source = ((SingleResultNode) lastSelectedSourceElement)\r
-                                                               .getNode();\r
-                                               String sourceParentPath = JcrUtils.parentPath(source\r
-                                                               .getPath());\r
-                                               if (tpNode.getPath().equals(sourceParentPath))\r
-                                                       doit = false;\r
-                                       }\r
-                                       if (doit) {\r
-                                               targetParentNode = tpNode;\r
-                                               validDrop = true;\r
-                                               lastSelectedTargetElement = (ResultParent) target;\r
-                                       }\r
+                                       targetParentNode = tpNode;\r
+                                       validDrop = true;\r
                                }\r
                        } catch (RepositoryException re) {\r
                                throw new SlcException(\r
@@ -601,66 +699,116 @@ public class JcrResultTreeView extends ViewPart {
                        // clear selection to prevent unwanted scrolling of the UI\r
                        resultTreeViewer.setSelection(null);\r
                        try {\r
-                               Node source = session.getNodeByIdentifier((String) data);\r
-\r
-                               // Check is a node with same name already exists at destination\r
-                               String name;\r
-                               if (source.hasProperty(Property.JCR_TITLE))\r
-                                       name = source.getProperty(Property.JCR_TITLE).getString();\r
-                               else if (source.hasProperty(SlcNames.SLC_TEST_CASE))\r
-                                       name = source.getProperty(SlcNames.SLC_TEST_CASE)\r
-                                                       .getString();\r
-                               else\r
-                                       name = source.getName();\r
+                               if (isResultFolder) {\r
+                                       // Sanity check : we cannot move a folder to one of its sub\r
+                                       // folder or neither move an object in the same parent\r
+                                       // folder\r
+                                       Node source = session.getNodeByIdentifier((String) data);\r
+                                       if (targetParentNode.getPath().startsWith(source.getPath())\r
+                                                       || source.getParent().getPath()\r
+                                                                       .equals(targetParentNode.getPath()))\r
+                                               return false;\r
 \r
-                               if (targetParentNode.hasNode(name)) {\r
-                                       ConfirmOverwriteWizard wizard = new ConfirmOverwriteWizard(\r
-                                                       name, targetParentNode);\r
-                                       WizardDialog dialog = new WizardDialog(Display.getDefault()\r
-                                                       .getActiveShell(), wizard);\r
+                                       // Move\r
+                                       String sourcePath = source.getPath();\r
+                                       String destPath = targetParentNode.getPath() + "/"\r
+                                                       + source.getName();\r
+                                       session.move(sourcePath, destPath);\r
+                                       // Update passed status of the parent source Node\r
+                                       ResultParentUtils.updatePassedStatus(\r
+                                                       session.getNode(JcrUtils.parentPath(sourcePath)),\r
+                                                       true);\r
+                                       // Node target = session.getNode(destPath);\r
+                                       session.save();\r
+                                       return true;\r
+                               }\r
 \r
-                                       if (dialog.open() == WizardDialog.CANCEL)\r
-                                               return true;\r
+                               String[] datas = ((String) data).split(";");\r
+                               nodesToCopy: for (String id : datas) {\r
+\r
+                                       Node source = session.getNodeByIdentifier(id);\r
+                                       String name;\r
+                                       if (source.hasProperty(Property.JCR_TITLE))\r
+                                               name = source.getProperty(Property.JCR_TITLE)\r
+                                                               .getString();\r
+                                       else if (source.hasProperty(SlcNames.SLC_TEST_CASE))\r
+                                               name = source.getProperty(SlcNames.SLC_TEST_CASE)\r
+                                                               .getString();\r
+                                       else\r
+                                               name = source.getName();\r
+\r
+                                       // Check if another copy of the same test instance already\r
+                                       // exists at target\r
+                                       NodeIterator ni = targetParentNode.getNodes();\r
+                                       String slcUid = source.getProperty(SlcNames.SLC_UUID)\r
+                                                       .getString();\r
+                                       while (ni.hasNext()) {\r
+                                               Node curr = ni.nextNode();\r
+                                               if (curr.hasProperty(SlcNames.SLC_UUID)\r
+                                                               && slcUid.equals(curr.getProperty(\r
+                                                                               SlcNames.SLC_UUID).getString())) {\r
+                                                       MessageDialog\r
+                                                                       .openWarning(\r
+                                                                                       PlatformUI.getWorkbench()\r
+                                                                                                       .getDisplay()\r
+                                                                                                       .getActiveShell(),\r
+                                                                                       "Duplicated instance.",\r
+                                                                                       "An instance of the same test case ("\r
+                                                                                                       + name\r
+                                                                                                       + ") exists at destination.\n "\r
+                                                                                                       + "This item will not be neither copied nor moved.");\r
+                                                       continue nodesToCopy;\r
 \r
-                                       if (wizard.overwrite()) {\r
-                                               targetParentNode.getNode(name).remove();\r
-                                               // session.save();\r
-                                       } else\r
-                                               name = wizard.newName();\r
-                               }\r
+                                               }\r
+                                       }\r
 \r
-                               Node target;\r
-                               if (!isActionUnderMyResult) {// Copy\r
-                                       target = targetParentNode.addNode(name, source\r
-                                                       .getPrimaryNodeType().getName());\r
-                                       JcrUtils.copy(source, target);\r
-                               } else {// move\r
-                                       String sourcePath = source.getPath();\r
-                                       String destPath = targetParentNode.getPath() + "/" + name;\r
-                                       session.move(sourcePath, destPath);\r
-                                       // session.save();\r
-                                       target = session.getNode(destPath);\r
+                                       Node target;\r
+                                       boolean passedStatus = false;\r
+                                       if (source.hasNode(SlcNames.SLC_AGGREGATED_STATUS))\r
+                                               passedStatus = source\r
+                                                               .getNode(SlcNames.SLC_AGGREGATED_STATUS)\r
+                                                               .getProperty(SlcNames.SLC_SUCCESS).getBoolean();\r
+\r
+                                       boolean isActionUnderMyResult = source.getPath()\r
+                                                       .startsWith(\r
+                                                                       SlcJcrResultUtils\r
+                                                                                       .getMyResultsBasePath(session));\r
+\r
+                                       if (!isActionUnderMyResult) {// Copy\r
+                                               target = targetParentNode.addNode(source.getName(),\r
+                                                               source.getPrimaryNodeType().getName());\r
+                                               JcrUtils.copy(source, target);\r
+                                       } else {// move\r
+                                               String sourcePath = source.getPath();\r
+                                               String destPath = targetParentNode.getPath() + "/"\r
+                                                               + name;\r
+                                               session.move(sourcePath, destPath);\r
+                                               // Update passed status of the parent source Node\r
+                                               ResultParentUtils\r
+                                                               .updatePassedStatus(session.getNode(JcrUtils\r
+                                                                               .parentPath(sourcePath)), true);\r
+                                               target = session.getNode(destPath);\r
+\r
+                                       }\r
+                                       if (!target.isNodeType(NodeType.MIX_TITLE))\r
+                                               target.addMixin(NodeType.MIX_TITLE);\r
+                                       target.setProperty(Property.JCR_TITLE, name);\r
+                                       ResultParentUtils.updatePassedStatus(target.getParent(),\r
+                                                       passedStatus);\r
+                                       session.save();\r
                                }\r
-                               if (!target.isNodeType(NodeType.MIX_TITLE))\r
-                                       target.addMixin(NodeType.MIX_TITLE);\r
-                               target.setProperty(Property.JCR_TITLE, name);\r
-                               ResultParentUtils\r
-                                               .updatePassedStatus(target,\r
-                                                               target.getNode(SlcNames.SLC_STATUS)\r
-                                                                               .getProperty(SlcNames.SLC_SUCCESS)\r
-                                                                               .getBoolean());\r
-                               session.save();\r
                        } catch (RepositoryException re) {\r
                                throw new SlcException(\r
                                                "unexpected error while copying dropped node", re);\r
+\r
                        }\r
                        return true;\r
                }\r
        }\r
 \r
-       class ResultObserver extends AsyncUiEventListener {\r
+       class MyResultsObserver extends AsyncUiEventListener {\r
 \r
-               public ResultObserver(Display display) {\r
+               public MyResultsObserver(Display display) {\r
                        super(display);\r
                }\r
 \r
@@ -673,42 +821,42 @@ public class JcrResultTreeView extends ViewPart {
 \r
                protected void onEventInUiThread(List<Event> events)\r
                                throws RepositoryException {\r
-                       refresh(lastSelectedSourceElementParent);\r
+                       List<Node> nodesToRefresh = new ArrayList<Node>();\r
+\r
+                       for (Event event : events) {\r
+                               String parPath = JcrUtils.parentPath(event.getPath());\r
+                               if (session.nodeExists(parPath)) {\r
+                                       Node node = session.getNode(parPath);\r
+                                       if (!nodesToRefresh.contains(node)) {\r
+                                               nodesToRefresh.add(node);\r
+                                       }\r
+                               }\r
+                       }\r
 \r
-                       // boolean wasRemoved = false;\r
-                       // boolean wasAdded = false;\r
-                       //\r
-                       // for (Event event : events) {\r
-                       // // if (log.isDebugEnabled())\r
-                       // // log.debug("Received event " + event);\r
-                       // int eventType = event.getType();\r
-                       // if (eventType == Event.NODE_REMOVED) {\r
-                       // String path = event.getPath();\r
-                       // String parPath = JcrUtils.parentPath(path);\r
-                       // if (session.nodeExists(parPath)) {\r
-                       // Node currNode = session.getNode(parPath);\r
-                       // if (currNode.isNodeType(NodeType.NT_UNSTRUCTURED)) {\r
-                       // // jcrRefresh(currNode);\r
-                       // wasRemoved = true;\r
-                       // }\r
-                       // }\r
-                       // } else if (eventType == Event.NODE_ADDED) {\r
-                       // // refresh(lastSelectedTargetElement);\r
-                       // String path = event.getPath();\r
-                       // if (session.nodeExists(path)) {\r
-                       // Node currNode = session.getNode(path);\r
-                       // if (currNode.isNodeType(SlcTypes.SLC_TEST_RESULT)\r
-                       // || currNode\r
-                       // .isNodeType(SlcTypes.SLC_RESULT_FOLDER)) {\r
-                       // // refresh(lastSelectedTargetElement);\r
-                       // wasAdded = true;\r
-                       // // resultTreeViewer.expandToLevel(\r
-                       // // lastSelectedTargetElement, 1);\r
-                       // }\r
-                       // }\r
-                       // }\r
-                       // }\r
-                       // if (wasRemoved || wasAdded)\r
+                       // Update check nodes\r
+                       for (Node node : nodesToRefresh)\r
+                               jcrRefresh(node);\r
+                       refresh(null);\r
+               }\r
+       }\r
+\r
+       class AllResultsObserver extends AsyncUiEventListener {\r
+\r
+               public AllResultsObserver(Display display) {\r
+                       super(display);\r
+               }\r
+\r
+               @Override\r
+               protected Boolean willProcessInUiThread(List<Event> events)\r
+                               throws RepositoryException {\r
+                       // unfiltered for the time being\r
+                       return true;\r
+               }\r
+\r
+               protected void onEventInUiThread(List<Event> events)\r
+                               throws RepositoryException {\r
+                       refresh(null);\r
+                       // if (lastSelectedSourceElementParent != null)\r
                        // refresh(lastSelectedSourceElementParent);\r
                }\r
        }\r
@@ -724,14 +872,29 @@ public class JcrResultTreeView extends ViewPart {
                public Object[] getElements(Object inputElement) {\r
                        try {\r
                                if (inputElement instanceof Node) {\r
-                                       List<Property> props = new ArrayList<Property>();\r
-                                       PropertyIterator pit = ((Node) inputElement)\r
-                                                       .getProperties();\r
-                                       while (pit.hasNext())\r
-                                               props.add(pit.nextProperty());\r
-                                       return props.toArray();\r
+                                       Node node = (Node) inputElement;\r
+                                       if (node.isNodeType(SlcTypes.SLC_TEST_RESULT)) {\r
+                                               List<Property> props = new ArrayList<Property>();\r
+                                               if (node.hasProperty(SlcNames.SLC_TEST_CASE))\r
+                                                       props.add(node.getProperty(SlcNames.SLC_TEST_CASE));\r
+                                               if (node.hasProperty(SlcNames.SLC_COMPLETED))\r
+                                                       props.add(node.getProperty(SlcNames.SLC_COMPLETED));\r
+                                               if (node.hasNode(SlcNames.SLC_AGGREGATED_STATUS)) {\r
+                                                       Node status = node\r
+                                                                       .getNode(SlcNames.SLC_AGGREGATED_STATUS);\r
+                                                       props.add(status.getProperty(SlcNames.SLC_SUCCESS));\r
+                                                       if (status.hasProperty(SlcNames.SLC_MESSAGE))\r
+                                                               props.add(status\r
+                                                                               .getProperty(SlcNames.SLC_MESSAGE));\r
+                                                       if (status.hasProperty(SlcNames.SLC_ERROR_MESSAGE))\r
+                                                               props.add(status\r
+                                                                               .getProperty(SlcNames.SLC_ERROR_MESSAGE));\r
+                                               }\r
+                                               return props.toArray();\r
+                                       }\r
                                }\r
                                return new Object[] {};\r
+\r
                        } catch (RepositoryException e) {\r
                                throw new ArgeoException("Cannot get element for "\r
                                                + inputElement, e);\r
@@ -752,28 +915,28 @@ public class JcrResultTreeView extends ViewPart {
                                else\r
                                        propertiesViewer.setInput(null);\r
                                // update cache for Drag & drop\r
-                               lastSelectedTargetElement = firstItem;\r
-                               lastSelectedSourceElement = firstItem;\r
-                               lastSelectedSourceElementParent = (ResultParent) firstItem\r
-                                               .getParent();\r
-                               String pPath = "";\r
-                               try {\r
-\r
-                                       if (firstItem instanceof ParentNodeFolder)\r
-                                               pPath = ((ParentNodeFolder) firstItem).getNode()\r
-                                                               .getPath();\r
-                                       else if (firstItem instanceof SingleResultNode)\r
-                                               pPath = ((SingleResultNode) firstItem).getNode()\r
-                                                               .getPath();\r
-                               } catch (RepositoryException e) {\r
-                                       throw new SlcException(\r
-                                                       "Unexpected error while checking parent UI tree", e);\r
-                               }\r
-                               if ((pPath.startsWith(SlcJcrResultUtils\r
-                                               .getMyResultsBasePath(session))))\r
-                                       isActionUnderMyResult = true;\r
-                               else\r
-                                       isActionUnderMyResult = false;\r
+                               // lastSelectedTargetElement = firstItem;\r
+                               // lastSelectedSourceElement = firstItem;\r
+                               // lastSelectedSourceElementParent = (ResultParent) firstItem\r
+                               // .getParent();\r
+                               // String pPath = "";\r
+                               // try {\r
+                               //\r
+                               // if (firstItem instanceof ParentNodeFolder)\r
+                               // pPath = ((ParentNodeFolder) firstItem).getNode()\r
+                               // .getPath();\r
+                               // else if (firstItem instanceof SingleResultNode)\r
+                               // pPath = ((SingleResultNode) firstItem).getNode()\r
+                               // .getPath();\r
+                               // } catch (RepositoryException e) {\r
+                               // throw new SlcException(\r
+                               // "Unexpected error while checking parent UI tree", e);\r
+                               // }\r
+                               // if ((pPath.startsWith(SlcJcrResultUtils\r
+                               // .getMyResultsBasePath(session))))\r
+                               // isActionUnderMyResult = true;\r
+                               // else\r
+                               // isActionUnderMyResult = false;\r
                        }\r
                }\r
        }\r
@@ -786,8 +949,23 @@ public class JcrResultTreeView extends ViewPart {
        }\r
 \r
        /* DEPENDENCY INJECTION */\r
+       @Deprecated\r
        public void setSession(Session session) {\r
                this.session = session;\r
        }\r
 \r
-}\r
+       public void dispose() {\r
+               // JcrUtils.unregisterQuietly(session.getWorkspace(), resultsObserver);\r
+               JcrUtils.logoutQuietly(session);\r
+               super.dispose();\r
+       }\r
+\r
+       public void setRepository(Repository repository) {\r
+               try {\r
+                       session = repository.login();\r
+               } catch (RepositoryException re) {\r
+                       throw new SlcException("Unable to log in Repository " + repository,\r
+                                       re);\r
+               }\r
+       }\r
+}
\ No newline at end of file