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

org.jibx.schema.codegen.extend.DefaultNameConverter Maven / Gradle / Ivy

/*
 * Copyright (c) 2006-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.schema.codegen.extend;

import java.util.ArrayList;

import org.jibx.schema.codegen.NameUtils;
import org.jibx.util.NameUtilities;

/**
 * Converter for working with Java names.
 * 
 * @author Dennis M. Sosnoski
 */
public class DefaultNameConverter implements NameConverter
{
    /** Camelcase field names flag. */
    private boolean m_camelCase;
    
    /** Prefix used for normal field names (non-null, may be empty). */
    private String m_fieldPrefix;
    
    /** Suffix used for normal field names (non-null, may be empty). */
    private String m_fieldSuffix;
    
    /** Prefix used for static field names (non-null, may be empty). */
    private String m_staticPrefix;
    
    /** Suffix used for static field names (non-null, may be empty). */
    private String m_staticSuffix;
    
    /** Use underscores in field names flag (as substitute for special characters, and to split words). */
    private boolean m_underscore;
    
    /** Uppercase initial letter of field names flag. */
    private boolean m_upperInitial;
    
    /** Set of XML name prefixes to be discarded in conversions. */
    private String[] m_stripPrefixes;
    
    /** Set of XML name suffixes to be discarded in conversions. */
    private String[] m_stripSuffixes;
    
    /** Reusable array for words in name. */
    private ArrayList m_wordList;
    
    /**
     * Constructor.
     */
    public DefaultNameConverter() {
        m_fieldPrefix = "";
        m_fieldSuffix = "";
        m_staticPrefix = "";
        m_staticSuffix = "";
        m_camelCase = true;
        m_stripPrefixes = m_stripSuffixes = new String[0];
        m_wordList = new ArrayList();
    }
    
    /**
     * Copy constructor.
     * 
     * @param base instance used to initialize settings
     */
    public DefaultNameConverter(DefaultNameConverter base) {
        m_fieldPrefix = base.m_fieldPrefix;
        m_fieldSuffix = base.m_fieldSuffix;
        m_staticPrefix = base.m_staticPrefix;
        m_staticSuffix = base.m_staticSuffix;
        m_camelCase = base.m_camelCase;
        m_stripPrefixes = base.m_stripPrefixes;
        m_stripSuffixes = base.m_stripSuffixes;
        m_wordList = new ArrayList();
    }
    
    /**
     * Convert singular name to plural form.
     * 
     * @param name base name
     * @return plural name
     */
    public String pluralize(String name) {
        return NameUtilities.pluralize(name);
    }
    
    /**
     * Convert potentially plural name to singular form.
     * 
     * @param name base name
     * @return singularized name
     */
    public String depluralize(String name) {
        return NameUtilities.depluralize(name);
    }
    
    /**
     * Get prefix text for normal field names.
     * 
     * @return field prefix (non-null, may be empty)
     */
    public String getFieldPrefix() {
        return m_fieldPrefix;
    }
    
    /**
     * Set prefix text for normal field names.
     * 
     * @param pref field prefix (non-null, may be empty)
     */
    public void setFieldPrefix(String pref) {
        m_fieldPrefix = pref;
    }
    
    /**
     * Get suffix text for normal field names.
     * 
     * @return field suffix (non-null, may be empty)
     */
    public String getFieldSuffix() {
        return m_fieldPrefix;
    }
    
    /**
     * Set suffix text for normal field names.
     * 
     * @param suff field suffix (non-null, may be empty)
     */
    public void setFieldSuffix(String suff) {
        m_fieldPrefix = suff;
    }
    
    /**
     * Get prefix text for static field names.
     * 
     * @return field prefix (non-null, may be empty)
     */
    public String getStaticPrefix() {
        return m_staticPrefix;
    }
    
    /**
     * Set prefix text for static field names.
     * 
     * @param pref field prefix (non-null, may be empty)
     */
    public void setStaticPrefix(String pref) {
        m_staticPrefix = pref;
    }
    
    /**
     * Get suffix text for static field names.
     * 
     * @return field suffix (non-null, may be empty)
     */
    public String getStaticSuffix() {
        return m_staticPrefix;
    }
    
    /**
     * Set suffix text for static field names.
     * 
     * @param suff field suffix (non-null, may be empty)
     */
    public void setStaticSuffix(String suff) {
        m_staticPrefix = suff;
    }

    /**
     * Get the prefixes to be stripped when converting XML names.
     *
     * @return prefixes
     */
    public String[] getStripPrefixes() {
        return m_stripPrefixes;
    }

    /**
     * Set the prefixes to be stripped when converting XML names.
     *
     * @param prefixes
     */
    public void setStripPrefixes(String[] prefixes) {
        m_stripPrefixes = prefixes;
    }

    /**
     * Get the suffixes to be stripped when converting XML names.
     *
     * @return suffixes
     */
    public String[] getStripSuffixes() {
        return m_stripSuffixes;
    }

    /**
     * Set the suffixes to be stripped when converting XML names.
     *
     * @param suffixes
     */
    public void setStripSuffixes(String[] suffixes) {
        m_stripSuffixes = suffixes;
    }

    /**
     * Trim specified prefixes and/or suffixes from an XML name.
     *
     * @param xname XML name
     * @return trimmed name, with specified prefixes and/or suffixes removed
     */
    public String trimXName(String xname) {
        
        // first trim off a prefix on name
        for (int i = 0; i < m_stripPrefixes.length; i++) {
            String prefix = m_stripPrefixes[i];
            if (xname.startsWith(prefix) && xname.length() > prefix.length()) {
                xname = xname.substring(prefix.length());
                break;
            }
        }
        
        // next trim off a suffix on name
        for (int i = 0; i < m_stripSuffixes.length; i++) {
            String suffix = m_stripSuffixes[i];
            if (xname.endsWith(suffix) && xname.length() > suffix.length()) {
                xname = xname.substring(0, xname.length() - suffix.length());
                break;
            }
        }
        return xname;
    }
    
    /**
     * Split an XML name into words. This splits first on the basis of separator characters ('.', '-', and '_') in the
     * name, and secondly based on case (an uppercase character immediately followed by one or more lowercase characters
     * is considered a word, and multiple uppercase characters not followed immediately by a lowercase character are
     * also considered a word). Characters which are not valid as parts of identifiers in Java are dropped from the XML
     * name before it is split, and words starting with initial uppercase characters have the upper case dropped for
     * consistency. Note that this method is not threadsafe.
     * 
     * @param name
     * @return array of words
     */
    public String[] splitXMLWords(String name) {
        
        // start by finding a valid start character
        int offset = 0;
        while (!Character.isJavaIdentifierStart(name.charAt(offset))) {
            if (++offset >= name.length()) {
                return new String[0];
            }
        }
        
        // accumulate the list of words
        StringBuffer word = new StringBuffer();
        m_wordList.clear();
        boolean lastupper = false;
        while (offset < name.length()) {
            
            // check next character of name
            char chr = name.charAt(offset++);
            if (chr == '-' || chr == '.' || chr == '_') {
                
                // force split at each splitting character
                if (word.length() > 0) {
                    m_wordList.add(word.toString());
                    word.setLength(0);
                }
                
            } else if (Character.isJavaIdentifierPart(chr)) {
                
                // check case of valid identifier part character
                if (Character.isUpperCase(chr)) {
                    if (!lastupper && word.length() > 0) {
                        
                        // upper after lower, split before upper
                        m_wordList.add(word.toString());
                        word.setLength(0);
                        
                    }
                    lastupper = true;
                    
                } else {
                    if (lastupper) {
                        if (word.length() > 1) {
                            
                            // multiple uppers followed by lower, split before last
                            int split = word.length() - 1;
                            m_wordList.add(word.substring(0, split));
                            char start = Character.toLowerCase(word.charAt(split));
                            word.setLength(0);
                            word.append(start);
                            
                        } else if (word.length() > 0) {
                            
                            // single upper followed by lower, convert upper to lower
                            word.setCharAt(0, Character.toLowerCase(word.charAt(0)));
                            
                        }
                    }
                    lastupper = false;
                }
                word.append(chr);
            }
        }
        if (word.length() > 0) {
            m_wordList.add(word.toString());
        }
        
        // return array of words
        return (String[])m_wordList.toArray(new String[m_wordList.size()]);
    }
    
