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

org.jibx.custom.classes.ClassCustom Maven / Gradle / Ivy

/*
 * Copyright (c) 2007-2009, Dennis M. Sosnoski. All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
 * following conditions are met:
 * 
 * Redistributions of source code must retain the above copyright notice, this list of conditions and the following
 * disclaimer. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
 * following disclaimer in the documentation and/or other materials provided with the distribution. Neither the name of
 * JiBX nor the names of its contributors may be used to endorse or promote products derived from this software without
 * specific prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package org.jibx.custom.classes;

import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.jibx.custom.CustomUtils;
import org.jibx.runtime.EnumSet;
import org.jibx.runtime.IUnmarshallingContext;
import org.jibx.runtime.QName;
import org.jibx.util.IClass;
import org.jibx.util.IClassItem;
import org.jibx.util.IClassLocator;
import org.jibx.util.InsertionOrderedMap;
import org.jibx.util.InsertionOrderedSet;
import org.jibx.util.StringArray;

/**
 * Class customization information. This supports direct class customizations (such as the corresponding element name,
 * when building a concrete mapping) and also acts as a container for individual fields and/or properties.
 * 
 * @author Dennis M. Sosnoski
 */
public class ClassCustom extends NestingBase implements IApply
{
    /** Enumeration of allowed attribute names */
    public static final StringArray s_allowedAttributes =
        new StringArray(new String[] { "create-type", "deserializer", "element-name", "enum-value-method", "excludes",
            "factory", "form", "includes", "name", "optionals", "requireds", "serializer", "type-name", "use-super" },
            NestingBase.s_allowedAttributes);
    
    /** Element name in XML customization file. */
    public static final String ELEMENT_NAME = "class";
    
    // value set information
    public static final int FORM_DEFAULT = 0;
    public static final int FORM_CONCRETE_MAPPING = 1;
    public static final int FORM_ABSTRACT_MAPPING = 2;
    public static final int FORM_STRING = 3;
    
    public static final EnumSet s_representationEnum =
        new EnumSet(FORM_DEFAULT, new String[] { "default", "concrete-mapping", "abstract-mapping", "simple" });
    
    // values specific to class level
    private String m_name;
    
    private String m_elementName;
    
    private String m_typeName;
    
    private String m_createType;
    
    private String m_factoryMethod;
    
    private String m_enumValueMethod;
    
    private int m_form;
    
    private String[] m_includes;
    
    private String[] m_excludes;
    
    private boolean m_useSuper;
    
    private String[] m_requireds;
    
    private String[] m_optionals;
    
    private String m_serializer;
    
    private String m_deserializer;
    
    // list of contained items
    private final ArrayList m_children;
    
    // values filled in by apply() method
    private boolean m_isApplied;
    
    private QName m_typeQName;
    
    private QName m_elementQName;
    
    private IClass m_classInformation;
    
    private InsertionOrderedMap m_memberMap;
    
    /**
     * Constructor.
     * 
     * @param parent
     * @param name class simple name (without package)
     */
    /* package */ClassCustom(NestingBase parent, String name) {
        super(parent);
        m_name = name;
        m_children = new ArrayList();
        m_useSuper = true;
        m_form = -1;
    }
    
    /**
     * Make sure all attributes are defined.
     * 
     * @param uctx unmarshalling context
     */
    private void preSet(IUnmarshallingContext uctx) {
        validateAttributes(uctx, s_allowedAttributes);
    }
    
    /**
     * Get fully-qualified class name.
     * 
     * @return class name
     */
    public String getName() {
        PackageCustom parent = (PackageCustom)getParent();
        String pack = parent.getName();
        if (pack.length() > 0) {
            return pack + '.' + m_name;
        } else {
            return m_name;
        }
    }
    
    /**
     * Get simple class name.
     * 
     * @return class name
     */
    public String getSimpleName() {
        return m_name;
    }
    
    /**
     * Get the element name to be used for this class in a concrete mapping.
     * 
     * @return element name
     */
    public String getElementName() {
        return m_elementName;
    }
    
