com.sun.faces.taglib.jsf_core.PhaseListenerTag Maven / Gradle / Ivy
/*
* 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.taglib.jsf_core;
import com.sun.faces.util.MessageUtils;
import com.sun.faces.util.Util;
import com.sun.faces.util.FacesLogger;
import javax.el.ValueExpression;
import javax.faces.component.UIComponent;
import javax.faces.component.UIViewRoot;
import javax.faces.event.AbortProcessingException;
import javax.faces.event.PhaseEvent;
import javax.faces.event.PhaseId;
import javax.faces.event.PhaseListener;
import javax.faces.webapp.UIComponentELTag;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;
import java.io.Serializable;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* Tag implementation that creates a {@link PhaseListener} instance
* and registers it on the {@link UIViewRoot} associated with our most
* immediate surrounding instance of a tag whose component
* is an instance of {@link UIViewRoot}. This tag creates no output to the
* page currently being created.
*
*/
public class PhaseListenerTag extends TagSupport {
private static final Logger LOGGER = FacesLogger.TAGLIB.getLogger();
// ------------------------------------------------------------- Attributes
/**
* The fully qualified class name of the {@link PhaseListener}
* instance to be created.
*/
private ValueExpression type = null;
/**
* The value binding expression used to create a listener
* instance and it is also used to wire up this listener to an
* {@link PhaseListener} property of a JavaBean class.
*/
private ValueExpression binding = null;
/**
* Set the fully qualified class name of the
* {@link PhaseListener} instance to be created.
*
* @param type The new class name
*/
public void setType(ValueExpression type) {
this.type = type;
}
/*
*
Set the value binding expression for this listener.
*
* @param binding The new value binding expression
*/
public void setBinding(ValueExpression binding) {
this.binding = binding;
}
// --------------------------------------------------------- Public Methods
/**
* Create a new instance of the specified {@link PhaseListener}
* class, and register it with the {@link UIComponent} instance associated
* with our most immediately surrounding {@link UIComponentELTag} instance, if
* the {@link UIComponent} instance was created by this execution of the
* containing JSP page.
*
* @throws JspException if a JSP error occurs
*/
public int doStartTag() throws JspException {
// find the viewTag
Tag parent = this;
UIComponentELTag tag = null;
while (null != (parent = parent.getParent())) {
if (parent instanceof UIComponentELTag) {
tag = (UIComponentELTag) parent;
}
}
if (tag == null) {
Object params[] = {this.getClass().getName()};
throw new JspException(
MessageUtils.getExceptionMessageString(
MessageUtils.NOT_NESTED_IN_FACES_TAG_ERROR_MESSAGE_ID, params));
}
// Nothing to do unless this tag created a component
if (!tag.getCreated()) {
return (SKIP_BODY);
}
UIViewRoot viewRoot = (UIViewRoot) tag.getComponentInstance();
if (viewRoot == null) {
throw new JspException(
MessageUtils.getExceptionMessageString(MessageUtils.NULL_COMPONENT_ERROR_MESSAGE_ID));
}
// If binding is null, type is set and is a literal value,
// then don't bother wrapping. Just instantiate and
// set.
PhaseListener listener;
if (binding == null && type != null && type.isLiteralText()) {
try {
listener = (PhaseListener)
Util.getListenerInstance(type, null);
} catch (Exception e) {
throw new JspException(e.getMessage(), e.getCause());
}
} else {
listener = new BindingPhaseListener(type, binding);
}
viewRoot.addPhaseListener(listener);
return (SKIP_BODY);
}
/**
* Release references to any acquired resources.
*/
public void release() {
this.type = null;
}
// ----------------------------------------------------------- Inner Classes
private static class BindingPhaseListener
implements PhaseListener, Serializable {
private ValueExpression type;
private ValueExpression binding;
// -------------------------------------------------------- Constructors
public BindingPhaseListener(ValueExpression type,
ValueExpression binding) {
this.type = type;
this.binding = binding;
}
// ------------------------------------------ Methods from PhaseListener
/**
*
Handle a notification that the processing for a particular
* phase has just been completed.
*/
public void afterPhase(PhaseEvent event) {
PhaseListener listener = getPhaseListener();
if (listener != null) {
listener.afterPhase(event);
}
}
/**
* Handle a notification that the processing for a particular
* phase of the request processing lifecycle is about to begin.
*/
public void beforePhase(PhaseEvent event) {
PhaseListener listener = getPhaseListener();
if (listener != null) {
listener.beforePhase(event);
}
}
/**
* Return the identifier of the request processing phase during
* which this listener is interested in processing {@link javax.faces.event.PhaseEvent}
* events. Legal values are the singleton instances defined by the
* {@link javax.faces.event.PhaseId} class, including PhaseId.ANY_PHASE
* to indicate an interest in being notified for all standard phases.
*/
public PhaseId getPhaseId() {
PhaseListener listener = getPhaseListener();
if (listener != null) {
return listener.getPhaseId();
}
return null;
}
/**
* Invoked when the value change described by the specified
* {@link javax.faces.event.ValueChangeEvent} occurs.
*
* @return a PhaseListener
instance
* @throws javax.faces.event.AbortProcessingException
* Signal the JavaServer Faces
* implementation that no further processing on the current event
* should be performed
*/
public PhaseListener getPhaseListener() throws AbortProcessingException {
PhaseListener instance = (PhaseListener)
Util.getListenerInstance(type, binding);
if (instance != null) {
return instance;
} else {
if (LOGGER.isLoggable(Level.WARNING)) {
// PENDING i18n
LOGGER.warning("PhaseListener will not be processed - " +
"both 'binding' and 'type' are null");
}
return null;
}
}
}
}