blob: e8e187a299ec3b0b94a79dc08e2a9b84cee0ecd1 [file] [log] [blame]
/* GENERATED SOURCE. DO NOT MODIFY. */
// © 2022 and later: Unicode, Inc. and others.
// License & terms of use: http://www.unicode.org/copyright.html
package android.icu.message2;
import java.util.Locale;
import java.util.Map;
/**
* {@code MessageFormatter} is the next iteration of {@link android.icu.text.MessageFormat}.
*
* <p>This new version builds on what we learned from using {@code MessageFormat} for 20 years
* in various environments, either exposed "as is" or as a base for other public APIs.</p>
*
* <p>It is more modular, easier to backport, and provides extension points to add new
* formatters and selectors without having to modify the specification.</p>
*
* <p>We will be able to add formatters for intervals, relative times, lists, measurement units,
* people names, and more, and support custom formatters implemented by developers
* outside of ICU itself, for company or even product specific needs.</p>
*
* <p>MessageFormat 2 will support more complex grammatical features, such as gender, inflections,
* and tagging parts of the message for style changes or speech.</p>
*
* <p>The reasoning for this effort is shared in the
* <a target="github" href="https://github.com/unicode-org/message-format-wg/blob/main/docs/why_mf_next.md">“Why
* MessageFormat needs a successor”</a> document.</p>
*
* <p>The “MessageFormat 2” project, which develops the new data model, semantics, and syntax,
* is hosted on <a target="github" href="https://github.com/unicode-org/message-format-wg">GitHub</a>.</p>
*
* <p>The current specification for the syntax and data model can be found
* <a target="github" href="https://github.com/unicode-org/message-format-wg/blob/main/spec/syntax.md">here</a>.</p>
*
* <p>This tech preview implements enough of the {@code MessageFormat} functions to be useful,
* but the final set of functions and the parameters accepted by those functions is not yet finalized.</p>
*
* <p>These are the functions interpreted right now:</p>
*
* <table border="1">
* <tr>
* <td rowspan="4">{@code datetime}</td>
* <td>Similar to the ICU {@code "date"} and {@code "time"}.</td>
* </tr>
*
* <tr><td>{@code datestyle} and {@code timestyle}<br>
* Similar to {@code argStyle : short | medium | long | full}.<br>
* Same values are accepted, but we can use both in one placeholder,
* for example <code>{$due :datetime datestyle=full timestyle=long}</code>.
* </td></tr>
*
* <tr><td>{@code pattern}<br>
* Similar to {@code argStyle = argStyleText}.<br>
* This is bad i18n practice, and will probably be dropped.<br>
* This is included just to support migration to MessageFormat 2.
* </td></tr>
*
* <tr><td>{@code skeleton}<br>
* Same as {@code argStyle = argSkeletonText}.<br>
* These are the date/time skeletons as supported by {@link android.icu.text.SimpleDateFormat}.
* </td></tr>
*
* <tr>
* <td rowspan="4">{@code number}</td>
* <td>Similar to the ICU "number".</td>
* </tr>
*
* <tr><td>{@code skeleton}<br>
* These are the number skeletons as supported by {@link android.icu.number.NumberFormatter}.</td></tr>
*
* <tr><td>{@code minimumFractionDigits}<br>
* Only implemented to be able to pass the unit tests from the ECMA tech preview implementation,
* which prefers options bags to skeletons.<br>
* TBD if the final {@number} function will support skeletons, option backs, or both.</td></tr>
*
* <tr><td>{@code offset}<br>
* Used to support plural with an offset.</td></tr>
*
* <tr><td >{@code identity}</td><td>Returns the direct string value of the argument (calling {@code toString()}).</td></tr>
*
* <tr>
* <td rowspan="3">{@code plural}</td>
* <td>Similar to the ICU {@code "plural"}.</td>
* </tr>
*
* <tr><td>{@code skeleton}<br>
* These are the number skeletons as supported by {@link android.icu.number.NumberFormatter}.<br>
* Can also be indirect, from a local variable of type {@code number} (recommended).</td></tr>
*
* <tr><td>{@code offset}<br>
* Used to support plural with an offset.<br>
* Can also be indirect, from a local variable of type {@code number} (recommended).</td></tr>
*
* <tr>
* <td>{@code selectordinal}</td>
* <td>Similar to the ICU {@code "selectordinal"}.<br>
* For now it accepts the same parameters as "plural", although there is no use case for them.<br>
* TBD if this will be merged into "plural" (with some {@code kind} option) or not.</td></tr>
*
* <tr><td>{@code select}</td><td>Literal match, same as the ICU4 {@code "select"}.</td></tr>
* </table>
*
* @deprecated This API is for technology preview only.
* @hide Only a subset of ICU is exposed in Android
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public class MessageFormatter {
private final Locale locale;
private final String pattern;
private final Mf2FunctionRegistry functionRegistry;
private final Mf2DataModel dataModel;
private final Mf2DataModelFormatter modelFormatter;
private MessageFormatter(Builder builder) {
this.locale = builder.locale;
this.functionRegistry = builder.functionRegistry;
if ((builder.pattern == null && builder.dataModel == null)
|| (builder.pattern != null && builder.dataModel != null)) {
throw new IllegalArgumentException("You need to set either a pattern, or a dataModel, but not both.");
}
if (builder.dataModel != null) {
this.dataModel = builder.dataModel;
this.pattern = Mf2Serializer.dataModelToString(this.dataModel);
} else {
this.pattern = builder.pattern;
Mf2Serializer tree = new Mf2Serializer();
Mf2Parser parser = new Mf2Parser(pattern, tree);
try {
parser.parse_Message();
dataModel = tree.build();
} catch (Mf2Parser.ParseException pe) {
throw new IllegalArgumentException(
"Parse error:\n"
+ "Message: <<" + pattern + ">>\n"
+ "Error:" + parser.getErrorMessage(pe) + "\n");
}
}
modelFormatter = new Mf2DataModelFormatter(dataModel, locale, functionRegistry);
}
/**
* Creates a builder.
*
* @return the Builder.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public static Builder builder() {
return new Builder();
}
/**
* Get the locale to use for all the formatting and selections in
* the current {@code MessageFormatter}.
*
* @return the locale.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public Locale getLocale() {
return locale;
}
/**
* Get the pattern (the serialized message in MessageFormat 2 syntax) of
* the current {@code MessageFormatter}.
*
* <p>If the {@code MessageFormatter} was created from an {@link Mf2DataModel}
* the this string is generated from that model.</p>
*
* @return the pattern.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public String getPattern() {
return pattern;
}
/**
* Give public access to the message data model.
*
* <p>This data model is similar to the functionality we have today
* in {@link android.icu.text.MessagePatternUtil} maybe even a bit more higher level.</p>
*
* <p>We can also imagine a model where one parses the string syntax, takes the data model,
* modifies it, and then uses that modified model to create a {@code MessageFormatter}.</p>
*
* @return the data model.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public Mf2DataModel getDataModel() {
return dataModel;
}
/**
* Formats a map of objects by iterating over the MessageFormat's pattern,
* with the plain text “as is” and the arguments replaced by the formatted objects.
*
* @param arguments a map of objects to be formatted and substituted.
* @return the string representing the message with parameters replaced.
*
* @throws IllegalArgumentException when something goes wrong
* (for example wrong argument type, or null arguments, etc.)
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public String formatToString(Map<String, Object> arguments) {
return modelFormatter.format(arguments);
}
/**
* Not yet implemented: formats a map of objects by iterating over the MessageFormat's
* pattern, with the plain text “as is” and the arguments replaced by the formatted objects.
*
* @param arguments a map of objects to be formatted and substituted.
* @return the {@link FormattedMessage} class representing the message with parameters replaced.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public FormattedMessage format(Map<String, Object> arguments) {
throw new RuntimeException("Not yet implemented.");
}
/**
* A {@code Builder} used to build instances of {@link MessageFormatter}.
*
* @deprecated This API is for technology preview only.
* @hide Only a subset of ICU is exposed in Android
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public static class Builder {
private Locale locale = Locale.getDefault(Locale.Category.FORMAT);
private String pattern = null;
private Mf2FunctionRegistry functionRegistry = Mf2FunctionRegistry.builder().build();
private Mf2DataModel dataModel = null;
// Prevent direct creation
private Builder() {
}
/**
* Sets the locale to use for all formatting and selection operations.
*
* @param locale the locale to set.
* @return the builder, for fluent use.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public Builder setLocale(Locale locale) {
this.locale = locale;
return this;
}
/**
* Sets the pattern (in MessageFormat 2 syntax) used to create the message.<br>
* It conflicts with the data model, so it will reset it (the last call on setter wins).
*
* @param pattern the pattern to set.
* @return the builder, for fluent use.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public Builder setPattern(String pattern) {
this.pattern = pattern;
this.dataModel = null;
return this;
}
/**
* Sets an instance of {@link Mf2FunctionRegistry} that should register any
* custom functions used by the message.
*
* <p>There is no need to do this in order to use standard functions
* (for example date / time / number formatting, plural / ordinal / literal selection).<br>
* The exact set of standard functions, with the types they format and the options
* they accept is still TBD.</p>
*
* @param functionRegistry the function registry to set.
* @return the builder, for fluent use.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public Builder setFunctionRegistry(Mf2FunctionRegistry functionRegistry) {
this.functionRegistry = functionRegistry;
return this;
}
/**
* Sets the data model used to create the message.<br>
* It conflicts with the pattern, so it will reset it (the last call on setter wins).
*
* @param dataModel the pattern to set.
* @return the builder, for fluent use.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public Builder setDataModel(Mf2DataModel dataModel) {
this.dataModel = dataModel;
this.pattern = null;
return this;
}
/**
* Builds an instance of {@link MessageFormatter}.
*
* @return the {@link MessageFormatter} created.
*
* @deprecated This API is for technology preview only.
* @hide draft / provisional / internal are hidden on Android
*/
@Deprecated
public MessageFormatter build() {
return new MessageFormatter(this);
}
}
}