]> gerrit.simantics Code Review - simantics/platform.git/blobdiff - bundles/org.simantics.scenegraph/src/org/simantics/scenegraph/g2d/events/EventHandlerReflection.java
Migrated source code from Simantics SVN
[simantics/platform.git] / bundles / org.simantics.scenegraph / src / org / simantics / scenegraph / g2d / events / EventHandlerReflection.java
diff --git a/bundles/org.simantics.scenegraph/src/org/simantics/scenegraph/g2d/events/EventHandlerReflection.java b/bundles/org.simantics.scenegraph/src/org/simantics/scenegraph/g2d/events/EventHandlerReflection.java
new file mode 100644 (file)
index 0000000..c3c51ea
--- /dev/null
@@ -0,0 +1,163 @@
+/*******************************************************************************\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
+/*\r
+ *\r
+ * @author Toni Kalajainen\r
+ */\r
+package org.simantics.scenegraph.g2d.events;\r
+\r
+import java.lang.annotation.ElementType;\r
+import java.lang.annotation.Retention;\r
+import java.lang.annotation.RetentionPolicy;\r
+import java.lang.annotation.Target;\r
+import java.lang.reflect.InvocationTargetException;\r
+import java.lang.reflect.Method;\r
+import java.util.ArrayList;\r
+import java.util.List;\r
+\r
+/**\r
+ * @author Toni Kalajainen\r
+ */\r
+public class EventHandlerReflection {\r
+\r
+    @Retention(RetentionPolicy.RUNTIME)\r
+    @Target(ElementType.METHOD)\r
+    public static @interface EventHandler {\r
+        int priority();\r
+    }\r
+\r
+    /**\r
+     * Scans an object with reflection for all event handler methods and returns\r
+     * an array of IEventHandler and priorities.\r
+     * <p>\r
+     * The corresponding class of obj must contain methods that meet the\r
+     * following criteria: 1) must return boolean 2) 1 argument, which is Event\r
+     * 3) has annotation EventHandler 4) may not throw any Exception 5) method\r
+     * must be accessible\r
+     * <p>\r
+     * Example:\r
+     * \r
+     * @EventHandler(priority = Integer.MAX_VALUE) public boolean\r
+     *                        handleEvent(Event e) { return false; }\r
+     * \r
+     * @param obj object to scan\r
+     * @return an array of painters and their priorities\r
+     */\r
+    @SuppressWarnings("unchecked")\r
+    public static EventHandlerDefinition[] getEventHandlers(final Object obj) {\r
+        List<EventHandlerDefinition> result = new ArrayList<EventHandlerDefinition>();\r
+        Class<?> clazz = obj.getClass();\r
+\r
+        for (final Method m : clazz.getMethods()) {\r
+            EventHandler anno = (EventHandler) m.getAnnotation(EventHandler.class);\r
+            if (anno == null)\r
+                continue;\r
+\r
+            Class<?> returnType = m.getReturnType();\r
+            if (!returnType.equals(boolean.class))\r
+                throw new RuntimeException(clazz.getName() + "." + m.getName() + " return type is invalid");\r
+\r
+            @SuppressWarnings("rawtypes")\r
+            Class[] argTypes = m.getParameterTypes();\r
+            if (argTypes.length != 1 || !Event.class.isAssignableFrom(argTypes[0]))\r
+                throw new RuntimeException(clazz.getName() + "." + m.getName() + " invalid arguments");\r
+\r
+            @SuppressWarnings("rawtypes")\r
+            Class argClass = argTypes[0];\r
+\r
+            @SuppressWarnings("rawtypes")\r
+            Class[] exceptionTypes = m.getExceptionTypes();\r
+            if (exceptionTypes.length != 0)\r
+                throw new RuntimeException(clazz.getName() + "." + m.getName() + " invalid exceptions");\r
+\r
+            int priority = anno.priority();\r
+\r
+            try {\r
+                m.setAccessible(true);\r
+            } catch (Throwable t) {\r
+                t.printStackTrace();\r
+                continue;\r
+            }\r
+\r
+            final int eventMask = EventTypes.toTypeMask(argClass);\r
+\r
+            IEventHandler eventHandler = new IEventHandler() {\r
+                @Override\r
+                public int getEventMask() {\r
+                    return eventMask;\r
+                }\r
+                @Override\r
+                public boolean handleEvent(Event e1) {\r
+                    try {\r
+                        return (Boolean) m.invoke(obj, e1);\r
+                    } catch (IllegalArgumentException e) {\r
+                        throw new Error(e);\r
+                    } catch (IllegalAccessException e) {\r
+                        throw new RuntimeException(e);\r
+                    } catch (InvocationTargetException e) {\r
+                        throw new RuntimeException(e);\r
+                    }\r
+                }\r
+            };\r
+            EventHandlerDefinition handler = new EventHandlerDefinition(obj, eventHandler, priority, argClass);\r
+            result.add(handler);\r
+        }\r
+\r
+        return result.toArray(new EventHandlerDefinition[0]);\r
+    }\r
+\r
+    public final static class EventHandlerDefinition {\r
+        public final Object        obj;\r
+        public final IEventHandler origEventHandler;\r
+        public final IEventHandler eventHandler;\r
+        public final int           priority;\r
+        public final Class<Event>  eventSuperClass;\r
+\r
+        public EventHandlerDefinition(Object obj, IEventHandler eventHandler, int priority, Class<Event> eventSuperClass) {\r
+            this.obj = obj;\r
+            this.priority = priority;\r
+            this.eventSuperClass = eventSuperClass;\r
+            this.origEventHandler = eventHandler;\r
+            final int eventMask = EventTypes.toTypeMask(eventSuperClass);\r
+            if (eventSuperClass.equals(Event.class)) {\r
+                this.eventHandler = eventHandler;\r
+            } else {\r
+                this.eventHandler = new IEventHandler() {\r
+                    @Override\r
+                    public int getEventMask() {\r
+                        return eventMask;\r
+                    }\r
+\r
+                    @Override\r
+                    public boolean handleEvent(Event e) {\r
+                        // Event masking already taken care of, no need to check eventMask\r
+                        if (EventHandlerDefinition.this.eventSuperClass.isAssignableFrom(e.getClass()))\r
+                            return EventHandlerDefinition.this.origEventHandler.handleEvent(e);\r
+                        return false;\r
+                    }\r
+\r
+                    @Override\r
+                    public String toString() {\r
+                        return EventHandlerDefinition.this.toString();\r
+                    }\r
+                };\r
+            }\r
+        }\r
+\r
+        @Override\r
+        public String toString() {\r
+            return String.format("[%11d] %s (%s)", priority, obj.getClass().getSimpleName(),\r
+                    eventSuperClass.getSimpleName());\r
+        }\r
+    }\r
+\r
+}\r