Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.google.gwt.i18n.rebind.MessagesMethodCreator Maven / Gradle / Ivy
/*
* Copyright 2008 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package com.google.gwt.i18n.rebind;
import com.google.gwt.core.ext.TreeLogger;
import com.google.gwt.core.ext.UnableToCompleteException;
import com.google.gwt.core.ext.typeinfo.JArrayType;
import com.google.gwt.core.ext.typeinfo.JClassType;
import com.google.gwt.core.ext.typeinfo.JEnumConstant;
import com.google.gwt.core.ext.typeinfo.JEnumType;
import com.google.gwt.core.ext.typeinfo.JField;
import com.google.gwt.core.ext.typeinfo.JMethod;
import com.google.gwt.core.ext.typeinfo.JParameter;
import com.google.gwt.core.ext.typeinfo.JParameterizedType;
import com.google.gwt.core.ext.typeinfo.JPrimitiveType;
import com.google.gwt.core.ext.typeinfo.JType;
import com.google.gwt.core.ext.typeinfo.TypeOracle;
import com.google.gwt.i18n.client.DateTimeFormat;
import com.google.gwt.i18n.client.DateTimeFormat.PredefinedFormat;
import com.google.gwt.i18n.client.Messages.AlternateMessage;
import com.google.gwt.i18n.client.Messages.Offset;
import com.google.gwt.i18n.client.Messages.Optional;
import com.google.gwt.i18n.client.Messages.PluralCount;
import com.google.gwt.i18n.client.Messages.PluralText;
import com.google.gwt.i18n.client.Messages.Select;
import com.google.gwt.i18n.client.NumberFormat;
import com.google.gwt.i18n.client.PluralRule;
import com.google.gwt.i18n.client.PluralRule.PluralForm;
import com.google.gwt.i18n.client.impl.plurals.DefaultRule;
import com.google.gwt.i18n.rebind.AbstractResource.MissingResourceException;
import com.google.gwt.i18n.rebind.AbstractResource.ResourceEntry;
import com.google.gwt.i18n.rebind.AbstractResource.ResourceList;
import com.google.gwt.i18n.rebind.MessageFormatParser.ArgumentChunk;
import com.google.gwt.i18n.rebind.MessageFormatParser.DefaultTemplateChunkVisitor;
import com.google.gwt.i18n.rebind.MessageFormatParser.StaticArgChunk;
import com.google.gwt.i18n.rebind.MessageFormatParser.StringChunk;
import com.google.gwt.i18n.rebind.MessageFormatParser.TemplateChunk;
import com.google.gwt.i18n.shared.GwtLocale;
import com.google.gwt.safehtml.shared.OnlyToBeUsedInGeneratedCodeStringBlessedAsSafeHtml;
import com.google.gwt.safehtml.shared.SafeHtml;
import com.google.gwt.safehtml.shared.SafeHtmlBuilder;
import com.google.gwt.user.rebind.AbstractGeneratorClassCreator;
import com.google.gwt.user.rebind.AbstractMethodCreator;
import com.google.gwt.user.rebind.SourceWriter;
import org.apache.tapestry.util.text.LocalizedPropertiesLoader;
import java.io.IOException;
import java.io.InputStream;
import java.text.ParseException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
/**
* Creator for methods of the Messages interface.
*/
@SuppressWarnings("deprecation") // for @PluralText
class MessagesMethodCreator extends AbstractMethodCreator {
private abstract static class AlternateFormSelector {
protected final int argNumber;
protected final JType argType;
public AlternateFormSelector(TreeLogger logger, int argNumber, JParameter[] params) {
this.argNumber = argNumber;
this.argType = params[argNumber].getType();
}
public abstract void generatePrepCode(SourceWriter out);
public abstract void generateSelectEnd(SourceWriter out);
public abstract void generateSelectMatchEnd(SourceWriter out, String value);
/**
* @param out
* @param logger
* @param value
* @throws UnableToCompleteException
*/
public abstract void generateSelectMatchStart(SourceWriter out,
TreeLogger logger, String value) throws UnableToCompleteException;
public abstract void generateSelectStart(SourceWriter out,
boolean exactMatches);
public abstract void issueWarnings(TreeLogger logger, JMethod m,
GwtLocale locale);
}
/**
* Implements {x,date...} references in MessageFormat.
*/
private static class DateFormatter implements ValueFormatter {
public boolean format(TreeLogger logger, GwtLocale locale,
StringGenerator out, Map formatArgs, String subformat,
String argName, JType argType, Parameters params) {
if (!"java.util.Date".equals(argType.getQualifiedSourceName())) {
logger.log(TreeLogger.ERROR, "Only java.util.Date acceptable for date format");
return true;
}
String tzParam = "";
String tzArg = formatArgs.get("tz");
if (tzArg != null) {
if (tzArg.startsWith("$")) {
int paramNum = params.getParameterIndex(tzArg.substring(1));
if (paramNum < 0) {
logger.log(TreeLogger.ERROR, "Unable to resolve tz argument " + tzArg);
return true;
} else if (!"com.google.gwt.i18n.client.TimeZone".equals(
params.getParameter(paramNum).getType().getQualifiedSourceName())) {
logger.log(TreeLogger.ERROR, "Currency code parameter must be TimeZone");
return true;
} else {
tzParam = ", arg" + paramNum;
}
} else {
tzParam = ", com.google.gwt.i18n.client.TimeZone.createTimeZone(" + tzArg + ")";
}
}
if (subformat == null || "medium".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getMediumDateFormat()" + ".format(" + argName + tzParam + ")");
} else if ("full".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getFullDateFormat().format(" + argName + tzParam + ")");
} else if ("long".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getLongDateFormat().format(" + argName + tzParam + ")");
} else if ("short".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getShortDateFormat()" + ".format(" + argName + tzParam + ")");
} else {
logger.log(TreeLogger.WARN, "Use localdatetime format instead");
out.appendStringValuedExpression(
dtFormatClassName + ".getFormat(" + wrap(subformat) + ").format(" + argName + tzParam
+ ")");
}
return false;
}
}
/**
* Comparator that ensures all exact value matches (=N) strings come before
* all non-exact matches.
*/
private static class ExactValueComparator implements Comparator {
private static int compareOne(String a, String b) {
boolean aExact = a.startsWith("=");
boolean bExact = a.startsWith("=");
if (aExact != bExact) {
return aExact ? -1 : 1;
}
if (aExact) {
return a.substring(1).compareTo(b.substring(1));
} else {
return a.compareTo(b);
}
}
public int compare(String a, String b) {
String[] aSplit = a.split("\\|");
String[] bSplit = b.split("\\|");
int c = 0;
for (int i = 0; c == 0 && i < aSplit.length && i < bSplit.length; ++i) {
c = compareOne(aSplit[i], bSplit[i]);
}
if (c == 0 && aSplit.length != bSplit.length) {
c = aSplit.length < bSplit.length ? -1 : 1;
}
return c;
}
}
/**
* An {@link AlternateFormSelector} used with {@link Select}.
*/
private static class GenericSelector extends AlternateFormSelector {
private final JEnumType enumType;
private final boolean isString;
private boolean startedIfChain;
/**
* @param logger
* @param m
* @param i
* @param params
* @throws UnableToCompleteException
*/
public GenericSelector(TreeLogger logger, JMethod m, int i,
JParameter[] params) throws UnableToCompleteException {
super(logger, i, params);
JPrimitiveType primType = argType.isPrimitive();
JClassType classType = argType.isClass();
JEnumType tempEnumType = null;
boolean tempIsString = false;
if (primType != null) {
if (primType == JPrimitiveType.DOUBLE
|| primType == JPrimitiveType.FLOAT) {
throw error(logger, m.getName() + ": @Select arguments may only be"
+ " integral primitives, boolean, enums, or String");
}
} else if (classType != null) {
tempEnumType = classType.isEnum();
tempIsString = "java.lang.String".equals(classType.getQualifiedSourceName());
if (tempEnumType == null && !tempIsString) {
throw error(logger, m.getName() + ": @Select arguments may only be"
+ " integral primitives, boolean, enums, or String");
}
} else {
throw error(logger, m.getName() + ": @Select arguments may only be"
+ " integral primitives, boolean, enums, or String");
}
enumType = tempEnumType;
isString = tempIsString;
}
@Override
public void generatePrepCode(SourceWriter out) {
if (enumType != null) {
out.println("int arg" + argNumber + "_ordinal = -1;");
out.println("if (arg" + argNumber + " != null) {");
out.indent();
out.println("arg" + argNumber + "_ordinal = arg" + argNumber
+ ".ordinal();");
out.outdent();
out.println("}");
}
}
@Override
public void generateSelectEnd(SourceWriter out) {
if (!startedIfChain) {
out.outdent();
}
out.println("}");
}
@Override
public void generateSelectMatchEnd(SourceWriter out, String value) {
if (!startedIfChain) {
out.println("break;");
}
out.outdent();
}
@Override
public void generateSelectMatchStart(SourceWriter out, TreeLogger logger,
String value) throws UnableToCompleteException {
if (isString) {
if (startedIfChain) {
out.print("} else ");
} else {
startedIfChain = true;
}
if ("other".equals(value)) {
out.println("{ // other");
} else {
value = value.replace("\"", "\\\"");
out.println("if (\"" + value + "\".equals(arg" + argNumber
+ ")) {");
}
} else {
if ("other".equals(value)) {
out.println("default: // other");
} else if (enumType != null) {
JField field = enumType.findField(value);
JEnumConstant enumConstant = null;
if (field != null) {
enumConstant = field.isEnumConstant();
}
if (field == null || enumConstant == null) {
throw error(logger, "'" + value + "' is not a valid value of "
+ enumType.getQualifiedSourceName() + " or 'other'");
}
out.println("case " + enumConstant.getOrdinal() + ": // " + value);
} else {
long longVal;
try {
longVal = Long.parseLong(value);
} catch (NumberFormatException e) {
throw error(logger, "'" + value + "' is not a valid numeric value",
e);
}
out.println("case " + longVal + ":");
}
}
out.indent();
}
@Override
public void generateSelectStart(SourceWriter out, boolean exactMatches) {
// ignore exactMatches, so "=VALUE" is the same as "VALUE"
if (isString) {
startedIfChain = false;
return;
}
String suffix = "";
if (enumType != null) {
suffix = "_ordinal";
}
out.println("switch (arg" + argNumber + suffix + ") {");
out.indent();
}
@Override
public void issueWarnings(TreeLogger logger, JMethod m, GwtLocale locale) {
// nothing to warn about
}
}
/**
* Interface used to abstract away differences between accessing an array and
* a list.
*/
private interface ListAccessor {
String getElement(String element);
String getSize();
}
/**
* Implementation of ListAccessor for an array.
*/
private static class ListAccessorArray implements ListAccessor {
private final int listArgNum;
public ListAccessorArray(int listArgNum) {
this.listArgNum = listArgNum;
}
public String getElement(String element) {
return "arg" + listArgNum + "[" + element + "]";
}
public String getSize() {
return "arg" + listArgNum + ".length";
}
}
/**
* Implementation of ListAccessor for a List.
*/
private static class ListAccessorList implements ListAccessor {
private final int listArgNum;
public ListAccessorList(int listArgNum) {
this.listArgNum = listArgNum;
}
public String getElement(String element) {
return "arg" + listArgNum + ".get(" + element + ")";
}
public String getSize() {
return "arg" + listArgNum + ".size()";
}
}
/**
* Implements {x,localdatetime,skeleton} references in MessageFormat.
*/
private static class LocalDateTimeFormatter implements ValueFormatter {
private static final String PREDEF = "predef:";
public boolean format(TreeLogger logger,
GwtLocale locale,
StringGenerator out,
Map formatArgs,
String subformat,
String argName,
JType argType,
Parameters params) {
if (!"java.util.Date".equals(argType.getQualifiedSourceName())) {
logger.log(TreeLogger.ERROR, "Only java.util.Date acceptable for localdatetime format");
return true;
}
if (subformat == null || subformat.length() == 0) {
logger.log(TreeLogger.ERROR, "localdatetime format requires a skeleton pattern");
return true;
}
String tzParam = "";
String tzArg = formatArgs.get("tz");
if (tzArg != null) {
if (tzArg.startsWith("$")) {
int paramNum = params.getParameterIndex(tzArg.substring(1));
if (paramNum < 0) {
logger.log(TreeLogger.ERROR, "Unable to resolve tz argument " + tzArg);
return true;
} else if (!"com.google.gwt.i18n.client.TimeZone".equals(
params.getParameter(paramNum).getType().getQualifiedSourceName())) {
logger.log(TreeLogger.ERROR, "tz parameter must be of type TimeZone");
return true;
} else {
tzParam = ", arg" + paramNum;
}
} else {
tzParam = ", com.google.gwt.i18n.client.TimeZone.createTimeZone(" + tzArg + ")";
}
}
if (subformat.startsWith(PREDEF)) {
// TODO(jat): error checking/logging
PredefinedFormat predef;
try {
predef = PredefinedFormat.valueOf(subformat.substring(PREDEF.length()));
} catch (IllegalArgumentException e) {
logger.log(TreeLogger.ERROR, "Unrecognized predefined format '" + subformat + "'");
return true;
}
out.appendStringValuedExpression(
dtFormatClassName + ".getFormat("
+ PredefinedFormat.class.getCanonicalName() + "."
+ predef.toString() + ").format(" + argName + tzParam + ")");
return false;
}
DateTimePatternGenerator dtpg = new DateTimePatternGenerator(locale);
try {
String pattern = dtpg.getBestPattern(subformat);
if (pattern == null) {
logger.log(
TreeLogger.ERROR, "Invalid localdatetime skeleton pattern \"" + subformat + "\"");
return true;
}
out.appendStringValuedExpression(
dtFormatClassName + ".getFormat(" + wrap(pattern) + ").format(" + argName + tzParam
+ ")");
} catch (IllegalArgumentException e) {
logger.log(TreeLogger.ERROR, "Unable to parse '" + subformat + ": " + e.getMessage());
return true;
}
return false;
}
}
/**
* Implements {x,number...} references in MessageFormat.
*/
private static class NumberFormatter implements ValueFormatter {
public boolean format(TreeLogger logger,
GwtLocale locale,
StringGenerator out,
Map formatArgs,
String subformat,
String argName,
JType argType,
Parameters params) {
JPrimitiveType argPrimType = argType.isPrimitive();
if (argPrimType != null) {
if (argPrimType == JPrimitiveType.BOOLEAN || argPrimType == JPrimitiveType.VOID) {
logger.log(TreeLogger.ERROR, "Illegal argument type for number format");
return true;
}
} else {
JClassType classType = argType.isClass();
if (classType == null) {
logger.log(TreeLogger.ERROR, "Unexpected argument type for number format");
return true;
}
TypeOracle oracle = classType.getOracle();
JClassType numberType = oracle.findType("java.lang.Number");
if (!classType.isAssignableTo(numberType)) {
logger.log(TreeLogger.ERROR, "Only Number subclasses may be formatted as a number");
return true;
}
}
String curCodeParam = "";
String curCode = formatArgs.get("curcode");
if (curCode != null) {
if (curCode.startsWith("$")) {
int paramNum = params.getParameterIndex(curCode.substring(1));
if (paramNum < 0) {
logger.log(TreeLogger.ERROR, "Unable to resolve curcode argument " + curCode);
return true;
} else if (!"java.lang.String".equals(
params.getParameter(paramNum).getType().getQualifiedSourceName())) {
logger.log(TreeLogger.ERROR, "Currency code parameter must be String");
return true;
} else {
curCodeParam = "arg" + paramNum;
}
} else {
curCodeParam = '"' + curCode + '"';
}
}
if (subformat == null) {
out.appendStringValuedExpression(
numFormatClassName + ".getDecimalFormat().format(" + argName + ")");
} else if ("integer".equals(subformat)) {
out.appendStringValuedExpression(
numFormatClassName + ".getIntegerFormat().format(" + argName + ")");
} else if ("currency".equals(subformat)) {
out.appendStringValuedExpression(
numFormatClassName + ".getCurrencyFormat(" + curCodeParam + ").format(" + argName
+ ")");
} else if ("percent".equals(subformat)) {
out.appendStringValuedExpression(
numFormatClassName + ".getPercentFormat().format(" + argName + ")");
} else {
if (curCodeParam.length() > 0) {
curCodeParam = ", " + curCodeParam;
}
out.appendStringValuedExpression(
numFormatClassName + ".getFormat(" + wrap(subformat) + curCodeParam + ").format("
+ argName + ")");
}
return false;
}
}
/**
* An interface for accessing parameters, giving the ability to record
* accesses.
*/
private interface Parameters {
/**
* Allow generated code to take advantage of plural offsets (see
* {@link Offset}).
*/
void enablePluralOffsets();
/**
* Return the count of parameters.
* @return the count of parameters
*/
int getCount();
/**
* Returns the given parameter.
*
* @param i index of the parameter to return, 0 .. getCount() - 1
* @return parameter or null if i is out of range
*/
JParameter getParameter(int i);
/**
* Returns the given parameter.
*
* @param name the name of the parameter to return
* @return parameter or null if the named parameter doesn't exist
*/
JParameter getParameter(String name);
/**
* Return an expression to get the value of the requested parameter. Note
* that for arrays or lists this will return an expression giving the count
* of items in the array or list.
*
* @param i index of the paramter, 0 .. getCount() - 1
* @return the source of code to access the parameter value
*/
String getParameterExpression(int i);
/**
* Find the index of a parameter by name.
*
* @param name
* @return index of requested parameter or -1 if not found
*/
int getParameterIndex(String name);
}
private static class ParametersImpl implements Parameters {
private JParameter[] params;
private boolean[] seenFlag;
private int[] offset;
private boolean[] isList;
private boolean[] isArray;
private boolean enablePluralOffsets;
public ParametersImpl(JParameter[] params, boolean[] seenFlag) {
this.params = params;
this.seenFlag = seenFlag;
int n = params.length;
offset = new int[n];
isList = new boolean[n];
isArray = new boolean[n];
for (int i = 0; i < n; ++i) {
Offset offsetAnnot = params[i].getAnnotation(Offset.class);
if (offsetAnnot != null) {
offset[i] = offsetAnnot.value();
}
JType type = params[i].getType();
if (type.isArray() != null) {
isArray[i] = true;
} else if (type.isInterface() != null) {
JClassType rawType = type.isInterface().getErasedType();
if ("java.util.List".equals(rawType.getQualifiedSourceName())) {
isList[i] = true;
}
}
}
}
public void enablePluralOffsets() {
enablePluralOffsets = true;
}
public int getCount() {
return params.length;
}
public JParameter getParameter(int i) {
if (i < 0 || i >= params.length) {
return null;
}
seenFlag[i] = true;
return params[i];
}
public JParameter getParameter(String name) {
return getParameter(getParameterIndex(name));
}
public String getParameterExpression(int i) {
if (i < 0 || i >= params.length) {
return null;
}
String argName = "arg" + i;
seenFlag[i] = true;
if (enablePluralOffsets && offset[i] != 0) {
return argName + "_count";
}
if (isArray[i]) {
return argName + ".length";
}
if (isList[i]) {
return argName + ".size()";
}
return argName;
}
public int getParameterIndex(String name) {
for (int i = 0; i < params.length; ++i) {
if (params[i].getName().equals(name)) {
return i;
}
}
return -1;
}
}
/**
* An {@link AlternateFormSelector} used with {@link PluralCount}.
*/
private static class PluralFormSelector extends AlternateFormSelector {
protected final String countSuffix;
protected final String listSuffix;
protected final Set missingPluralForms;
protected final int pluralOffset;
protected final PluralRule pluralRule;
private boolean hasExactMatches;
private boolean inExactMatches;
// used to generate unique case values for bogus plural forms
private int bogusCaseValue = 1000;
public PluralFormSelector(TreeLogger logger, JMethod method, int argNumber,
JParameter[] params, GwtLocale locale)
throws UnableToCompleteException {
super(logger, argNumber, params);
PluralCount pluralCount = params[argNumber].getAnnotation(
PluralCount.class);
Class extends PluralRule> ruleClass = pluralCount.value();
if (ruleClass == PluralRule.class) {
ruleClass = DefaultRule.class;
}
pluralRule = createLocalizedPluralRule(logger,
method.getEnclosingType().getOracle(), ruleClass, locale);
missingPluralForms = new HashSet();
for (PluralForm form : pluralRule.pluralForms()) {
if (form.getWarnIfMissing() && !"other".equals(form.getName())) {
missingPluralForms.add(form.getName());
}
}
Offset offsetAnnot = params[argNumber].getAnnotation(Offset.class);
int offset = 0;
if (offsetAnnot != null) {
offset = offsetAnnot.value();
}
this.pluralOffset = offset;
boolean isArray = false;
boolean isList = false;
JPrimitiveType primType = argType.isPrimitive();
JClassType classType = argType.isInterface();
if (classType != null) {
classType = classType.getErasedType();
if ("java.util.List".equals(classType.getQualifiedSourceName())) {
isList = true;
} else {
classType = null;
}
}
JArrayType arrayType = argType.isArray();
if (arrayType != null) {
isArray = true;
}
if (!isList && !isArray && (primType == null
|| (primType != JPrimitiveType.INT
&& primType != JPrimitiveType.SHORT))) {
throw error(logger, method.getName()
+ ": PluralCount parameter must be int, short, array, or List");
}
String tempListSuffix = "";
if (isList) {
tempListSuffix = ".size()";
} else if (isArray) {
tempListSuffix = ".length";
}
String tempCountSuffix = tempListSuffix;
if (isList || isArray || offset != 0) {
tempCountSuffix = "_count";
}
listSuffix = tempListSuffix;
countSuffix = tempCountSuffix;
}
@Override
public void generatePrepCode(SourceWriter out) {
// save a value with the count value, applying an offset if present
if (countSuffix.length() > 0) {
out.print("int arg" + argNumber + countSuffix + " = arg" + argNumber
+ listSuffix);
if (pluralOffset != 0) {
out.print(" - " + pluralOffset);
}
out.println(";");
}
// save the selected plural form
// TODO(jat): cache instances of the same plural rule?
out.println("int arg" + argNumber + "_form = new "
+ pluralRule.getClass().getCanonicalName()
+ "().select(arg" + argNumber + countSuffix + ");");
}
@Override
public void generateSelectEnd(SourceWriter out) {
if (hasExactMatches && !inExactMatches) {
// undo extra nesting level
out.outdent();
out.println("}");
out.println("break;");
out.outdent();
}
out.outdent();
out.println("}");
}
@Override
public void generateSelectMatchEnd(SourceWriter out, String value) {
out.println("break;");
out.outdent();
}
@Override
public void generateSelectMatchStart(SourceWriter out, TreeLogger logger,
String value) throws UnableToCompleteException {
missingPluralForms.remove(value);
if (value.startsWith("=")) {
try {
long val = Long.parseLong(value.substring(1));
out.println("case " + val + ": // " + value);
} catch (NumberFormatException e) {
throw error(logger, "Exact match value '" + value
+ "' must be integral", e);
}
out.indent();
return;
}
if (inExactMatches) {
/*
* If this is the first non-exact value, create a nested select that
* chooses the message based on the plural form only if no exact values
* matched.
*/
inExactMatches = false;
out.println("default: // non-exact matches");
out.indent();
out.println("switch (arg" + argNumber + "_form) {");
out.indent();
}
if ("other".equals(value)) {
out.println("default: // other");
out.indent();
return;
}
PluralForm[] pluralForms = pluralRule.pluralForms();
for (int i = 0; i < pluralForms.length; ++i) {
if (pluralForms[i].getName().equals(value)) {
out.println("case " + i + ": // " + value);
out.indent();
return;
}
}
logger.log(TreeLogger.WARN, "Plural form '" + value + "' unknown in "
+ pluralRule.getClass().getCanonicalName() + ": ignoring");
// TODO(jat): perhaps return a failure instead, and let the called skip
// the nested selector code? It gets complicated really quick though.
out.println("case " + (bogusCaseValue++) + ": // unknown plural form '"
+ value + "'");
out.indent();
}
@Override
public void generateSelectStart(SourceWriter out, boolean hasExactMatches) {
this.hasExactMatches = hasExactMatches;
inExactMatches = hasExactMatches;
String suffix = hasExactMatches ? listSuffix : "_form";
out.println("switch (arg" + argNumber + suffix + ") {");
out.indent();
}
public PluralForm[] getPluralForms() {
return pluralRule.pluralForms();
}
@Override
public void issueWarnings(TreeLogger logger, JMethod m, GwtLocale locale) {
if (!missingPluralForms.isEmpty()) {
// TODO(jat): avoid giving warnings for values that are not necessary
// due to exact value matches. For example, in English there is no need
// for ONE if the =1 value was given, and it may be important to have
// the =1 value across all locales.
logger.log(TreeLogger.WARN, "In locale '" + locale
+ "', required plural forms are missing: " + missingPluralForms);
}
}
}
/**
* Implements {x,time...} references in MessageFormat.
*/
private static class TimeFormatter implements ValueFormatter {
public boolean format(TreeLogger logger,
GwtLocale locale,
StringGenerator out,
Map formatArgs,
String subformat,
String argName,
JType argType,
Parameters params) {
if (!"java.util.Date".equals(argType.getQualifiedSourceName())) {
logger.log(TreeLogger.ERROR, "Only java.util.Date acceptable for date format");
return true;
}
String tzParam = "";
String tzArg = formatArgs.get("tz");
if (tzArg != null) {
if (tzArg.startsWith("$")) {
int paramNum = params.getParameterIndex(tzArg.substring(1));
if (paramNum < 0) {
logger.log(TreeLogger.ERROR, "Unable to resolve tz argument " + tzArg);
return true;
} else if (!"com.google.gwt.i18n.client.TimeZone".equals(
params.getParameter(paramNum).getType().getQualifiedSourceName())) {
logger.log(TreeLogger.ERROR, "Currency code parameter must be TimeZone");
return true;
} else {
tzParam = ", arg" + paramNum;
}
} else {
tzParam = ", com.google.gwt.i18n.client.TimeZone.createTimeZone(" + tzArg + ")";
}
}
if (subformat == null || "medium".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getMediumTimeFormat().format(" + argName + tzParam + ")");
} else if ("full".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getFullTimeFormat().format(" + argName + tzParam + ")");
} else if ("long".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getLongTimeFormat().format(" + argName + tzParam + ")");
} else if ("short".equals(subformat)) {
out.appendStringValuedExpression(
dtFormatClassName + ".getShortTimeFormat().format(" + argName + tzParam + ")");
} else {
logger.log(TreeLogger.WARN, "Use localdatetime format instead");
out.appendStringValuedExpression(
dtFormatClassName + ".getFormat(" + wrap(subformat) + ").format(" + argName + tzParam
+ ")");
}
return false;
}
}
private interface ValueFormatter {
/**
* Creates code to format a value according to a format string.
*
* @param logger
* @param locale current locale
* @param out StringBuffer to append to
* @param formatArgs format-specific arguments
* @param subformat the remainder of the format string
* @param argName the name of the argument to use in the generated code
* @param argType the type of the argument
* @param params argument list or null
* @return true if a fatal error occurred (which will already be logged)
*/
boolean format(TreeLogger logger,
GwtLocale locale,
StringGenerator out,
Map formatArgs,
String subformat,
String argName,
JType argType,
Parameters params);
}
/**
* Class names, in a refactor-friendly manner.
*/
private static final String dtFormatClassName = DateTimeFormat.class.getCanonicalName();
/**
* Fully-qualified class name of the SafeHtml interface.
*/
public static final String SAFE_HTML_FQCN = SafeHtml.class.getCanonicalName();
/**
* Fully-qualified class name of the SafeHtmlBuilder class.
*/
public static final String SAFE_HTML_BUILDER_FQCN = SafeHtmlBuilder.class.getCanonicalName();
/**
* Map of supported formats.
*/
private static Map formatters = new HashMap();
private static final String numFormatClassName = NumberFormat.class.getCanonicalName();
/*
* Register supported formats.
*/
static {
formatters.put("date", new DateFormatter());
formatters.put("number", new NumberFormatter());
formatters.put("time", new TimeFormatter());
formatters.put("localdatetime", new LocalDateTimeFormatter());
}
/**
* Creates an instance of a locale-specific plural rule implementation.
*
* Note that this uses TypeOracle's ability to find all subclasses of the
* supplied parent class, then uses reflection to actually load the class.
* This works because PluralRule instances are required to be translatable,
* since part of them is executed at runtime and part at compile time.
*
* @param logger TreeLogger instance
* @param oracle TypeOracle instance to use
* @param ruleClass PluralRule implementation to localize
* @param locale current locale we are compiling for
* @return an instance of a PluralRule implementation. If an appropriate
* implementation of the requested class cannot be found, an instance
* of DefaultRule is used instead as a default of last resort.
* @throws UnableToCompleteException if findDerivedClasses fails
*
* TODO: consider impact of possibly having multiple TypeOracles
*/
private static PluralRule createLocalizedPluralRule(
TreeLogger logger, TypeOracle oracle, Class extends PluralRule> ruleClass, GwtLocale locale)
throws UnableToCompleteException {
String baseName = ruleClass.getCanonicalName();
JClassType ruleJClassType = oracle.findType(baseName);
Map matchingClasses =
LocalizableLinkageCreator.findDerivedClasses(logger, ruleJClassType);
for (GwtLocale search : locale.getCompleteSearchList()) {
JClassType localizedType = matchingClasses.get(search.toString());
if (localizedType != null) {
try {
Class> testClass = Class.forName(
localizedType.getQualifiedBinaryName(), false, PluralRule.class.getClassLoader());
if (PluralRule.class.isAssignableFrom(testClass)) {
return (PluralRule) testClass.newInstance();
}
} catch (ClassCastException e) {
// ignore classes of the wrong type
} catch (ClassNotFoundException e) {
// ignore missing classes
} catch (InstantiationException e) {
// skip classes we can't instantiate
} catch (IllegalAccessException e) {
// ignore inaccessible classes
}
}
}
// default of last resort
return new DefaultRule();
}
private final Map> listPatternCache;
private SourceWriter writer;
/**
* Constructor for MessagesMethodCreator
.
*
* @param classCreator associated class creator
* @param writer
*/
public MessagesMethodCreator(AbstractGeneratorClassCreator classCreator,
SourceWriter writer) {
super(classCreator);
listPatternCache = new HashMap>();
this.writer = writer;
}
@Override
public void createMethodFor(TreeLogger logger, JMethod m, String key,
ResourceList resourceList, GwtLocale locale)
throws UnableToCompleteException {
ResourceEntry resourceEntry = resourceList.getEntry(key);
if (resourceEntry == null) {
throw new MissingResourceException(key, resourceList);
}
JParameter[] params = m.getParameters();
boolean seenPluralCount = false;
boolean seenSelect = false;
int numParams = params.length;
List selectors = new ArrayList();
// See if any parameter is tagged as a PluralCount or Select parameter.
for (int i = 0; i < numParams; ++i) {
PluralCount pluralCount = params[i].getAnnotation(PluralCount.class);
Select select = params[i].getAnnotation(Select.class);
if (pluralCount != null && select != null) {
throw error(logger, params[i].getName() + " cannot be both @PluralCount"
+ " and @Select");
}
AlternateFormSelector selector = null;
if (select != null) {
selector = new GenericSelector(logger, m, i, params);
seenSelect = true;
} else if (pluralCount != null) {
PluralFormSelector pluralSelector = new PluralFormSelector(logger, m, i,
params, locale);
selector = pluralSelector;
if (!seenPluralCount) {
// TODO(jat): what if we have different plural rules on the different
// forms?
resourceList.setPluralForms(key, pluralSelector.getPluralForms());
}
seenPluralCount = true;
}
if (selector != null) {
selectors.add(selector);
}
}
boolean[] seenFlags = new boolean[numParams];
final Parameters paramsAccessor = new ParametersImpl(params, seenFlags);
boolean isSafeHtml = m.getReturnType().getQualifiedSourceName().equals(
SAFE_HTML_FQCN);
String template = resourceEntry.getForm(null);
if (template == null) {
logger.log(TreeLogger.ERROR,"No default form for method " + m.getName()
+ "' in " + m.getEnclosingType() + " for locale " + locale, null);
throw new UnableToCompleteException();
}
// Generate code to format any lists
// TODO(jat): handle messages with different list formats in alternate forms
try {
for (TemplateChunk chunk : MessageFormatParser.parse(template)) {
if (chunk instanceof ArgumentChunk) {
ArgumentChunk argChunk = (ArgumentChunk) chunk;
if (argChunk.isList()) {
ListAccessor listAccessor = null;
int listArgNum = argChunk.getArgumentNumber();
JType listType = params[listArgNum].getType();
JClassType classType = listType.isInterface();
JType elemType = null;
if (classType != null) {
if ("java.util.List".equals(
classType.getErasedType().getQualifiedSourceName())) {
listAccessor = new ListAccessorList(listArgNum);
} else {
logger.log(TreeLogger.ERROR, "Parameters formatted as lists "
+ "must be declared as java.util.List or arrays in "
+ m.getEnclosingType().getSimpleSourceName() + "."
+ m.getName());
throw new UnableToCompleteException();
}
JParameterizedType paramType = classType.isParameterized();
if (paramType != null) {
elemType = paramType.getTypeArgs()[0];
} else {
elemType = classType.getOracle().getJavaLangObject();
}
} else {
JArrayType arrayType = listType.isArray();
if (arrayType != null) {
elemType = arrayType.getComponentType();
listAccessor = new ListAccessorArray(listArgNum);
}
}
generateListFormattingCode(logger, locale, argChunk,
elemType, isSafeHtml, listAccessor, paramsAccessor);
}
}
}
} catch (ParseException pe) {
throw error(logger, "Error parsing '" + template + "'", pe);
}
if (!seenPluralCount && !seenSelect
&& (m.getAnnotation(AlternateMessage.class) != null
|| m.getAnnotation(PluralText.class) != null)) {
logger.log(TreeLogger.WARN, "Unused @AlternateMessage or @PluralText on "
+ m.getEnclosingType().getSimpleSourceName() + "." + m.getName()
+ "; did you intend to mark a @Select or @PluralCount parameter?",
null);
}
Collection resourceForms = resourceEntry.getForms();
if (seenPluralCount) {
paramsAccessor.enablePluralOffsets();
writer.println(m.getReturnType().getParameterizedQualifiedSourceName()
+ " returnVal = null;");
for (AlternateFormSelector selector : selectors) {
selector.generatePrepCode(writer);
}
// sort forms so that all exact-value forms come first
String[] forms = resourceForms.toArray(new String[resourceForms.size()]);
Arrays.sort(forms, new ExactValueComparator());
generateMessageSelectors(logger, m, locale,
resourceEntry, selectors, paramsAccessor, isSafeHtml, forms);
for (AlternateFormSelector selector : selectors) {
selector.issueWarnings(logger, m, locale);
}
writer.println("if (returnVal != null) {");
writer.indent();
writer.println("return returnVal;");
writer.outdent();
writer.println("}");
}
writer.print("return ");
generateString(logger, locale, template, paramsAccessor, writer,
isSafeHtml);
writer.println(";");
// Generate an error if any required parameter was not used somewhere.
for (int i = 0; i < numParams; ++i) {
if (!seenFlags[i]) {
Optional optional = params[i].getAnnotation(Optional.class);
Select select = params[i].getAnnotation(Select.class);
if (optional == null && select == null) {
throw error(logger, "Required argument " + i + " not present: "
+ template);
}
}
}
}
private void formatArg(TreeLogger logger, GwtLocale locale,
StringGenerator buf, ArgumentChunk argChunk, String argExpr,
JType paramType, Parameters params) throws UnableToCompleteException {
String format = argChunk.getFormat();
if (format != null) {
String subformat = argChunk.getSubFormat();
ValueFormatter formatter = formatters.get(format);
if (formatter != null) {
if (formatter.format(logger, locale, buf, argChunk.getFormatArgs(),
subformat, argExpr, paramType, params)) {
throw new UnableToCompleteException();
}
return;
}
}
// no format specified or unknown format
// have to ensure that the result is stringified if necessary
boolean isSafeHtmlTyped = SAFE_HTML_FQCN.equals(paramType.getQualifiedSourceName());
boolean isPrimitiveType = (paramType.isPrimitive() != null);
boolean needsConversionToString =
!("java.lang.String".equals(paramType.getQualifiedSourceName()));
buf.appendExpression(argExpr, isSafeHtmlTyped, isPrimitiveType, needsConversionToString);
}
/**
* Generate code for one list pattern.
*
* @param logger logger to use for error/warning messages
* @param locale locale we are generating code for
* @param listArg the {n,list,...} argument in the original format pattern
* @param val0 the expression defining the {0} argument in the list pattern
* @param val1 the expression defining the {1} argument in the list pattern
* @param elemType the element type of the list/array being rendered as a list
* * @param isSafeHtml true if the resulting string is SafeHtml
* @param listPattern the list pattern to generate code for, ie "{0}, {1}"
* @param formatSecond true if the {1} parameter needs to be formatted
* @param params parameters passed to the Messages method call
* @return a constructed string containing the code to implement the given
* list pattern
* @throws UnableToCompleteException
*/
private CharSequence formatListPattern(final TreeLogger logger,
final GwtLocale locale, final ArgumentChunk listArg, final String val0,
final String val1, final JType elemType, final boolean isSafeHtml,
String listPattern, final boolean formatSecond, final Parameters params)
throws UnableToCompleteException {
final StringBuffer buf = new StringBuffer();
final StringGenerator gen = new StringGenerator(buf, isSafeHtml);
try {
List chunks = MessageFormatParser.parse(listPattern);
for (TemplateChunk chunk : chunks) {
chunk.accept(new DefaultTemplateChunkVisitor() {
@Override
public void visit(ArgumentChunk argChunk) throws UnableToCompleteException {
// The {0} argument in the list pattern always needs formatting,
// but the {1} argument is the part of the list already rendered
// (in either String of SafeHtml form) unless formatSecond is true.
if (argChunk.getArgumentNumber() == 0 || formatSecond) {
formatArg(logger, locale, gen, listArg,
argChunk.getArgumentNumber() == 0 ? val0 : val1, elemType,
params);
} else {
gen.appendExpression(val1, isSafeHtml, false, false);
}
}
@Override
public void visit(StringChunk stringChunk)
throws UnableToCompleteException {
gen.appendStringLiteral(stringChunk.getString());
}
});
}
} catch (ParseException e) {
logger.log(TreeLogger.ERROR,
"Internal error: can't parse list pattern '" + listPattern
+ "' for locale " + locale, e);
throw new UnableToCompleteException();
}
gen.completeString();
return buf;
}
/**
* Generates code to format a list in a format pattern.
*
* @param logger logger to use for error/warning messages
* @param locale locale we are generating code for
* @param listArg the {n,list,...} argument in the original format pattern
* @param elemType the element type of the list/array being rendered as a list
* @param isSafeHtml true if the resulting string is SafeHtml
* @param listAccessor a way to access elements of the list type supplied by
* the user
* @param params parameters passed to the Messages method call
* @throws UnableToCompleteException
*/
private void generateListFormattingCode(TreeLogger logger, GwtLocale locale,
ArgumentChunk listArg, JType elemType,
boolean isSafeHtml, ListAccessor listAccessor, Parameters params)
throws UnableToCompleteException {
Map listPatternParts = getListPatternParts(logger, locale);
int listArgNum = listArg.getArgumentNumber();
writer.println("int arg" + listArgNum + "_size = " + listAccessor.getSize()
+ ";");
if (isSafeHtml) {
writer.println(SafeHtml.class.getCanonicalName() + " arg" + listArgNum
+ "_list = new "
+ OnlyToBeUsedInGeneratedCodeStringBlessedAsSafeHtml.class.getCanonicalName()
+ "(\"\");");
} else {
writer.println("String arg" + listArgNum + "_list = \"\";");
}
writer.println("switch (arg" + listArgNum + "_size) {");
writer.indent();
// TODO(jat): add support for special-cases besides 2 if CLDR ever adds them
String pairPattern = listPatternParts.get("2");
if (pairPattern != null) {
writer.println("case 2:");
writer.indent();
writer.println(" arg" + listArgNum + "_list = "
+ formatListPattern(logger, locale, listArg,
listAccessor.getElement("0"), listAccessor.getElement("1"), elemType,
isSafeHtml, pairPattern, true, params) + ";");
writer.println("break;");
writer.outdent();
}
writer.println("default:");
writer.indent();
writer.println("int i = arg" + listArgNum + "_size;");
writer.println("if (i > 0) {");
writer.indent();
StringBuffer outbuf = new StringBuffer();
StringGenerator buf = new StringGenerator(outbuf, isSafeHtml);
formatArg(logger, locale, buf, listArg, listAccessor.getElement("--i"),
elemType, params);
buf.completeString();
writer.println("arg" + listArgNum + "_list = " + outbuf + ";");
writer.outdent();
writer.println("}");
writer.println("if (i > 0) {");
writer.indent();
writer.println("arg" + listArgNum + "_list = "
+ formatListPattern(logger, locale, listArg,
listAccessor.getElement("--i"), "arg" + listArgNum + "_list", elemType,
isSafeHtml, listPatternParts.get("end"), false, params) + ";");
writer.outdent();
writer.println("}");
writer.println("while (i > 1) {");
writer.indent();
writer.println("arg" + listArgNum + "_list = "
+ formatListPattern(logger, locale, listArg,
listAccessor.getElement("--i"), "arg" + listArgNum + "_list", elemType,
isSafeHtml, listPatternParts.get("middle"), false, params) + ";");
writer.outdent();
writer.println("}");
writer.println("if (i > 0) {");
writer.indent();
writer.println("arg" + listArgNum + "_list = "
+ formatListPattern(logger, locale, listArg,
listAccessor.getElement("--i"), "arg" + listArgNum + "_list", elemType,
isSafeHtml, listPatternParts.get("start"), false, params) + ";");
writer.outdent();
writer.println("}");
writer.println("break;");
writer.outdent();
writer.outdent();
writer.println("}");
}
/**
* @param logger
* @param m
* @param locale
* @param resourceEntry
* @param selectors
* @param paramsAccessor
* @param isSafeHtml
* @param forms
* @throws UnableToCompleteException
*/
private void generateMessageSelectors(TreeLogger logger, JMethod m,
GwtLocale locale, ResourceEntry resourceEntry,
List selectors, Parameters paramsAccessor,
boolean isSafeHtml, String[] forms)
throws UnableToCompleteException {
int numSelectors = selectors.size();
String[] lastForm = new String[numSelectors];
for (String form : forms) {
String[] splitForms = form.split("\\|");
if (splitForms.length != numSelectors) {
throw error(logger, "Incorrect number of selector forms for "
+ m.getName() + " - '" + form + "'");
}
boolean allOther = true;
for (String splitForm : splitForms) {
if (splitForm.startsWith("=")) {
allOther = false;
} else if (!"other".equals(splitForm)) {
allOther = false;
}
}
if (allOther) {
// don't process the all-other case, that is the default return value
logger.log(TreeLogger.WARN, "Ignoring supplied alternate form with all"
+ " 'other' values, @DefaultMessage will be used");
continue;
}
// find where the changes are
int firstDifferent = 0;
while (firstDifferent < numSelectors
&& splitForms[firstDifferent].equals(lastForm[firstDifferent])) {
firstDifferent++;
}
// close nested selects deeper than where the change was
for (int i = numSelectors; i-- > firstDifferent; ) {
if (lastForm[i] != null) {
selectors.get(i).generateSelectMatchEnd(writer, lastForm[i]);
if (i > firstDifferent) {
selectors.get(i).generateSelectEnd(writer);
}
}
}
// open all the nested selects from here
for (int i = firstDifferent; i < numSelectors; ++i) {
if (i > firstDifferent || lastForm[i] == null) {
selectors.get(i).generateSelectStart(writer,
splitForms[i].startsWith("="));
}
selectors.get(i).generateSelectMatchStart(writer, logger,
splitForms[i]);
lastForm[i] = splitForms[i];
}
writer.print("returnVal = ");
generateString(logger, locale, resourceEntry.getForm(form),
paramsAccessor, writer, isSafeHtml);
writer.println(";");
}
for (int i = numSelectors; i-- > 0; ) {
if (lastForm[i] != null) {
selectors.get(i).generateSelectMatchEnd(writer, lastForm[i]);
selectors.get(i).generateSelectEnd(writer);
}
}
}
/**
* Generate a Java string for a given MessageFormat string.
*
* @param logger
* @param template
* @param paramsAccessor
* @param writer
* @throws UnableToCompleteException
*/
private void generateString(final TreeLogger logger,final GwtLocale locale,
final String template,final Parameters paramsAccessor,
SourceWriter writer, final boolean isSafeHtml)
throws UnableToCompleteException {
StringBuffer outputBuf = new StringBuffer();
final StringGenerator buf = new StringGenerator(outputBuf, isSafeHtml);
final int n = paramsAccessor.getCount();
try {
for (TemplateChunk chunk : MessageFormatParser.parse(template)) {
chunk.accept(new DefaultTemplateChunkVisitor() {
@Override
public void visit(ArgumentChunk argChunk) throws UnableToCompleteException {
int argNumber = argChunk.getArgumentNumber();
if (argNumber >= n) {
throw error(
logger, "Argument " + argNumber + " beyond range of arguments: " + template);
}
JParameter param = paramsAccessor.getParameter(argNumber);
String arg = "arg" + argNumber;
if (argChunk.isList()) {
buf.appendExpression(arg + "_list", isSafeHtml, false, false);
} else {
JType paramType = param.getType();
formatArg(logger, locale, buf, argChunk,
paramsAccessor.getParameterExpression(argNumber), paramType,
paramsAccessor);
}
}
@Override
public void visit(StaticArgChunk staticArgChunk) throws UnableToCompleteException {
buf.appendStringLiteral(staticArgChunk.getReplacement());
}
@Override
public void visit(StringChunk stringChunk) {
buf.appendStringLiteral(stringChunk.getString());
}
});
}
} catch (ParseException e) {
throw error(logger, e);
}
buf.completeString();
writer.print(outputBuf.toString());
}
private Map getListPatternParts(TreeLogger logger, GwtLocale locale) {
Map map = listPatternCache.get(locale);
if (map == null) {
// TODO(jat): get these from ResourceOracle instead
String baseName = MessagesMethodCreator.class.getPackage().getName().replace('.', '/')
+ "/cldr/ListPatterns_";
ClassLoader cl = MessagesMethodCreator.class.getClassLoader();
for (GwtLocale search : locale.getCompleteSearchList()) {
String propFile = baseName + search.getAsString() + ".properties";
InputStream stream = cl.getResourceAsStream(propFile);
if (stream != null) {
try {
LocalizedPropertiesLoader loader = new LocalizedPropertiesLoader(stream, "UTF-8");
map = new HashMap();
loader.load(map);
break;
} catch (IOException e) {
logger.log(TreeLogger.WARN, "Ignoring error reading file " + propFile, e);
} finally {
try {
stream.close();
} catch (IOException e) {
}
}
}
}
listPatternCache.put(locale, map);
}
return map;
}
}