All Downloads are FREE. Search and download functionalities are using the official Maven repository.

jaxx.runtime.JAXXUtil Maven / Gradle / Ivy

There is a newer version: 3.0-alpha-6
Show newest version
/*
 * *##% 
 * JAXX Runtime
 * Copyright (C) 2008 - 2009 CodeLutin
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation, either version 3 of the
 * License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Lesser Public License for more details.
 *
 * You should have received a copy of the GNU General Lesser Public
 * License along with this program.  If not, see
 * .
 * ##%*
 */
package jaxx.runtime;

import jaxx.runtime.context.JAXXContextEntryDef;
import jaxx.runtime.context.JAXXInitialContext;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeListenerProxy;
import java.io.IOException;
import java.lang.ref.WeakReference;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.*;

public class JAXXUtil {

    /** Logger */
    static private final Log log = LogFactory.getLog(JAXXUtil.class);

    public static final String PARENT = "parent";

    // Maps root objects to lists of event listeners

    private static Map>>
            eventListeners = new WeakHashMap>>();

    private static Map>>
            dataBindingUpdateListeners = new WeakHashMap>>();

    private static final PropertyChangeListener[] EMPTY_ARRAY_PROPERTY_CHANGE_LISTENERS = new PropertyChangeListener[0];

    private static class EventListenerDescriptor {

        Class listenerClass;

        String listenerMethodName;

        String methodName;

        Object eventListener;
    }

    /**
     * Decodes the serialized representation of a JAXXObjectDescriptor.  The
     * string must be a byte-to-character mapping of the binary serialization
     * data for a JAXXObjectDescriptor.  See the comments in
     * JAXXCompiler.createJAXXObjectDescriptorField for the rationale behind
     * this (admittedly ugly) approach.
     *
     * @param descriptor descriptor to decode
     * @return the dedoced descriptor
     */
    public static JAXXObjectDescriptor decodeJAXXObjectDescriptor(
            String descriptor) {
        try {
            return (JAXXObjectDescriptor) Base64Coder.deserialize(descriptor,
                                                                  false);
            /*byte[] data = new byte[descriptor.length()];
            // copy low-order bytes into the array.  The high-order bytes should all be zero.
            System.arraycopy(descriptor.getBytes(), 0, data, 0, data.length);
            //descriptor.getBytes(0, descriptor.length(), data, 0);
            ObjectInputStream in = new ObjectInputStream(new ByteArrayInputStream(data));
            return (JAXXObjectDescriptor) in.readObject();*/
        } catch (IOException e) {
            throw new RuntimeException("Internal error: can't-happen error", e);
        } catch (ClassNotFoundException e) {
            throw new RuntimeException("Internal error: can't-happen error", e);
        }
    }

    public static JAXXObjectDescriptor decodeCompressedJAXXObjectDescriptor(
            String descriptor) {
        try {
            return (JAXXObjectDescriptor) Base64Coder.deserialize(descriptor,
                                                                  true);

            /*byte[] data = new byte[descriptor.length()];
            // copy low-order bytes into the array.  The high-order bytes should all be zero.
            System.arraycopy(descriptor.getBytes(), 0, data, 0, data.length);
            //descriptor.getBytes(0, descriptor.length(), data, 0); 
            ObjectInputStream in = new ObjectInputStream(new GZIPInputStream(new ByteArrayInputStream(data)));
            return (JAXXObjectDescriptor) in.readObject();*/
        } catch (IOException e) {
            throw new RuntimeException("Internal error: can't-happen error", e);
        } catch (ClassNotFoundException e) {
            throw new RuntimeException("Internal error: can't-happen error", e);
        }
    }

    public static  JAXXContextEntryDef newContextEntryDef(
            Class klass) {
        return newContextEntryDef(null, klass);
    }

    public static  JAXXContextEntryDef newContextEntryDef(
            String name, Class klass) {
        return new JAXXContextEntryDef(name, klass);
    }

    public static  JAXXContextEntryDef> newListContextEntryDef() {
        return newListContextEntryDef(null);
    }

    public static  JAXXContextEntryDef> newListContextEntryDef(
            String name) {
        Class> castList = castList();
        JAXXContextEntryDef> contextEntryDef =
                new JAXXContextEntryDef>(name, castList);
        return contextEntryDef;
    }

    @SuppressWarnings({"unchecked"})
    protected static  Class> castList() {
        return (Class>) Collections.emptyList().getClass();
    }

    /**
     * Method to initialize the context of a ui.
     *
     * @param ui            the ui
     * @param parentContext the context to set in ui
     */
    public static void initContext(JAXXObject ui, JAXXContext parentContext) {

        if (parentContext instanceof JAXXInitialContext) {
            ((JAXXInitialContext) parentContext).to(ui);
        } else {
            ui.setContextValue(parentContext);
        }
        // if parentContext is a JAXXObject then
        // add an special parent entry to can go up
        if (parentContext instanceof JAXXObject) {
            ui.setContextValue(parentContext, PARENT);
        }
    }

    public static  E getEventListener(
            Class listenerClass,
            final String listenerMethodName,
            final Object methodContainer,
            String methodName) {

        WeakReference> ref =
                eventListeners.get(methodContainer);
        List descriptors = ref != null ?
                                                    ref.get() : null;
        if (descriptors == null) {
            descriptors = new ArrayList();
            eventListeners.put(
                    methodContainer,
                    new WeakReference>(
                            descriptors));
        } else {
            for (EventListenerDescriptor descriptor : descriptors) {
                if (descriptor.listenerClass.equals(listenerClass) &&
                    (listenerMethodName == null ?
                     descriptor.listenerMethodName == null :
                     listenerMethodName.equals(
                             descriptor.listenerMethodName)) &&
                    methodName.equals(descriptor.methodName)) {
                    return (E) descriptor.eventListener;
                }
            }
        }

        // else we need to create a new listener
        final EventListenerDescriptor descriptor = new EventListenerDescriptor();
        descriptor.listenerClass = listenerClass;
        descriptor.listenerMethodName = listenerMethodName;
        descriptor.methodName = methodName;
        try {
            final List listenerMethods =
                    Arrays.asList(listenerClass.getMethods());
            Method listenerMethod = null;
            if (listenerMethodName != null) {
                for (Method listenerMethod1 : listenerMethods) {
                    if (listenerMethod1.getName().equals(listenerMethodName)) {
                        listenerMethod = listenerMethod1;
                        break;
                    }
                }
            }
            if (listenerMethodName != null && listenerMethod == null) {
                throw new IllegalArgumentException(
                        "no method named " + listenerMethodName +
                        " found in class " + listenerClass.getName());
            }
            Class[] parameterTypes =
                    listenerMethods.get(0).getParameterTypes();
            Class methodContainerClass = methodContainer.getClass();
            final Method targetMethod =
                    methodContainerClass.getMethod(methodName, parameterTypes);
            descriptor.eventListener =
                    Proxy.newProxyInstance(listenerClass.getClassLoader(),
                                           new Class[]{listenerClass},
                                           new InvocationHandler() {

                                               @Override
                                               public Object invoke(Object proxy,
                                                                    Method method,
                                                                    Object[] args) {
                                                   String methodName = method.getName();
                                                   if ((listenerMethodName == null &&
                                                        listenerMethods.contains(method)) ||
                                                       methodName.equals(listenerMethodName)) {
                                                       try {
                                                           targetMethod.setAccessible(true);
                                                           return targetMethod.invoke(
                                                                   methodContainer, args);
                                                       } catch (IllegalAccessException e) {
                                                           throw new RuntimeException(
                                                                   "could not invoke on container " +
                                                                   methodContainer, e);
                                                       } catch (InvocationTargetException e) {
                                                           throw new RuntimeException(e);
                                                       }
                                                   }
                                                   if (methodName.equals("toString")) {
                                                       return toString();
                                                   }
                                                   if (methodName.equals("equals")) {
                                                       return descriptor.eventListener == args[0];
                                                   }
                                                   if (methodName.equals("hashCode")) {
                                                       return hashCode();
                                                   }
                                                   return null;
                                               }
                                           });
            descriptors.add(descriptor);
            return (E) descriptor.eventListener;
        } catch (NoSuchMethodException e) {
            throw new RuntimeException(e);
        }
    }

    public static  E getEventListener(
            Class listenerClass,
            Object methodContainer,
            String methodName) {
        return getEventListener(listenerClass, null, methodContainer,
                                methodName);
    }

    public static DataBindingUpdateListener getDataBindingUpdateListener(
            JAXXObject object, String bindingName) {
        WeakReference> ref =
                dataBindingUpdateListeners.get(object);
        List listeners = ref == null ? null :
                                                    ref.get();
        if (listeners == null) {
            listeners = new ArrayList();
            dataBindingUpdateListeners.put(
                    object,
                    new WeakReference>(
                            listeners));
        } else {
            for (DataBindingUpdateListener listener : listeners) {
                if (bindingName.equals(listener.getBindingName())) {
                    return listener;
                }
            }
        }
        DataBindingUpdateListener listener =
                new DataBindingUpdateListener(object, bindingName);
        listeners.add(listener);
        return listener;
    }

//    public static void setComponentWidth(Component component, int width) {
//        component.setSize(width, component.getHeight());
//        if (component instanceof JComponent) {
//            JComponent jcomponent = (JComponent) component;
//            jcomponent.setPreferredSize(new Dimension(width, jcomponent.getPreferredSize().height));
//            jcomponent.setMinimumSize(new Dimension(width, jcomponent.getPreferredSize().height));
//            if (jcomponent.isDisplayable()) {
//                jcomponent.revalidate();
//            }
//        }
//    }

//    public static void setComponentHeight(Component component, int height) {
//        component.setSize(component.getWidth(), height);
//        if (component instanceof JComponent) {
//            JComponent jcomponent = (JComponent) component;
//            jcomponent.setPreferredSize(new Dimension(jcomponent.getPreferredSize().width, height));
//            jcomponent.setMinimumSize(new Dimension(jcomponent.getPreferredSize().width, height));
//            if (jcomponent.isDisplayable()) {
//                jcomponent.revalidate();
//            }
//        }
//    }

