Package org.apache.myfaces.trinidad.webapp

Source Code of org.apache.myfaces.trinidad.webapp.UIXComponentTag

/*
*  Licensed to the Apache Software Foundation (ASF) under one
*  or more contributor license agreements.  See the NOTICE file
*  distributed with this work for additional information
*  regarding copyright ownership.  The ASF licenses this file
*  to you 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 org.apache.myfaces.trinidad.webapp;

import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;

import java.util.ArrayList;
import java.util.Date;
import java.util.Iterator;

import javax.el.MethodExpression;

import javax.faces.component.UIComponent;
import javax.faces.component.UIViewRoot;
import javax.faces.context.FacesContext;
import javax.faces.el.MethodBinding;
import javax.faces.el.ValueBinding;
import javax.faces.webapp.UIComponentTag;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.TagSupport;

import org.apache.myfaces.trinidad.bean.FacesBean;
import org.apache.myfaces.trinidad.bean.PropertyKey;
import org.apache.myfaces.trinidad.change.AddComponentChange;
import org.apache.myfaces.trinidad.change.AttributeComponentChange;
import org.apache.myfaces.trinidad.change.ComponentChange;
import org.apache.myfaces.trinidad.component.UIXComponent;
import org.apache.myfaces.trinidad.context.RequestContext;
import org.apache.myfaces.trinidad.event.AttributeChangeEvent;
import org.apache.myfaces.trinidad.logging.TrinidadLogger;

/**
* Subclass of UIComponentTag to add convenience methods,
* and optimize where appropriate.
*/
abstract public class UIXComponentTag extends UIComponentTag
{
  public UIXComponentTag()
  {
  }

  @Override
  public void setId(String id)
  {
    _id = id;
  }

  @Override
  public void setRendered(String rendered)
  {
    _rendered = rendered;
  }

  public void setAttributeChangeListener(String attributeChangeListener)
  {
    _attributeChangeListener = attributeChangeListener;
  }

  @Override
  public int doStartTag() throws JspException
  {
    _parentELContext = (ELContextTag)
       TagSupport.findAncestorWithClass(this, ELContextTag.class);

    // Transform "rendered" on behalf of the UIComponentTag
    String rendered = _rendered;
    if (rendered != null)
    {
      if ((_parentELContext != null) && isValueReference(rendered))
        rendered = _parentELContext.transformExpression(rendered);

      super.setRendered(rendered);
    }


    String id = _id;
    if (id != null)
    {
      if (_parentELContext != null)
        id = _parentELContext.transformId(id);

      super.setId(id);
    }

    int retVal = super.doStartTag();

    //pu: There could have been some validation error during property setting
    //  on the bean, this is the closest opportunity to burst out.
    if (_validationError != null)
      throw new JspException(_validationError);

    return retVal;
  }

  @Override
  public int doEndTag() throws JspException
  {
    UIComponent component = getComponentInstance();
    if (isSuppressed() && getCreated())
      _applyChanges(getFacesContext(), component);
    return super.doEndTag();
  }

  @Override
  protected void encodeBegin() throws java.io.IOException
  {
    UIComponent component = getComponentInstance();
    if (!isSuppressed() && getCreated())
      _applyChanges(getFacesContext(), component);
    super.encodeBegin();
  }

  @Override
  protected final void setProperties(UIComponent component)
  {
    if (component instanceof UIViewRoot)
    {
      throw new IllegalStateException(_LOG.getMessage(
        "VIEW_TAG_NOT_PRESENT", this));
    }

    super.setProperties(component);
   
    UIXComponent uixComponent = (UIXComponent) component;

    if (_attributeChangeListener != null)
    {
      MethodExpression me = getFacesContext().getApplication().
         getExpressionFactory().createMethodExpression(
             getFacesContext().getELContext(),
             _attributeChangeListener,
             null,
             new Class[]{AttributeChangeEvent.class});

      uixComponent.setAttributeChangeListener(me);
    }


    setProperties(uixComponent.getFacesBean());
  }

