All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.wicket.model.StringResourceModel Maven / Gradle / Ivy

Go to download

Pax Wicket Service is an OSGi extension of the Wicket framework, allowing for dynamic loading and unloading of Wicket components and pageSources.

There is a newer version: 5.0.0
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.wicket.model;

import java.text.MessageFormat;
import java.util.Arrays;
import java.util.Locale;

import org.apache.wicket.Application;
import org.apache.wicket.Component;
import org.apache.wicket.Localizer;
import org.apache.wicket.Session;
import org.apache.wicket.resource.loader.ComponentStringResourceLoader;
import org.apache.wicket.util.string.Strings;
import org.apache.wicket.util.string.interpolator.PropertyVariableInterpolator;


/**
 * This model class encapsulates the full power of localization support within the Wicket framework.
 * It combines the flexible Wicket resource loading mechanism with property expressions, property
 * models and standard Java MessageFormat substitutions. This combination should be
 * able to solve any dynamic localization requirement that a project has.
 * 

* The model should be created with four parameters, which are described in detail below: *

    *
  • resourceKey - This is the most important parameter as it contains the key that should * be used to obtain resources from any string resource loaders. This parameter is mandatory: a null * value will throw an exception. Typically it will contain an ordinary string such as * "label.username". To add extra power to the key functionality the key may also contain * a property expression which will be evaluated if the model parameter (see below) is not null. * This allows keys to be changed dynamically as the application is running. For example, the key * could be "product.${product.id}" which prior to rendering will call * model.getObject().getProduct().getId() and substitute this value into the resource key before is * is passed to the loader. *
  • component - This parameter should be a component that the string resource is relative * to. In a simple application this will usually be the Page on which the component resides. For * reusable components/containers that are packaged with their own string resource bundles it should * be the actual component/container rather than the page. For more information on this please see * {@link org.apache.wicket.resource.loader.ComponentStringResourceLoader}. The relative component * may actually be {@code null} if this model is wrapped on assignment ( * {@link IComponentAssignedModel}) or when all resource loading is to be done from a global * resource loader. However, we recommend that a relative component is still supplied even in the * latter case in order to 'future proof' your application with regards to changing resource loading * strategies. *
  • model - This parameter is mandatory if either the resourceKey, the found string * resource (see below) or any of the substitution parameters (see below) contain property * expressions. Where property expressions are present they will all be evaluated relative to this * model object. If there are no property expressions present then this model parameter may be * null *
  • parameters - The parameters parameter allows an array of objects to be passed for * substitution on the found string resource (see below) using a standard * java.text.MessageFormat object. Each parameter may be an ordinary Object, in which * case it will be processed by the standard formatting rules associated with * java.text.MessageFormat. Alternatively, the parameter may be an instance of * IModel in which case the getObject() method will be applied prior to * the parameter being passed to the java.text.MessageFormat. This allows such features * dynamic parameters that are obtained using a PropertyModel object or even nested * string resource models. *
* As well as the supplied parameters, the found string resource can contain formatting information. * It may contain property expressions in which case these are evaluated using the model object * supplied when the string resource model is created. The string resource may also contain * java.text.MessageFormat style markup for replacement of parameters. Where a string * resource contains both types of formatting information then the property expression will be * applied first. *

* Example 1 *

* In its simplest form, the model can be used as follows: * *

 * public MyPage extends WebPage<Void>
 * {
 *    public MyPage(final PageParameters parameters)
 *    {
 *        add(new Label("username", new StringResourceModel("label.username", this, null)));
 *    }
 * }
 * 
* * Where the resource bundle for the page contains the entry label.username=Username *

* Example 2 *

* In this example, the resource key is selected based on the evaluation of a property expression: * *

 * public MyPage extends WebPage<Void>
 * {
 *     public MyPage(final PageParameters parameters)
 *     {
 *         WeatherStation ws = new WeatherStation();
 *         add(new Label("weatherMessage",
 *             new StringResourceModel("weather.${currentStatus}", this, new Model<String>(ws)));
 *     }
 * }
 * 
* * Which will call the WeatherStation.getCurrentStatus() method each time the string resource model * is used and where the resource bundle for the page contains the entries: * *
 * weather.sunny=Don't forget sunscreen!
 * weather.raining=You might need an umbrella
 * weather.snowing=Got your skis?
 * weather.overcast=Best take a coat to be safe
 * 
* *

* Example 3 *

* In this example the found resource string contains a property expression that is substituted via * the model: * *

 * public MyPage extends WebPage<Void>
 * {
 *     public MyPage(final PageParameters parameters)
 *     {
 *         WeatherStation ws = new WeatherStation();
 *         add(new Label("weatherMessage",
 *             new StringResourceModel("weather.message", this, new Model<String>(ws)));
 *     }
 * }
 * 
* * Where the resource bundle contains the entry weather.message=Weather station reports that * the temperature is ${currentTemperature} ${units} *

* Example 4 *

* In this example, the use of substitution parameters is employed to format a quite complex message * string. This is an example of the most complex and powerful use of the string resource model: * *

 * public MyPage extends WebPage<Void>
 * {
 *     public MyPage(final PageParameters parameters)
 *     {
 *         WeatherStation ws = new WeatherStation();
 *         IModel<WeatherStation> model = new Model<WeatherStation>(ws);
 *         add(new Label("weatherMessage",
 *             new StringResourceModel(
 *                 "weather.detail", this, model,
 *                     new Object[]
 *                     {
 *                         new Date(),
 *                         new PropertyModel<?>(model, "currentStatus"),
 *                         new PropertyModel<?>(model, "currentTemperature"),
 *                         new PropertyModel<?>(model, "units")
 *         }));
 *     }
 * }
 * 
* * And where the resource bundle entry is: * *
 * weather.detail=The report for {0,date}, shows the temperature as {2,number,###.##} {3} \
 *     and the weather to be {1}
 * 
* * @see ComponentStringResourceLoader for additional information especially on the component search * order * * @author Chris Turner */ public class StringResourceModel extends LoadableDetachableModel implements IComponentAssignedModel { private static final long serialVersionUID = 1L; /** The wrapped model. */ private final IModel model; /** Optional parameters. */ private final Object[] parameters; /** The relative component used for lookups. */ private final Component component; /** The key of message to get. */ private final String resourceKey; /** The default value of the message. */ private final String defaultValue; public IWrapModel wrapOnAssignment(Component component) { return new AssignmentWrapper(component); } private class AssignmentWrapper extends LoadableDetachableModel implements IWrapModel { private static final long serialVersionUID = 1L; private final Component component; /** * Construct. * * @param component */ public AssignmentWrapper(Component component) { this.component = component; } @Override public void detach() { super.detach(); StringResourceModel.this.detach(); } @Override protected void onDetach() { if (StringResourceModel.this.component == null) { StringResourceModel.this.onDetach(); } } @Override protected String load() { if (StringResourceModel.this.component != null) { // ignore assignment if component was specified explicitely return StringResourceModel.this.getObject(); } else { return getString(component); } } @Override public void setObject(String object) { StringResourceModel.this.setObject(object); } public IModel getWrappedModel() { return StringResourceModel.this; } } /** * Creates a new string resource model using the supplied parameters. *

* The relative component parameter should generally be supplied, as without it resources can * not be obtained from resource bundles that are held relative to a particular component or * page. However, for application that use only global resources then this parameter may be * null. *

* The model parameter is also optional and only needs to be supplied if value substitutions are * to take place on either the resource key or the actual resource strings. *

* The parameters parameter is also optional and is used for substitutions. * * @param resourceKey * The resource key for this string resource * @param component * The component that the resource is relative to * @param model * The model to use for property substitutions * @param parameters * The parameters to substitute using a Java MessageFormat object */ public StringResourceModel(final String resourceKey, final Component component, final IModel model, final Object... parameters) { this(resourceKey, component, model, null, parameters); } /** * Creates a new string resource model using the supplied parameters. *

* The relative component parameter should generally be supplied, as without it resources can * not be obtained from resource bundles that are held relative to a particular component or * page. However, for application that use only global resources then this parameter may be * null. *

* The model parameter is also optional and only needs to be supplied if value substitutions are * to take place on either the resource key or the actual resource strings. *

* The parameters parameter is also optional and is used for substitutions. * * @param resourceKey * The resource key for this string resource * @param component * The component that the resource is relative to * @param model * The model to use for property substitutions * @param defaultValue * The default value if the resource key is not found. * @param parameters * The parameters to substitute using a Java MessageFormat object */ public StringResourceModel(final String resourceKey, final Component component, final IModel model, final String defaultValue, final Object... parameters) { if (resourceKey == null) { throw new IllegalArgumentException("Resource key must not be null"); } this.resourceKey = resourceKey; this.component = component; this.model = model; this.defaultValue = defaultValue; this.parameters = parameters; } /** * Creates a new string resource model using the supplied parameters. *

* The model parameter is also optional and only needs to be supplied if value substitutions are * to take place on either the resource key or the actual resource strings. *

* The parameters parameter is also optional and is used for substitutions. * * @param resourceKey * The resource key for this string resource * @param model * The model to use for property substitutions * @param parameters * The parameters to substitute using a Java MessageFormat object */ public StringResourceModel(final String resourceKey, final IModel model, final Object... parameters) { this(resourceKey, null, model, null, parameters); } /** * Creates a new string resource model using the supplied parameters. *

* The model parameter is also optional and only needs to be supplied if value substitutions are * to take place on either the resource key or the actual resource strings. *

* The parameters parameter is also optional and is used for substitutions. * * @param resourceKey * The resource key for this string resource * @param model * The model to use for property substitutions * @param parameters * The parameters to substitute using a Java MessageFormat object * @param defaultValue * The default value if the resource key is not found. */ public StringResourceModel(final String resourceKey, final IModel model, final String defaultValue, final Object... parameters) { this(resourceKey, null, model, defaultValue, parameters); } /** * Gets the localizer that is being used by this string resource model. * * @return The localizer */ public Localizer getLocalizer() { return Application.get().getResourceSettings().getLocalizer(); } /** * Gets the string currently represented by this model. The string that is returned may vary for * each call to this method depending on the values contained in the model and an the parameters * that were passed when this string resource model was created. * * @return The string */ public final String getString() { return getString(component); } private String getString(Component component) { final Localizer localizer = getLocalizer(); final Locale locale; if (component != null) { locale = component.getLocale(); } else { locale = Session.exists() ? Session.get().getLocale() : Locale.getDefault(); } String value; // Substitute any parameters if necessary Object[] parameters = getParameters(); if (parameters == null || parameters.length == 0) { // Get the string resource, doing any property substitutions as part // of the get operation value = localizer.getString(getResourceKey(), component, model, defaultValue); if (value == null) { value = defaultValue; } } else { // Get the string resource, doing not any property substitutions // that has to be done later after MessageFormat value = localizer.getString(getResourceKey(), component, null, defaultValue); if (value == null) { value = defaultValue; } if (value != null) { // Build the real parameters Object[] realParams = new Object[parameters.length]; for (int i = 0; i < parameters.length; i++) { if (parameters[i] instanceof IModel) { realParams[i] = ((IModel)parameters[i]).getObject(); } else if (model != null && parameters[i] instanceof String) { realParams[i] = PropertyVariableInterpolator.interpolate( (String)parameters[i], model.getObject()); } else { realParams[i] = parameters[i]; } } // Escape all single quotes outside {..} if (value.indexOf('\'') != -1) { value = escapeQuotes(value); } if (model != null) { // First escape all substitute properties so that message format doesn't try to // parse that. value = Strings.replaceAll(value, "${", "$'{'").toString(); } // Apply the parameters final MessageFormat format = new MessageFormat(value, locale); value = format.format(realParams); if (model != null) { // un escape the substitute properties value = Strings.replaceAll(value, "$'{'", "${").toString(); // now substitute the properties value = localizer.substitutePropertyExpressions(component, value, model); } } } // Return the string resource return value; } /** * Replace "'" with "''" outside of "{..}" * * @param value * @return escaped message format */ private String escapeQuotes(final String value) { StringBuilder newValue = new StringBuilder(value.length() + 10); int count = 0; for (int i = 0; i < value.length(); i++) { char ch = value.charAt(i); if (ch == '{') { count += 1; } else if (ch == '}') { count -= 1; } newValue.append(ch); if ((ch == '\'') && (count == 0)) { // Escape "'" newValue.append(ch); } } return newValue.toString(); } /** * This method just returns debug information, so it won't return the localized string. Please * use getString() for that. * * @return The string for this model object */ @Override public String toString() { StringBuilder sb = new StringBuilder("StringResourceModel["); sb.append("key:"); sb.append(resourceKey); sb.append(",default:"); sb.append(defaultValue); sb.append(",params:"); if (parameters != null) { sb.append(Arrays.asList(parameters)); } sb.append("]"); return sb.toString(); } /** * Gets the Java MessageFormat substitution parameters. * * @return The substitution parameters */ protected Object[] getParameters() { return parameters; } /** * Gets the resource key for this string resource. If the resource key contains property * expressions and the model is not null then the returned value is the actual resource key with * all substitutions undertaken. * * @return The (possibly substituted) resource key */ protected final String getResourceKey() { if (model != null) { return PropertyVariableInterpolator.interpolate(resourceKey, model.getObject()); } else { return resourceKey; } } /** * Gets the string that this string resource model currently represents. */ @Override protected String load() { return getString(); } /** * @see org.apache.wicket.model.IDetachable#detach() */ @Override protected final void onDetach() { super.onDetach(); // detach any model if (model != null) { model.detach(); } // some parameters can be detachable if (parameters != null) { for (Object parameter : parameters) { if (parameter instanceof IDetachable) { ((IDetachable)parameter).detach(); } } } } @Override public void setObject(String object) { throw new UnsupportedOperationException(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy