Package com.googlecode.wicketwebbeans.model

Source Code of com.googlecode.wicketwebbeans.model.ElementMetaData

/*---
   Copyright 2006-2007 Visual Systems Corporation.
   http://www.vscorp.com

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at
  
        http://www.apache.org/licenses/LICENSE-2.0
  
   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.
---*/
package com.googlecode.wicketwebbeans.model;

import java.io.Serializable;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.Collection;
import java.util.Iterator;

import org.apache.wicket.Component;
import org.apache.wicket.behavior.AttributeAppender;
import org.apache.wicket.markup.html.basic.Label;
import org.apache.wicket.model.Model;
import org.apache.wicket.util.lang.PropertyResolver;

import com.googlecode.wicketwebbeans.fields.ImageLabel;


/**
* Represents the metadata for a single property of a bean or an action.
*/
public final class ElementMetaData extends MetaData implements Serializable
{
    private static final long serialVersionUID = 1842855853283371120L;

    public static final String PARAM_ELEMENT_TYPE = "elementType";
    public static final String PARAM_VIEW_ONLY = "viewOnly";
    public static final String PARAM_LABEL_IMAGE = "labelImage";
    public static final String PARAM_LABEL = "label";
    public static final String PARAM_FIELD_TYPE = "fieldType";
    public static final String PARAM_REQUIRED = "required";
    public static final String PARAM_MAX_LENGTH = "maxlength";
    public static final String PARAM_DEFAULT_VALUE = "defaultValue";
    public static final String PARAM_ROWS = "rows";
    public static final String PARAM_COLUMNS = "cols";

    public static final String CSS_REQUIRED_FIELD_CLASS = "beanRequiredField";
    public static final int DEFAULT_ORDER = Integer.MAX_VALUE;
   
    private BeanMetaData beanMetaData;
    private String propertyName;
    private Type propertyType;
    private int order;
    private String tabId;
    private boolean isAction = false;
    private boolean actionSpecifiedInProps = false;
    private boolean specifiedInProps = false;
   
    ElementMetaData(BeanMetaData beanMetaData, String propertyName, String label, Type propertyType)
    {
        super(beanMetaData.getComponent());
       
        this.beanMetaData = beanMetaData;
        this.propertyName = propertyName;
        this.propertyType = propertyType;
        setLabel(label);
        this.order = DEFAULT_ORDER;
       
        consumeParameter(PARAM_FIELD_TYPE);
        consumeParameter(PARAM_LABEL);
        consumeParameter(PARAM_LABEL_IMAGE);
        consumeParameter(PARAM_VIEW_ONLY);
        consumeParameter(PARAM_ELEMENT_TYPE);
    }
   
    public String getFieldType()
    {
        String fieldType = getParameter(PARAM_FIELD_TYPE);
        if (fieldType != null && fieldType.indexOf('.') < 0) {
            // Base class name provided. Try to resolve it from the BeanMetaData's ComponentRegistry.
            String fullFieldType = beanMetaData.getComponentRegistry().findMatchingFieldClass(fieldType);
            if (fullFieldType != null) {
                // Store it now that we know it.
                setFieldType(fullFieldType);
                return fullFieldType;
            }
        }
       
        return fieldType;
    }
   
    public void setFieldType(String fieldType)
    {
        setParameter(PARAM_FIELD_TYPE, fieldType);
    }

    public String getTabId()
    {
        return tabId;
    }
   
    public void setTabId(String groupId)
    {
        this.tabId = groupId;
    }

    public String getLabel()
    {
        return getParameter(PARAM_LABEL);
    }
   
    public void setLabel(String label)
    {
        setParameter(PARAM_LABEL, label);
    }
   
    public String getDefaultValue()
    {
        return getParameter(PARAM_DEFAULT_VALUE);
    }
   
    public void setDefaultValue(String value)
    {
        if (value != null) {
            setParameter(PARAM_DEFAULT_VALUE, value);
        }
    }
   
    /**
     * @return the maximum length for the field, or null if no maxmimum length.
     */
    public Integer getMaxLength()
    {
        return getIntegerParameter(PARAM_MAX_LENGTH);
    }
   
    /**
     * Sets the maximum length.
     *
     * @param maxLength the maximum length. If null, nothing is set.
     */
    public void setMaxLength(Integer maxLength)
    {
        if (maxLength != null && maxLength != 0) {
            setParameter(PARAM_MAX_LENGTH, maxLength.toString());
        }
    }
   
    public boolean isRequired()
    {
        return getBooleanParameter(PARAM_REQUIRED);
    }
   
    public void setRequired(boolean required)
    {
        setParameter(PARAM_REQUIRED, String.valueOf(required));
    }
   
    /**
     * Gets the labelImage.
     *
     * @return the labelImage.
     */
    public String getLabelImage()
    {
        return getParameter(PARAM_LABEL_IMAGE);
    }

    /**
     * @param wicketId
     * @return the Component used to render the label. If a label image was specified, this is the
     *  image, otherwise a plain-text label.
     */
    public Component getLabelComponent(String wicketId)
    {
        Component component;
        if (getLabelImage() == null) {
            component = new Label(wicketId, getLabel());
        }
        else {
            component = new ImageLabel(wicketId, getBeanMetaData().getComponent().getClass(), getLabelImage(), getLabel());
        }
       
        if (isRequired()) {
            component.add( new AttributeAppender("class", new Model<String>(CSS_REQUIRED_FIELD_CLASS), " ") );
        }
       
        return component;
    }
   
    public int getOrder()
    {
        return order;
    }
   
    public void setOrder(int order)
    {
        this.order = order;
    }
   
    public String getPropertyName()
    {
        return propertyName;
    }
   
    public void setPropertyName(String propertyName)
    {
        this.propertyName = propertyName;
    }
   
    public boolean isViewOnly()
    {
        return getBooleanParameter(PARAM_VIEW_ONLY);
    }
   
    public void setViewOnly(boolean viewOnly)
    {
        setParameter(PARAM_VIEW_ONLY, String.valueOf(viewOnly));
    }
   
    @Override
    public int hashCode()
    {
        int result = 1;
        result = 31 * result + ((propertyName == null) ? 0 : propertyName.hashCode());
        return result;
    }

    @Override
    public boolean equals(Object obj)
    {
        if (this == obj) {
            return true;
        }
       
        if (obj == null || this.getClass() != obj.getClass()) {
            return false;
        }
       
        final ElementMetaData other = (ElementMetaData) obj;
        if (propertyName == null) {
            if (other.propertyName != null) {
                return false;
            }
        }

        return propertyName.equals(other.propertyName);
    }
   
    public Class getPropertyType()
    {
        if (propertyType == null) {
            return null;
        } else if (propertyType instanceof ParameterizedType) {
            ParameterizedType parameterizedType = (ParameterizedType) propertyType;
            return (Class) parameterizedType.getRawType();
        } else if (propertyType instanceof Class) {
            return (Class) propertyType;
        } else {
            throw new RuntimeException("Unknown subclass of element type: " + propertyType.getClass().getName());
        }
    }

    /**
     * Gets the beanMetaData.
     *
     * @return a BeanMetaData.
     */
    public BeanMetaData getBeanMetaData()
    {
        return beanMetaData;
    }

    public String getElementTypeName()
    {
        String elementType = getParameter(PARAM_ELEMENT_TYPE);
        if (elementType != null && elementType.indexOf('.') < 0) {
            // Base class name provided. Assume that it is the bean's package.
            String beanPkg = beanMetaData.getBeanClass().getPackage().getName();
            if (beanPkg.length() > 0) {
                elementType = beanPkg + '.' + elementType;
                setParameter(PARAM_ELEMENT_TYPE, elementType);
            }
        }
       
        return elementType;
    }
   
    /**
     * Gets the array/collection element type of the property. If elementTypeName is
     * not defined and the property type is an array, the element type of the array is returned.
     * If elementTypeName is
     * not defined and value is not null and it is
     * a Collection and is not empty, the element type is derived from the first element of the collection.
     *
     * Patch submitted by Kent in issue 31:
     * If the Collection is empty, use reflection at runtime to obtain the elementType
     *
     * @param value the property's current value. May be null to ignore runtime type detection.
     *
     * @return the elementType or null if not defined.
     */
    public Class getElementType(Object value)
    {
        Class elementType = null;
        String elementTypeName = getElementTypeName();
        if (elementTypeName == null) {
            // This only returns the element type if the property is an array, otherwise null.
            elementType = getPropertyType().getComponentType();
        }
       
        if (elementType == null) {
            // Try to detect it from first element if non-empty.
            if (value instanceof Collection) {
                Iterator iter = ((Collection)value).iterator();
                if (iter.hasNext()) {
                    elementType = iter.next().getClass();
                }
                else {
                    Class compileTimeElementType = getCompileTimeElementType();
                    if (compileTimeElementType != null) {
                        elementType = compileTimeElementType;
                    } else {
                        // If empty - just use Object.
                        elementType = new Serializable() {
                                private static final long serialVersionUID = 1L;
                            }.getClass();
                    }
                }
            }
        }

        if (elementType == null && elementTypeName != null) {
            try {
                return Class.forName(elementTypeName);
            }
            catch (ClassNotFoundException e) {
                throw new RuntimeException("Cannot load class for elementType: " + elementTypeName + " for property " + getPropertyName(), e);
            }
        }
       
        return elementType;
    }