  protected void setProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      bean.setProperty(key, value);
    }
  }

  /**
   * Set a property of type java.lang.Boolean.  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed with Integer.valueOf().
   * Null values are ignored.
   */
  protected void setBooleanProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      bean.setProperty(key, Boolean.valueOf(value));
    }
  }

  /**
   * Set a property of type java.lang.Number.  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed with Integer.valueOf() or Double.valueOf() .
   * Null values are ignored.
   */
  protected void setNumberProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      if(value.indexOf('.') == -1)
        bean.setProperty(key, Integer.valueOf(value));
      else
        bean.setProperty(key, Double.valueOf(value));
    }
  }
/**
  * Set a property of type java.lang.Integer.  If the value
  * is an EL expression, it will be stored as a ValueBinding.
  * Otherwise, it will parsed with Integer.valueOf().
  * Null values are ignored.
  */
protected void setIntegerProperty(
   FacesBean   bean,
   PropertyKey key,
   String      value)
{
   if (value == null)
     return;

   if (isValueReference(value))
   {
     bean.setValueBinding(key, createValueBinding(value));
   }
   else
   {
     bean.setProperty(key, Integer.valueOf(value));
   }
}


  /**
   * Set a property of type java.lang.Character.  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, its first character will be stored (unless
   * it is an empty string, in which case it will be ignored).
   * Null values are ignored.
   */
  protected void setCharacterProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      if (value.length() >= 1)
        bean.setProperty(key, Character.valueOf(value.charAt(0)));
    }
  }

  /**
   * Set a property of type java.lang.Long.  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed with Long.valueOf().
   * Null values are ignored.
   */
  protected void setLongProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      bean.setProperty(key, Long.valueOf(value));
    }
  }

  /**
   * Set a property of type java.lang.Double.  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed with Double.valueOf().
   * Null values are ignored.
   */
  protected void setDoubleProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      bean.setProperty(key, Double.valueOf(value));
    }
  }

  /**
   * Set a property of type java.lang.Float.  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed with Float.valueOf().
   * Null values are ignored.
   */
  protected void setFloatProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      bean.setProperty(key, Float.valueOf(value));
    }
  }



  /**
   * Set a property of type java.lang.String[].  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed as a whitespace-separated series
   * of strings.
   * Null values are ignored.
   */
  protected void setStringArrayProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      bean.setProperty(key, _parseNameTokens(value));
    }
  }

  /**
   * Set a property of type int[].  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed as a whitespace-separated series
   * of ints.
   * Null values are ignored.
   */
  protected void setIntArrayProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      String[] strings = _parseNameTokens(value);
      final int[] ints;
      if (strings != null)
      {
        try
        {
          ints = new int[strings.length];
          for(int i=0; i<strings.length; i++)
          {
            int j = Integer.parseInt(strings[i]);
            ints[i] = j;
          }
        }
        catch (NumberFormatException e)
        {
          _LOG.severe("CANNOT_CONVERT_INTO_INT_ARRAY",value);
          _LOG.severe(e);
          return;
        }
      }
      else
        ints = null;

      bean.setProperty(key, ints);
    }
  }

  /**
   * Set a property of type java.util.Date.  If the value
   * is an EL expression, it will be stored as a ValueBinding.
   * Otherwise, it will parsed as an ISO 8601 date (yyyy-MM-dd).
   * Null values are ignored.
   */
  protected void setDateProperty(
    FacesBean   bean,
    PropertyKey key,
    String      value)
  {
    if (value == null)
      return;

    if (isValueReference(value))
    {
      bean.setValueBinding(key, createValueBinding(value));
    }
    else
    {
      bean.setProperty(key, _parseISODate(value));
    }
  }


  // TODO Handle syntax exceptions gracefully?
  protected final ValueBinding createValueBinding(String string)
  {
    if (_parentELContext != null)
      string = _parentELContext.transformExpression(string);

    return getFacesContext().getApplication().createValueBinding(string);
  }

  // TODO Handle syntax exceptions gracefully?
  protected final MethodBinding createMethodBinding(
    String   string,
    Class [] types)
  {
    if (_parentELContext != null)
      string = _parentELContext.transformExpression(string);

    return getFacesContext().getApplication().createMethodBinding(string,
                                                                  types);
  }

  protected void setProperties(FacesBean bean)
  {
    // Could be abstract, but it's easier to *always* call super.setProperties(),
    // and perhaps we'll have something generic in here, esp. if we take
    // over "rendered" from UIComponentTag
  }

  /**
   * Sets any fatal validation error that could have happened during property
   *  setting. If this is set, tag execution aborts with a JspException at the
   *  end of doStartTag().
   * @param validationError
   */
  protected void setValidationError(String validationError)
  {
    _validationError = validationError;
  }

  /**
   * Parse a string into a java.util.Date object.  The
   * string must be in ISO 9601 format (yyyy-MM-dd).
   */
  static private final Date _parseISODate(String stringValue)
  {
    try
    {
      return _ISO_DATE_FORMAT.parse(stringValue);
    }
    catch (ParseException pe)
    {
      _LOG.info("CANNOT_PARSE_VALUE_INTO_DATE", stringValue);
      return null;
    }
  }

  /**
   * Parses a whitespace separated series of name tokens.
   * @param stringValue the full string
   * @return an array of each constituent value, or null
   *  if there are no tokens (that is, the string is empty or
   *  all whitespace)
   */
  // TODO Move to utility function somewhere
  static private final String[] _parseNameTokens(String stringValue)
  {
    if (stringValue == null)
      return null;

    ArrayList<String> list = new ArrayList<String>(5);

    int     length = stringValue.length();
    boolean inSpace = true;
    int     start = 0;
    for (int i = 0; i < length; i++)
    {
      char ch = stringValue.charAt(i);

      // We're in whitespace;  if we've just departed
      // a run of non-whitespace, append a string.
      // Now, why do we use the supposedly deprecated "Character.isSpace()"
      // function instead of "isWhitespace"?  We're following XML rules
      // here for the meaning of whitespace, which specifically
      // EXCLUDES general Unicode spaces.
      if (Character.isWhitespace(ch))
      {
        if (!inSpace)
        {
          list.add(stringValue.substring(start, i));
          inSpace = true;
        }
      }
      // We're out of whitespace;  if we've just departed
      // a run of whitespace, start keeping track of this string
      else
      {
        if (inSpace)
        {
          start = i;
          inSpace = false;
        }
      }
    }

    if (!inSpace)
      list.add(stringValue.substring(start));

    if (list.isEmpty())
      return null;

    return list.toArray(new String[list.size()]);
  }

  private static void _applyChanges(
    FacesContext facesContext,
    UIComponent uiComponent)
  {
    RequestContext afc = RequestContext.getCurrentInstance();
    Iterator<ComponentChange> changeIter =
                  afc.getChangeManager().getComponentChanges(facesContext, uiComponent);

    if (changeIter == null)
      return;
    while (changeIter.hasNext())
    {
      ComponentChange change = changeIter.next();
     
      change.changeComponent(uiComponent);
     
      //pu: In case this Change has added a new component/facet, the added
      //  component could have its own Changes, that may need to be applied here.
      if (change instanceof AddComponentChange)
      {
        UIComponent newAddedComponent =
          ( (AddComponentChange)change ).getComponent();

        if (newAddedComponent != null)
        {
          _applyChanges(facesContext, newAddedComponent);
        }
      }
    }
  }

  private static final TrinidadLogger _LOG = TrinidadLogger.createTrinidadLogger(UIXComponentTag.class);

  // We rely strictly on ISO 8601 formats
  private static DateFormat  _ISO_DATE_FORMAT =
    new SimpleDateFormat("yyyy-MM-dd");

  private String       _rendered;
  private String       _id;
  private String       _attributeChangeListener;
  private String       _validationError;
  private ELContextTag _parentELContext;
}
TOP

Related Classes of org.apache.myfaces.trinidad.webapp.UIXComponentTag

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.