    /**
     * Get the qualified element name to be used for this class in a concrete mapping.
     * 
     * @return element name
     */
    public QName getElementQName() {
        return m_elementQName;
    }
    
    /**
     * Get the type name to be used for this class in an abstract mapping.
     * 
     * @return type name
     */
    public String getTypeName() {
        return m_typeName;
    }
    
    /**
     * Get the type name to be used when creating an instance of this class.
     * 
     * @return type name
     */
    public String getCreateType() {
        return m_createType;
    }
    
    /**
     * Set the type name to be used when creating an instance of this class.
     * 
     * @param type
     */
    public void setCreateType(String type) {
        m_createType = type;
    }
    
    /**
     * Get serializer method.
     *
     * @return serializer
     */
    public String getSerializer() {
        return m_serializer;
    }
    
    /**
     * Get deserializer method.
     *
     * @return deserializer
     */
    public String getDeserializer() {
        return m_deserializer;
    }
    
    /**
     * Get the method used to retrieve the text value for an enum class.
     * 
     * @return method name
     */
    public String getEnumValueMethod() {
        return m_enumValueMethod;
    }
    
    /**
     * Get the factory method to be used when creating an instance of this class.
     * 
     * @return method name
     */
    public String getFactoryMethod() {
        return m_factoryMethod;
    }
    
    /**
     * Get the qualified type name to be used for this class in an abstract mapping.
     * 
     * @return type qname
     */
    public QName getTypeQName() {
        return m_typeQName;
    }
    
    /**
     * Get the representation code.
     * 
     * @return value from {@link #s_representationEnum} enumeration
     */
    public int getForm() {
        return m_form;
    }
    
    /**
     * Get list of names to be excluded from class representation.
     * 
     * @return excludes (null if none)
     */
    public String[] getExcludes() {
        return m_excludes;
    }
    
    /**
     * Get list of names to be included in class representation.
     * 
     * @return includes (null if none)
     */
    public String[] getIncludes() {
        return m_includes;
    }
    
    /**
     * Check for superclass to be included in binding.
     * 
     * @return true if superclass included, false if not
     */
    public boolean isUseSuper() {
        return m_useSuper;
    }
    
    /**
     * Check if this is a directly instantiable class (not an interface, and not abstract)
     * 
     * @return true if instantiable, false if not
     */
    public boolean isConcreteClass() {
        return !(m_classInformation.isAbstract() || m_classInformation.isInterface());
    }
    
    /**
     * Check if class represents a simple text value.
     *
     * @return text value flag
     */
    public boolean isSimpleValue() {
        return m_form == FORM_STRING;
    }
    
    /**
     * Check if abstract mapping required for class.
     * 
     * @return abstract mapping flag
     */
    public boolean isAbstractMappingForced() {
        return m_form == FORM_ABSTRACT_MAPPING;
    }
    
    /**
     * Check if concrete mapping required for class. If a 'form' setting is defined for the class it returns the flag
     * based on that setting, and otherwise returns based on the nesting abstract mapping flag.
     * 
     * @return abstract mapping flag
     */
    public boolean isConcreteMappingForced() {
        return m_form == FORM_CONCRETE_MAPPING;
    }
    
    /**
     * Get list of children.
     * 
     * @return list
     */
    public List getChildren() {
        return m_children;
    }
    
    /**
     * Add child.
     * 
     * @param child
     */
    protected void addChild(CustomBase child) {
        if (child.getParent() == this) {
            m_children.add(child);
        } else {
            throw new IllegalStateException("Internal error: child not linked");
        }
    }
    
    /**
     * Form set text method. This is intended for use during unmarshalling. TODO: add validation
     * 
     * @param text
     * @param ictx
     */
    private void setFormText(String text, IUnmarshallingContext ictx) {
        m_form = s_representationEnum.getValue(text);
    }
    
    /**
     * Form get text method. This is intended for use during marshalling.
     * 
     * @return text
     */
    private String getFormText() {
        return s_representationEnum.getName(m_form);
    }
    