    /**
     * Uses reflection at runtime to get the elementType of an empty Collection
     *
     * @return the actual elementType of the Collection at runtime
     */
    private Class getCompileTimeElementType()
    {
        if (propertyType instanceof ParameterizedType) {
            ParameterizedType parameterizedType = (ParameterizedType) propertyType;
            Type[] typeArgs = parameterizedType.getActualTypeArguments();
            if (typeArgs.length == 1) {
                Type firstTypeArg = typeArgs[0];
                if (firstTypeArg instanceof Class) {
                    return (Class) typeArgs[0];
                }
            }
        }
        return null;
    }

    /**
     * Determines whether this element is an action.
     *
     * @return true if this is an action, else false.
     */
    public boolean isAction()
    {
        return isAction;
    }

    /**
     * Sets whether this element is an action.
     *
     * @param isAction true if this is an action, else false.
     */
    public void setAction(boolean isAction)
    {
        this.isAction = isAction;
    }
   
    /**
     * Gets the actionSpecifiedInProps.
     *
     * @return a boolean.
     */
    public boolean isActionSpecifiedInProps()
    {
        return actionSpecifiedInProps;
    }
   
    /**
     * Sets actionSpecifiedInProps.
     *
     * @param actionSpecifiedInProps a boolean.
     */
    public void setActionSpecifiedInProps(boolean actionSpecifiedInProps)
    {
        this.actionSpecifiedInProps = actionSpecifiedInProps;
    }
   
    public boolean isSpecifiedInProps()
    {
  return specifiedInProps;
    }

    public void setSpecifiedInProps(boolean specifiedInProps)
    {
        this.specifiedInProps = specifiedInProps;
    }

    /**
     * If this is an action, return the action's method name.
     *
     * @return the action method name, or null if this is not an action.
     */
    public String getActionMethodName()
    {
        if (isAction()) {
            return getPropertyName().substring(BeanMetaData.ACTION_PROPERTY_PREFIX.length());
        }
       
        return null;
    }

    /**
     * Gets the value for this property from the given bean.
     *
     * @param bean the bean, which may be null.
     *
     * @return the property's value, or null if bean is null.
     */
    public Object getPropertyValue(Object bean)
    {
        if (bean == null) {
            return null;
        }
       
        return PropertyResolver.getValue(getPropertyName(), bean);
    }

    /**
     * Creates a new BeanMetaData based on this property's type.
     *
     * @param viewOnly
     *
     * @return a new BeanMetaData.
     */
    public BeanMetaData createBeanMetaData(boolean viewOnly)
    {
        return createBeanMetaData(getPropertyType(), viewOnly);
    }
   
    /**
     * Creates a new BeanMetaData for the given beanType based on this property. Context and ComponentRegistry are
     * inherited from this element's BeanMetaData (the parent).
     *
     * @param beanType
     * @param viewOnly
     *
     * @return a new BeanMetaData.
     */
    public BeanMetaData createBeanMetaData(Class beanType, boolean viewOnly)
    {
        // Compose a keyPrefix from the original bean meta data, the base bean class name, and this property name.
        BeanMetaData parentMetaData = getBeanMetaData();
           
        return new BeanMetaData(beanType, parentMetaData.getContext(), parentMetaData.getBeansMetaData(), parentMetaData.getMetaDataClass(), parentMetaData.getComponent(),
                        parentMetaData.getComponentRegistry(), viewOnly, true);
    }
   
    /**
     * @return a new instance of the object represented by this property's type.
     */
    public Object createInstance()
    {
        // TODO - handle arrays???
        try {
            return getPropertyType().newInstance();
        }
        catch (Exception e) {
            throw new RuntimeException("Error creating instance of " + getPropertyType());
        }
    }
}
TOP

Related Classes of com.googlecode.wicketwebbeans.model.ElementMetaData

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.