com.sun.faces.facelets.tag.jsf.ValidatorTagHandlerDelegateImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jsf-impl Show documentation
Show all versions of jsf-impl Show documentation
This is the master POM file for Oracle's Implementation of the JSF 2.2 Specification.
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 1997-2010 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package com.sun.faces.facelets.tag.jsf;
import com.sun.faces.component.validator.ComponentValidators;
import com.sun.faces.facelets.tag.MetaRulesetImpl;
import com.sun.faces.util.Util;
import com.sun.faces.util.RequestStateManager;
import javax.el.ValueExpression;
import javax.faces.component.EditableValueHolder;
import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import javax.faces.validator.Validator;
import javax.faces.view.AttachedObjectHandler;
import javax.faces.view.facelets.*;
import java.io.IOException;
import java.util.Set;
import java.util.HashSet;
public class ValidatorTagHandlerDelegateImpl extends TagHandlerDelegate implements AttachedObjectHandler {
protected final ValidatorHandler owner;
private final boolean wrapping;
// ------------------------------------------------------------ Constructors
public ValidatorTagHandlerDelegateImpl(ValidatorHandler owner) {
this.owner = owner;
wrapping = isWrapping();
}
// ----------------------------------------- Methods from TagHandlerDelegate
@Override
public void apply(FaceletContext ctx, UIComponent parent)
throws IOException {
if (wrapping) {
applyWrapping(ctx, parent);
} else {
applyNested(ctx, parent);
}
}
@Override
public MetaRuleset createMetaRuleset(Class type) {
Util.notNull("type", type);
MetaRuleset m = new MetaRulesetImpl(owner.getTag(), type);
return m.ignore("binding").ignore("disabled").ignore("for");
}
// -------------------------------------- Methods from AttachedObjectHandler
@SuppressWarnings({"unchecked"})
public void applyAttachedObject(FacesContext context, UIComponent parent) {
FaceletContext ctx = (FaceletContext) context.getAttributes().get(FaceletContext.FACELET_CONTEXT_KEY);
EditableValueHolder evh = (EditableValueHolder) parent;
if (owner.isDisabled(ctx)) {
Set disabledIds = (Set)
RequestStateManager.get(context, RequestStateManager.DISABLED_VALIDATORS);
if (disabledIds == null) {
disabledIds = new HashSet(3);
RequestStateManager.set(context,
RequestStateManager.DISABLED_VALIDATORS,
disabledIds);
}
disabledIds.add(owner.getValidatorId(ctx));
return;
}
ValueExpression ve = null;
Validator v = null;
if (owner.getBinding() != null) {
ve = owner.getBinding().getValueExpression(ctx, Validator.class);
v = (Validator) ve.getValue(ctx);
}
if (v == null) {
v = this.createValidator(ctx);
if (ve != null) {
ve.setValue(ctx, v);
}
}
if (v == null) {
throw new TagException(owner.getTag(), "No Validator was created");
}
owner.setAttributes(ctx, v);
evh.addValidator(v);
}
public String getFor() {
String result = null;
TagAttribute attr = owner.getTagAttribute("for");
if (null != attr) {
if (attr.isLiteral()) {
result = attr.getValue();
} else {
FacesContext context = FacesContext.getCurrentInstance();
FaceletContext ctx = (FaceletContext) context.getAttributes().get(FaceletContext.FACELET_CONTEXT_KEY);
result = (String)attr.getValueExpression(ctx, String.class).getValue(ctx);
}
}
return result;
}
// ------------------------------------------------------- Protected Methods
protected ComponentValidators.ValidatorInfo createValidatorInfo(FaceletContext ctx) {
return new ComponentValidators.ValidatorInfo(ctx, owner);
}
// --------------------------------------------------------- Private Methods
// Tests whether the valiator tag is wrapping other tags.
private boolean isWrapping() {
// Would be nice if there was some easy way to determine whether
// we are a leaf handler. However, even leaf handlers have a
// non-null nextHandler - the CompilationUnit.LEAF instance.
// We assume that if we've got a TagHandler or CompositeFaceletHandler
// as our nextHandler, we are not a leaf.
return ((owner.getValidatorConfig().getNextHandler() instanceof TagHandler) ||
(owner.getValidatorConfig().getNextHandler() instanceof CompositeFaceletHandler));
}
private void applyWrapping(FaceletContext ctx,
UIComponent parent) throws IOException {
ComponentValidators validators = ComponentValidators.getValidators(ctx.getFacesContext(), true);
validators.pushValidatorInfo(createValidatorInfo(ctx));
owner.getValidatorConfig().getNextHandler().apply(ctx, parent);
validators.popValidatorInfo();
}
private void applyNested(FaceletContext ctx,
UIComponent parent) {
// only process if it's been created
if (!ComponentHandler.isNew(parent)) {
return;
}
if (parent instanceof EditableValueHolder) {
applyAttachedObject(ctx.getFacesContext(), parent);
} else if (UIComponent.isCompositeComponent(parent)) {
if (null == owner.getFor()) {
// PENDING(): I18N
throw new TagException(owner.getTag(),
"validator tags nested within composite components must have a non-null \"for\" attribute");
}
// Allow the composite component to know about the target
// component.
CompositeComponentTagHandler.getAttachedObjectHandlers(parent).add(owner);
} else {
throw new TagException(owner.getTag(),
"Parent not an instance of EditableValueHolder: " + parent);
}
}
/**
* Template method for creating a Validator instance
*
* @param ctx
* FaceletContext to use
* @return a new Validator instance
*/
private Validator createValidator(FaceletContext ctx) {
String id = owner.getValidatorId(ctx);
if (id == null) {
throw new TagException(
owner.getTag(),
"A validator id was not specified. Typically the validator id is set in the constructor ValidateHandler(ValidatorConfig)");
}
return ctx.getFacesContext().getApplication().createValidator(id);
}
}