aboutsummaryrefslogtreecommitdiff
path: root/src/com/ibm/icu/text/SelectFormat.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/com/ibm/icu/text/SelectFormat.java')
-rw-r--r--src/com/ibm/icu/text/SelectFormat.java384
1 files changed, 384 insertions, 0 deletions
diff --git a/src/com/ibm/icu/text/SelectFormat.java b/src/com/ibm/icu/text/SelectFormat.java
new file mode 100644
index 0000000..c062744
--- /dev/null
+++ b/src/com/ibm/icu/text/SelectFormat.java
@@ -0,0 +1,384 @@
+/*
+ *******************************************************************************
+ * Copyright (C) 2004-2011, International Business Machines Corporation and *
+ * others. All Rights Reserved. *
+ * Copyright (C) 2009 , Yahoo! Inc. *
+ *******************************************************************************
+ */
+package com.ibm.icu.text;
+
+import java.io.IOException;
+import java.io.ObjectInputStream;
+import java.text.FieldPosition;
+import java.text.Format;
+import java.text.ParsePosition;
+
+import com.ibm.icu.impl.PatternProps;
+
+/**
+ * <p><code>SelectFormat</code> supports the creation of internationalized
+ * messages by selecting phrases based on keywords. The pattern specifies
+ * how to map keywords to phrases and provides a default phrase. The
+ * object provided to the format method is a string that's matched
+ * against the keywords. If there is a match, the corresponding phrase
+ * is selected; otherwise, the default phrase is used.</p>
+ *
+ * <h4>Using <code>SelectFormat</code> for Gender Agreement</h4>
+ *
+ * <p>Note: Typically, select formatting is done via <code>MessageFormat</code>
+ * with a <code>select</code> argument type,
+ * rather than using a stand-alone <code>SelectFormat</code>.</p>
+ *
+ * <p>The main use case for the select format is gender based inflection.
+ * When names or nouns are inserted into sentences, their gender can affect pronouns,
+ * verb forms, articles, and adjectives. Special care needs to be
+ * taken for the case where the gender cannot be determined.
+ * The impact varies between languages:</p>
+ *
+ * <ul>
+ * <li>English has three genders, and unknown gender is handled as a special
+ * case. Names use the gender of the named person (if known), nouns referring
+ * to people use natural gender, and inanimate objects are usually neutral.
+ * The gender only affects pronouns: "he", "she", "it", "they".
+ *
+ * <li>German differs from English in that the gender of nouns is rather
+ * arbitrary, even for nouns referring to people ("M&#u00E4;dchen", girl, is neutral).
+ * The gender affects pronouns ("er", "sie", "es"), articles ("der", "die",
+ * "das"), and adjective forms ("guter Mann", "gute Frau", "gutes M&#u00E4;dchen").
+ *
+ * <li>French has only two genders; as in German the gender of nouns
+ * is rather arbitrary - for sun and moon, the genders
+ * are the opposite of those in German. The gender affects
+ * pronouns ("il", "elle"), articles ("le", "la"),
+ * adjective forms ("bon", "bonne"), and sometimes
+ * verb forms ("all&#u00E9;", "all&#u00E9e;").
+ *
+ * <li>Polish distinguishes five genders (or noun classes),
+ * human masculine, animate non-human masculine, inanimate masculine,
+ * feminine, and neuter.
+ * </ul>
+ *
+ * <p>Some other languages have noun classes that are not related to gender,
+ * but similar in grammatical use.
+ * Some African languages have around 20 noun classes.</p>
+ *
+ * <p><b>Note:</b>For the gender of a <i>person</i> in a given sentence,
+ * we usually need to distinguish only between female, male and other/unknown.</p>
+ *
+ * <p>To enable localizers to create sentence patterns that take their
+ * language's gender dependencies into consideration, software has to provide
+ * information about the gender associated with a noun or name to
+ * <code>MessageFormat</code>.
+ * Two main cases can be distinguished:</p>
+ *
+ * <ul>
+ * <li>For people, natural gender information should be maintained for each person.
+ * Keywords like "male", "female", "mixed" (for groups of people)
+ * and "unknown" could be used.
+ *
+ * <li>For nouns, grammatical gender information should be maintained for
+ * each noun and per language, e.g., in resource bundles.
+ * The keywords "masculine", "feminine", and "neuter" are commonly used,
+ * but some languages may require other keywords.
+ * </ul>
+ *
+ * <p>The resulting keyword is provided to <code>MessageFormat</code> as a
+ * parameter separate from the name or noun it's associated with. For example,
+ * to generate a message such as "Jean went to Paris", three separate arguments
+ * would be provided: The name of the person as argument 0, the gender of
+ * the person as argument 1, and the name of the city as argument 2.
+ * The sentence pattern for English, where the gender of the person has
+ * no impact on this simple sentence, would not refer to argument 1 at all:</p>
+ *
+ * <pre>{0} went to {2}.</pre>
+ *
+ * <p><b>Note:</b> The entire sentence should be included (and partially repeated)
+ * inside each phrase. Otherwise translators would have to be trained on how to
+ * move bits of the sentence in and out of the select argument of a message.
+ * (The examples below do not follow this recommendation!)</p>
+ *
+ * <p>The sentence pattern for French, where the gender of the person affects
+ * the form of the participle, uses a select format based on argument 1:</p>
+ *
+ * <pre>{0} est {1, select, female {all&#u00E9;e} other {all&#u00E9;}} &#u00E0; {2}.</pre>
+ *
+ * <p>Patterns can be nested, so that it's possible to handle interactions of
+ * number and gender where necessary. For example, if the above sentence should
+ * allow for the names of several people to be inserted, the following sentence
+ * pattern can be used (with argument 0 the list of people's names,
+ * argument 1 the number of people, argument 2 their combined gender, and
+ * argument 3 the city name):</p>
+ *
+ * <pre>{0} {1, plural,
+ * one {est {2, select, female {all&#u00E9;e} other {all&#u00E9;}}}
+ * other {sont {2, select, female {all&#u00E9;es} other {all&#u00E9;s}}}
+ * }&#u00E0; {3}.</pre>
+ *
+ * <h4>Patterns and Their Interpretation</h4>
+ *
+ * <p>The <code>SelectFormat</code> pattern string defines the phrase output
+ * for each user-defined keyword.
+ * The pattern is a sequence of (keyword, message) pairs.
+ * A keyword is a "pattern identifier": [^[[:Pattern_Syntax:][:Pattern_White_Space:]]]+</p>
+ *
+ * <p>Each message is a MessageFormat pattern string enclosed in {curly braces}.</p>
+ *
+ * <p>You always have to define a phrase for the default keyword
+ * <code>other</code>; this phrase is returned when the keyword
+ * provided to
+ * the <code>format</code> method matches no other keyword.
+ * If a pattern does not provide a phrase for <code>other</code>, the method
+ * it's provided to returns the error <code>U_DEFAULT_KEYWORD_MISSING</code>.
+ * <br/>
+ * Pattern_White_Space between keywords and messages is ignored.
+ * Pattern_White_Space within a message is preserved and output.</p>
+ *
+ * <p><pre>Example:
+ * MessageFormat msgFmt = new MessageFormat("{0} est " +
+ * "{1, select, female {all&#u00E9;e} other {all&#u00E9;}} &#u00E0; Paris.",
+ * new ULocale("fr"));
+ * Object args[] = {"Kirti","female"};
+ * System.out.println(msgFmt.format(args));
+ * </pre>
+ * <p>
+ * Produces the output:<br/>
+ * <code>Kirti est all&#u00E9;e &#u00E0; Paris.</code>
+ * </p>
+ *
+ * @stable ICU 4.4
+ */
+
+public class SelectFormat extends Format{
+ // Generated by serialver from JDK 1.5
+ private static final long serialVersionUID = 2993154333257524984L;
+
+ /*
+ * The applied pattern string.
+ */
+ private String pattern = null;
+
+ /**
+ * The MessagePattern which contains the parsed structure of the pattern string.
+ */
+ transient private MessagePattern msgPattern;
+
+ /**
+ * Creates a new <code>SelectFormat</code> for a given pattern string.
+ * @param pattern the pattern for this <code>SelectFormat</code>.
+ * @stable ICU 4.4
+ */
+ public SelectFormat(String pattern) {
+ applyPattern(pattern);
+ }
+
+ /*
+ * Resets the <code>SelectFormat</code> object.
+ */
+ private void reset() {
+ pattern = null;
+ if(msgPattern != null) {
+ msgPattern.clear();
+ }
+ }
+
+ /**
+ * Sets the pattern used by this select format.
+ * Patterns and their interpretation are specified in the class description.
+ *
+ * @param pattern the pattern for this select format.
+ * @throws IllegalArgumentException when the pattern is not a valid select format pattern.
+ * @stable ICU 4.4
+ */
+ public void applyPattern(String pattern) {
+ this.pattern = pattern;
+ if (msgPattern == null) {
+ msgPattern = new MessagePattern();
+ }
+ try {
+ msgPattern.parseSelectStyle(pattern);
+ } catch(RuntimeException e) {
+ reset();
+ throw e;
+ }
+ }
+
+ /**
+ * Returns the pattern for this <code>SelectFormat</code>
+ *
+ * @return the pattern string
+ * @stable ICU 4.4
+ */
+ public String toPattern() {
+ return pattern;
+ }
+
+ /**
+ * Finds the SelectFormat sub-message for the given keyword, or the "other" sub-message.
+ * @param pattern A MessagePattern.
+ * @param partIndex the index of the first SelectFormat argument style part.
+ * @param keyword a keyword to be matched to one of the SelectFormat argument's keywords.
+ * @return the sub-message start part index.
+ */
+ public static int findSubMessage(MessagePattern pattern, int partIndex, String keyword) {
+ int count=pattern.countParts();
+ int msgStart=0;
+ // Iterate over (ARG_SELECTOR, message) pairs until ARG_LIMIT or end of select-only pattern.
+ do {
+ MessagePattern.Part part=pattern.getPart(partIndex++);
+ MessagePattern.Part.Type type=part.getType();
+ if(type==MessagePattern.Part.Type.ARG_LIMIT) {
+ break;
+ }
+ assert type==MessagePattern.Part.Type.ARG_SELECTOR;
+ // part is an ARG_SELECTOR followed by a message
+ if(pattern.partSubstringMatches(part, keyword)) {
+ // keyword matches
+ return partIndex;
+ } else if(msgStart==0 && pattern.partSubstringMatches(part, "other")) {
+ msgStart=partIndex;
+ }
+ partIndex=pattern.getLimitPartIndex(partIndex);
+ } while(++partIndex<count);
+ return msgStart;
+ }
+
+ /**
+ * Selects the phrase for the given keyword.
+ *
+ * @param keyword a phrase selection keyword.
+ * @return the string containing the formatted select message.
+ * @throws IllegalArgumentException when the given keyword is not a "pattern identifier"
+ * @stable ICU 4.4
+ */
+ public final String format(String keyword) {
+ //Check for the validity of the keyword
+ if (!PatternProps.isIdentifier(keyword)) {
+ throw new IllegalArgumentException("Invalid formatting argument.");
+ }
+ // If no pattern was applied, throw an exception
+ if (msgPattern == null || msgPattern.countParts() == 0) {
+ throw new IllegalStateException("Invalid format error.");
+ }
+
+ // Get the appropriate sub-message.
+ int msgStart = findSubMessage(msgPattern, 0, keyword);
+ if (!msgPattern.jdkAposMode()) {
+ int msgLimit = msgPattern.getLimitPartIndex(msgStart);
+ return msgPattern.getPatternString().substring(msgPattern.getPart(msgStart).getLimit(),
+ msgPattern.getPatternIndex(msgLimit));
+ }
+ // JDK compatibility mode: Remove SKIP_SYNTAX.
+ StringBuilder result = null;
+ int prevIndex = msgPattern.getPart(msgStart).getLimit();
+ for (int i = msgStart;;) {
+ MessagePattern.Part part = msgPattern.getPart(++i);
+ MessagePattern.Part.Type type = part.getType();
+ int index = part.getIndex();
+ if (type == MessagePattern.Part.Type.MSG_LIMIT) {
+ if (result == null) {
+ return pattern.substring(prevIndex, index);
+ } else {
+ return result.append(pattern, prevIndex, index).toString();
+ }
+ } else if (type == MessagePattern.Part.Type.SKIP_SYNTAX) {
+ if (result == null) {
+ result = new StringBuilder();
+ }
+ result.append(pattern, prevIndex, index);
+ prevIndex = part.getLimit();
+ } else if (type == MessagePattern.Part.Type.ARG_START) {
+ if (result == null) {
+ result = new StringBuilder();
+ }
+ result.append(pattern, prevIndex, index);
+ prevIndex = index;
+ i = msgPattern.getLimitPartIndex(i);
+ index = msgPattern.getPart(i).getLimit();
+ MessagePattern.appendReducedApostrophes(pattern, prevIndex, index, result);
+ prevIndex = index;
+ }
+ }
+ }
+
+ /**
+ * Selects the phrase for the given keyword.
+ * and appends the formatted message to the given <code>StringBuffer</code>.
+ * @param keyword a phrase selection keyword.
+ * @param toAppendTo the selected phrase will be appended to this
+ * <code>StringBuffer</code>.
+ * @param pos will be ignored by this method.
+ * @throws IllegalArgumentException when the given keyword is not a String
+ * or not a "pattern identifier"
+ * @return the string buffer passed in as toAppendTo, with formatted text
+ * appended.
+ * @stable ICU 4.4
+ */
+ public StringBuffer format(Object keyword, StringBuffer toAppendTo,
+ FieldPosition pos) {
+ if (keyword instanceof String) {
+ toAppendTo.append(format( (String)keyword));
+ }else{
+ throw new IllegalArgumentException("'" + keyword + "' is not a String");
+ }
+ return toAppendTo;
+ }
+
+ /**
+ * This method is not supported by <code>SelectFormat</code>.
+ * @param source the string to be parsed.
+ * @param pos defines the position where parsing is to begin,
+ * and upon return, the position where parsing left off. If the position
+ * has not changed upon return, then parsing failed.
+ * @return nothing because this method is not supported.
+ * @throws UnsupportedOperationException thrown always.
+ * @stable ICU 4.4
+ */
+ public Object parseObject(String source, ParsePosition pos) {
+ throw new UnsupportedOperationException();
+ }
+
+ /**
+ * {@inheritDoc}
+ * @stable ICU 4.4
+ */
+ @Override
+ public boolean equals(Object obj) {
+ if(this == obj) {
+ return true;
+ }
+ if(obj == null || getClass() != obj.getClass()) {
+ return false;
+ }
+ SelectFormat sf = (SelectFormat) obj;
+ return msgPattern == null ? sf.msgPattern == null : msgPattern.equals(sf.msgPattern);
+ }
+
+ /**
+ * {@inheritDoc}
+ * @stable ICU 4.4
+ */
+ @Override
+ public int hashCode() {
+ if (pattern != null) {
+ return pattern.hashCode();
+ }
+ return 0;
+ }
+
+ /**
+ * {@inheritDoc}
+ * @stable ICU 4.4
+ */
+ @Override
+ public String toString() {
+ return "pattern='" + pattern + "'";
+ }
+
+ private void readObject(ObjectInputStream in)
+ throws IOException, ClassNotFoundException {
+ in.defaultReadObject();
+ if (pattern != null) {
+ applyPattern(pattern);
+ }
+ }
+}