    /**
     * Build map from member names to read access methods. This assumes that each public, non-static, no-argument
     * method which returns a value and has a name beginning with "get" or "is" is a property read access method. It
     * maps the corresponding property name to the method, and returns the map.
     * 
     * @param methods
     * @param inclset set of member names to be included (null if not specified)
     * @param exclset set of member names to be excluded (null if not specified, ignored if inclset is
     * non-null)
     * @return map
     */
    private Map mapPropertyReadMethods(IClassItem[] methods, Set inclset, Set exclset) {
        
        // check all methods for property read access matches
        InsertionOrderedMap getmap = new InsertionOrderedMap();
        for (int i = 0; i < methods.length; i++) {
            IClassItem item = methods[i];
            String name = item.getName();
            int flags = item.getAccessFlags();
            if (item.getArgumentCount() == 0 && (flags & Modifier.STATIC) == 0 && (flags & Modifier.PUBLIC) != 0 && 
                ((name.startsWith("get") && !item.getTypeName().equals("void")) || (name.startsWith("is") &&
                item.getTypeName().equals("boolean")))) {
                
                // have what appears to be a getter, check if it should be used
                String memb = ValueCustom.memberNameFromGetMethod(name);
                boolean use = true;
                if (inclset != null) {
                    use = inclset.contains(memb.toLowerCase());
                } else if (exclset != null) {
                    use = !exclset.contains(memb.toLowerCase());
                }
                if (use) {
                    getmap.put(memb, item);
                }
                
            }
        }
        return getmap;
    }
    
    /**
     * Build map from member names to write access methods. This assumes that each public, non-static, single-argument
     * method which returns void and has a name beginning with "set" is a property write access method. It maps the
     * corresponding property name to the method, and returns the map.
     * 
     * @param methods
     * @param inclset set of member names to be included (null if not specified)
     * @param exclset set of member names to be excluded (null if not specified, ignored if inclset is
     * non-null)
     * @return map
     */
    private Map mapPropertyWriteMethods(IClassItem[] methods, Set inclset, Set exclset) {
        
        // check all methods for property write access matches
        InsertionOrderedMap setmap = new InsertionOrderedMap();
        for (int i = 0; i < methods.length; i++) {
            IClassItem item = methods[i];
            String name = item.getName();
            int flags = item.getAccessFlags();
            if (item.getArgumentCount() == 1 && (flags & Modifier.STATIC) == 0 && (flags & Modifier.PUBLIC) != 0 &&
                name.startsWith("set") && item.getTypeName().equals("void")) {
                
                // have what appears to be a setter, check if it should be used
                String memb = ValueCustom.memberNameFromSetMethod(name);
                boolean use = true;
                if (inclset != null) {
                    use = inclset.contains(memb.toLowerCase());
                } else if (exclset != null) {
                    use = !exclset.contains(memb.toLowerCase());
                }
                if (use) {
                    setmap.put(memb, item);
                }
                
            }
        }
        return setmap;
    }
    
    /**
     * Build map from member names to fields. This includes all non-static and non-transient fields of the class.
     * 
     * @param fields
     * @param prefs prefixes to be stripped in deriving names
     * @param suffs suffixes to be stripped in deriving names
     * @param inclset set of member names to be included (null if not specified)
     * @param exclset set of member names to be excluded (null if not specified, ignored if inclset is
     * non-null)
     * @return map
     */
    private Map mapFields(IClassItem[] fields, String[] prefs, String[] suffs, Set inclset, Set exclset) {
        
        // check all fields for use as members
        InsertionOrderedMap fieldmap = new InsertionOrderedMap();
        for (int i = 0; i < fields.length; i++) {
            IClassItem item = fields[i];
            if ((item.getAccessFlags() & (Modifier.FINAL | Modifier.STATIC | Modifier.TRANSIENT)) == 0) {
                
                // not final, static, or transient, so check if it should be used
                String name = item.getName();
                String memb = ValueCustom.memberNameFromField(name, prefs, suffs);
                boolean use = true;
                if (inclset != null) {
                    use = inclset.contains(memb.toLowerCase());
                } else if (exclset != null) {
                    use = !exclset.contains(memb.toLowerCase());
                }
                if (use) {
                    fieldmap.put(memb, item);
                }
            }
        }
        return fieldmap;
    }
    