    /**
     * Check if a name needs to be converted from XML form due to invalid characters or embedded underscores.
     * 
     * @param name
     * @return true if invalid name, false if valid
     */
    protected static boolean isConversionNeeded(String name) {
        int length = name.length();
        if (length > 0) {
            if (Character.isJavaIdentifierStart(name.charAt(0))) {
                for (int i = 1; i < length; i++) {
                    char chr = name.charAt(i);
                    if (chr == '_' || !Character.isJavaIdentifierPart(chr)) {
                        return true;
                    }
                }
                return false;
            }
        }
        return true;
    }
    
    /**
     * Convert a base name to a legal Java class name. This implementation avoids using class names from the default
     * import classes in java.lang, to avoid conflicts with the import handling in generated code.
     * 
     * @param name base name
     * @return converted name
     */
    public String toJavaClassName(String name) {
        
        // make sure initial letter is uppercase
        int index = 0;
        char chr;
        while ((chr = name.charAt(index)) == '_') {
            index++;
        }
        if (Character.isLowerCase(chr)) {
            name = name.substring(0, index) + Character.toUpperCase(chr) + name.substring(index+1);
        }
        
        // prevent conflicts with default import classes
        while (NameUtils.isDefaultImport(name)) {
            name = "_" + name;
        }
        return name;
    }
    
    /**
     * Convert an XML name to a Java value base name. The base name is in normalized camelcase form with leading lower
     * case (unless the first word of the name is all uppercase).
     * 
     * @param xname XML name
     * @return converted name
     */
    public String toBaseName(String xname) {
        String name = trimXName(xname);
        if (isConversionNeeded(name) || Character.isUpperCase(name.charAt(0))) {
            
            // split trimmed name into a series of words
            String[] words = splitXMLWords(name);
            if (words.length == 0) {
                return "x";
            }
            
            // use single underscore for empty result
            if (words.length == 0) {
                words = new String[] { "_" };
            }
            
            // form name by concatenating words with configured style
            StringBuffer buff = new StringBuffer();
            for (int i = 0; i < words.length; i++) {
                String word = words[i];
                if (i > 0 && m_underscore) {
                    buff.append('_');
                }
                if ((i == 0 && m_upperInitial) || (i > 0 && m_camelCase)) {
                    buff.append(Character.toUpperCase(word.charAt(0)));
                    if (word.length() > 1) {
                        buff.append(word.substring(1, word.length()));
                    }
                } else {
                    buff.append(word);
                }
            }
            name = buff.toString();
        }
        return name;
    }
    
    /**
     * Convert text to constant name. The returned name is guaranteed not to match a Java keyword.
     *
     * @param text raw text to be converted
     * @return constant name
     */
    public String toConstantName(String text) {
        
        // insert underscores to separate words of name, and convert invalid characters
        StringBuffer buff = new StringBuffer(text.length());
        boolean lastup = false;
        boolean multup = false;
        char lastchar = 0;
        for (int index = 0; index < text.length(); index++) {
            char chr = text.charAt(index);
            if (index == 0 && !Character.isJavaIdentifierStart(chr)) {
                buff.append('_');
            }
            if (Character.isJavaIdentifierPart(chr)) {
                if (lastup) {
                    if (Character.isUpperCase(chr)) {
                        multup = true;
                    } else {
                        lastup = false;
                        if (chr == '_') {
                            multup = false;
                        } else if (multup) {
                            buff.insert(buff.length()-1, '_');
                            multup = false;
                        }
                    }
                } else if (Character.isUpperCase(chr)) {
                    if (index > 0 && lastchar != '_') {
                        buff.append('_');
                    }
                    lastup = true;
                    multup = false;
                }
                lastchar = Character.toUpperCase(chr);
                buff.append(lastchar);
            }
        }
        if (buff.length() == 0) {
            buff.append('_');
        }
        String name = buff.toString();
        if (NameUtils.isReserved(name)) {
            name = "_" + name;
        }
        return name;
    }

