6983207: API: MessageFormat quote handling discrepancy

Reviewed-by: peytoia
This commit is contained in:
Masayoshi Okutsu 2010-12-17 18:06:28 +09:00
parent c1d0a22eac
commit 62f4413b7b

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 1996, 2008, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1996, 2010, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -93,73 +93,65 @@ import java.util.Locale;
* currency * currency
* percent * percent
* <i>SubformatPattern</i> * <i>SubformatPattern</i>
*
* <i>String:</i>
* <i>StringPart<sub>opt</sub></i>
* <i>String</i> <i>StringPart</i>
*
* <i>StringPart:</i>
* ''
* ' <i>QuotedString</i> '
* <i>UnquotedString</i>
*
* <i>SubformatPattern:</i>
* <i>SubformatPatternPart<sub>opt</sub></i>
* <i>SubformatPattern</i> <i>SubformatPatternPart</i>
*
* <i>SubFormatPatternPart:</i>
* ' <i>QuotedPattern</i> '
* <i>UnquotedPattern</i>
* </pre></blockquote> * </pre></blockquote>
* *
* <p> * <p>Within a <i>String</i>, a pair of single quotes can be used to
* Within a <i>String</i>, <code>"''"</code> represents a single * quote any arbitrary characters except single quotes. For example,
* quote. A <i>QuotedString</i> can contain arbitrary characters * pattern string <code>"'{0}'"</code> represents string
* except single quotes; the surrounding single quotes are removed. * <code>"{0}"</code>, not a <i>FormatElement</i>. A single quote itself
* An <i>UnquotedString</i> can contain arbitrary characters * must be represented by doubled single quotes {@code ''} throughout a
* except single quotes and left curly brackets. Thus, a string that * <i>String</i>. For example, pattern string <code>"'{''}'"</code> is
* should result in the formatted message "'{0}'" can be written as * interpreted as a sequence of <code>'{</code> (start of quoting and a
* <code>"'''{'0}''"</code> or <code>"'''{0}'''"</code>. * left curly brace), <code>''</code> (a single quote), and
* <p> * <code>}'</code> (a right curly brace and end of quoting),
* Within a <i>SubformatPattern</i>, different rules apply. * <em>not</em> <code>'{'</code> and <code>'}'</code> (quoted left and
* A <i>QuotedPattern</i> can contain arbitrary characters * right curly braces): representing string <code>"{'}"</code>,
* except single quotes; but the surrounding single quotes are * <em>not</em> <code>"{}"</code>.
* <strong>not</strong> removed, so they may be interpreted by the *
* subformat. For example, <code>"{1,number,$'#',##}"</code> will * <p>A <i>SubformatPattern</i> is interpreted by its corresponding
* produce a number format with the pound-sign quoted, with a result * subformat, and subformat-dependent pattern rules apply. For example,
* such as: "$#31,45". * pattern string <code>"{1,number,<u>$'#',##</u>}"</code>
* An <i>UnquotedPattern</i> can contain arbitrary characters * (<i>SubformatPattern</i> with underline) will produce a number format
* except single quotes, but curly braces within it must be balanced. * with the pound-sign quoted, with a result such as: {@code
* For example, <code>"ab {0} de"</code> and <code>"ab '}' de"</code> * "$#31,45"}. Refer to each {@code Format} subclass documentation for
* are valid subformat patterns, but <code>"ab {0'}' de"</code> and * details.
* <code>"ab } de"</code> are not. *
* <p>Any unmatched quote is treated as closed at the end of the given
* pattern. For example, pattern string {@code "'{0}"} is treated as
* pattern {@code "'{0}'"}.
*
* <p>Any curly braces within an unquoted pattern must be balanced. For
* example, <code>"ab {0} de"</code> and <code>"ab '}' de"</code> are
* valid patterns, but <code>"ab {0'}' de"</code>, <code>"ab } de"</code>
* and <code>"''{''"</code> are not.
*
* <p> * <p>
* <dl><dt><b>Warning:</b><dd>The rules for using quotes within message * <dl><dt><b>Warning:</b><dd>The rules for using quotes within message
* format patterns unfortunately have shown to be somewhat confusing. * format patterns unfortunately have shown to be somewhat confusing.
* In particular, it isn't always obvious to localizers whether single * In particular, it isn't always obvious to localizers whether single
* quotes need to be doubled or not. Make sure to inform localizers about * quotes need to be doubled or not. Make sure to inform localizers about
* the rules, and tell them (for example, by using comments in resource * the rules, and tell them (for example, by using comments in resource
* bundle source files) which strings will be processed by MessageFormat. * bundle source files) which strings will be processed by {@code MessageFormat}.
* Note that localizers may need to use single quotes in translated * Note that localizers may need to use single quotes in translated
* strings where the original version doesn't have them. * strings where the original version doesn't have them.
* </dl> * </dl>
* <p> * <p>
* The <i>ArgumentIndex</i> value is a non-negative integer written * The <i>ArgumentIndex</i> value is a non-negative integer written
* using the digits '0' through '9', and represents an index into the * using the digits {@code '0'} through {@code '9'}, and represents an index into the
* <code>arguments</code> array passed to the <code>format</code> methods * {@code arguments} array passed to the {@code format} methods
* or the result array returned by the <code>parse</code> methods. * or the result array returned by the {@code parse} methods.
* <p> * <p>
* The <i>FormatType</i> and <i>FormatStyle</i> values are used to create * The <i>FormatType</i> and <i>FormatStyle</i> values are used to create
* a <code>Format</code> instance for the format element. The following * a {@code Format} instance for the format element. The following
* table shows how the values map to Format instances. Combinations not * table shows how the values map to {@code Format} instances. Combinations not
* shown in the table are illegal. A <i>SubformatPattern</i> must * shown in the table are illegal. A <i>SubformatPattern</i> must
* be a valid pattern string for the Format subclass used. * be a valid pattern string for the {@code Format} subclass used.
* <p> * <p>
* <table border=1 summary="Shows how FormatType and FormatStyle values map to Format instances"> * <table border=1 summary="Shows how FormatType and FormatStyle values map to Format instances">
* <tr> * <tr>
* <th id="ft">Format Type * <th id="ft" class="TableHeadingColor">FormatType
* <th id="fs">Format Style * <th id="fs" class="TableHeadingColor">FormatStyle
* <th id="sc">Subformat Created * <th id="sc" class="TableHeadingColor">Subformat Created
* <tr> * <tr>
* <td headers="ft"><i>(none)</i> * <td headers="ft"><i>(none)</i>
* <td headers="fs"><i>(none)</i> * <td headers="fs"><i>(none)</i>
@ -167,61 +159,61 @@ import java.util.Locale;
* <tr> * <tr>
* <td headers="ft" rowspan=5><code>number</code> * <td headers="ft" rowspan=5><code>number</code>
* <td headers="fs"><i>(none)</i> * <td headers="fs"><i>(none)</i>
* <td headers="sc"><code>NumberFormat.getInstance(getLocale())</code> * <td headers="sc">{@link NumberFormat#getInstance(Locale) NumberFormat.getInstance}{@code (getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>integer</code> * <td headers="fs"><code>integer</code>
* <td headers="sc"><code>NumberFormat.getIntegerInstance(getLocale())</code> * <td headers="sc">{@link NumberFormat#getIntegerInstance(Locale) NumberFormat.getIntegerInstance}{@code (getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>currency</code> * <td headers="fs"><code>currency</code>
* <td headers="sc"><code>NumberFormat.getCurrencyInstance(getLocale())</code> * <td headers="sc">{@link NumberFormat#getCurrencyInstance(Locale) NumberFormat.getCurrencyInstance}{@code (getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>percent</code> * <td headers="fs"><code>percent</code>
* <td headers="sc"><code>NumberFormat.getPercentInstance(getLocale())</code> * <td headers="sc">{@link NumberFormat#getPercentInstance(Locale) NumberFormat.getPercentInstance}{@code (getLocale())}
* <tr> * <tr>
* <td headers="fs"><i>SubformatPattern</i> * <td headers="fs"><i>SubformatPattern</i>
* <td headers="sc"><code>new DecimalFormat(subformatPattern, DecimalFormatSymbols.getInstance(getLocale()))</code> * <td headers="sc">{@code new} {@link DecimalFormat#DecimalFormat(String,DecimalFormatSymbols) DecimalFormat}{@code (subformatPattern,} {@link DecimalFormatSymbols#getInstance(Locale) DecimalFormatSymbols.getInstance}{@code (getLocale()))}
* <tr> * <tr>
* <td headers="ft" rowspan=6><code>date</code> * <td headers="ft" rowspan=6><code>date</code>
* <td headers="fs"><i>(none)</i> * <td headers="fs"><i>(none)</i>
* <td headers="sc"><code>DateFormat.getDateInstance(DateFormat.DEFAULT, getLocale())</code> * <td headers="sc">{@link DateFormat#getDateInstance(int,Locale) DateFormat.getDateInstance}{@code (}{@link DateFormat#DEFAULT}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>short</code> * <td headers="fs"><code>short</code>
* <td headers="sc"><code>DateFormat.getDateInstance(DateFormat.SHORT, getLocale())</code> * <td headers="sc">{@link DateFormat#getDateInstance(int,Locale) DateFormat.getDateInstance}{@code (}{@link DateFormat#SHORT}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>medium</code> * <td headers="fs"><code>medium</code>
* <td headers="sc"><code>DateFormat.getDateInstance(DateFormat.DEFAULT, getLocale())</code> * <td headers="sc">{@link DateFormat#getDateInstance(int,Locale) DateFormat.getDateInstance}{@code (}{@link DateFormat#DEFAULT}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>long</code> * <td headers="fs"><code>long</code>
* <td headers="sc"><code>DateFormat.getDateInstance(DateFormat.LONG, getLocale())</code> * <td headers="sc">{@link DateFormat#getDateInstance(int,Locale) DateFormat.getDateInstance}{@code (}{@link DateFormat#LONG}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>full</code> * <td headers="fs"><code>full</code>
* <td headers="sc"><code>DateFormat.getDateInstance(DateFormat.FULL, getLocale())</code> * <td headers="sc">{@link DateFormat#getDateInstance(int,Locale) DateFormat.getDateInstance}{@code (}{@link DateFormat#FULL}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><i>SubformatPattern</i> * <td headers="fs"><i>SubformatPattern</i>
* <td headers="sc"><code>new SimpleDateFormat(subformatPattern, getLocale())</code> * <td headers="sc">{@code new} {@link SimpleDateFormat#SimpleDateFormat(String,Locale) SimpleDateFormat}{@code (subformatPattern, getLocale())}
* <tr> * <tr>
* <td headers="ft" rowspan=6><code>time</code> * <td headers="ft" rowspan=6><code>time</code>
* <td headers="fs"><i>(none)</i> * <td headers="fs"><i>(none)</i>
* <td headers="sc"><code>DateFormat.getTimeInstance(DateFormat.DEFAULT, getLocale())</code> * <td headers="sc">{@link DateFormat#getTimeInstance(int,Locale) DateFormat.getTimeInstance}{@code (}{@link DateFormat#DEFAULT}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>short</code> * <td headers="fs"><code>short</code>
* <td headers="sc"><code>DateFormat.getTimeInstance(DateFormat.SHORT, getLocale())</code> * <td headers="sc">{@link DateFormat#getTimeInstance(int,Locale) DateFormat.getTimeInstance}{@code (}{@link DateFormat#SHORT}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>medium</code> * <td headers="fs"><code>medium</code>
* <td headers="sc"><code>DateFormat.getTimeInstance(DateFormat.DEFAULT, getLocale())</code> * <td headers="sc">{@link DateFormat#getTimeInstance(int,Locale) DateFormat.getTimeInstance}{@code (}{@link DateFormat#DEFAULT}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>long</code> * <td headers="fs"><code>long</code>
* <td headers="sc"><code>DateFormat.getTimeInstance(DateFormat.LONG, getLocale())</code> * <td headers="sc">{@link DateFormat#getTimeInstance(int,Locale) DateFormat.getTimeInstance}{@code (}{@link DateFormat#LONG}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><code>full</code> * <td headers="fs"><code>full</code>
* <td headers="sc"><code>DateFormat.getTimeInstance(DateFormat.FULL, getLocale())</code> * <td headers="sc">{@link DateFormat#getTimeInstance(int,Locale) DateFormat.getTimeInstance}{@code (}{@link DateFormat#FULL}{@code , getLocale())}
* <tr> * <tr>
* <td headers="fs"><i>SubformatPattern</i> * <td headers="fs"><i>SubformatPattern</i>
* <td headers="sc"><code>new SimpleDateFormat(subformatPattern, getLocale())</code> * <td headers="sc">{@code new} {@link SimpleDateFormat#SimpleDateFormat(String,Locale) SimpleDateFormat}{@code (subformatPattern, getLocale())}
* <tr> * <tr>
* <td headers="ft"><code>choice</code> * <td headers="ft"><code>choice</code>
* <td headers="fs"><i>SubformatPattern</i> * <td headers="fs"><i>SubformatPattern</i>
* <td headers="sc"><code>new ChoiceFormat(subformatPattern)</code> * <td headers="sc">{@code new} {@link ChoiceFormat#ChoiceFormat(String) ChoiceFormat}{@code (subformatPattern)}
* </table> * </table>
* <p> * <p>
* *
@ -321,7 +313,7 @@ import java.util.Locale;
* </pre></blockquote> * </pre></blockquote>
* *
* <p> * <p>
* Likewise, parsing with a MessageFormat object using patterns containing * Likewise, parsing with a {@code MessageFormat} object using patterns containing
* multiple occurrences of the same argument would return the last match. For * multiple occurrences of the same argument would return the last match. For
* example, * example,
* <blockquote><pre> * <blockquote><pre>
@ -343,7 +335,11 @@ import java.util.Locale;
* @see Format * @see Format
* @see NumberFormat * @see NumberFormat
* @see DecimalFormat * @see DecimalFormat
* @see DecimalFormatSymbols
* @see ChoiceFormat * @see ChoiceFormat
* @see DateFormat
* @see SimpleDateFormat
*
* @author Mark Davis * @author Mark Davis
*/ */