    /**
     * Find the most specific type for a property based on the access methods.
     * 
     * @param gmeth read access method (null if not defined)
     * @param smeth write access method (null if not defined)
     * @param icl
     * @return most specific type name
     */
    private String findPropertyType(IClassItem gmeth, IClassItem smeth, IClassLocator icl) {
        String type;
        if (gmeth == null) {
            if (smeth == null) {
                throw new IllegalArgumentException("Internal error: no access methods known");
            } else {
                type = smeth.getArgumentType(0);
            }
        } else if (smeth == null) {
            type = gmeth.getTypeName();
        } else {
            String gtype = gmeth.getTypeName();
            String stype = smeth.getArgumentType(0);
            IClass gclas = icl.getRequiredClassInfo(gtype);
            if (gclas.isSuperclass(stype) || gclas.isImplements(stype)) {
                type = gtype;
            } else {
                type = stype;
            }
        }
        return type;
    }
    
    /**
     * Utility method to strip any leading non-alphanumeric characters from an array of name strings.
     * 
     * @param names (null if none)
     * @return array of stripped names (null if none)
     */
    private static String[] stripNames(String[] names) {
        if (names == null) {
            return null;
        } else {
            String[] strips = null;
            for (int i = 0; i < names.length; i++) {
                String name = names[i];
                int index = 0;
                while (index < name.length()) {
                    char chr = name.charAt(index);
                    if (Character.isLetter(chr) || Character.isDigit(chr)) {
                        break;
                    } else {
                        index++;
                    }
                }
                if (index > 0) {
                    if (strips == null) {
                        strips = new String[names.length];
                        System.arraycopy(names, 0, strips, 0, names.length);
                    }
                    strips[i] = name.substring(index);
                }
            }
            return (strips == null) ? names : strips;
        }
    }
    
    /**
     * Classify an array of names as elements or attributes, based on leading flag characters ('@' for an attribute,
     * '<' for an element).
     *
     * @param names (null if none)
     * @param elems set of element names
     * @param attrs set of attribute names
     */
    private static void classifyNames(String[] names, Set elems, Set attrs) {
        if (names != null) {
            for (int i = 0; i < names.length; i++) {
                String name = names[i];
                if (name.length() > 0) {
                    char chr = name.charAt(0);
                    Set addset = null;
                    Set altset = null;
                    if (chr == '@') {
                        addset = attrs;
                        altset = elems;
                    } else if (chr == '/') {
                        addset = elems;
                        altset = attrs;
                    }
                    if (addset != null) {
                        name = name.substring(1).toLowerCase();
                        addset.add(name);
                        if (altset.contains(name)) {
                            throw new IllegalArgumentException("Name " + name + " used as both element and attribute");
                        }
                    }
                }
            }
        }
    }
    
