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

org.metawidget.faces.component.html.widgetbuilder.HtmlWidgetBuilder Maven / Gradle / Ivy

// Metawidget
//
// This file is dual licensed under both the LGPL
// (http://www.gnu.org/licenses/lgpl-2.1.html) and the EPL
// (http://www.eclipse.org/org/documents/epl-v10.php). As a
// recipient of Metawidget, you may choose to receive it under either
// the LGPL or the EPL.
//
// Commercial licenses are also available. See http://metawidget.org
// for details.

package org.metawidget.faces.component.html.widgetbuilder;

import static org.metawidget.inspector.InspectionResultConstants.*;
import static org.metawidget.inspector.faces.FacesInspectionResultConstants.*;

import java.util.Collection;
import java.util.Date;
import java.util.List;
import java.util.Map;

import javax.faces.application.Application;
import javax.faces.component.UIColumn;
import javax.faces.component.UICommand;
import javax.faces.component.UIComponent;
import javax.faces.component.UIData;
import javax.faces.component.UISelectItem;
import javax.faces.component.UISelectItems;
import javax.faces.component.UISelectMany;
import javax.faces.component.ValueHolder;
import javax.faces.component.html.HtmlColumn;
import javax.faces.component.html.HtmlCommandButton;
import javax.faces.component.html.HtmlCommandLink;
import javax.faces.component.html.HtmlDataTable;
import javax.faces.component.html.HtmlInputSecret;
import javax.faces.component.html.HtmlInputText;
import javax.faces.component.html.HtmlInputTextarea;
import javax.faces.component.html.HtmlOutputText;
import javax.faces.component.html.HtmlSelectBooleanCheckbox;
import javax.faces.component.html.HtmlSelectManyCheckbox;
import javax.faces.component.html.HtmlSelectOneMenu;
import javax.faces.component.html.HtmlSelectOneRadio;
import javax.faces.context.FacesContext;
import javax.faces.convert.Converter;
import javax.faces.el.MethodBinding;
import javax.faces.model.DataModel;
import javax.faces.model.SelectItem;

import org.metawidget.faces.FacesUtils;
import org.metawidget.faces.component.UIMetawidget;
import org.metawidget.faces.component.UIStub;
import org.metawidget.faces.component.html.HtmlMetawidget;
import org.metawidget.faces.component.widgetprocessor.ConverterProcessor;
import org.metawidget.faces.component.widgetprocessor.StandardBindingProcessor;
import org.metawidget.util.ArrayUtils;
import org.metawidget.util.CollectionUtils;
import org.metawidget.util.WidgetBuilderUtils;
import org.metawidget.util.XmlUtils;
import org.metawidget.util.simple.StringUtils;
import org.metawidget.widgetbuilder.iface.WidgetBuilder;
import org.metawidget.widgetbuilder.iface.WidgetBuilderException;
import org.metawidget.widgetprocessor.iface.WidgetProcessor;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

/**
 * WidgetBuilder for Java Server Faces environments.
 * 

* Creates native JSF HTML UIComponents, such as HtmlInputText and * HtmlSelectOneMenu, to suit the inspected fields. * * @author Richard Kennard */ @SuppressWarnings( "deprecation" ) public class HtmlWidgetBuilder implements WidgetBuilder { // // Private statics // private static final String DATATABLE_ROW_ACTION = "dataTableRowAction"; /** * The number of items in a multi-select lookup at which it should change from being a * 'lineDirection' to a 'pageDirection' layout. The latter is generally the safer choice, as it * stops the Metawidget blowing out horizontally. */ private static final int SHORT_LOOKUP_SIZE = 3; // // Private members // private final String mDataTableStyleClass; private final String[] mDataTableColumnClasses; private final String[] mDataTableRowClasses; private final int mMaximumColumnsInDataTable; // // Constructor // public HtmlWidgetBuilder() { this( new HtmlWidgetBuilderConfig() ); } public HtmlWidgetBuilder( HtmlWidgetBuilderConfig config ) { mDataTableStyleClass = config.getDataTableStyleClass(); mDataTableColumnClasses = config.getDataTableColumnClasses(); mDataTableRowClasses = config.getDataTableRowClasses(); mMaximumColumnsInDataTable = config.getMaximumColumnsInDataTable(); } // // Public methods // /** * Purely creates the widget. Does not concern itself with the widget's id, value binding or * preparing metadata for the renderer. * * @return the widget to use in non-read-only scenarios */ public UIComponent buildWidget( String elementName, Map attributes, UIMetawidget metawidget ) { FacesContext context = FacesContext.getCurrentInstance(); Application application = context.getApplication(); // Hidden if ( TRUE.equals( attributes.get( HIDDEN ) ) ) { return application.createComponent( UIStub.COMPONENT_TYPE ); } // Overridden component UIComponent component = null; String componentName = attributes.get( FACES_COMPONENT ); if ( componentName != null ) { component = application.createComponent( componentName ); } // Action if ( ACTION.equals( elementName ) ) { if ( component == null ) { component = application.createComponent( HtmlCommandButton.COMPONENT_TYPE ); } ( (UICommand) component ).setValue( metawidget.getLabelString( attributes ) ); return component; } // Lookup the class Class clazz = WidgetBuilderUtils.getActualClassOrType( attributes, String.class ); // Faces Lookups String facesLookup = attributes.get( FACES_LOOKUP ); if ( facesLookup != null && !"".equals( facesLookup ) ) { if ( component == null ) { // UISelectMany... if ( clazz != null && ( List.class.isAssignableFrom( clazz ) || clazz.isArray() ) ) { component = application.createComponent( HtmlSelectManyCheckbox.COMPONENT_TYPE ); } else { // ...otherwise just a UISelectOne component = application.createComponent( HtmlSelectOneMenu.COMPONENT_TYPE ); } } initFacesSelect( component, facesLookup, attributes, metawidget ); return component; } // clazz may be null, if type is symbolic (eg. type="Login Screen") if ( clazz != null ) { // Support mandatory Booleans (can be rendered as a checkbox, even though they have a // Lookup) if ( component == null && Boolean.class.equals( clazz ) && TRUE.equals( attributes.get( REQUIRED ) ) ) { return application.createComponent( HtmlSelectBooleanCheckbox.COMPONENT_TYPE ); } // String Lookups String lookup = attributes.get( LOOKUP ); if ( lookup != null && !"".equals( lookup ) ) { if ( component == null ) { // UISelectMany... if ( List.class.isAssignableFrom( clazz ) || clazz.isArray() ) { component = application.createComponent( HtmlSelectManyCheckbox.COMPONENT_TYPE ); } else { // ...otherwise just a UISelectOne component = application.createComponent( HtmlSelectOneMenu.COMPONENT_TYPE ); } } initStaticSelect( component, lookup, clazz, attributes, metawidget ); } // If no component specified yet, pick one if ( component == null ) { if ( boolean.class.equals( clazz ) ) { component = application.createComponent( HtmlSelectBooleanCheckbox.COMPONENT_TYPE ); } else if ( char.class.equals( clazz ) || Character.class.isAssignableFrom( clazz ) ) { component = application.createComponent( HtmlInputText.COMPONENT_TYPE ); ( (HtmlInputText) component ).setMaxlength( 1 ); } else if ( clazz.isPrimitive() ) { component = application.createComponent( HtmlInputText.COMPONENT_TYPE ); } else if ( Date.class.isAssignableFrom( clazz ) ) { // Support Date as standard, so that StandardConverterProcessor can attach a // DateTimeConverter component = application.createComponent( HtmlInputText.COMPONENT_TYPE ); } else if ( Number.class.isAssignableFrom( clazz ) ) { component = application.createComponent( HtmlInputText.COMPONENT_TYPE ); } else if ( String.class.equals( clazz ) ) { if ( TRUE.equals( attributes.get( MASKED ) ) ) { component = application.createComponent( HtmlInputSecret.COMPONENT_TYPE ); } else if ( TRUE.equals( attributes.get( LARGE ) ) ) { component = application.createComponent( HtmlInputTextarea.COMPONENT_TYPE ); // XHTML requires the 'cols' and 'rows' attributes be set, even though // most people override them with CSS widths and heights. The default is // generally 20 columns by 2 rows ( (HtmlInputTextarea) component ).setCols( 20 ); ( (HtmlInputTextarea) component ).setRows( 2 ); } else { component = application.createComponent( HtmlInputText.COMPONENT_TYPE ); } } else if ( List.class.isAssignableFrom( clazz ) || DataModel.class.isAssignableFrom( clazz ) || clazz.isArray() ) { // Supported Collections return createDataTableComponent( elementName, attributes, metawidget ); } else if ( Collection.class.isAssignableFrom( clazz ) ) { // Unsupported Collections return application.createComponent( UIStub.COMPONENT_TYPE ); } } setMaximumLength( component, attributes ); if ( component != null ) { return component; } } // Not simple, but don't expand if ( TRUE.equals( attributes.get( DONT_EXPAND ) ) ) { return application.createComponent( HtmlInputText.COMPONENT_TYPE ); } // Nested Metawidget return null; } // // Protected methods // protected void initFacesSelect( UIComponent component, String facesLookup, Map attributes, UIMetawidget metawidget ) { // (pageDirection is a 'safer' default for anything but short lists) if ( component instanceof HtmlSelectManyCheckbox ) { ( (HtmlSelectManyCheckbox) component ).setLayout( "pageDirection" ); } else if ( component instanceof HtmlSelectOneRadio ) { ( (HtmlSelectOneRadio) component ).setLayout( "pageDirection" ); } addSelectItems( component, facesLookup, attributes, metawidget ); } protected void initStaticSelect( UIComponent component, String lookup, Class clazz, Map attributes, UIMetawidget metawidget ) { FacesContext context = FacesContext.getCurrentInstance(); Application application = context.getApplication(); // (pageDirection is a 'safer' default for anything but short lists) List values = CollectionUtils.fromString( lookup ); if ( values.size() > SHORT_LOOKUP_SIZE ) { if ( component instanceof HtmlSelectManyCheckbox ) { ( (HtmlSelectManyCheckbox) component ).setLayout( "pageDirection" ); } else if ( component instanceof HtmlSelectOneRadio ) { ( (HtmlSelectOneRadio) component ).setLayout( "pageDirection" ); } } // Convert values of SelectItems (eg. from Strings to ints)... List valuesAfterConversion = values; if ( component instanceof ValueHolder ) { // ...using the specified converter (call setConverter prematurely so // we can find out what Converter to use)... Converter converter = null; ConverterProcessor processor = metawidget.getWidgetProcessor( ConverterProcessor.class ); if ( processor != null ) { converter = processor.getConverter( (ValueHolder) component, attributes ); } // ...(setConverter doesn't do application-wide converters)... if ( converter == null ) { // ...(we don't try a 'clazz' converter for a UISelectMany, // because the 'clazz' will generally be a Collection. setConverter // will have already tried PARAMETERIZED_TYPE)... if ( !( component instanceof UISelectMany ) ) { converter = application.createConverter( clazz ); } } // ...if any if ( converter != null ) { int size = valuesAfterConversion.size(); List convertedValues = CollectionUtils.newArrayList( size ); for ( int loop = 0; loop < size; loop++ ) { // Note: the component at this point will not have a ValueBinding, as // that gets added in addWidget. This can scupper clever Converters that // try to determine the type based on the ValueBinding. For those, we // recommend overriding 'Application.createConverter' and passing the // type in the Converter's constructor instead Object convertedValue = converter.getAsObject( context, component, (String) values.get( loop ) ); convertedValues.add( convertedValue ); } valuesAfterConversion = convertedValues; } } addSelectItems( component, valuesAfterConversion, CollectionUtils.fromString( attributes.get( LOOKUP_LABELS ) ), attributes, metawidget ); } protected void setMaximumLength( UIComponent component, Map attributes ) { String maximumLength = attributes.get( MAXIMUM_LENGTH ); if ( maximumLength != null && !"".equals( maximumLength ) ) { if ( component instanceof HtmlInputText ) { ( (HtmlInputText) component ).setMaxlength( Integer.parseInt( maximumLength ) ); } else if ( component instanceof HtmlInputSecret ) { ( (HtmlInputSecret) component ).setMaxlength( Integer.parseInt( maximumLength ) ); } } } /** * @param elementName * such as ENTITY or PROPERTY. Can be useful in determining how to construct the EL * for the table. */ protected UIComponent createDataTableComponent( String elementName, Map attributes, UIMetawidget metawidget ) { FacesContext context = FacesContext.getCurrentInstance(); Application application = context.getApplication(); HtmlDataTable dataTable = (HtmlDataTable) application.createComponent( HtmlDataTable.COMPONENT_TYPE ); dataTable.setVar( "_item" ); // CSS dataTable.setStyleClass( mDataTableStyleClass ); dataTable.setColumnClasses( ArrayUtils.toString( mDataTableColumnClasses ) ); dataTable.setRowClasses( ArrayUtils.toString( mDataTableRowClasses ) ); // Inspect component type String componentType = WidgetBuilderUtils.getComponentType( attributes ); String inspectedType = null; if ( componentType != null ) { inspectedType = metawidget.inspect( null, componentType ); } // If there is no type... NodeList elements; if ( inspectedType == null ) { elements = null; } else { Element root = XmlUtils.documentFromString( inspectedType ).getDocumentElement(); elements = root.getFirstChild().getChildNodes(); } if ( elements == null || elements.getLength() == 0 ) { // ...resort to a single column table... Map columnAttributes = CollectionUtils.newHashMap(); columnAttributes.put( NAME, attributes.get( NAME ) ); addColumnComponent( dataTable, attributes, ENTITY, columnAttributes, metawidget ); } else { // ...otherwise, iterate over the component type and add multiple columns addColumnComponents( dataTable, attributes, elements, metawidget ); } // Add an 'action' column (if requested) String rowActionParameter = metawidget.getParameter( DATATABLE_ROW_ACTION ); if ( rowActionParameter != null ) { HtmlCommandLink rowAction = (HtmlCommandLink) application.createComponent( HtmlCommandLink.COMPONENT_TYPE ); rowAction.setId( FacesUtils.createUniqueId() ); // (dataTableRowAction cannot be wrapped when used on the JSP page) if ( FacesUtils.isExpression( rowActionParameter ) ) { throw WidgetBuilderException.newException( DATATABLE_ROW_ACTION + " must be an unwrapped JSF expression (eg. foo.bar, not #{foo.bar})" ); } String actionName = StringUtils.substringAfterLast( rowActionParameter, "." ); String localizedKey = metawidget.getLocalizedKey( actionName ); if ( localizedKey == null ) { rowAction.setValue( StringUtils.uncamelCase( actionName ) ); } else { rowAction.setValue( localizedKey ); } MethodBinding binding = application.createMethodBinding( FacesUtils.wrapExpression( rowActionParameter ), null ); rowAction.setAction( binding ); UIColumn column = (UIColumn) application.createComponent( HtmlColumn.COMPONENT_TYPE ); column.setId( FacesUtils.createUniqueId() ); column.getChildren().add( rowAction ); dataTable.getChildren().add( column ); // Put a blank header, so that CSS styling (such as border-bottom) still applies HtmlOutputText headerText = (HtmlOutputText) application.createComponent( HtmlOutputText.COMPONENT_TYPE ); headerText.setId( FacesUtils.createUniqueId() ); headerText.setValue( "
" ); headerText.setEscape( false ); column.setHeader( headerText ); } return dataTable; } /** * Adds column components to the given UIData. *

* Clients can override this method to add additional columns, such as a 'Delete' button. */ protected void addColumnComponents( UIData dataTable, Map attributes, NodeList elements, UIMetawidget metawidget ) { // At first, try to add columns for just the 'required' fields boolean onlyRequired = true; while ( true ) { // For each property... for ( int loop = 0, length = elements.getLength(); loop < length; loop++ ) { Node node = elements.item( loop ); if ( !( node instanceof Element ) ) { continue; } Element element = (Element) node; // ...(not action)... if ( ACTION.equals( element.getNodeName() ) ) { continue; } // ...that is visible... if ( TRUE.equals( element.getAttribute( HIDDEN ) ) ) { continue; } // ...and is required... // // Note: this is a controversial choice. Our logic is that a) we need to limit // the number of columns somehow, and b) displaying all the required fields should // be enough to uniquely identify the row to the user. However, users may wish // to override this default behaviour if ( onlyRequired && !TRUE.equals( element.getAttribute( REQUIRED ) ) ) { continue; } // ...add a column... addColumnComponent( dataTable, attributes, PROPERTY, XmlUtils.getAttributesAsMap( element ), metawidget ); // ...up to a sensible maximum if ( dataTable.getChildren().size() == mMaximumColumnsInDataTable ) { break; } } // If we couldn't add any 'required' columns, try again for every field if ( !dataTable.getChildren().isEmpty() || !onlyRequired ) { break; } onlyRequired = false; } } /** * Create a UIColumn component for the given attributes, to the given UIData. *

* Clients can override this method to modify the column contents. For example, to place a link * around the text. * * @param tableAttributes * the metadata attributes used to render the parent table. May be useful for * determining the overall type of the row */ protected void addColumnComponent( UIData dataTable, Map tableAttributes, String elementName, Map columnAttributes, UIMetawidget metawidget ) { FacesContext context = FacesContext.getCurrentInstance(); Application application = context.getApplication(); UIColumn column = (UIColumn) application.createComponent( HtmlColumn.COMPONENT_TYPE ); column.setId( FacesUtils.createUniqueId() ); // Make the column contents... // // Note: this cannot be implemented as a nested Metawidget until // http://java.net/jira/browse/JAVASERVERFACES-2089 UIComponent columnText = application.createComponent( HtmlOutputText.COMPONENT_TYPE ); columnText.setId( FacesUtils.createUniqueId() ); HtmlMetawidget dummyMetawidget = new HtmlMetawidget(); dummyMetawidget.setValueBinding( "value", application.createValueBinding( FacesUtils.wrapExpression( dataTable.getVar() ) ) ); // ...process them... WidgetProcessor bindingProcessor = metawidget.getWidgetProcessor( StandardBindingProcessor.class ); if ( bindingProcessor != null ) { bindingProcessor.processWidget( columnText, elementName, columnAttributes, dummyMetawidget ); } @SuppressWarnings( "unchecked" ) WidgetProcessor converterProcessor = (WidgetProcessor) metawidget.getWidgetProcessor( ConverterProcessor.class ); if ( converterProcessor != null ) { converterProcessor.processWidget( columnText, elementName, columnAttributes, dummyMetawidget ); } column.getChildren().add( columnText ); // ...with a localized header HtmlOutputText headerText = (HtmlOutputText) application.createComponent( HtmlOutputText.COMPONENT_TYPE ); headerText.setId( FacesUtils.createUniqueId() ); headerText.setValue( metawidget.getLabelString( columnAttributes ) ); column.setHeader( headerText ); dataTable.getChildren().add( column ); } // // Private methods // private void addSelectItems( UIComponent component, List values, List labels, Map attributes, UIMetawidget metawidget ) { if ( values == null ) { return; } // Empty option if ( component instanceof HtmlSelectOneMenu && WidgetBuilderUtils.needsEmptyLookupItem( attributes ) ) { addSelectItem( component, null, null, metawidget ); } // See if we're using labels // // (note: where possible, it is better to use a Converter than a hard-coded label) if ( labels != null && !labels.isEmpty() && labels.size() != values.size() ) { throw WidgetBuilderException.newException( "Labels list must be same size as values list" ); } // Add the select items for ( int loop = 0, length = values.size(); loop < length; loop++ ) { Object value = values.get( loop ); String label = null; if ( labels != null && !labels.isEmpty() ) { label = labels.get( loop ); } addSelectItem( component, value, label, metawidget ); } } private void addSelectItem( UIComponent component, Object value, String label, UIMetawidget metawidget ) { FacesContext context = FacesContext.getCurrentInstance(); Application application = context.getApplication(); UISelectItem selectItem = (UISelectItem) application.createComponent( UISelectItem.COMPONENT_TYPE ); selectItem.setId( FacesUtils.createUniqueId() ); // JSF 1.1 doesn't allow 'null' as the item value, but JSF 1.2 requires // it for proper behaviour (see // https://javaserverfaces.dev.java.net/issues/show_bug.cgi?id=795) if ( value == null ) { try { UISelectItem.class.getMethod( "getValueExpression", String.class ); selectItem.setValue( new SelectItem( null, "" ) ); } catch ( NoSuchMethodException e ) { selectItem.setItemValue( "" ); } } else { selectItem.setItemValue( value ); } if ( label == null ) { // If no label, make it the same as the value. For JSF-RI, this is needed for // labels next to UISelectMany checkboxes. See // https://javaserverfaces.dev.java.net/issues/show_bug.cgi?id=913 selectItem.setItemLabel( StringUtils.quietValueOf( value ) ); } else { // Label may be a value reference (eg. into a bundle) if ( FacesUtils.isExpression( label ) ) { selectItem.setValueBinding( "itemLabel", application.createValueBinding( label ) ); } else { // Label may be localized String localizedLabel = metawidget.getLocalizedKey( StringUtils.camelCase( label ) ); if ( localizedLabel != null ) { selectItem.setItemLabel( localizedLabel ); } else { selectItem.setItemLabel( label ); } } } component.getChildren().add( selectItem ); } private void addSelectItems( UIComponent component, String binding, Map attributes, UIMetawidget metawidget ) { if ( binding == null ) { return; } FacesContext context = FacesContext.getCurrentInstance(); Application application = context.getApplication(); // Empty option if ( component instanceof HtmlSelectOneMenu && WidgetBuilderUtils.needsEmptyLookupItem( attributes ) ) { addSelectItem( component, null, null, metawidget ); } UISelectItems selectItems = (UISelectItems) application.createComponent( UISelectItems.COMPONENT_TYPE ); selectItems.setId( FacesUtils.createUniqueId() ); component.getChildren().add( selectItems ); if ( !FacesUtils.isExpression( binding ) ) { throw WidgetBuilderException.newException( "Lookup '" + binding + "' is not of the form #{...}" ); } selectItems.setValueBinding( "value", application.createValueBinding( binding ) ); // Optional attributes String var = attributes.get( FACES_LOOKUP_VAR ); if ( var != null ) { selectItems.getAttributes().put( "var", var ); } String itemLabelBinding = attributes.get( FACES_LOOKUP_ITEM_LABEL ); if ( itemLabelBinding != null ) { selectItems.setValueBinding( "itemLabel", application.createValueBinding( itemLabelBinding ) ); } String itemValueBinding = attributes.get( FACES_LOOKUP_ITEM_VALUE ); if ( itemValueBinding != null ) { selectItems.setValueBinding( "itemValue", application.createValueBinding( itemValueBinding ) ); } } }