    /**
     * Build a field name using supplied prefix and/or suffix.
     *
     * @param base normalized camelcase base name
     * @param prefix text to be added at start of name
     * @param suffix text to be added at end of name
     * @return field name
     */
    private String buildFieldName(String base, String prefix, String suffix) {
        
        // check for any change needed
        String name = base;
        int added = prefix.length() + suffix.length();
        boolean toupper = m_upperInitial && !Character.isUpperCase(name.charAt(0));
        if (added == 0) {
            
            // not adding prefix or suffix, check for case conversion
            if (toupper) {
                
                // convert first character to uppercase
                StringBuffer buff = new StringBuffer(name);
                buff.setCharAt(0, Character.toUpperCase(buff.charAt(0)));
                name = buff.toString();
                
            }
            
        } else {
            
            // append prefix and/or suffix, with case conversion if needed
            StringBuffer buff = new StringBuffer(name.length() + added);
            buff.append(prefix);
            int offset = buff.length();
            buff.append(name);
            if (toupper) {
                buff.setCharAt(offset, Character.toUpperCase(name.charAt(0)));
            }
            buff.append(suffix);
            name = buff.toString();
            
        }
        
        // make sure the result doesn't match a reserved name
        if (NameUtils.isReserved(name)) {
            name = "_" + name;
        }
        return name;
    }
    
    /**
     * Convert base name to normal field name. The returned name is guaranteed not to match a Java keyword.
     * 
     * @param base normalized camelcase base name
     * @return field name
     */
    public String toFieldName(String base) {
        String prefix = m_fieldPrefix;
        String suffix = m_fieldSuffix;
        return buildFieldName(base, prefix, suffix);
    }
    
    /**
     * Convert base name to static field name. The returned name is guaranteed not to match a Java keyword.
     * 
     * @param base normalized camelcase base name
     * @return field name
     */
    public String toStaticFieldName(String base) {
        String prefix = m_staticPrefix;
        String suffix = m_staticSuffix;
        return buildFieldName(base, prefix, suffix);
    }

    /**
     * Convert base name to property name (used for all method names). The property name is always in initial-upper
     * camelcase form.
     *
     * @param base normalized camelcase base name
     * @return property name in initial-upper camelcase form
     */
    public String toPropertyName(String base) {
        if (!Character.isUpperCase(base.charAt(0))) {
            
            // convert first character to uppercase
            StringBuffer buff = new StringBuffer(base);
            buff.setCharAt(0, Character.toUpperCase(buff.charAt(0)));
            return buff.toString();
            
        } else {
            
            // no change needed, just use base name directly
            return base;
            
        }
    }
    
    /**
     * Convert property name to read access method name.
     *
     * @param prop property name in initial-upper camelcase form
     * @return read access method name
     */
    public String toReadAccessMethodName(String prop) {
        return "get" + prop;
    }
    
    /**
     * Convert property name to write access method name.
     *
     * @param prop property name in initial-upper camelcase form
     * @return write access method name
     */
    public String toWriteAccessMethodName(String prop) {
        return "set" + prop;
    }
    
    /**
     * Convert property name to write access method name.
     *
     * @param prop property name in initial-upper camelcase form
     * @return write access method name
     */
    public String toTestAccessMethodName(String prop) {
        return "is" + prop;
    }
    
    /**
     * Convert property name to if set access method name (for value in set of alternatives).
     *
     * @param prop property name in initial-upper camelcase form
     * @return if set access method name
     */
    public String toIfSetAccessMethodName(String prop) {
        return "if" + prop;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy