-/*******************************************************************************\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.getCause());\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
+/*******************************************************************************
+ * Copyright (c) 2007, 2010 Association for Decentralized Information Management
+ * in Industry THTH ry.
+ * All rights reserved. This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution, and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ * VTT Technical Research Centre of Finland - initial API and implementation
+ *******************************************************************************/
+/*
+ *
+ * @author Toni Kalajainen
+ */
+package org.simantics.scenegraph.g2d.events;
+
+import java.lang.annotation.ElementType;
+import java.lang.annotation.Retention;
+import java.lang.annotation.RetentionPolicy;
+import java.lang.annotation.Target;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * @author Toni Kalajainen
+ */
+public class EventHandlerReflection {
+
+ @Retention(RetentionPolicy.RUNTIME)
+ @Target(ElementType.METHOD)
+ public static @interface EventHandler {
+ int priority();
+ }
+
+ /**
+ * Scans an object with reflection for all event handler methods and returns
+ * an array of IEventHandler and priorities.
+ * <p>
+ * The corresponding class of obj must contain methods that meet the
+ * following criteria: 1) must return boolean 2) 1 argument, which is Event
+ * 3) has annotation EventHandler 4) may not throw any Exception 5) method
+ * must be accessible
+ * <p>
+ * Example:
+ *
+ * @EventHandler(priority = Integer.MAX_VALUE) public boolean
+ * handleEvent(Event e) { return false; }
+ *
+ * @param obj object to scan
+ * @return an array of painters and their priorities
+ */
+ @SuppressWarnings("unchecked")
+ public static EventHandlerDefinition[] getEventHandlers(final Object obj) {
+ List<EventHandlerDefinition> result = new ArrayList<EventHandlerDefinition>();
+ Class<?> clazz = obj.getClass();
+
+ for (final Method m : clazz.getMethods()) {
+ EventHandler anno = (EventHandler) m.getAnnotation(EventHandler.class);
+ if (anno == null)
+ continue;
+
+ Class<?> returnType = m.getReturnType();
+ if (!returnType.equals(boolean.class))
+ throw new RuntimeException(clazz.getName() + "." + m.getName() + " return type is invalid");
+
+ @SuppressWarnings("rawtypes")
+ Class[] argTypes = m.getParameterTypes();
+ if (argTypes.length != 1 || !Event.class.isAssignableFrom(argTypes[0]))
+ throw new RuntimeException(clazz.getName() + "." + m.getName() + " invalid arguments");
+
+ @SuppressWarnings("rawtypes")
+ Class argClass = argTypes[0];
+
+ @SuppressWarnings("rawtypes")
+ Class[] exceptionTypes = m.getExceptionTypes();
+ if (exceptionTypes.length != 0)
+ throw new RuntimeException(clazz.getName() + "." + m.getName() + " invalid exceptions");
+
+ int priority = anno.priority();
+
+ try {
+ m.setAccessible(true);
+ } catch (Throwable t) {
+ t.printStackTrace();
+ continue;
+ }
+
+ final int eventMask = EventTypes.toTypeMask(argClass);
+
+ IEventHandler eventHandler = new IEventHandler() {
+ @Override
+ public int getEventMask() {
+ return eventMask;
+ }
+ @Override
+ public boolean handleEvent(Event e1) {
+ try {
+ return (Boolean) m.invoke(obj, e1);
+ } catch (IllegalArgumentException e) {
+ throw new Error(e);
+ } catch (IllegalAccessException e) {
+ throw new RuntimeException(e);
+ } catch (InvocationTargetException e) {
+ throw new RuntimeException(e.getCause());
+ }
+ }
+ };
+ EventHandlerDefinition handler = new EventHandlerDefinition(obj, eventHandler, priority, argClass);
+ result.add(handler);
+ }
+
+ return result.toArray(new EventHandlerDefinition[0]);
+ }
+
+ public final static class EventHandlerDefinition {
+ public final Object obj;
+ public final IEventHandler origEventHandler;
+ public final IEventHandler eventHandler;
+ public final int priority;
+ public final Class<Event> eventSuperClass;
+
+ public EventHandlerDefinition(Object obj, IEventHandler eventHandler, int priority, Class<Event> eventSuperClass) {
+ this.obj = obj;
+ this.priority = priority;
+ this.eventSuperClass = eventSuperClass;
+ this.origEventHandler = eventHandler;
+ final int eventMask = EventTypes.toTypeMask(eventSuperClass);
+ if (eventSuperClass.equals(Event.class)) {
+ this.eventHandler = eventHandler;
+ } else {
+ this.eventHandler = new IEventHandler() {
+ @Override
+ public int getEventMask() {
+ return eventMask;
+ }
+
+ @Override
+ public boolean handleEvent(Event e) {
+ // Event masking already taken care of, no need to check eventMask
+ if (EventHandlerDefinition.this.eventSuperClass.isAssignableFrom(e.getClass()))
+ return EventHandlerDefinition.this.origEventHandler.handleEvent(e);
+ return false;
+ }
+
+ @Override
+ public String toString() {
+ return EventHandlerDefinition.this.toString();
+ }
+ };
+ }
+ }
+
+ @Override
+ public String toString() {
+ return String.format("[%11d] %s (%s)", priority, obj.getClass().getSimpleName(),
+ eventSuperClass.getSimpleName());
+ }
+ }
+
+}