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

javax.faces.component.UIViewParameter Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 1997, 2018 Oracle and/or its affiliates. All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */

package javax.faces.component;

import java.io.IOException;
import java.util.Iterator;
import javax.el.ValueExpression;
import javax.faces.FactoryFinder;
import javax.faces.application.FacesMessage;
import javax.faces.context.FacesContext;
import javax.faces.convert.Converter;
import javax.faces.convert.ConverterException;
import javax.faces.render.RenderKit;
import javax.faces.render.RenderKitFactory;
import javax.faces.render.Renderer;
import javax.faces.validator.RequiredValidator;
import javax.faces.validator.Validator;

/**
 * 

UIViewParameter represents a * binding between a request parameter and a model property or {@link UIViewRoot} * property. This is a bi-directional binding.

* *
*

The {@link javax.faces.view.ViewDeclarationLanguage} * implementation must cause an instance of this component to appear in * the view for each occurrence of an <f:viewParam /> * element placed inside of an <f:metadata /> * element. The user must place this facet within the * UIViewRoot.

*

Because this class extends UIInput any actions that * one would normally take on a UIInput instance are valid * for instances of this class. Instances of this class participate in * the regular JSF lifecycle, including on Ajax requests.

*
* * @since 2.0 */ public class UIViewParameter extends UIInput { // ------------------------------------------------------ Manifest Constants /** *

The standard component type for this component.

*/ public static final String COMPONENT_TYPE = "javax.faces.ViewParameter"; /** *

The standard component family for this component.

*/ public static final String COMPONENT_FAMILY = "javax.faces.ViewParameter"; enum PropertyKeys { name, submittedValue } // ------------------------------------------------------ Instance Variables private Renderer inputTextRenderer = null; private transient Boolean emptyStringIsNull; // ------------------------------------------------------------ Constructors /** *

Create a new {@link UIViewParameter} instance with default * property values.

*/ public UIViewParameter() { super(); setRendererType(null); } // ------------------------------------------------------ Instance Variables /** *

The raw value is the "implicit" binding for this view parameter. This property * maintains the submitted value of the view parameter for the duration of the request. * If the view parameter does not explicitly specify a value expression, then when the * request ends, this value is stored with the state of this component to use as the * submitted value on an ensuing postback.

*/ private String rawValue; // -------------------------------------------------------------- Properties @Override public String getFamily() { return (COMPONENT_FAMILY); } /** *

Return the request parameter name * from which the value is retrieved.

* * @return the name. * @since 2.0 */ public String getName() { return (String) getStateHelper().eval(PropertyKeys.name); } /** *

Set the request parameter name from * which the value is retrieved.

* * @param name The new request parameter name. * @since 2.0 */ public void setName(String name) { getStateHelper().put(PropertyKeys.name, name); } /** *

Return false. The * immediate setting is not relevant for view parameters and must be * assumed to be false.

* * @return true if immediate, false otherwise. * @since 2.0 */ @Override public boolean isImmediate() { return false; } /** *

Assume * that the submitted value is * always a string, but the return type * from this method is Object..

* * @return the submitted value. * @since 2.0 */ @Override public Object getSubmittedValue() { return getStateHelper().get(PropertyKeys.submittedValue); } /** * PENDING (docs) Interesting that submitted value isn't saved by the parent * @param submittedValue The new submitted value */ @Override public void setSubmittedValue(Object submittedValue) { getStateHelper().put(PropertyKeys.submittedValue, submittedValue); } // ----------------------------------------------------- UIComponent Methods // This is the "Apply Request Phase" step // QUESTION should we just override processDecodes() directly? // ANSWER: In this case, no. We don't want to take responsibility for // traversing any children we may have in the future. /** *

Override behavior from superclass to * pull a value from the incoming request parameter map under the * name given by {@link #getName} and store it with a call to {@link * UIInput#setSubmittedValue}.

* @since 2.0 */ @Override public void decode(FacesContext context) { if (context == null) { throw new NullPointerException(); } // QUESTION can we move forward and support an array? no different than UISelectMany; perhaps need to know // if the value expression is single or multi-valued // ANSWER: I'd rather not right now. String paramValue = context.getExternalContext().getRequestParameterMap().get(getName()); // submitted value will stay as previous value (null on initial request) if a parameter is absent if (paramValue != null) { setSubmittedValue(paramValue); } rawValue = (String) getSubmittedValue(); setValid(true); } /** *

Specialize * superclass behavior to treat * null differently. In this class, a null value * along with the "required" flag being set to true will * cause a validation failure. Otherwise, * If the {@link UIInput#EMPTY_STRING_AS_NULL_PARAM_NAME} * context parameter is true and the value is {@code null}, call * {@link UIInput#setSubmittedValue} passing the empty string * as the argument. This will cause the normal validation processing * to happen, including bean validation.

* * @param context the Faces context. * @since 2.0 */ @Override public void processValidators(FacesContext context) { if (context == null) { throw new NullPointerException(); } // Skip processing if our rendered flag is false if (!isRendered()) { return; } Object submittedValue = getSubmittedValue(); // we have to override since UIInput assumes that a null value means don't check if (submittedValue == null && myIsRequired()) { String requiredMessageStr = getRequiredMessage(); FacesMessage message; if (null != requiredMessageStr) { message = new FacesMessage(FacesMessage.SEVERITY_ERROR, requiredMessageStr, requiredMessageStr); } else { message = MessageFactory.getMessage(context, REQUIRED_MESSAGE_ID, MessageFactory.getLabel( context, this)); } context.addMessage(getClientId(context), message); setValid(false); context.validationFailed(); context.renderResponse(); } else { if (myConsiderEmptyStringNull(context)) { // JAVASERVERFACES_SPEC_PUBLIC-1329: If the EMPTY_STRING_SUBMITTED_VALUES_AS_NULL // config is set, ensure that logic gets a chance to be executed // in UIInput.processValidators(). if (null == submittedValue) { setSubmittedValue(""); } } super.processValidators(context); } } private boolean myConsiderEmptyStringNull(FacesContext ctx) { if (emptyStringIsNull == null) { String val = ctx.getExternalContext().getInitParameter(EMPTY_STRING_AS_NULL_PARAM_NAME); emptyStringIsNull = Boolean.valueOf(val); } return emptyStringIsNull; } private boolean myIsRequired() { return super.isRequired() || isRequiredViaNestedRequiredValidator(); } /* JAVASERVERFACES-3058. Handle the nested requiredValidator case * explicitly in the case of . * */ private boolean isRequiredViaNestedRequiredValidator() { boolean result = false; if (null == validators) { return result; } Iterator iter = validators.iterator(); while (iter.hasNext()) { if (iter.next() instanceof RequiredValidator) { // See JAVASERVERFACES-2526. Note that we can assume // that at this point the validator is not disabled, // so the mere existence of the validator implies it is // enabled. result = true; Object submittedValue = getSubmittedValue(); if (submittedValue == null) { // JAVASERVERFACES-3058 asserts that view parameters // should be treated differently than form parameters // if they are not submitted. I'm not sure if that's // correct, but let's put this in and see how // the community responds. this.setSubmittedValue(""); } break; } } return result; } /** *

Call through to superclass {@link * UIInput#updateModel} then take the additional action of pushing * the value into request scope if and only if the value is not a * value expression, is valid, and the local value was set on this * lifecycle execution.

* @since 2.0 */ @Override public void updateModel(FacesContext context) { super.updateModel(context); if (!hasValueExpression() && isValid() && isLocalValueSet()) { // QUESTION should this be done even when a value expression is present? // ANSWER: I don't see why not. context.getExternalContext().getRequestMap().put(getName(), getLocalValue()); } } // This is called during the real "Render Response" phase /** *

Called specially by {@link * UIViewRoot#encodeEnd}, this method simply sets the submitted * value to be the return from {@link #getStringValue}.

* * @throws IOException when an I/O error occurs. * @since 2.0 */ @Override public void encodeAll(FacesContext context) throws IOException { if (context == null) { throw new NullPointerException(); } // if there is a value expression, update view parameter w/ latest value after render // QUESTION is it okay that a null string value may be suppressing the view parameter value? // ANSWER: I'm not sure. setSubmittedValue(getStringValue(context)); } /** *

If the value of this parameter comes * from a ValueExpression return the value of the * expression, otherwise, return the local value.

* * @param context the Faces context. * @return the string value. * @since 2.0 */ public String getStringValue(FacesContext context) { String result = null; if (hasValueExpression()) { result = getStringValueFromModel(context); } else { result = (null != rawValue) ? rawValue : (String) getValue(); } return result; } /** *

Manually perform standard conversion * steps to get a string value from the value expression.

* * @param context the Faces context. * @return the string value from the model. * @since 2.0 */ public String getStringValueFromModel(FacesContext context) throws ConverterException { ValueExpression ve = getValueExpression("value"); if (ve == null) { return null; } Object currentValue = ve.getValue(context.getELContext()); // If there is a converter attribute, use it to to ask application // instance for a converter with this identifer. Converter c = getConverter(); if (c == null) { // if value is null and no converter attribute is specified, then // return null (null has meaning for a view parameters; it means remove it). if (currentValue == null) { return null; } // Do not look for "by-type" converters for Strings if (currentValue instanceof String) { return (String) currentValue; } // if converter attribute set, try to acquire a converter // using its class type. Class converterType = currentValue.getClass(); c = context.getApplication().createConverter(converterType); // if there is no default converter available for this identifier, // assume the model type to be String. if (c == null) { return currentValue.toString(); } } return c.getAsString(context, this, currentValue); } /** *

Because this class has no {@link * Renderer}, leverage the one from the standard HTML_BASIC {@link * RenderKit} with component-family: javax.faces.Input * and renderer-type: javax.faces.Text and call its * {@link Renderer#getConvertedValue} method.

* * @param submittedValue the submitted value. * @return the converted value. * @since 2.0 */ @Override protected Object getConvertedValue(FacesContext context, Object submittedValue) throws ConverterException { return getInputTextRenderer(context).getConvertedValue(context, this, submittedValue); } private Renderer getInputTextRenderer(FacesContext context) { if (null == inputTextRenderer) { RenderKitFactory rkf = (RenderKitFactory) FactoryFinder.getFactory(FactoryFinder.RENDER_KIT_FACTORY); RenderKit standard = rkf.getRenderKit(context, RenderKitFactory.HTML_BASIC_RENDER_KIT); inputTextRenderer = standard. getRenderer("javax.faces.Input", "javax.faces.Text"); } assert(null != inputTextRenderer); return inputTextRenderer; } // ----------------------------------------------------- Helper Methods private boolean hasValueExpression() { return null != getValueExpression("value"); } /** *

Inner class to encapsulate a * UIViewParameter instance so that it may be safely * referenced regardless of whether or not the current view is the * same as the view in which this UIViewParameter * resides.

* @since 2.0 */ public static class Reference { private StateHolderSaver saver; private int indexInParent; private String viewIdAtTimeOfConstruction; /** *

Construct a reference to a * UIViewParameter. This constructor cause the * {@link StateHolder#saveState} method to be called on * argument UIViewParameter.

* * @param context the FacesContextfor this request * @param param the UIViewParameter. * @param indexInParent the index of the * UIViewParameter in its parent * UIPanel. * @param viewIdAtTimeOfConstruction the viewId of the view in * which the UIViewParameter is included. This may * not be the same as the viewId from the context * argument. * * @since 2.0 */ public Reference(FacesContext context, UIViewParameter param, int indexInParent, String viewIdAtTimeOfConstruction) { this.saver = new StateHolderSaver(context, param); this.indexInParent = indexInParent; this.viewIdAtTimeOfConstruction = viewIdAtTimeOfConstruction; } /** *

Return the * UIViewParameter to which this instance refers. * If the current viewId is the same as the viewId passed to our * constructor, use the index passed to the constructor to find * the actual UIViewParameter instance and return * it. Otherwise, call {@link StateHolder#restoreState} on the * saved state and return the result.

* * @param context the FacesContextfor this request * @return the UIViewParameter. * @since 2.0 */ public UIViewParameter getUIViewParameter(FacesContext context) { UIViewParameter result = null; UIViewRoot root = context.getViewRoot(); // If the view root is the same as when we were constructed... if (this.viewIdAtTimeOfConstruction.equals(root.getViewId())) { // get the actual view parameter from the tree... UIComponent metadataFacet = root.getFacet(UIViewRoot.METADATA_FACET_NAME); result = (UIViewParameter) metadataFacet.getChildren().get(indexInParent); } else { // otherwise, use the saved one result = (UIViewParameter) this.saver.restore(context); } return result; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy