001    // Copyright 2007, 2008, 2009, 2010 The Apache Software Foundation
002    //
003    // Licensed under the Apache License, Version 2.0 (the "License");
004    // you may not use this file except in compliance with the License.
005    // You may obtain a copy of the License at
006    //
007    // http://www.apache.org/licenses/LICENSE-2.0
008    //
009    // Unless required by applicable law or agreed to in writing, software
010    // distributed under the License is distributed on an "AS IS" BASIS,
011    // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
012    // See the License for the specific language governing permissions and
013    // limitations under the License.
014    
015    package org.apache.tapestry5.internal;
016    
017    import org.apache.tapestry5.*;
018    import org.apache.tapestry5.dom.Element;
019    import org.apache.tapestry5.services.Environment;
020    import org.apache.tapestry5.services.FormSupport;
021    
022    /**
023     * Default implementation that writes an attribute into fields or labels that are in error.
024     */
025    public final class DefaultValidationDecorator extends BaseValidationDecorator
026    {
027        private final Environment environment;
028    
029        private final Asset spacerAsset;
030    
031        private final MarkupWriter markupWriter;
032    
033        /**
034         * @param environment
035         *            used to locate objects and services during the render
036         * @param spacerAsset
037         *            asset for a one-pixel spacer image used as a placeholder for the error marker icon
038         * @param markupWriter
039         */
040        public DefaultValidationDecorator(Environment environment, Asset spacerAsset, MarkupWriter markupWriter)
041        {
042            this.environment = environment;
043            this.spacerAsset = spacerAsset;
044            this.markupWriter = markupWriter;
045        }
046    
047        @Override
048        public void insideField(Field field)
049        {
050            if (inError(field))
051                addErrorClassToCurrentElement();
052        }
053    
054        @Override
055        public void insideLabel(Field field, Element element)
056        {
057            if (field == null)
058                return;
059    
060            if (inError(field))
061                element.addClassName(CSSClassConstants.ERROR);
062        }
063    
064        /**
065         * Writes an icon for field after the field. The icon has the same id as the field, with ":icon" appended. This is
066         * expected by the default client-side JavaScript. The icon's src is a blank spacer image (this is to allow the
067         * image displayed to be overridden via CSS). The icon's CSS class is "t-error-icon", with "t-invisible" added
068         * if the field is not in error when rendered. If client validation is not enabled for the form containing the
069         * field and the field is not in error, then the error icon itself is not rendered.
070         * 
071         * @param field
072         *            which just completed rendering itself
073         */
074        @Override
075        public void afterField(Field field)
076        {
077            boolean inError = inError(field);
078    
079            boolean clientValidationEnabled = getFormSupport().isClientValidationEnabled();
080    
081            if (inError || clientValidationEnabled)
082            {
083                String iconId = field.getClientId() + "_icon";
084    
085                String cssClass = inError ? "t-error-icon" : "t-error-icon t-invisible";
086    
087                markupWriter.element("img", "src", spacerAsset.toClientURL(), "alt", "", "class", cssClass, "id", iconId);
088                markupWriter.end();
089            }
090    
091        }
092    
093        private FormSupport getFormSupport()
094        {
095            return environment.peekRequired(FormSupport.class);
096        }
097    
098        private boolean inError(Field field)
099        {
100            ValidationTracker tracker = environment.peekRequired(ValidationTracker.class);
101    
102            return tracker.inError(field);
103        }
104    
105        private void addErrorClassToCurrentElement()
106        {
107            markupWriter.getElement().addClassName(CSSClassConstants.ERROR);
108        }
109    }