    /**
     * Apply customizations to class to fill out members. The name handling gets tricky, because of the variety of
     * options provided. The user can specify value names to be included and/or excluded for the class, and also value
     * names to be treated as optional or required. If an 'includes' list is given, only the values on that list are
     * processed from the class, along with any values with their own child elements. If an 'excludes' list is given,
     * only the names not on that list are processed. If neither an 'includes' nor an 'excludes' list is present, all
     * values defined by the class will be processed. It is an error if the same name occurs on both an 'excludes' list
     * and any other list, or if a name on the 'excludes' list has a child element present. It is also an error if the
     * same name occurs on both the 'optionals' and 'requireds' list. Each list name can also be flagged with a leading
     * indicator character to say whether the value should be represented as an attribute ('@') or element ('<'). The
     * order of child elements is also partially determined by the lists, with the 'includes' list processed first, then
     * the 'requireds' list, then the 'optionals' list, then the child elements, then any values not yet processed.
     * 
     * @param icl class locator
     */
    public void apply(IClassLocator icl) {
        
        // check for repeated call (happens due to package-level apply()
        if (m_isApplied) {
            return;
        }
        m_isApplied = true;
        
        // initialize class information
        m_classInformation = icl.getClassInfo(getName());
        if (m_classInformation == null) {
            throw new IllegalStateException("Internal error: unable to find class " + m_name);
        }
        
        // skip any member handling if simple value
        if (m_form == FORM_STRING) {
            return;
        }
        
        // inherit namespace directly from package level, if not specified
        String ns = getSpecifiedNamespace();
        if (ns == null) {
            ns = getParent().getNamespace();
        }
        setNamespace(ns);
        
        // set the name(s) to be used if mapped
        String cname = convertName(getSimpleName());
        if (m_elementName == null) {
            m_elementName = cname;
        }
        m_elementQName = new QName(getNamespace(), m_elementName);
        if (m_typeName == null) {
            m_typeName = cname;
        }
        m_typeQName = new QName(getNamespace(), m_typeName);
        
        // initialize maps with existing member customizations
        HashMap namemap = new HashMap();
        for (Iterator iter = getChildren().iterator(); iter.hasNext();) {
            ValueCustom memb = (ValueCustom)iter.next();
            String name = memb.getBaseName();
            if (name != null) {
                namemap.put(name, memb);
            }
        }
        
        // generate sets of names to be included or ignored, and optional/requires, elements/attributes
        InsertionOrderedSet nameset = new InsertionOrderedSet();
        Set exclset = null;
        Set inclset = null;
        Set elemset = Collections.EMPTY_SET;
        Set attrset = Collections.EMPTY_SET;
        Set optset = Collections.EMPTY_SET;
        Set reqset = Collections.EMPTY_SET;
        if (m_includes != null || m_optionals != null || m_requireds != null) {
            
            // initialize the ordered set of names to be included
            String[] optionals = stripNames(m_optionals);
            String[] requireds = stripNames(m_requireds);
            if (m_includes != null) {
                nameset.addAll(stripNames(m_includes));
            } else {
                if (requireds != null) {
                    nameset.addAll(requireds);
                }
                if (optionals != null) {
                    nameset.addAll(optionals);
                }
            }
            
            // build sets of includeds, optionals, and requireds
            inclset = CustomUtils.noCaseNameSet(stripNames(m_includes));
            if (optionals != null) {
                optset = CustomUtils.noCaseNameSet(optionals);
            }
            if (requireds != null) {
                reqset = CustomUtils.noCaseNameSet(requireds);
            }
            
            // classify names as elements or attributes
            elemset = new HashSet();
            attrset = new HashSet();
            classifyNames(m_requireds, elemset, attrset);
            classifyNames(m_optionals, elemset, attrset);
            classifyNames(m_includes, elemset, attrset);
            
        }
        if (m_excludes != null) {
            exclset = CustomUtils.noCaseNameSet(stripNames(m_excludes));
        }
        
        // first find members from property access methods
        Map getmap = null;
        Map setmap = null;
        IClassItem[] methods = m_classInformation.getMethods();
        GlobalCustom global = getGlobal();
        if (global.isOutput()) {
            
            // find properties using read access methods
            getmap = mapPropertyReadMethods(methods, inclset, exclset);
            if (global.isInput()) {
                
                // find properties using write access methods
                setmap = mapPropertyWriteMethods(methods, inclset, exclset);
                
                // discard any read-only properties
                for (Iterator iter = getmap.keySet().iterator(); iter.hasNext();) {
                    if (!setmap.containsKey(iter.next())) {
                        iter.remove();
                    }
                }
                
            }
        }
        if (global.isInput()) {
            
            // find properties using write access methods
            setmap = mapPropertyWriteMethods(methods, inclset, exclset);
            
        }
        
        // find members from fields
        IClassItem[] fields = m_classInformation.getFields();
        Map fieldmap = mapFields(fields, getStripPrefixes(), getStripSuffixes(), inclset, exclset);
        
        // get list of names selected for use by options
        if (m_includes == null) {
            if (isPropertyAccess()) {
                if (global.isOutput()) {
                    nameset.addAll(getmap.keySet());
                } else {
                    nameset.addAll(setmap.keySet());
                }
            } else {
                for (Iterator iter = fieldmap.keySet().iterator(); iter.hasNext();) {
                    String name = (String)iter.next();
                    IClassItem field = (IClassItem)fieldmap.get(name);
                    int access = field.getAccessFlags();
                    if (!Modifier.isStatic(access) && !Modifier.isTransient(access)) {
                        nameset.add(name);
                    }
                }
            }
        }
        
        // process all members found in class
        m_memberMap = new InsertionOrderedMap();
        boolean auto = !getName().startsWith("java.") && !getName().startsWith("javax.");
        List names = nameset.asList();
        for (Iterator iter = names.iterator(); iter.hasNext();) {
            
            // get basic member information
            String name = (String)iter.next();
            String lcname = name.toLowerCase();
            ValueCustom cust = null;
            IClassItem gmeth = (IClassItem)(getmap == null ? null : getmap.get(name));
            IClassItem smeth = (IClassItem)(setmap == null ? null : setmap.get(name));
            IClassItem field = (IClassItem)(fieldmap == null ? null : fieldmap.get(name));
            
            // find the optional/required setting
            Boolean isreq = null;
            if (optset.contains(lcname)) {
                isreq = Boolean.FALSE;
            }
            if (reqset.contains(lcname)) {
                isreq = Boolean.TRUE;
            }
            
            // find the style setting
            Integer style = null;
            if (attrset.contains(lcname)) {
                style = ATTRIBUTE_STYLE_INTEGER;
            } else if (elemset.contains(lcname)) {
                style = ELEMENT_STYLE_INTEGER;
            }
            
            // check for existing customization
            if (namemap.containsKey(name)) {
                
                // fill in data missing from existing member customization
                cust = (ValueCustom)namemap.get(name);
                
            } else if (auto) {
                if (isPropertyAccess()) {
                    if (gmeth != null || smeth != null) {
                        
                        // create value information from methods
                        cust = new ValueCustom(this, name);
                        
                    }
                } else if (field != null) {
                    
                    // create value information from field
                    cust = new ValueCustom(this, name);
                    
                }
            }
            
            // add customization to map
            if (cust != null) {
                if (isPropertyAccess()) {
                    cust.fillDetails(null, gmeth, smeth, icl, isreq, style);
                } else {
                    cust.fillDetails(field, null, null, icl, isreq, style);
                }
                m_memberMap.put(name, cust);
            }
        }
        
        // check for any supplied customizations that haven't been matched
        for (Iterator iter = namemap.keySet().iterator(); iter.hasNext();) {
            String name = (String)iter.next();
            String lcname = name.toLowerCase();
            if (!m_memberMap.containsKey(name)) {
                
                // find the optional/required setting
                Boolean req = null;
                if (optset != null && optset.contains(lcname)) {
                    req = Boolean.FALSE;
                }
                if (reqset != null && reqset.contains(lcname)) {
                    req = Boolean.TRUE;
                }
                
                // find the style setting
                Integer style = null;
                if (attrset.contains(lcname)) {
                    style = ATTRIBUTE_STYLE_INTEGER;
                } else if (elemset.contains(lcname)) {
                    style = ELEMENT_STYLE_INTEGER;
                }
                
                // complete the customization and add to map
                ValueCustom cust = (ValueCustom)namemap.get(name);
                cust.fillDetails(m_classInformation, req, style);
                m_memberMap.put(name, cust);
                
            }
        }
    }
    
    /**
     * Get customization information for a member by name. This method may only be called after
     * {@link #apply(IClassLocator)}.
     * 
     * @param name
     * @return customization, or null if none
     */
    public ValueCustom getMember(String name) {
        return (ValueCustom)m_memberMap.get(name);
    }
    
    /**
     * Get actual class information. This method may only be called after {@link #apply(IClassLocator)}.
     * 
     * @return class information
     */
    public IClass getClassInformation() {
        return m_classInformation;
    }
    
    /**
     * Get collection of members in class.
     * 
     * @return members
     */
    public Collection getMembers() {
        return m_memberMap.values();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy