]> gerrit.simantics Code Review - simantics/platform.git/blobdiff - bundles/org.simantics.diagram/src/org/simantics/diagram/ui/WorkbenchSelectionProvider.java
Migrated source code from Simantics SVN
[simantics/platform.git] / bundles / org.simantics.diagram / src / org / simantics / diagram / ui / WorkbenchSelectionProvider.java
diff --git a/bundles/org.simantics.diagram/src/org/simantics/diagram/ui/WorkbenchSelectionProvider.java b/bundles/org.simantics.diagram/src/org/simantics/diagram/ui/WorkbenchSelectionProvider.java
new file mode 100644 (file)
index 0000000..096fd7a
--- /dev/null
@@ -0,0 +1,208 @@
+/*******************************************************************************\r
+ * Copyright (c) 2007, 2010 Association for Decentralized Information Management\r
+ * in Industry THTH ry.\r
+ * All rights reserved. This program and the accompanying materials\r
+ * are made available under the terms of the Eclipse Public License v1.0\r
+ * which accompanies this distribution, and is available at\r
+ * http://www.eclipse.org/legal/epl-v10.html\r
+ *\r
+ * Contributors:\r
+ *     VTT Technical Research Centre of Finland - initial API and implementation\r
+ *******************************************************************************/\r
+package org.simantics.diagram.ui;\r
+\r
+import java.util.ArrayList;\r
+import java.util.Collections;\r
+import java.util.concurrent.CopyOnWriteArrayList;\r
+import java.util.concurrent.TimeUnit;\r
+\r
+import org.eclipse.jface.viewers.IPostSelectionProvider;\r
+import org.eclipse.jface.viewers.ISelection;\r
+import org.eclipse.jface.viewers.ISelectionChangedListener;\r
+import org.eclipse.jface.viewers.SelectionChangedEvent;\r
+import org.eclipse.jface.viewers.StructuredSelection;\r
+import org.eclipse.ui.IWorkbenchPartSite;\r
+import org.simantics.diagram.elements.AdaptableImmutableProxyElement;\r
+import org.simantics.g2d.canvas.ICanvasContext;\r
+import org.simantics.g2d.canvas.impl.AbstractCanvasParticipant;\r
+import org.simantics.g2d.canvas.impl.DependencyReflection.Dependency;\r
+import org.simantics.g2d.canvas.impl.HintReflection.HintListener;\r
+import org.simantics.g2d.diagram.participant.Selection;\r
+import org.simantics.g2d.element.ElementHints;\r
+import org.simantics.g2d.element.IElement;\r
+import org.simantics.utils.datastructures.hints.IHintObservable;\r
+import org.simantics.utils.datastructures.hints.IHintContext.Key;\r
+import org.simantics.utils.threads.IThreadWorkQueue;\r
+import org.simantics.utils.threads.ThreadUtils;\r
+\r
+/**\r
+ * A canvas participant that listens to the #0 mouse selection and provides it\r
+ * forward through the {@link IPostSelectionProvider} interface.\r
+ * \r
+ * @author Tuukka Lehtonen\r
+ */\r
+public class WorkbenchSelectionProvider extends AbstractCanvasParticipant implements IPostSelectionProvider {\r
+\r
+    private static final long                                 POST_SELECTION_DELAY = 300;\r
+\r
+    @Dependency protected Selection selection;\r
+\r
+    protected IThreadWorkQueue                                swt;\r
+    protected IWorkbenchPartSite                              site;\r
+    protected ISelection                                      currentSelection = StructuredSelection.EMPTY;\r
+    protected CopyOnWriteArrayList<ISelectionChangedListener> listeners        = new CopyOnWriteArrayList<ISelectionChangedListener>();\r
+    protected CopyOnWriteArrayList<ISelectionChangedListener> postListeners    = new CopyOnWriteArrayList<ISelectionChangedListener>();\r
+\r
+    public WorkbenchSelectionProvider(IThreadWorkQueue swt) {\r
+        this(swt, null);\r
+    }\r
+\r
+    public WorkbenchSelectionProvider(IThreadWorkQueue swt, IWorkbenchPartSite site) {\r
+        this.swt = swt;\r
+        this.site = site;\r
+        if (site != null)\r
+            site.setSelectionProvider(this);\r
+    }\r
+\r
+    @Override\r
+    public void addedToContext(ICanvasContext ctx) {\r
+        super.addedToContext(ctx);\r
+        if (site != null && site.getSelectionProvider() != this) {\r
+            swt.asyncExec(new Runnable() {\r
+                @Override\r
+                public void run() {\r
+                    if (site.getSelectionProvider() != WorkbenchSelectionProvider.this)\r
+                        site.setSelectionProvider(WorkbenchSelectionProvider.this);\r
+                }\r
+            });\r
+        }\r
+    }\r
+\r
+    @Override\r
+    public void removedFromContext(ICanvasContext ctx) {\r
+        if (site != null && site.getSelectionProvider() == this) {\r
+            swt.asyncExec(new Runnable() {\r
+                @Override\r
+                public void run() {\r
+                    if (site.getSelectionProvider() == WorkbenchSelectionProvider.this)\r
+                        site.setSelectionProvider(null);\r
+                }\r
+            });\r
+        }\r
+        super.removedFromContext(ctx);\r
+    }\r
+\r
+    @HintListener(Class = Selection.class, Field = "SELECTION0")\r
+    public void hintChanged(IHintObservable sender, Key key, Object oldValue, Object newValue) {\r
+        Iterable<?> selection = (Iterable<?>) newValue;\r
+        final ISelection s = constructAdaptableSelection(selection);\r
+        swt.asyncExec(new Runnable() {\r
+            @Override\r
+            public void run() {\r
+                currentSelection = s;\r
+                fireSelectionChanged(s);\r
+                schedulePostSelectionChanged(s);\r
+            }\r
+        });\r
+    }\r
+\r
+    @HintListener(Class = Selection.class, Field = "SELECTION0")\r
+    public void hintRemoved(IHintObservable sender, Key key, Object oldValue) {\r
+        //System.out.println("selection removed: " + oldValue);\r
+        final ISelection s = constructAdaptableSelection(Collections.emptyList());\r
+        swt.asyncExec(new Runnable() {\r
+            @Override\r
+            public void run() {\r
+                currentSelection = s;\r
+                fireSelectionChanged(s);\r
+                schedulePostSelectionChanged(s);\r
+            }\r
+        });\r
+    }\r
+\r
+    // Post selection changed scheduling helper.\r
+    private int modCount = 0;\r
+\r
+    protected void schedulePostSelectionChanged(final ISelection s) {\r
+        final int count = ++modCount;\r
+        ThreadUtils.getNonBlockingWorkExecutor().schedule(new Runnable() {\r
+            @Override\r
+            public void run() {\r
+                int newCount = modCount;\r
+                if (count != newCount)\r
+                    return;\r
+                if (isRemoved())\r
+                    return;\r
+                swt.asyncExec(new Runnable() {\r
+                    @Override\r
+                    public void run() {\r
+                        if (isRemoved())\r
+                            return;\r
+                        firePostSelectionChanged(s);\r
+                    }\r
+                });\r
+            }\r
+        }, POST_SELECTION_DELAY, TimeUnit.MILLISECONDS);\r
+    }\r
+\r
+    protected ISelection constructAdaptableSelection(Iterable<?> selection) {\r
+        ArrayList<Object> objects = new ArrayList<Object>();\r
+        for (Object o : selection) {\r
+            if (o instanceof IElement) {\r
+                IElement e = (IElement) o;\r
+                Object object = e.getHint(ElementHints.KEY_OBJECT);\r
+                if (object != null) {\r
+                    objects.add(new AdaptableImmutableProxyElement(e));\r
+                } else {\r
+                    System.out.println("  discarding element from selection, null object for " + e);\r
+                }\r
+            } else {\r
+                System.out.println("  unrecognized selection: " + o.getClass() + ": " + o);\r
+            }\r
+        }\r
+        return new StructuredSelection(objects);\r
+    }\r
+\r
+    void fireSelectionChanged(ISelection selection) {\r
+        SelectionChangedEvent e = new SelectionChangedEvent(this, selection);\r
+        for (ISelectionChangedListener l : listeners)\r
+            l.selectionChanged(e);\r
+    }\r
+\r
+    void firePostSelectionChanged(ISelection selection) {\r
+        SelectionChangedEvent e = new SelectionChangedEvent(this, selection);\r
+        for (ISelectionChangedListener l : postListeners)\r
+            l.selectionChanged(e);\r
+    }\r
+\r
+    @Override\r
+    public void addPostSelectionChangedListener(ISelectionChangedListener listener) {\r
+        postListeners.add(listener);\r
+    }\r
+\r
+    @Override\r
+    public void removePostSelectionChangedListener(ISelectionChangedListener listener) {\r
+        postListeners.remove(listener);\r
+    }\r
+\r
+    @Override\r
+    public void addSelectionChangedListener(ISelectionChangedListener listener) {\r
+        listeners.add(listener);\r
+    }\r
+\r
+    @Override\r
+    public void removeSelectionChangedListener(ISelectionChangedListener listener) {\r
+        listeners.remove(listener);\r
+    }\r
+\r
+    @Override\r
+    public ISelection getSelection() {\r
+        return currentSelection;\r
+    }\r
+\r
+    @Override\r
+    public void setSelection(ISelection selection) {\r
+        System.out.println("WorkbenchSelectionProvider: TODO: set selection: " + selection);\r
+    }\r
+\r
+}\r