/* * � Copyright IBM Corp. 2011 * * 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. */ /* * Author: Maire Kehoe (mkehoe@ie.ibm.com) * Date: 23 Aug 2011 * UIVarPublisherBase.java */ package com.ibm.xsp.extlib.component.layout; import java.io.IOException; import java.util.List; import javax.faces.FacesException; import javax.faces.component.ContextCallback; import javax.faces.component.UIComponentBase; import javax.faces.component.visit.VisitCallback; import javax.faces.component.visit.VisitContext; import javax.faces.context.FacesContext; import javax.faces.event.AbortProcessingException; import javax.faces.event.FacesEvent; import com.ibm.xsp.component.FacesComponent; import com.ibm.xsp.component.FacesDataProvider; import com.ibm.xsp.component.FacesEventWrapper; import com.ibm.xsp.context.FacesContextEx; import com.ibm.xsp.page.FacesComponentBuilder; import com.ibm.xsp.util.DataPublisher; import com.ibm.xsp.util.DataPublisher.ShadowedObject; /** * This is not actually a family of controls, just a useful base class for controls * that need to publish some var, as it invokes the {@link #publishControlData(FacesContext)} * and {@link #revokeControlData(List, FacesContext)} methods * at all the appropriate times in the JSF lifecycle. * @author Maire Kehoe (mkehoe@ie.ibm.com) */ public abstract class UIVarPublisherBase extends UIComponentBase implements FacesComponent{ private transient List<ShadowedObject> _shadowedData; public UIVarPublisherBase() { super(); } /** * Available to be overridden in the subclasses to publish other vars, * besides those supported by default when the control implements {@link FacesDataProvider} * or one of the other interfaces with implicit supoort in {@link DataPublisher}. * @see DataPublisher#publishControlData(javax.faces.component.UIComponent) * @param context * @return */ protected List<ShadowedObject> publishControlData(FacesContext context) { DataPublisher dataPublisher = ((FacesContextEx)context).getDataPublisher(); return dataPublisher.publishControlData(this); } /** * Available to be overridden in the subclasses to do other revoke and clean-up code. * @see DataPublisher#revokeControlData(List, javax.faces.component.UIComponent) * @param shadowedData * @param context */ protected void revokeControlData(List<ShadowedObject> shadowedData, FacesContext context) { DataPublisher dataPublisher = ((FacesContextEx)context).getDataPublisher(); dataPublisher.revokeControlData(shadowedData, this); } /** * @see DataPublisher#isAllowCreateViewPublish(javax.faces.component.UIComponent) * @param context * @return */ protected boolean isAllowCreateViewPublish(FacesContext context) { return ((FacesContextEx)context).getDataPublisher().isAllowCreateViewPublish(this); } public void initBeforeContents(FacesContext context) throws FacesException { // make the published object available to the children if( isAllowCreateViewPublish(context) ) { _shadowedData = publishControlData(context); } } public void buildContents(FacesContext context, FacesComponentBuilder builder) throws FacesException { // default implementation builder.buildAll(context, this, true); } public void initAfterContents(FacesContext context) throws FacesException { if( null != _shadowedData ){ revokeControlData(_shadowedData, context); _shadowedData = null; } } @Override public void encodeBegin(FacesContext context) throws IOException { _shadowedData = publishControlData(context); super.encodeBegin(context); } @Override public void encodeEnd(FacesContext context) throws IOException { super.encodeEnd(context); revokeControlData(_shadowedData, context); _shadowedData = null; } @Override public void processDecodes(FacesContext context) { try { _shadowedData = publishControlData(context); super.processDecodes(context); } finally { revokeControlData(_shadowedData, context); _shadowedData = null; } } @Override public void processUpdates(FacesContext context) { try { _shadowedData = publishControlData(context); super.processUpdates(context); } finally { revokeControlData(_shadowedData, context); _shadowedData = null; } } @Override public void processValidators(FacesContext context) { try { _shadowedData = publishControlData(context); super.processValidators(context); } finally { revokeControlData(_shadowedData, context); _shadowedData = null; } } @Override public boolean invokeOnComponent(FacesContext context, String clientId, ContextCallback callback) throws FacesException { try { _shadowedData = publishControlData(context); return super.invokeOnComponent(context, clientId, callback); } finally { revokeControlData(_shadowedData, context); _shadowedData = null; } } @Override public boolean visitTree(VisitContext context, VisitCallback callback) { try { _shadowedData = publishControlData(context.getFacesContext()); return super.visitTree(context, callback); } finally { revokeControlData(_shadowedData, context.getFacesContext()); _shadowedData = null; } } @Override public void queueEvent(FacesEvent event) { super.queueEvent(new FacesEventWrapper(this, event)); } @Override public void broadcast(FacesEvent event) throws AbortProcessingException { if (event instanceof FacesEventWrapper) { FacesEventWrapper wrapper = (FacesEventWrapper) event; FacesContext context = getFacesContext(); try { _shadowedData = publishControlData(context); FacesEvent original = wrapper.getFacesEvent(); original.getComponent().broadcast(original); } finally { revokeControlData(_shadowedData, context); _shadowedData = null; } } else { // original event was queued on this control super.broadcast(event); } } }