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

com.sun.faces.facelets.el.ContextualCompositeMethodExpression Maven / Gradle / Ivy

Go to download

This is the master POM file for Oracle's Implementation of the JSF 2.2 Specification.

There is a newer version: 2.4.0
Show newest version
/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright (c) 1997-2013 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.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.
 *
 *
 * This file incorporates work covered by the following copyright and
 * permission notice:
 *
 * Copyright 2005-2007 The Apache Software Foundation
 *
 * 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.sun.faces.facelets.el;

import com.sun.faces.component.CompositeComponentStackManager;

import com.sun.faces.util.FacesLogger;

import java.util.logging.Level;
import java.util.logging.Logger;

import javax.el.MethodExpression;
import javax.el.ValueExpression;
import javax.el.MethodInfo;
import javax.el.ELContext;
import javax.el.ELException;
import javax.el.MethodNotFoundException;
import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import javax.faces.event.AbortProcessingException;
import javax.faces.event.ComponentSystemEvent;
import javax.faces.event.ComponentSystemEventListener;
import javax.faces.event.PostAddToViewEvent;
import javax.faces.validator.ValidatorException;
import javax.faces.view.Location;

/**
 * 

* This specialized MethodExpression enables the evaluation of * composite component expressions. Instances of this expression will be created * when {@link com.sun.faces.facelets.tag.TagAttributeImpl#getValueExpression(javax.faces.view.facelets.FaceletContext, Class)} * is invoked and the expression represents a composite component expression (i.e. #{cc.[properties]}). *

* *

* It's important to note that these MethodExpressions are context * sensitive in that they leverage the location in which they were referenced * in order to push the proper composite component to the evaluation context * prior to evaluating the expression itself. *

* * Here's an example: * *
 * Using Page test.xhtml
 * ---------------------------------
 *    <ez:comp1 do="#{bean.action}" />
 *
 *
 * comp1.xhtml
 * ---------------------------------
 * <composite:interface>
 *    <composite:attribute name="do" method-signature="String f()" required="true" />
 * </composite:interface>
 * <composite:implementation>
 *    <ez:nesting>
 *       <h:commandButton value="Click Me!" action="#{cc.attrs.do} />
 *    </ez:nesting>
 * </composite:implementation>
 *
 * nesting.xhtml
 * ---------------------------------
 * <composite:interface />
 * <composite:implementation>
 *    <composite:insertChildren>
 * </composite:implementation>
 * 
* * When commandButton is clicked, the ContextualCompositeMethodExpression * first is looked up by {@link com.sun.faces.facelets.tag.TagAttributeImpl.AttributeLookupMethodExpression} * which results an instance of ContextualCompositeMethodExpression. * When this ContextualCompositeMethodExpression is invoked, the {@link javax.faces.view.Location} * object necessary to find the proper composite component will be derived from * source ValueExpression provided at construction time. Using the * derived {@link javax.faces.view.Location}, we can find the composite component * that matches 'owns' the template in which the expression was defined in by * comparing the path of the Location with the name and library of the {@link javax.faces.application.Resource} * instance associated with each composite component. If a matching composite * component is found, it will be made available to the EL by calling {@link CompositeComponentStackManager#push(javax.faces.component.UIComponent)}. *

*/ public class ContextualCompositeMethodExpression extends MethodExpression { private static final long serialVersionUID = -6281398928485392830L; // Log instance for this class private static final Logger LOGGER = FacesLogger.FACELETS_EL.getLogger(); private final MethodExpression delegate; private final ValueExpression source; private final Location location; private String ccClientId; // -------------------------------------------------------- Constructors public ContextualCompositeMethodExpression(ValueExpression source, MethodExpression delegate) { this.delegate = delegate; this.source = source; this.location = null; FacesContext ctx = FacesContext.getCurrentInstance(); UIComponent cc = UIComponent.getCurrentCompositeComponent(ctx); cc.subscribeToEvent(PostAddToViewEvent.class, new SetClientIdListener(this)); } public ContextualCompositeMethodExpression(Location location, MethodExpression delegate) { this.delegate = delegate; this.location = location; this.source = null; FacesContext ctx = FacesContext.getCurrentInstance(); UIComponent cc = UIComponent.getCurrentCompositeComponent(ctx); cc.subscribeToEvent(PostAddToViewEvent.class, new SetClientIdListener(this)); } // ------------------------------------------- Methods from MethodExpression @Override public MethodInfo getMethodInfo(ELContext elContext) { return delegate.getMethodInfo(elContext); } @Override public Object invoke(ELContext elContext, Object[] objects) { FacesContext ctx = (FacesContext) elContext.getContext(FacesContext.class); try { boolean pushed = pushCompositeComponent(ctx); try { return delegate.invoke(elContext, objects); } finally { if (pushed) { popCompositeComponent(ctx); } } } catch (ELException ele) { /* * If we got a validator exception it is actually correct to * immediately bubble it up. */ if (ele.getCause() != null && ele.getCause() instanceof ValidatorException) { throw (ValidatorException) ele.getCause(); } if (source != null && ele instanceof MethodNotFoundException) { // special handling when an ELException handling. This is necessary // when there are multiple levels of composite component nesting. // When this happens, we need to evaluate the source expression // to find and invoke the MethodExpression at the next highest // nesting level. Is there a cleaner way to detect this case? try { Object fallback = source.getValue(elContext); if (fallback != null && fallback instanceof MethodExpression) { return ((MethodExpression) fallback).invoke(elContext, objects); } } catch(ELException ex) { /* * If we got a validator exception it is actually correct to * immediately bubble it up. */ if (ex.getCause() != null && ex.getCause() instanceof ValidatorException) { throw (ValidatorException) ex.getCause(); } if (LOGGER.isLoggable(Level.WARNING)) { LOGGER.log(Level.WARNING, ele.toString()); LOGGER.log(Level.WARNING, "jsf.facelets.el.method.expression.invoke.error: {0} {1}", new Object[] { ex.toString(), source.getExpressionString() }); } if (!(ex instanceof MethodNotFoundException)) { throw ex; } } } throw ele; } } // ------------------------------------------------- Methods from Expression @Override public String getExpressionString() { return delegate.getExpressionString(); } @SuppressWarnings({"EqualsWhichDoesntCheckParameterClass"}) @Override public boolean equals(Object o) { return delegate.equals(o); } @Override public int hashCode() { return delegate.hashCode(); } @Override public boolean isLiteralText() { return delegate.isLiteralText(); } // ----------------------------------------------------- Private Methods private boolean pushCompositeComponent(FacesContext ctx) { CompositeComponentStackManager manager = CompositeComponentStackManager.getManager(ctx); UIComponent foundCc = null; if (location != null) { foundCc = manager.findCompositeComponentUsingLocation(ctx, location); } else { // We need to obtain the Location of the source expression in order // to find the composite component that needs to be available within // the evaluation stack. if (source instanceof TagValueExpression) { ValueExpression orig = ((TagValueExpression) source).getWrapped(); if (orig instanceof ContextualCompositeValueExpression) { foundCc = manager.findCompositeComponentUsingLocation(ctx, ((ContextualCompositeValueExpression) orig).getLocation()); } } } if (null == foundCc) { foundCc = ctx.getViewRoot().findComponent(ccClientId); } return manager.push(foundCc); } private void popCompositeComponent(FacesContext ctx) { CompositeComponentStackManager manager = CompositeComponentStackManager.getManager(ctx); manager.pop(); } private class SetClientIdListener implements ComponentSystemEventListener { private ContextualCompositeMethodExpression ccME; public SetClientIdListener() { } public SetClientIdListener(ContextualCompositeMethodExpression ccME) { this.ccME = ccME; } @Override public void processEvent(ComponentSystemEvent event) throws AbortProcessingException { ccME.ccClientId = event.getComponent().getClientId(); event.getComponent().unsubscribeFromEvent(PostAddToViewEvent.class, this); } } } // END ContextualCompositeMethodExpression




© 2015 - 2024 Weber Informatics LLC | Privacy Policy