br.ojimarcius.commons.text.EnhancedStringBuilder.java Source code

Java tutorial

Introduction

Here is the source code for br.ojimarcius.commons.text.EnhancedStringBuilder.java

Source

/*
 * Copyright (C) 2013 Marcius da Silva da Fonseca.
 *
 * This library 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 2.1 of the License, or (at your option) any later version.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
 * MA 02110-1301 USA
 */
package br.ojimarcius.commons.text;

import br.ojimarcius.commons.util.CharSequenceUtils;
import br.ojimarcius.commons.util.LatinCharacterUtils;
import br.ojimarcius.commons.constants.Constants;
import br.ojimarcius.commons.util.ArgumentUtils;
import br.ojimarcius.commons.util.ArrayUtils;
import br.ojimarcius.commons.util.CollectionUtils;
import br.ojimarcius.commons.util.DateUtils;
import br.ojimarcius.commons.util.NumberUtils;
import br.ojimarcius.commons.util.ObjectUtils;
import java.io.Serializable;
import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Pattern;

/**
 * A StringBuilder Wrapper that adds some new useful methods to the Java's standard {@link StringBuilder}.
 * <p/>
 * Uses the Delegation design pattern, where every method is dispatched to the delegate object, witch can
 * be accessed via <tt>this.getDelegate()</tt>.
 *
 * @author Marcius da Silva da Fonseca (sf.marcius@gmail.com)
 * @version 1.0
 */
public class EnhancedStringBuilder
        implements Iterable<Character>, CharSequence, Appendable, Serializable, Comparable<CharSequence> {

    private static final long serialVersionUID = -8028376141448717457L;
    private static final String CAPITALIZE_ALL_REGEX = "[\\s\\p{P}\\p{S}]+";
    private static final String CAPITALIZE_FIRST_REGEX = "^[\\s\\p{P}\\p{S}]+|[.!?]+[\\s\\p{P}\\p{S}]*";
    private static final String TOKENIZER_REGEX = "(?<!(^|[\\p{Lu}0-9]))(?=[\\p{Lu}0-9])"
            + "|(?<!(^|[^\\p{Lu}]))(?=[0-9])" + "|(?<!(^|[^0-9]))(?=[\\p{Lu}\\p{Ll}])"
            + "|(?<!^)(?=[\\p{Lu}][\\p{Ll}])" + "|([_]+)" + "|([\\s]+)";

    /**
     * A list of conjunctions on the Brazilian Portuguese language, witch are ignored on the capitalize process.
     */
    static final Collection<String> IGNORE_LIST_PT = Arrays.asList("o", "a", "os", "as", "do", "da", "dos", "das",
            "em", "no", "na", "nos", "nas", "de");
    /**
     * A list of conjunctions on the English language, witch are ignored on the capitalize process.
     */
    static final Collection<String> IGNORE_LIST_EN = Arrays.asList("the", "of", "in", "on", "at", "as");
    /**
     * Regex pattern used to left trim the sequence as a whole (SL stands for Single Line).
     */
    static final Pattern SL_LTRIM = Pattern.compile("^[\\s]+");
    /**
     * Regex pattern used to right trim the sequence as a whole (SL stands for Single Line).
     */
    static final Pattern SL_RTRIM = Pattern.compile("[\\s]+$");
    /**
     * Regex pattern used to left and right trim the sequence as a whole (SL stands for Single Line).
     */
    static final Pattern SL_LRTRIM = Pattern.compile("^[\\s]+|[\\s]+$");
    /**
     * Regex pattern used to middle trim (process also known as space normalization) the sequence as a whole (SL stands for Single Line).
     */
    static final Pattern SL_MTRIM = Pattern.compile("(?<=\\S)(\\s+)(?=\\S)");
    /**
     * Regex pattern used to left trim the sequence, line by line (ML stands for Multi Line).
     */
    static final Pattern ML_LTRIM = Pattern.compile("^[ \\f\\t]+", Pattern.MULTILINE);
    /**
     * Regex pattern used to right trim the sequence, line by line (ML stands for Multi Line).
     */
    static final Pattern ML_RTRIM = Pattern.compile("[ \\f\\t]+$", Pattern.MULTILINE);
    /**
     * Regex pattern used to left and right trim the sequence, line by line (ML stands for Multi Line).
     */
    static final Pattern ML_LRTRIM = Pattern.compile("^[ \\f\\t]+|[ \\f\\t]+$", Pattern.MULTILINE);
    /**
     * Regex pattern used to middle trim (process also known as space normalization) the sequence, line by line (ML stands for Multi Line).
     */
    static final Pattern ML_MTRIM = Pattern.compile("(?<=\\S)([ \\f\\t]+)(?=\\S)", Pattern.MULTILINE);

    /**
     * The underlying StringBuilder, whose all know methods are delegated and the new ones takes place.
     */
    protected final StringBuilder delegate;
    /**
     * The line break mode, used by <tt>this.appendln()</tt> methods.
     */
    protected LineBreakMode lineBreakMode = LineBreakMode.SYSTEM_DETECT;

    /**
     * Default constructor.
     * <p/>
     * Creates a new, empty, underlying StringBuilder.
     */
    public EnhancedStringBuilder() {
        this.delegate = new StringBuilder();
    }

    /**
     * Constructor that defines the initial capacity.
     * <p/>
     * Creates a new, empty, underlying StringBuilder, with the given capacity.
     *
     * @param capacity The initial capacity to the underlying StringBuilder.
     */
    public EnhancedStringBuilder(final int capacity) {
        ArgumentUtils.rejectIfLessEquals(capacity, 0);
        this.delegate = new StringBuilder(capacity);
    }

    /**
     * Constructor that defines the initial content.
     * <p/>
     * Creates a new underlying StringBuilder, with the given content.
     *
     * @param string The initial content of the underlying StringBuilder, in String format.
     */
    public EnhancedStringBuilder(final Object string) {
        this.delegate = (string != null) ? new StringBuilder(format(string)) : new StringBuilder();
    }

    /**
     * Constructor that defines the wrapped StringBuilder.
     * <p/>
     * Sets the given StringBuilder as the underlying.
     * <p/>
     * The given StringBuilder will be modified by calls to this wrapper.
     *
     * @param delegate The StringBuilder to be used as wrapped object.
     */
    public EnhancedStringBuilder(final StringBuilder delegate) {
        this.delegate = (delegate != null) ? delegate : new StringBuilder();
    }

    /**
     * Returns the underlying StringBuilder.
     *
     * @return The underlying, wrapped StringBuilder.
     */
    public StringBuilder getDelegate() {
        return delegate;
    }

    public String getLineBreak() {
        return lineBreakMode.getLineBreakString();
    }

    public EnhancedStringBuilder setLineBreakMode(final LineBreakMode lineBreakMode) {
        this.lineBreakMode = lineBreakMode;
        return this;
    }

    public EnhancedStringBuilder trim() {
        return trim(false);
    }

    public EnhancedStringBuilder ltrim() {
        return ltrim(false);
    }

    public EnhancedStringBuilder rtrim() {
        return rtrim(false);
    }

    public EnhancedStringBuilder mtrim() {
        return mtrim(false);
    }

    public EnhancedStringBuilder trim(final boolean multiLine) {
        return (this.isEmpty()) ? this : deletePattern(multiLine ? ML_LRTRIM : SL_LRTRIM);
    }

    public EnhancedStringBuilder ltrim(final boolean multiLine) {
        return (this.isEmpty()) ? this : deletePattern(multiLine ? ML_LTRIM : SL_LTRIM);
    }

    public EnhancedStringBuilder rtrim(final boolean multiLine) {
        return (this.isEmpty()) ? this : deletePattern(multiLine ? ML_RTRIM : SL_RTRIM);
    }

    public EnhancedStringBuilder mtrim(final boolean multiLine) {
        return (this.isEmpty()) ? this : replacePattern(multiLine ? ML_MTRIM : SL_MTRIM, " ");
    }

    public EnhancedStringBuilder clear() {
        delegate.delete(0, length());
        return this;
    }

    public boolean isEmpty() {
        return CharSequenceUtils.isEmpty(delegate);
    }

    public boolean isNotEmpty() {
        return CharSequenceUtils.isNotEmpty(delegate);
    }

    public boolean isBlank() {
        return CharSequenceUtils.isBlank(delegate);
    }

    public boolean isNotBlank() {
        return CharSequenceUtils.isNotBlank(delegate);
    }

    public boolean isSmallerThan(final CharSequence sequence) {
        return CharSequenceUtils.isSmaller(delegate, sequence);
    }

    public boolean isBiggerThan(final CharSequence sequence) {
        return CharSequenceUtils.isBigger(delegate, sequence);
    }

    public boolean isSameOrSmallerThan(final CharSequence sequence) {
        return CharSequenceUtils.isSameOrSmaller(delegate, sequence);
    }

    public boolean isSameOrBiggerThan(final CharSequence sequence) {
        return CharSequenceUtils.isSameOrBigger(delegate, sequence);
    }

    public boolean isSameLengthThan(final CharSequence sequence) {
        return CharSequenceUtils.isSameLength(delegate, sequence);
    }

    public boolean isLengthBetween(final int low, final int high) {
        return CharSequenceUtils.isLengthBetween(delegate, low, high);
    }

    public boolean isInsideBounds(final int index) {
        return CharSequenceUtils.isInsideBounds(delegate, index);
    }

    public boolean containsPattern(final CharSequence regex) {
        return CharSequenceUtils.containsPattern(regex, delegate);
    }

    public boolean containsPattern(final Pattern pattern) {
        return CharSequenceUtils.containsPattern(pattern, delegate);
    }

    public boolean hasAccents() {
        return CharSequenceUtils.hasAccents(delegate);
    }

    public EnhancedStringBuilder append(final CharSequence sequence) {
        return append((Object) sequence);
    }

    public EnhancedStringBuilder append(final CharSequence sequence, final int start, final int end) {
        return append((Object) sequence, start, end);
    }

    public EnhancedStringBuilder append(final char c) {
        return append((Object) c);
    }

    public EnhancedStringBuilder append(final Object... sequences) {
        if (sequences != null) {
            for (Object sequence : sequences) {
                if (sequence != null) {
                    delegate.append(format(sequence));
                }
            }
        }
        return this;
    }

    public EnhancedStringBuilder append(final Object sequence, final int start, final int end) {
        if (sequence != null) {
            delegate.append(format(sequence), start, end);
        }
        return this;
    }

    public EnhancedStringBuilder append(final char[] chars, final int offset, final int len) {
        if (chars != null && chars.length > 0) {
            delegate.append(chars, offset, len);
        }
        return this;
    }

    public EnhancedStringBuilder append(final boolean condition, final Object sequenceIfTrue,
            final Object sequenceIfFalse) {
        return condition ? this.append(sequenceIfTrue) : this.append(sequenceIfFalse);
    }

    public EnhancedStringBuilder appendIfTrue(final boolean condition, final Object... sequences) {
        return condition ? this.append(sequences) : this;
    }

    public EnhancedStringBuilder appendIfFalse(final boolean condition, final Object... sequences) {
        return !condition ? this.append(sequences) : this;
    }

    public EnhancedStringBuilder appendIfNotBlank(final CharSequence sequence) {
        return (CharSequenceUtils.isBlankOrNull(sequence)) ? this : this.append(sequence);
    }

    public EnhancedStringBuilder appendIfDontEndWith(final CharSequence sequence) {
        if (CharSequenceUtils.isNotEmpty(sequence)) {
            if (isSmallerThan(sequence) || !endsWith(sequence)) {
                this.append(sequence);
            }
        }
        return this;
    }

    public EnhancedStringBuilder appendln(final Object... sequences) {
        return this.append(sequences).append(lineBreakMode.getLineBreakString());
    }

    public EnhancedStringBuilder appendln(final Object sequence, final int start, final int end) {
        return this.append(sequence, start, end).appendln();

    }

    public EnhancedStringBuilder appendln(final char[] chars, final int offset, final int len) {
        return this.append(chars, offset, len).appendln();
    }

    public EnhancedStringBuilder appendln(final boolean condition, final Object sequenceIfTrue,
            final Object sequenceIfFalse) {
        return condition ? this.appendln(sequenceIfTrue) : this.appendln(sequenceIfFalse);
    }

    public EnhancedStringBuilder appendlnIfTrue(final boolean condition, final Object... sequences) {
        return condition ? this.appendln(sequences) : this;
    }

    public EnhancedStringBuilder appendlnIfFalse(final boolean condition, final Object... sequences) {
        return !condition ? this.appendln(sequences) : this;
    }

    public EnhancedStringBuilder appendlnIfNotBlank(final CharSequence sequence) {
        return (CharSequenceUtils.isBlankOrNull(sequence)) ? this : this.appendln(sequence);
    }

    public EnhancedStringBuilder appendlnIfDontEndWith(final CharSequence sequence) {
        if (CharSequenceUtils.isNotEmpty(sequence)) {
            if (isSmallerThan(sequence) || !endsWith(sequence)) {
                this.appendln(sequence);
            }
        }
        return this;
    }

    public EnhancedStringBuilder appendCodePoint(final int codePoint) {
        delegate.appendCodePoint(codePoint);
        return this;
    }

    public EnhancedStringBuilder putLeading(final Object leading, final int n) {
        if (n <= 0 || CharSequenceUtils.isEmptyOrNull(format(leading))) {
            return this;
        }
        final StringBuilder leadingSeq = new StringBuilder(n);
        for (int i = 0; i < n; i++) {
            leadingSeq.append(leading);
        }
        return this.insert(0, leadingSeq);
    }

    public EnhancedStringBuilder putTrailing(final Object trailing, final int n) {
        if (n <= 0 || CharSequenceUtils.isEmptyOrNull(format(trailing))) {
            return this;
        }
        final StringBuilder trailingSeq = new StringBuilder(n);
        for (int i = 0; i < n; i++) {
            trailingSeq.append(trailing);
        }
        return this.append(trailingSeq);
    }

    public EnhancedStringBuilder insert(final int offset, final Object obj) {
        if (obj != null) {
            delegate.insert(offset, obj);
        }
        return this;
    }

    public EnhancedStringBuilder insert(final int offset, final Object sequence, final int start, final int end) {
        if (sequence != null) {
            delegate.insert(offset, format(sequence), start, end);
        }
        return this;
    }

    public EnhancedStringBuilder insert(final int index, final char[] chars, final int offset, final int len) {
        if (chars != null && chars.length > 0) {
            delegate.insert(index, chars, offset, len);
        }
        return this;
    }

    public EnhancedStringBuilder removeAccents() {
        for (int i = 0; i < delegate.length(); i++) {
            delegate.setCharAt(i, LatinCharacterUtils.undecorateLetter(delegate.charAt(i)));
        }
        return this;
    }

    public EnhancedStringBuilder delete(final int start, final int end) {
        delegate.delete(start, end);
        return this;
    }

    public EnhancedStringBuilder delete(final CharSequence toDelete) {
        return delete(toDelete, Constants.CASE_SENSITIVE);
    }

    public EnhancedStringBuilder delete(final CharSequence toDelete, final Boolean caseSensitive) {
        if (CharSequenceUtils.isEmptyOrNull(toDelete) || this.isEmpty()) {
            return this;
        }
        final List<MatchEntry> occurrences = new ArrayList<MatchEntry>(find(toDelete, caseSensitive));
        Collections.reverse(occurrences);
        for (MatchEntry occurrence : occurrences) {
            delegate.delete(occurrence.getStart(), occurrence.getEnd());
        }
        return this;
    }

    public EnhancedStringBuilder deleteCharAt(final int index) {
        delegate.deleteCharAt(index);
        return this;
    }

    public EnhancedStringBuilder deletePattern(final CharSequence regex) {
        return (CharSequenceUtils.isEmptyOrNull(regex) || this.isEmpty()) ? this
                : deletePattern(Pattern.compile(regex.toString()));
    }

    public EnhancedStringBuilder deletePattern(final Pattern pattern) {
        if (pattern == null || this.isEmpty()) {
            return this;
        }
        final List<MatchEntry> occurrences = new ArrayList<MatchEntry>(findPattern(pattern));
        Collections.reverse(occurrences);
        for (MatchEntry occurrence : occurrences) {
            delegate.delete(occurrence.getStart(), occurrence.getEnd());
        }
        return this;
    }

    public EnhancedStringBuilder replace(final int start, final int end, final Object string) {
        delegate.replace(start, end, format(string).toString());
        return this;
    }

    public EnhancedStringBuilder replace(final MatchEntry matchEntry, final Object replacement) {
        ArgumentUtils.rejectIfNull(matchEntry);
        if (!matchEntry.isValidFor(delegate)) {
            throw new IllegalArgumentException("MatchEntry not valid.");
        }
        delegate.replace(matchEntry.getStart(), matchEntry.getEnd(), replacement.toString());
        return this;
    }

    public EnhancedStringBuilder replace(final CharSequence paramName, final Object replacement,
            final String startDelimiter, final String endDelimiter) {
        if (CharSequenceUtils.isEmptyOrNull(paramName) || this.isEmpty()) {
            return this;
        }
        // TODO : cross-reference resolving, when has delimiters
        // merge delimiters on the param key, if applicable
        int keyLen = paramName.length() + length(startDelimiter, endDelimiter);
        final String key = new StringBuilder(keyLen).append(startDelimiter).append(paramName).append(endDelimiter)
                .toString();
        // get the param value as a String, to do the replacement
        final CharSequence value = format(replacement);
        int idx = indexOf(key);
        while (idx >= 0) {
            replace(idx, idx + key.length(), value);
            idx = indexOf(key, idx + value.length());
        }
        return this;
    }

    public EnhancedStringBuilder replace(final String startDelimiter, final String endDelimiter,
            final Map<? extends CharSequence, ?> params) {
        if (this.isEmpty() || CollectionUtils.isEmptyOrNull(params)) {
            return this;
        }
        for (Map.Entry<? extends CharSequence, ?> entry : params.entrySet()) {
            replace(entry.getKey(), format(entry.getValue()), startDelimiter, endDelimiter);
        }
        return this;
    }

    public EnhancedStringBuilder replace(final String startDelimiter, final String endDelimiter,
            final Object[] params) {
        if (this.isEmpty() || ArrayUtils.isEmptyOrNull(params)) {
            return this;
        }
        for (int i = 0; i < params.length; i++) {
            replace(Integer.toString(i), format(params[i]), startDelimiter, endDelimiter);
        }
        return this;
    }

    public EnhancedStringBuilder replacePlain(final CharSequence original, final Object replacement) {
        return replace(original, replacement, null, null);
    }

    public EnhancedStringBuilder replacePlain(final Map<? extends CharSequence, ?> params) {
        return replace(null, null, params);
    }

    public EnhancedStringBuilder replaceParam(final CharSequence paramName, final Object value) {
        final Map<CharSequence, Object> params = new LinkedHashMap<CharSequence, Object>();
        params.put(paramName, value);
        return this.replaceParams(params);
    }

    public EnhancedStringBuilder replaceParams(final Map<? extends CharSequence, ?> params) {
        return replace(Constants.DEFAULT_PARAM_START, Constants.DEFAULT_PARAM_END, params);
    }

    public EnhancedStringBuilder replaceParams(final Object[] params) {
        return replace(Constants.DEFAULT_PARAM_START, Constants.DEFAULT_PARAM_END, params);
    }

    public EnhancedStringBuilder replacePattern(final CharSequence regex, final Object replacement) {
        return (CharSequenceUtils.isEmptyOrNull(regex) || this.isEmpty()) ? this
                : replacePattern(Pattern.compile(regex.toString()), format(replacement));
    }

    public EnhancedStringBuilder replacePattern(final Pattern pattern, final Object replacement) {
        if (pattern == null || this.isEmpty()) {
            return this;
        }
        final List<MatchEntry> occurrences = new ArrayList<MatchEntry>(findPattern(pattern));
        Collections.reverse(occurrences);
        for (MatchEntry occurrence : occurrences) {
            replace(occurrence.getStart(), occurrence.getEnd(), format(replacement));
        }
        return this;
    }

    public EnhancedStringBuilder replacePattern(final Map<CharSequence, ?> patterns) {
        if (this.isEmpty() || CollectionUtils.isEmptyOrNull(patterns)) {
            return this;
        }
        for (Map.Entry<CharSequence, ?> entry : patterns.entrySet()) {
            replacePattern(entry.getKey(), format(entry.getValue()));
        }
        return this;
    }

    public EnhancedStringBuilder reverse() {
        delegate.reverse();
        return this;
    }

    public List<String> split(final CharSequence regex) {
        return CharSequenceUtils.split(delegate, regex);
    }

    public List<String> split(final Pattern pattern) {
        return CharSequenceUtils.split(delegate, pattern);
    }

    public List<String> split(final CharSequence regex, final boolean ignoreBlank) {
        return CharSequenceUtils.split(delegate, regex, ignoreBlank);
    }

    public List<String> split(final Pattern pattern, final boolean ignoreBlank) {
        return CharSequenceUtils.split(delegate, pattern, ignoreBlank);
    }

    public List<MatchEntry> find(final CharSequence toSearch) {
        return CharSequenceUtils.find(toSearch, delegate);
    }

    public List<MatchEntry> find(final CharSequence toSearch, final Boolean caseSensitive) {
        return CharSequenceUtils.find(toSearch, delegate, caseSensitive);
    }

    @SuppressWarnings("unchecked")
    public List<MatchEntry> findPattern(final CharSequence regex) {
        return CharSequenceUtils.findPattern(regex, delegate);
    }

    @SuppressWarnings("unchecked")
    public List<MatchEntry> findPattern(final Pattern pattern) {
        return CharSequenceUtils.findPattern(pattern, delegate);
    }

    public Set<String> listParams() {
        return CharSequenceUtils.listParams(delegate);
    }

    @SuppressWarnings("unchecked")
    public Set<String> listParams(final String startDelimiter, final String endDelimiter) {
        return CharSequenceUtils.listParams(delegate, startDelimiter, endDelimiter);
    }

    public EnhancedStringBuilder toUpperCase() {
        return toUpperCase(0, length());
    }

    public EnhancedStringBuilder toUpperCase(final int start, final int end) {
        ArgumentUtils.rejectIfLessThan(start, 0);
        ArgumentUtils.rejectIfGreaterEqual(start, end);
        for (int i = start; i < length() && i < end; i++) {
            setCharAt(i, Character.toUpperCase(charAt(i)));
        }
        return this;
    }

    public EnhancedStringBuilder toLowerCase() {
        return toLowerCase(0, length());
    }

    public EnhancedStringBuilder toLowerCase(final int start, final int end) {
        ArgumentUtils.rejectIfLessThan(start, 0);
        ArgumentUtils.rejectIfGreaterEqual(start, end);
        for (int i = start; i < length() && i < end; i++) {
            setCharAt(i, Character.toLowerCase(charAt(i)));
        }
        return this;
    }

    public EnhancedStringBuilder toCamelCase() {
        return toCamelCase(true, true);
    }

    public EnhancedStringBuilder toCamelCase(final boolean removeAccents, final boolean capitalizeFirst) {
        if (CharSequenceUtils.isNotEmpty(delegate)) {
            // copy to a tmp
            final EnhancedStringBuilder tmp = new EnhancedStringBuilder(this);
            // find delimiters
            final List<MatchEntry> occurrences = tmp.findPattern(TOKENIZER_REGEX);
            if (CollectionUtils.isEmptyOrNull(occurrences)) {
                // single word. no tokenizing...
                if (removeAccents) {
                    removeAccents();
                }
                if (capitalizeFirst) {
                    capitalizeFirst();
                }
            } else {
                // clear this
                clear();
                if (removeAccents) {
                    tmp.removeAccents();
                }
                int start = 0;
                for (final MatchEntry occurrence : occurrences) {
                    // get token
                    final EnhancedStringBuilder token = tmp.subSequence(start, occurrence.getStart());
                    appendCamelcaseToken(token, capitalizeFirst);
                    // update offset
                    start = occurrence.getEnd();
                }
                final MatchEntry last = CollectionUtils.getLast(occurrences);
                final EnhancedStringBuilder token = tmp.subSequence(last.getEnd());
                appendCamelcaseToken(token, capitalizeFirst);
            }
        }
        return this;
    }

    private void appendCamelcaseToken(final EnhancedStringBuilder token, final boolean capitalizeFirst) {
        if (CharSequenceUtils.isNotEmpty(token)) {
            // uppercase first
            if (!this.isEmpty() || capitalizeFirst) {
                token.capitalizeFirst();
            }
            // append token
            this.append(token);
        }
    }

    public EnhancedStringBuilder toUnderscore() {
        return toUnderscore(true, true, false);
    }

    public EnhancedStringBuilder toUnderscore(final boolean removeAccents, final boolean upperCased,
            final boolean separateNumbers) {
        if (CharSequenceUtils.isNotEmpty(delegate)) {
            // copy to a tmp
            final EnhancedStringBuilder tmp = new EnhancedStringBuilder(this);
            // find delimiters
            final List<MatchEntry> occurrences = tmp.findPattern(TOKENIZER_REGEX);
            if (CollectionUtils.isEmptyOrNull(occurrences)) {
                // single word. no tokenizing...
                if (removeAccents) {
                    removeAccents();
                }
                if (upperCased) {
                    toUpperCase();
                }
            } else {
                // clear this
                clear();
                if (removeAccents) {
                    tmp.removeAccents();
                }
                // find delimiters
                int start = 0;
                for (final MatchEntry occurrence : occurrences) {
                    // get token
                    final EnhancedStringBuilder token = tmp.subSequence(start, occurrence.getStart());
                    appendUnderscoreToken(token, upperCased, separateNumbers);
                    start = occurrence.getEnd();
                }
                final MatchEntry last = CollectionUtils.getLast(occurrences);
                final EnhancedStringBuilder token = tmp.subSequence(last.getEnd());
                appendUnderscoreToken(token, upperCased, separateNumbers);
            }
        }
        return this;
    }

    private void appendUnderscoreToken(final EnhancedStringBuilder token, final boolean upper,
            final boolean separateNumbers) {
        if (CharSequenceUtils.isNotEmpty(token)) {
            if (this.isNotEmpty() && (!CharSequenceUtils.isNumber(token) || separateNumbers)) {
                this.append('_');
            }
            this.append(upper, token.toUpperCase(), token);
        }
    }

    public EnhancedStringBuilder capitalizeFirst() {
        return capitalize(CAPITALIZE_FIRST_REGEX);
    }

    public EnhancedStringBuilder capitalizeFirst(final CharSequence delimiterRegex) {
        return capitalize(delimiterRegex);
    }

    public EnhancedStringBuilder capitalizeAll() {
        return capitalize(CAPITALIZE_ALL_REGEX);
    }

    public EnhancedStringBuilder capitalizeAll(final CharSequence delimiterRegex) {
        return capitalize(delimiterRegex);
    }

    private EnhancedStringBuilder capitalize(final CharSequence delimiterRegex) {
        if (CharSequenceUtils.isNotEmpty(delegate)) {
            // copy to a tmp
            final EnhancedStringBuilder tmp = new EnhancedStringBuilder(this);
            // find delimiters
            final List<MatchEntry> occurrences = tmp.findPattern(
                    CharSequenceUtils.isNotEmpty(delimiterRegex) ? delimiterRegex : CAPITALIZE_ALL_REGEX);
            if (CollectionUtils.isEmptyOrNull(occurrences)) {
                // single word. no tokenizing...
                return toUpperCase(0, 1);
            } else {
                // find delimiters
                final MatchEntry first = CollectionUtils.getFirst(occurrences);
                if (first.getStart() > 0) {
                    // doesnt starts with delimiter
                    toUpperCase(0, 1);
                }
                for (final MatchEntry occurrence : occurrences) {
                    int i = occurrence.getEnd();
                    toUpperCase(i, i + 1);
                }
            }
        }
        return this;
    }

    public int indexOf(final CharSequence toSearch) {
        return CharSequenceUtils.indexOf(toSearch, delegate);
    }

    public int indexOf(final CharSequence toSearch, final boolean caseSensitive) {
        return CharSequenceUtils.indexOf(toSearch, delegate, caseSensitive);
    }

    public int indexOf(final CharSequence toSearch, final int fromIndex) {
        return CharSequenceUtils.indexOf(toSearch, fromIndex, delegate);
    }

    public int indexOf(final CharSequence toSearch, final int fromIndex, final boolean caseSensitive) {
        return CharSequenceUtils.indexOf(toSearch, fromIndex, delegate, caseSensitive);
    }

    public int lastIndexOf(final CharSequence toSearch) {
        return CharSequenceUtils.lastIndexOf(toSearch, delegate);
    }

    public int lastIndexOf(final CharSequence toSearch, final boolean caseSensitive) {
        return CharSequenceUtils.lastIndexOf(toSearch, delegate, caseSensitive);
    }

    public int lastIndexOf(final CharSequence toSearch, final int fromIndex) {
        return CharSequenceUtils.lastIndexOf(toSearch, fromIndex, delegate);
    }

    public int lastIndexOf(final CharSequence toSearch, final int fromIndex, final boolean caseSensitive) {
        return CharSequenceUtils.lastIndexOf(toSearch, fromIndex, delegate, caseSensitive);
    }

    public int indexOf(final char toSearch) {
        return CharSequenceUtils.indexOf(toSearch, delegate);
    }

    public int indexOf(final char toSearch, final boolean caseSensitive) {
        return CharSequenceUtils.indexOf(toSearch, delegate, caseSensitive);
    }

    public int indexOf(final char toSearch, final int fromIndex) {
        return CharSequenceUtils.indexOf(toSearch, fromIndex, delegate);
    }

    public int indexOf(final char toSearch, final int fromIndex, final boolean caseSensitive) {
        return CharSequenceUtils.indexOf(toSearch, fromIndex, delegate, caseSensitive);
    }

    public int lastIndexOf(final char toSearch) {
        return CharSequenceUtils.lastIndexOf(toSearch, delegate);
    }

    public int lastIndexOf(final char toSearch, final boolean caseSensitive) {
        return CharSequenceUtils.lastIndexOf(toSearch, delegate, caseSensitive);
    }

    public int lastIndexOf(final char toSearch, final int fromIndex) {
        return CharSequenceUtils.lastIndexOf(toSearch, fromIndex, delegate);
    }

    public int lastIndexOf(final char toSearch, final int fromIndex, final boolean caseSensitive) {
        return CharSequenceUtils.lastIndexOf(toSearch, fromIndex, delegate, caseSensitive);
    }

    public int indexOfLastChar() {
        return CharSequenceUtils.indexOfLastChar(delegate);
    }

    public int countOccurrencesOf(final CharSequence toCount) {
        return CharSequenceUtils.countOccurrencesOf(toCount, delegate);
    }

    public int countOccurrencesOf(final CharSequence toCount, final int fromIndex) {
        return CharSequenceUtils.countOccurrencesOf(toCount, fromIndex, delegate);
    }

    public int countOccurrencesOf(final CharSequence toCount, final int fromIndex, final Boolean caseSensitive) {
        return CharSequenceUtils.countOccurrencesOf(toCount, fromIndex, delegate, caseSensitive);
    }

    public int countOccurrencesOf(final char toCount) {
        return CharSequenceUtils.countOccurrencesOf(toCount, delegate);
    }

    public int countOccurrencesOf(final char toCount, final int fromIndex) {
        return CharSequenceUtils.countOccurrencesOf(toCount, fromIndex, delegate);
    }

    public int countOccurrencesOf(final char toCount, final int fromIndex, final Boolean caseSensitive) {
        return CharSequenceUtils.countOccurrencesOf(toCount, fromIndex, delegate, caseSensitive);
    }

    public Map<Character, Integer> countCharOccurrences() {
        return CharSequenceUtils.getCharCount(delegate);
    }

    @SuppressWarnings("unchecked")
    public Map<Character, Integer> countCharOccurrences(final int fromIndex) {
        return CharSequenceUtils.getCharCount(fromIndex, delegate);
    }

    public int capacity() {
        return delegate.capacity();
    }

    public void ensureCapacity(final int minimumCapacity) {
        delegate.ensureCapacity(minimumCapacity);
    }

    public void trimToSize() {
        delegate.trimToSize();
    }

    public void setLength(final int newLength) {
        delegate.setLength(newLength);
    }

    public int codePointAt(final int index) {
        return delegate.codePointAt(index);
    }

    public int codePointBefore(final int index) {
        return delegate.codePointBefore(index);
    }

    public int codePointCount(final int start, final int end) {
        return delegate.codePointCount(start, end);
    }

    public int offsetByCodePoints(final int index, final int codePointOffset) {
        return delegate.offsetByCodePoints(index, codePointOffset);
    }

    public void getChars(final int start, final int end, final char[] dest, final int destStart) {
        delegate.getChars(start, end, dest, destStart);
    }

    public void setCharAt(final int index, final char c) {
        delegate.setCharAt(index, c);
    }

    public EnhancedStringBuilder subSequence(final int start) {
        return subSequence(start, length());
    }

    public EnhancedStringBuilder subSequence(final int start, final int end) {
        int len = end - start;
        if (start < 0) {
            throw new StringIndexOutOfBoundsException(start);
        } else if (end > delegate.length()) {
            throw new StringIndexOutOfBoundsException(end);
        } else if (len < 0) {
            throw new StringIndexOutOfBoundsException(len);
        }
        final EnhancedStringBuilder builder = new EnhancedStringBuilder(len + 100);
        for (int i = start; i < end; i++) {
            builder.append(delegate.charAt(i));
        }
        return builder;
    }

    public String substring(final int start) {
        return delegate.substring(start);
    }

    public String substring(final int start, final int end) {
        return delegate.substring(start, end);
    }

    public String substring(final MatchEntry matchEntry) {
        return delegate.substring(matchEntry.getStart(), matchEntry.getEnd());
    }

    public String substringBeforeFirst(final CharSequence toSearch) {
        return substringBeforeFirst(toSearch, true);
    }

    public String substringBeforeFirst(final CharSequence toSearch, final Boolean caseSensitive) {
        final int idx = indexOf(toSearch, caseSensitive);
        return (idx < 0) ? null : substring(0, idx);
    }

    public String substringBeforeLast(final CharSequence toSearch) {
        return substringBeforeLast(toSearch, true);
    }

    public String substringBeforeLast(final CharSequence toSearch, final Boolean caseSensitive) {
        final int idx = lastIndexOf(toSearch, caseSensitive);
        return (idx < 0) ? null : substring(0, idx);
    }

    public String substringAfterFirst(final CharSequence toSearch) {
        return substringAfterFirst(toSearch, true);
    }

    public String substringAfterFirst(final CharSequence toSearch, final Boolean caseSensitive) {
        final int idx = indexOf(toSearch, caseSensitive);
        return (idx < 0) ? null : substring(idx + toSearch.length(), length());
    }

    public String substringAfterLast(final CharSequence toSearch) {
        return substringAfterLast(toSearch, true);
    }

    public String substringAfterLast(final CharSequence toSearch, final Boolean caseSensitive) {
        final int idx = lastIndexOf(toSearch, caseSensitive);
        return (idx < 0) ? null : substring(idx + toSearch.length(), length());
    }

    public boolean contains(final CharSequence s) {
        return indexOf(s) >= 0;
    }

    public boolean startsWith(final CharSequence preffix) {
        return startsWith(preffix, 0, delegate);
    }

    public boolean endsWith(final CharSequence suffix) {
        return startsWith(suffix, delegate.length() - suffix.length(), delegate);
    }

    public boolean matches(final CharSequence regex) {
        return matches(Pattern.compile(regex.toString(), 0));
    }

    public boolean matches(final CharSequence regex, final int flags) {
        return matches(Pattern.compile(regex.toString(), flags));
    }

    public boolean matches(final Pattern pattern) {
        if (pattern == null || CharSequenceUtils.isEmptyOrNull(delegate)) {
            return false;
        }
        return pattern.matcher(delegate).matches();
    }

    public boolean matchesAny(final CharSequence[] regexes) {
        return matchesAny(Arrays.asList(regexes), 0);
    }

    public boolean matchesAny(final CharSequence[] regexes, final int flags) {
        return matchesAny(Arrays.asList(regexes), flags);
    }

    public boolean matchesAny(final Collection<? extends CharSequence> regexes) {
        return matchesAny(regexes, 0);
    }

    public boolean matchesAny(final Collection<? extends CharSequence> regexes, final int flags) {
        for (final CharSequence regex : regexes) {
            if (matches(regex, flags)) {
                return true;
            }
        }
        return false;
    }

    public char firstChar() {
        return delegate.charAt(0);
    }

    public char lastChar() {
        return delegate.charAt(indexOfLastChar());
    }

    public int length() {
        return delegate.length();
    }

    public char charAt(final int index) {
        return delegate.charAt(index);
    }

    public Iterator<Character> iterator() {
        return new CharIterator();
    }

    public String toString() {
        return delegate.toString();
    }

    public int compareTo(final CharSequence o) {
        return compareTo(o, Constants.CASE_SENSITIVE);
    }

    public int compareTo(final CharSequence o, final Boolean caseSensitive) {
        return new CharSequenceComparator(caseSensitive, true).compare(delegate, o);
    }

    protected static int length(final CharSequence... sequences) {
        int length = 0;
        if (ArrayUtils.isNotEmpty(sequences)) {
            for (final CharSequence item : sequences) {
                if (CharSequenceUtils.isNotEmpty(item)) {
                    length += item.length();
                }
            }
        }
        return length;
    }

    protected static boolean startsWith(final CharSequence prefix, final int offset, final CharSequence sequence) {
        if (CharSequenceUtils.isEmptyOrNull(prefix) || CharSequenceUtils.isEmptyOrNull(sequence)
                || prefix.length() > sequence.length()) {
            return false;
        }
        int i = offset, j = 0;
        int pLen = prefix.length();
        while (--pLen >= 0) {
            if (sequence.charAt(i++) != prefix.charAt(j++)) {
                return false;
            }
        }
        return true;
    }

    protected static CharSequence format(final Object object) {
        final StringBuilder builder = new StringBuilder();
        format(object, builder);
        return builder;
    }

    protected static void format(final Object object, final StringBuilder builder) {
        if (object == null) {
            return;
        }
        if (ObjectUtils.isCharSequence(object)) {
            builder.append((CharSequence) object);
        } else if (ObjectUtils.isCharacter(object)) {
            builder.append((Character) object);
        } else if (ObjectUtils.isNumber(object)) {
            builder.append(NumberUtils.format((Number) object));
        } else if (ObjectUtils.isDate(object) || ObjectUtils.isCalendar(object)) {
            builder.append(DateUtils.formatDateTime(object));
        } else if (ObjectUtils.isArray(object)) {
            int size = Array.getLength(object);
            builder.ensureCapacity(builder.length() + (size));
            for (int i = 0; i < size; i++) {
                format(Array.get(object, i), builder);
            }
        } else if (ObjectUtils.isCollection(object)) {
            Collection<?> collection = (Collection) object;
            for (Object item : collection) {
                format(item, builder);
            }
        } else {
            builder.append(object);
        }
    }

    private class CharIterator implements Iterator<Character> {

        int cursor; // index of next element to return
        int lastRet = -1; // index of last element returned; -1 if no such

        public boolean hasNext() {
            return length() > 0 && cursor < length();
        }

        public Character next() {
            if (cursor >= length()) {
                throw new ArrayIndexOutOfBoundsException();
            }
            return charAt(cursor++);
        }

        public void remove() {
            if (lastRet < 0) {
                throw new IllegalStateException();
            }
            deleteCharAt(lastRet);
            cursor = lastRet;
            lastRet = -1;
        }
    }

    public enum LineBreakMode {

        /**
         * The system line break string.
         * <p/>
         * On Windows systems it will be "\r\n"; On Unix systems it will be just "\n".
         */
        SYSTEM_DETECT(org.apache.commons.io.IOUtils.LINE_SEPARATOR),
        /**
         * The Unix line break string.
         * <p/>
         * value = "\n".
         */
        UNIX(org.apache.commons.io.IOUtils.LINE_SEPARATOR_UNIX),
        /**
         * The Windows line break string.
         * <p/>
         * value = "\r\n".
         */
        WINDOWS(org.apache.commons.io.IOUtils.LINE_SEPARATOR_WINDOWS);

        private final String lineBreakString;

        /**
         * Defines the line break String.
         * <p/>
         * @param newLineString The line break String.
         */
        private LineBreakMode(final String newLineString) {
            this.lineBreakString = newLineString;
        }

        /**
         * Returns the line break String.
         * <p/>
         * @return The line break String.
         */
        public String getLineBreakString() {
            return lineBreakString;
        }

        /**
         * Returns the same as {@link #getLineBreakString()}.
         * <p/>
         * @return The line break String.
         */
        @Override
        public String toString() {
            return getLineBreakString();
        }
    }
}