SentenceFormatter.java
// Generated by delombok at Mon Jan 06 07:19:11 UTC 2025
package de.larssh.utils.text;
import static java.util.Arrays.asList;
import static java.util.Collections.emptyList;
import java.util.ArrayList;
import java.util.List;
import java.util.function.Function;
/**
* Class to format and parse a list of words.
*
* <p>
* Constants define instances of common formatters. The following is an example
* using the words {@code abc}, {@code def} and {@code ghi}.
* <table>
* <caption>Examples</caption>
* <tr>
* <th>Formatter</th>
* <th>Example</th>
* </tr>
* <tr>
* <td>{@link #LOWER_CAMEL_CASE}</td>
* <td>abcDefGhi</td>
* </tr>
* <tr>
* <td>{@link #LOWER_KEBAB_CASE}</td>
* <td>abc-def-ghi</td>
* </tr>
* <tr>
* <td>{@link #LOWER_SNAKE_CASE}</td>
* <td>abc_def_ghi</td>
* </tr>
* <tr>
* <td>{@link #LOWER_WHITE_SPACE}</td>
* <td>Abc def ghi</td>
* </tr>
* <tr>
* <td>{@link #UPPER_CAMEL_CASE}</td>
* <td>AbcDefGhi</td>
* </tr>
* <tr>
* <td>{@link #UPPER_KEBAB_CASE}</td>
* <td>ABC-DEF-GHI</td>
* </tr>
* <tr>
* <td>{@link #UPPER_SNAKE_CASE}</td>
* <td>ABC_DEF_GHI</td>
* </tr>
* <tr>
* <td>{@link #UPPER_WHITE_SPACE}</td>
* <td>Abc Def Ghi</td>
* </tr>
* </table>
*/
public class SentenceFormatter {
/**
* Word separator for kebab space
*/
private static final String SEPARATOR_KEBAB_CASE = "-";
/**
* Word separator for snake space
*/
private static final String SEPARATOR_SNAKE_CASE = "_";
/**
* Word separator for white space
*/
private static final String SEPARATOR_WHITE_SPACE = " ";
/**
* Lower Camel Case formatter
*
* <p>
* Example: abcDefGhi
*
* <p>
* Words are translated to title case, except the first words, which is
* translated to lower case.
*/
public static final SentenceFormatter LOWER_CAMEL_CASE = new SentenceFormatter(Strings::toLowerCaseNeutral, "", Strings::toTitleCaseNeutral);
/**
* Lower Kebab Case formatter
*
* <p>
* Example: abc-def-ghi
*
* <p>
* Words are translated to lower case and are separated by minus character.
*/
public static final SentenceFormatter LOWER_KEBAB_CASE = new SentenceFormatter(Strings::toLowerCaseNeutral, SEPARATOR_KEBAB_CASE, Strings::toLowerCaseNeutral);
/**
* Lower Snake Case formatter
*
* <p>
* Example: abc_def_ghi
*
* <p>
* Words are translated to lower case and are separated by underscore character.
*/
public static final SentenceFormatter LOWER_SNAKE_CASE = new SentenceFormatter(Strings::toLowerCaseNeutral, SEPARATOR_SNAKE_CASE, Strings::toLowerCaseNeutral);
/**
* Lower White Space formatter
*
* <p>
* Example: Abc def ghi
*
* <p>
* Words are translated to lower case, except the first word, which is
* translated to title case, and are separated by white space character.
*/
public static final SentenceFormatter LOWER_WHITE_SPACE = new SentenceFormatter(Strings::toTitleCaseNeutral, SEPARATOR_WHITE_SPACE, Strings::toLowerCaseNeutral);
/**
* Upper Camel Case formatter
*
* <p>
* Example: AbcDefGhi
*
* <p>
* Words are translated to title case.
*/
public static final SentenceFormatter UPPER_CAMEL_CASE = new SentenceFormatter(Strings::toTitleCaseNeutral, "", Strings::toTitleCaseNeutral);
/**
* Upper Kebab Case formatter
*
* <p>
* Example: ABC-DEF-GHI
*
* <p>
* Words are translated to upper case and are separated by minus character.
*/
public static final SentenceFormatter UPPER_KEBAB_CASE = new SentenceFormatter(Strings::toUpperCaseNeutral, SEPARATOR_KEBAB_CASE, Strings::toUpperCaseNeutral);
/**
* Upper Snake Case formatter
*
* <p>
* Example: ABC_DEF_GHI
*
* <p>
* Words are translated to upper case and are separated by underscore character.
*/
public static final SentenceFormatter UPPER_SNAKE_CASE = new SentenceFormatter(Strings::toUpperCaseNeutral, SEPARATOR_SNAKE_CASE, Strings::toUpperCaseNeutral);
/**
* Upper White Space formatter
*
* <p>
* Example: Abc Def Ghi
*
* <p>
* Words are translated to title case and are separated by white space
* character.
*/
public static final SentenceFormatter UPPER_WHITE_SPACE = new SentenceFormatter(Strings::toTitleCaseNeutral, SEPARATOR_WHITE_SPACE, Strings::toTitleCaseNeutral);
/**
* Function to convert the first word
*/
private final Function<String, String> convertFirstWord;
/**
* Separator character between words
*/
private final String separator;
/**
* Function to convert all words except the first word
*/
private final Function<String, String> convertSubsequentWords;
/**
* Formats the list of given words using the specified converter functions and
* separator.
*
* @param words list of words
* @return formatted sentence
*/
public String format(final String... words) {
return format(asList(words));
}
/**
* Formats the list of given words using the specified converter functions and
* separator.
*
* @param words list of words
* @return formatted sentence
*/
public String format(final List<String> words) {
if (words.isEmpty()) {
return "";
}
final StringBuilder builder = new StringBuilder();
builder.append(getConvertFirstWord().apply(words.get(0)));
final int size = words.size();
for (int index = 1; index < size; index += 1) {
builder.append(getSeparator());
builder.append(getConvertSubsequentWords().apply(words.get(index)));
}
return builder.toString();
}
/**
* Splits the given sentence into words using the separator. If the separator
* string is empty, the sentence is splitted by title characters. A leading
* title character does not generate an empty leading word.
*
* @param sentence the sentence to be splitted
* @return list of splitted words
*/
public List<String> parse(final String sentence) {
if (sentence.isEmpty()) {
return emptyList();
}
if (getSeparator().isEmpty()) {
return splitByTitleCharacters(sentence);
}
return splitBySeparator(sentence);
}
/**
* Splits the given sentence into words using the separator string.
*
* @param sentence the sentence to be splitted
* @return list of splitted words
*/
private List<String> splitBySeparator(final String sentence) {
final int sentenceLength = sentence.length();
final int separatorLength = getSeparator().length();
final int length = sentenceLength - separatorLength;
int beginIndex = 0;
final List<String> words = new ArrayList<>();
for (int index = 0; index < length; index += 1) {
if (sentence.regionMatches(index, getSeparator(), 0, separatorLength)) {
words.add(sentence.substring(beginIndex, index));
beginIndex = index + separatorLength;
}
}
words.add(sentence.substring(beginIndex, sentenceLength));
return words;
}
/**
* Splits the given sentence into words by title characters. A leading title
* character does not generate an empty leading word.
*
* @param sentence the sentence to be splitted
* @return list of splitted words
*/
private List<String> splitByTitleCharacters(final String sentence) {
final int length = sentence.length();
int beginIndex = 0;
final List<String> words = new ArrayList<>();
for (int index = 1; index < length; index += 1) {
if (Character.isUpperCase(sentence.charAt(index))) {
words.add(sentence.substring(beginIndex, index));
beginIndex = index;
}
}
words.add(sentence.substring(beginIndex, length));
return words;
}
/**
* Function to convert the first word
*
* @return convert function for the first word of a sentence
*/
@java.lang.SuppressWarnings("all")
@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
@lombok.Generated
public Function<String, String> getConvertFirstWord() {
return this.convertFirstWord;
}
/**
* Separator character between words
*
* @return separator
*/
@java.lang.SuppressWarnings("all")
@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
@lombok.Generated
public String getSeparator() {
return this.separator;
}
/**
* Function to convert all words except the first word
*
* @return convert function for subsequent words
*/
@java.lang.SuppressWarnings("all")
@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
@lombok.Generated
public Function<String, String> getConvertSubsequentWords() {
return this.convertSubsequentWords;
}
/**
* Creates a new {@code SentenceFormatter} instance.
*
* @param convertFirstWord Function to convert the first word
* @param separator Separator character between words
* @param convertSubsequentWords Function to convert all words except the first word
*/
@java.lang.SuppressWarnings("all")
@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
@lombok.Generated
public SentenceFormatter(final Function<String, String> convertFirstWord, final String separator, final Function<String, String> convertSubsequentWords) {
this.convertFirstWord = convertFirstWord;
this.separator = separator;
this.convertSubsequentWords = convertSubsequentWords;
}
}