1 /*
2  *******************************************************************************
3  * Copyright (C) 2004-2011, International Business Machines Corporation and    *
4  * others. All Rights Reserved.                                                *
5  * Copyright (C) 2009 , Yahoo! Inc.                                            *
6  *******************************************************************************
7  */
8 package com.ibm.icu.text;
9 
10 import java.io.IOException;
11 import java.io.ObjectInputStream;
12 import java.text.FieldPosition;
13 import java.text.Format;
14 import java.text.ParsePosition;
15 
16 import com.ibm.icu.impl.PatternProps;
17 
18 /**
19  * <p><code>SelectFormat</code> supports the creation of  internationalized
20  * messages by selecting phrases based on keywords. The pattern  specifies
21  * how to map keywords to phrases and provides a default phrase. The
22  * object provided to the format method is a string that's matched
23  * against the keywords. If there is a match, the corresponding phrase
24  * is selected; otherwise, the default phrase is used.</p>
25  *
26  * <h4>Using <code>SelectFormat</code> for Gender Agreement</h4>
27  *
28  * <p>Note: Typically, select formatting is done via <code>MessageFormat</code>
29  * with a <code>select</code> argument type,
30  * rather than using a stand-alone <code>SelectFormat</code>.</p>
31  *
32  * <p>The main use case for the select format is gender based  inflection.
33  * When names or nouns are inserted into sentences, their gender can  affect pronouns,
34  * verb forms, articles, and adjectives. Special care needs to be
35  * taken for the case where the gender cannot be determined.
36  * The impact varies between languages:</p>
37  *
38  * <ul>
39  * <li>English has three genders, and unknown gender is handled as a  special
40  * case. Names use the gender of the named person (if known), nouns  referring
41  * to people use natural gender, and inanimate objects are usually  neutral.
42  * The gender only affects pronouns: "he", "she", "it", "they".
43  *
44  * <li>German differs from English in that the gender of nouns is  rather
45  * arbitrary, even for nouns referring to people ("M&#u00E4;dchen", girl, is  neutral).
46  * The gender affects pronouns ("er", "sie", "es"), articles ("der",  "die",
47  * "das"), and adjective forms ("guter Mann", "gute Frau", "gutes  M&#u00E4;dchen").
48  *
49  * <li>French has only two genders; as in German the gender of nouns
50  * is rather arbitrary - for sun and moon, the genders
51  * are the opposite of those in German. The gender affects
52  * pronouns ("il", "elle"), articles ("le", "la"),
53  * adjective forms ("bon", "bonne"), and sometimes
54  * verb forms ("all&#u00E9;", "all&#u00E9e;").
55  *
56  * <li>Polish distinguishes five genders (or noun classes),
57  * human masculine, animate non-human masculine, inanimate masculine,
58  * feminine, and neuter.
59  * </ul>
60  *
61  * <p>Some other languages have noun classes that are not related to  gender,
62  * but similar in grammatical use.
63  * Some African languages have around 20 noun classes.</p>
64  *
65  * <p><b>Note:</b>For the gender of a <i>person</i> in a given sentence,
66  * we usually need to distinguish only between female, male and other/unknown.</p>
67  *
68  * <p>To enable localizers to create sentence patterns that take their
69  * language's gender dependencies into consideration, software has to  provide
70  * information about the gender associated with a noun or name to
71  * <code>MessageFormat</code>.
72  * Two main cases can be distinguished:</p>
73  *
74  * <ul>
75  * <li>For people, natural gender information should be maintained  for each person.
76  * Keywords like "male", "female", "mixed" (for groups of people)
77  * and "unknown" could be used.
78  *
79  * <li>For nouns, grammatical gender information should be maintained  for
80  * each noun and per language, e.g., in resource bundles.
81  * The keywords "masculine", "feminine", and "neuter" are commonly  used,
82  * but some languages may require other keywords.
83  * </ul>
84  *
85  * <p>The resulting keyword is provided to <code>MessageFormat</code>  as a
86  * parameter separate from the name or noun it's associated with. For  example,
87  * to generate a message such as "Jean went to Paris", three separate  arguments
88  * would be provided: The name of the person as argument 0, the  gender of
89  * the person as argument 1, and the name of the city as argument 2.
90  * The sentence pattern for English, where the gender of the person has
91  * no impact on this simple sentence, would not refer to argument 1  at all:</p>
92  *
93  * <pre>{0} went to {2}.</pre>
94  *
95  * <p><b>Note:</b> The entire sentence should be included (and partially repeated)
96  * inside each phrase. Otherwise translators would have to be trained on how to
97  * move bits of the sentence in and out of the select argument of a message.
98  * (The examples below do not follow this recommendation!)</p>
99  *
100  * <p>The sentence pattern for French, where the gender of the person affects
101  * the form of the participle, uses a select format based on argument 1:</p>
102  *
103  * <pre>{0} est {1, select, female {all&#u00E9;e} other {all&#u00E9;}} &#u00E0; {2}.</pre>
104  *
105  * <p>Patterns can be nested, so that it's possible to handle  interactions of
106  * number and gender where necessary. For example, if the above  sentence should
107  * allow for the names of several people to be inserted, the  following sentence
108  * pattern can be used (with argument 0 the list of people's names,
109  * argument 1 the number of people, argument 2 their combined gender, and
110  * argument 3 the city name):</p>
111  *
112  * <pre>{0} {1, plural,
113  * one {est {2, select, female {all&#u00E9;e} other  {all&#u00E9;}}}
114  * other {sont {2, select, female {all&#u00E9;es} other {all&#u00E9;s}}}
115  * }&#u00E0; {3}.</pre>
116  *
117  * <h4>Patterns and Their Interpretation</h4>
118  *
119  * <p>The <code>SelectFormat</code> pattern string defines the phrase  output
120  * for each user-defined keyword.
121  * The pattern is a sequence of (keyword, message) pairs.
122  * A keyword is a "pattern identifier": [^[[:Pattern_Syntax:][:Pattern_White_Space:]]]+</p>
123  *
124  * <p>Each message is a MessageFormat pattern string enclosed in {curly braces}.</p>
125  *
126  * <p>You always have to define a phrase for the default keyword
127  * <code>other</code>; this phrase is returned when the keyword
128  * provided to
129  * the <code>format</code> method matches no other keyword.
130  * If a pattern does not provide a phrase for <code>other</code>, the  method
131  * it's provided to returns the error  <code>U_DEFAULT_KEYWORD_MISSING</code>.
132  * <br/>
133  * Pattern_White_Space between keywords and messages is ignored.
134  * Pattern_White_Space within a message is preserved and output.</p>
135  *
136  * <p><pre>Example:
137  * MessageFormat msgFmt = new MessageFormat("{0} est " +
138  *     "{1, select, female {all&#u00E9;e} other {all&#u00E9;}} &#u00E0; Paris.",
139  *     new ULocale("fr"));
140  * Object args[] = {"Kirti","female"};
141  * System.out.println(msgFmt.format(args));
142  * </pre>
143  * <p>
144  * Produces the output:<br/>
145  * <code>Kirti est all&#u00E9;e &#u00E0; Paris.</code>
146  * </p>
147  *
148  * @stable ICU 4.4
149  */
150 
151 public class SelectFormat extends Format{
152     // Generated by serialver from JDK 1.5
153     private static final long serialVersionUID = 2993154333257524984L;
154 
155     /*
156      * The applied pattern string.
157      */
158     private String pattern = null;
159 
160     /**
161      * The MessagePattern which contains the parsed structure of the pattern string.
162      */
163     transient private MessagePattern msgPattern;
164 
165     /**
166      * Creates a new <code>SelectFormat</code> for a given pattern string.
167      * @param  pattern the pattern for this <code>SelectFormat</code>.
168      * @stable ICU 4.4
169      */
SelectFormat(String pattern)170     public SelectFormat(String pattern) {
171         applyPattern(pattern);
172     }
173 
174     /*
175      * Resets the <code>SelectFormat</code> object.
176      */
reset()177     private void reset() {
178         pattern = null;
179         if(msgPattern != null) {
180             msgPattern.clear();
181         }
182     }
183 
184     /**
185      * Sets the pattern used by this select format.
186      * Patterns and their interpretation are specified in the class description.
187      *
188      * @param pattern the pattern for this select format.
189      * @throws IllegalArgumentException when the pattern is not a valid select format pattern.
190      * @stable ICU 4.4
191      */
applyPattern(String pattern)192     public void applyPattern(String pattern) {
193         this.pattern = pattern;
194         if (msgPattern == null) {
195             msgPattern = new MessagePattern();
196         }
197         try {
198             msgPattern.parseSelectStyle(pattern);
199         } catch(RuntimeException e) {
200             reset();
201             throw e;
202         }
203     }
204 
205     /**
206      * Returns the pattern for this <code>SelectFormat</code>
207      *
208      * @return the pattern string
209      * @stable ICU 4.4
210      */
toPattern()211     public String toPattern() {
212         return pattern;
213     }
214 
215     /**
216      * Finds the SelectFormat sub-message for the given keyword, or the "other" sub-message.
217      * @param pattern A MessagePattern.
218      * @param partIndex the index of the first SelectFormat argument style part.
219      * @param keyword a keyword to be matched to one of the SelectFormat argument's keywords.
220      * @return the sub-message start part index.
221      */
findSubMessage(MessagePattern pattern, int partIndex, String keyword)222     public static int findSubMessage(MessagePattern pattern, int partIndex, String keyword) {
223         int count=pattern.countParts();
224         int msgStart=0;
225         // Iterate over (ARG_SELECTOR, message) pairs until ARG_LIMIT or end of select-only pattern.
226         do {
227             MessagePattern.Part part=pattern.getPart(partIndex++);
228             MessagePattern.Part.Type type=part.getType();
229             if(type==MessagePattern.Part.Type.ARG_LIMIT) {
230                 break;
231             }
232             assert type==MessagePattern.Part.Type.ARG_SELECTOR;
233             // part is an ARG_SELECTOR followed by a message
234             if(pattern.partSubstringMatches(part, keyword)) {
235                 // keyword matches
236                 return partIndex;
237             } else if(msgStart==0 && pattern.partSubstringMatches(part, "other")) {
238                 msgStart=partIndex;
239             }
240             partIndex=pattern.getLimitPartIndex(partIndex);
241         } while(++partIndex<count);
242         return msgStart;
243     }
244 
245     /**
246      * Selects the phrase for the given keyword.
247      *
248      * @param keyword a phrase selection keyword.
249      * @return the string containing the formatted select message.
250      * @throws IllegalArgumentException when the given keyword is not a "pattern identifier"
251      * @stable ICU 4.4
252      */
format(String keyword)253     public final String format(String keyword) {
254         //Check for the validity of the keyword
255         if (!PatternProps.isIdentifier(keyword)) {
256             throw new IllegalArgumentException("Invalid formatting argument.");
257         }
258         // If no pattern was applied, throw an exception
259         if (msgPattern == null || msgPattern.countParts() == 0) {
260             throw new IllegalStateException("Invalid format error.");
261         }
262 
263         // Get the appropriate sub-message.
264         int msgStart = findSubMessage(msgPattern, 0, keyword);
265         if (!msgPattern.jdkAposMode()) {
266             int msgLimit = msgPattern.getLimitPartIndex(msgStart);
267             return msgPattern.getPatternString().substring(msgPattern.getPart(msgStart).getLimit(),
268                                                            msgPattern.getPatternIndex(msgLimit));
269         }
270         // JDK compatibility mode: Remove SKIP_SYNTAX.
271         StringBuilder result = null;
272         int prevIndex = msgPattern.getPart(msgStart).getLimit();
273         for (int i = msgStart;;) {
274             MessagePattern.Part part = msgPattern.getPart(++i);
275             MessagePattern.Part.Type type = part.getType();
276             int index = part.getIndex();
277             if (type == MessagePattern.Part.Type.MSG_LIMIT) {
278                 if (result == null) {
279                     return pattern.substring(prevIndex, index);
280                 } else {
281                     return result.append(pattern, prevIndex, index).toString();
282                 }
283             } else if (type == MessagePattern.Part.Type.SKIP_SYNTAX) {
284                 if (result == null) {
285                     result = new StringBuilder();
286                 }
287                 result.append(pattern, prevIndex, index);
288                 prevIndex = part.getLimit();
289             } else if (type == MessagePattern.Part.Type.ARG_START) {
290                 if (result == null) {
291                     result = new StringBuilder();
292                 }
293                 result.append(pattern, prevIndex, index);
294                 prevIndex = index;
295                 i = msgPattern.getLimitPartIndex(i);
296                 index = msgPattern.getPart(i).getLimit();
297                 MessagePattern.appendReducedApostrophes(pattern, prevIndex, index, result);
298                 prevIndex = index;
299             }
300         }
301     }
302 
303     /**
304      * Selects the phrase for the given keyword.
305      * and appends the formatted message to the given <code>StringBuffer</code>.
306      * @param keyword a phrase selection keyword.
307      * @param toAppendTo the selected phrase will be appended to this
308      *        <code>StringBuffer</code>.
309      * @param pos will be ignored by this method.
310      * @throws IllegalArgumentException when the given keyword is not a String
311      *         or not a "pattern identifier"
312      * @return the string buffer passed in as toAppendTo, with formatted text
313      *         appended.
314      * @stable ICU 4.4
315      */
format(Object keyword, StringBuffer toAppendTo, FieldPosition pos)316     public StringBuffer format(Object keyword, StringBuffer toAppendTo,
317             FieldPosition pos) {
318         if (keyword instanceof String) {
319             toAppendTo.append(format( (String)keyword));
320         }else{
321             throw new IllegalArgumentException("'" + keyword + "' is not a String");
322         }
323         return toAppendTo;
324     }
325 
326     /**
327      * This method is not supported by <code>SelectFormat</code>.
328      * @param source the string to be parsed.
329      * @param pos defines the position where parsing is to begin,
330      * and upon return, the position where parsing left off.  If the position
331      * has not changed upon return, then parsing failed.
332      * @return nothing because this method is not supported.
333      * @throws UnsupportedOperationException thrown always.
334      * @stable ICU 4.4
335      */
parseObject(String source, ParsePosition pos)336     public Object parseObject(String source, ParsePosition pos) {
337         throw new UnsupportedOperationException();
338     }
339 
340     /**
341      * {@inheritDoc}
342      * @stable ICU 4.4
343      */
344     @Override
equals(Object obj)345     public boolean equals(Object obj) {
346         if(this == obj) {
347             return true;
348         }
349         if(obj == null || getClass() != obj.getClass()) {
350             return false;
351         }
352         SelectFormat sf = (SelectFormat) obj;
353         return msgPattern == null ? sf.msgPattern == null : msgPattern.equals(sf.msgPattern);
354     }
355 
356     /**
357      * {@inheritDoc}
358      * @stable ICU 4.4
359      */
360     @Override
hashCode()361     public int hashCode() {
362         if (pattern != null) {
363             return pattern.hashCode();
364         }
365         return 0;
366     }
367 
368     /**
369      * {@inheritDoc}
370      * @stable ICU 4.4
371      */
372     @Override
toString()373     public String toString() {
374         return "pattern='" + pattern + "'";
375     }
376 
readObject(ObjectInputStream in)377     private void readObject(ObjectInputStream in)
378         throws IOException, ClassNotFoundException {
379         in.defaultReadObject();
380         if (pattern != null) {
381             applyPattern(pattern);
382         }
383     }
384 }
385