    public static boolean assignment(boolean value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static byte assignment(byte value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static short assignment(short value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static int assignment(int value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static long assignment(long value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static float assignment(float value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static double assignment(double value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static char assignment(char value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    public static Object assignment(Object value, String name, JAXXObject src) {
        src.firePropertyChange(name.trim(), null, "dummy value");
        return value;
    }

    /**
     * Compute the string representation of an object.
     * 

* Return empty string if given object is null * * @param value the value to write * @return the string representation of the given object or an empty string * if object is null. */ public static String getStringValue(Object value) { String result; result = value == null ? "" : value.toString(); return result; } /** * Test if some entries exists in a given context and throw an * IllegalArgumentException if not. *

* * @param context the context to test * @param defs the definitions of entries to seek in context * @throws IllegalArgumentException if the entry is not found in context. */ public static void checkJAXXContextEntries(JAXXContext context, JAXXContextEntryDef... defs) throws IllegalArgumentException { for (JAXXContextEntryDef def : defs) { Object value = def.getContextValue(context); if (value == null) { throw new IllegalArgumentException( "the context entry [" + def + "] ] was not found in " + "context " + context); } } } /** * Test if a type of entry exists in a given context and throw an * IllegalArgumentException if not found. *

* If entry is found, return his value in context. * * @param the type of required data * @param context the context to test * @param def the definition of the entry to seek in context * @return the value from the context * @throws IllegalArgumentException if the entry is not found in context. */ public static T checkJAXXContextEntry(JAXXContext context, JAXXContextEntryDef def) throws IllegalArgumentException { T value = def.getContextValue(context); if (value == null) { throw new IllegalArgumentException( "the context entry [" + def + "] ] was not found in " + "context " + context); } return value; } /** * Convinient method to apply more than one binding on a JAXX ui. * * @param src the ui to treate * @param bindings the list of binding to process. */ public static void applyDataBinding(JAXXObject src, String... bindings) { for (String binding : bindings) { src.applyDataBinding(binding); } } /** * Convinient method to apply more than one binding on a JAXX ui. * * @param src the ui to treate * @param bindings the list of binding to process. */ public static void applyDataBinding(JAXXObject src, Collection bindings) { for (String binding : bindings) { src.applyDataBinding(binding); } } /** * Convinient method to process more than one binding on a JAXX ui. * * @param src the ui to treate * @param bindings the list of binding to process. */ public static void processDataBinding(JAXXObject src, String... bindings) { for (String binding : bindings) { src.processDataBinding(binding); } } /** * Convinient method to remove more than one binding on a JAXX ui. * * @param src the ui to treate * @param bindings the list of binding to process. */ public static void removeDataBinding(JAXXObject src, String... bindings) { for (String binding : bindings) { src.removeDataBinding(binding); } } /** * detects all PropertychangedListener added by Jaxx uis (should be a {@link * DataBindingListener} * * @param propertyNames the array of property names to find * @param listeners the array of listeners to filter * @return the filtered listeners */ public static PropertyChangeListener[] findJaxxPropertyChangeListener( String[] propertyNames, PropertyChangeListener... listeners) { if (listeners == null || listeners.length == 0) { return EMPTY_ARRAY_PROPERTY_CHANGE_LISTENERS; } List pNames = Arrays.asList(propertyNames); List toRemove = new ArrayList(); for (PropertyChangeListener listener : listeners) { String pName = null; PropertyChangeListenerProxy plistener = null; if (listener instanceof PropertyChangeListenerProxy) { plistener = (PropertyChangeListenerProxy) listener; if (!pNames.contains(plistener.getPropertyName())) { // not on the good property continue; } listener = (PropertyChangeListener) plistener.getListener(); pName = plistener.getPropertyName(); } if (plistener != null && pName != null && listener instanceof DataBindingListener) { if (log.isDebugEnabled()) { log.debug("find config listener to remove [" + pName + "] : " + listener); } toRemove.add(plistener); //toRemove.add(listener); } } return toRemove.toArray(new PropertyChangeListener[toRemove.size()]); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy