Coverage Report - org.kuali.student.common.ui.client.configurable.mvc.FieldDescriptor
 
Classes in this File Line Coverage Branch Coverage Complexity
FieldDescriptor
0%
0/132
0%
0/82
2.303
 
 1  
 /**
 2  
  * Copyright 2010 The Kuali Foundation Licensed under the
 3  
  * Educational Community License, Version 2.0 (the "License"); you may
 4  
  * not use this file except in compliance with the License. You may
 5  
  * obtain a copy of the License at
 6  
  *
 7  
  * http://www.osedu.org/licenses/ECL-2.0
 8  
  *
 9  
  * Unless required by applicable law or agreed to in writing,
 10  
  * software distributed under the License is distributed on an "AS IS"
 11  
  * BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
 12  
  * or implied. See the License for the specific language governing
 13  
  * permissions and limitations under the License.
 14  
  */
 15  
 
 16  
 package org.kuali.student.common.ui.client.configurable.mvc;
 17  
 
 18  
 
 19  
 import org.kuali.student.common.assembly.data.Metadata;
 20  
 import org.kuali.student.common.assembly.data.MetadataInterrogator;
 21  
 import org.kuali.student.common.ui.client.application.Application;
 22  
 import org.kuali.student.common.ui.client.configurable.mvc.binding.ModelWidgetBinding;
 23  
 import org.kuali.student.common.ui.client.configurable.mvc.binding.MultiplicityCompositeBinding;
 24  
 import org.kuali.student.common.ui.client.configurable.mvc.multiplicity.MultiplicityComposite;
 25  
 import org.kuali.student.common.ui.client.mvc.Callback;
 26  
 import org.kuali.student.common.ui.client.mvc.HasCrossConstraints;
 27  
 import org.kuali.student.common.ui.client.mvc.HasDataValue;
 28  
 import org.kuali.student.common.ui.client.widgets.KSCheckBox;
 29  
 import org.kuali.student.common.ui.client.widgets.KSTextBox;
 30  
 import org.kuali.student.common.ui.client.widgets.RichTextEditor;
 31  
 import org.kuali.student.common.ui.client.widgets.field.layout.element.FieldElement;
 32  
 import org.kuali.student.common.ui.client.widgets.field.layout.element.MessageKeyInfo;
 33  
 import org.kuali.student.common.ui.client.widgets.list.KSSelectItemWidgetAbstract;
 34  
 
 35  
 import com.google.gwt.user.client.ui.HasText;
 36  
 import com.google.gwt.user.client.ui.HasValue;
 37  
 import com.google.gwt.user.client.ui.Widget;
 38  
 
 39  
 /**
 40  
  * This is a field descriptor that defines ui fields.
 41  
  * A field descriptor is defined by its fieldKey and its metadata, the field key is the key that is used
 42  
  * to identify it during validation and save processing and maps directly to a field which exists in the
 43  
  * model definition for the model it relates to.  The fieldKey must match something within the model to
 44  
  * be a valid field.  <br>The metadata is its piece of information in the model definition which is used to
 45  
  * generate the field's widget and determine what the validation on this field is (also if this field
 46  
  * is backed by a search it will use the lookup metadata defined in its metadata to generate the appropriate
 47  
  * search widget).  It also determines if a requiredness indicator appears (constraint minOccurs > 1).
 48  
  * <br><br>
 49  
  * A field descriptor can be fully customized to any need, the widget can be chosen to be
 50  
  * defined directly instead of using one that is auto generated by the metadata - if this is done make sure
 51  
  * the data type it is using works with what is in the metadata and that it implements one of these standard 
 52  
  * interfaces:<br>
 53  
  * HasText<br>
 54  
  * KSSelectItemWidgetAbstract<br>
 55  
  * HasDataValue<br>
 56  
  * HasValue<br>
 57  
  * or, you MUST override the ModelWidgetBinding by calling setModelWidgetBinding
 58  
  * <br>
 59  
  * Setting the ModelWidgetBinding allows you to manipulate the widget's data and model data as you see fit
 60  
  * before a save when model data is loaded into the widget.
 61  
  * <br>
 62  
  * <br>
 63  
  * General layout of a field generated by FieldDescriptor, all elements are conditional based on the field's
 64  
  * configuration: <br>
 65  
  * <b>[label][requiredness indicator][help]<br>
 66  
  * [input widget]<br>
 67  
  * [constraint text]<br></b>
 68  
  * <br>
 69  
  * The messageKeyInfo passed in is used to generate the messages needed for a field these include:<br>
 70  
  * Field Label<br>
 71  
  * Help text<br>
 72  
  * Instructions<br>
 73  
  * Example text<br>
 74  
  * Constraint text<br>
 75  
  * Watermark text - only if the widget one that accepts text input<br><br>
 76  
  * These are generated by a single key, the additional text is determined by special suffixes on keys within
 77  
  * the messages data - example - you pass in "sampleField" for the message key - it is automatically determined
 78  
  * that if there is a message in the message data named "sampleField-instruct", instructions will be added to the field
 79  
  * in the appropriate location.<br>
 80  
  * List of the appended keys for use in messages data:<br>
 81  
  * "-help" for help text<br>
 82  
  * "-instruct" for instructions<br>
 83  
  * "-examples" for examples<br>
 84  
  * "-constraints" for constraint text<br>
 85  
  * "-watermark" for watermark text<br>
 86  
  * 
 87  
  * @author Kuali Student Team
 88  
  * @see FieldElement
 89  
  * @see Section
 90  
  * @see BaseSection
 91  
  * @see Configurer
 92  
  */
 93  
 public class FieldDescriptor {
 94  
     protected String fieldKey;
 95  
         protected Metadata metadata;
 96  
     @SuppressWarnings("unchecked")
 97  
         private ModelWidgetBinding modelWidgetBinding;
 98  
     private Callback<Boolean> validationRequestCallback;
 99  0
     private boolean dirty = false;
 100  0
     private boolean hasHadFocus = false;
 101  
     private final FieldElement fieldElement;
 102  
     private String modelId;
 103  
     private MessageKeyInfo messageKey;
 104  0
     private boolean optional = false;
 105  0
     private boolean ignoreShowRequired = false; 
 106  
 
 107  
     /**
 108  
      * @param fieldKey - key for this field which matches a field in the overall model definition that this
 109  
      * field will be used for
 110  
      * @param messageKey - key object used for determing field labels
 111  
      * @param metadata - metadata used to determine requiredness, validation, and autogenerated widget
 112  
      */
 113  0
     public FieldDescriptor(String fieldKey, MessageKeyInfo messageKey, Metadata metadata) {
 114  0
             this.fieldKey = fieldKey;
 115  0
             this.metadata = metadata;
 116  0
             if(messageKey == null){
 117  0
                     messageKey = new MessageKeyInfo("");
 118  
             }
 119  0
             setMessageKey(messageKey);
 120  0
             fieldElement = new FieldElement(fieldKey, messageKey, createFieldWidget(), metadata);
 121  0
             setupField();
 122  
             
 123  
             //Add mapping from path to field definition
 124  0
             if((getFieldWidget() instanceof HasDataValue || getFieldWidget() instanceof KSTextBox || getFieldWidget() instanceof HasValue)&&!(this instanceof FieldDescriptorReadOnly)){
 125  0
                     Application.getApplicationContext().putPathToFieldMapping(null, Application.getApplicationContext().getParentPath()+fieldKey, this);
 126  
                 }
 127  
 
 128  
             //Add cross constraints
 129  0
             if(fieldElement.getFieldWidget() instanceof HasCrossConstraints){
 130  0
                     HasCrossConstraints crossConstraintWidget = (HasCrossConstraints) fieldElement.getFieldWidget();
 131  0
                     if(crossConstraintWidget!=null&&crossConstraintWidget.getCrossConstraints()!=null){
 132  0
                             for(String path:crossConstraintWidget.getCrossConstraints()){
 133  0
                                 Application.getApplicationContext().putCrossConstraint(null, path, crossConstraintWidget);
 134  
                             }
 135  
                     }
 136  
             }
 137  0
     }
 138  
 
 139  
     /**
 140  
      * @param fieldKey - key for this field which matches a field in the overall model definition that this
 141  
      * field will be used for
 142  
      * @param messageKey - key object used for determing field labels
 143  
      * @param metadata - metadata used to determine requiredness and validation
 144  
      * @param fieldWidget - widget to use instead of an automatically determined one
 145  
      */
 146  0
     public FieldDescriptor(String fieldKey, MessageKeyInfo messageKey, Metadata metadata, Widget fieldWidget){
 147  0
             this.fieldKey = fieldKey;
 148  0
             this.metadata = metadata;
 149  0
             if(messageKey == null){
 150  0
                     messageKey = new MessageKeyInfo("");
 151  
             }
 152  0
         setMessageKey(messageKey);
 153  0
             addStyleToWidget(fieldWidget);
 154  0
             fieldElement = new FieldElement(fieldKey, messageKey, fieldWidget, metadata);
 155  0
         setupField();
 156  
             
 157  
             //Add mapping from path to field definition if the definition has a data value
 158  0
             if((fieldWidget instanceof HasDataValue || fieldWidget instanceof KSTextBox) &&!(this instanceof FieldDescriptorReadOnly)){
 159  0
                     Application.getApplicationContext().putPathToFieldMapping(null, Application.getApplicationContext().getParentPath()+fieldKey, this);
 160  
                 }
 161  
             
 162  
             //Add cross constraints
 163  0
             if(fieldElement.getFieldWidget() instanceof HasCrossConstraints){
 164  0
                     HasCrossConstraints crossConstraintWidget = (HasCrossConstraints) fieldElement.getFieldWidget();
 165  0
                     if(crossConstraintWidget!=null&&crossConstraintWidget.getCrossConstraints()!=null){
 166  0
                             for(String path:crossConstraintWidget.getCrossConstraints()){
 167  0
                                 Application.getApplicationContext().putCrossConstraint(null, path, crossConstraintWidget);
 168  
                             }
 169  
                     }
 170  
             }
 171  
             
 172  0
     }
 173  
 
 174  
     protected void addStyleToWidget(Widget w){
 175  0
             if(fieldKey != null && !fieldKey.isEmpty() && w != null){
 176  0
                     String style = this.fieldKey.replaceAll("/", "-");
 177  0
                     w.addStyleName(style);
 178  
             }
 179  0
     }
 180  
 
 181  
     protected void setupField() {
 182  0
             if(metadata != null){
 183  0
                     if(MetadataInterrogator.isRequired(metadata)){
 184  0
                             fieldElement.setRequiredString("requiredMarker", "ks-form-module-elements-required");
 185  
                     }
 186  0
                     else if(MetadataInterrogator.isRequiredForNextState(metadata)){
 187  0
                             String nextState = MetadataInterrogator.getNextState(metadata);
 188  0
                             if(nextState != null){
 189  0
                                     if(nextState.equalsIgnoreCase("SUBMITTED")){
 190  0
                                             fieldElement.setRequiredString("requiredOnSubmit", "ks-form-required-for-submit");
 191  
                                     }
 192  0
                                     else if(nextState.equalsIgnoreCase("APPROVED")){
 193  0
                                             fieldElement.setRequiredString("reqApproval", "ks-form-required-for-submit");
 194  
                                     }
 195  0
                                         else if(nextState.equalsIgnoreCase("ACTIVE")){
 196  0
                                                 fieldElement.setRequiredString("reqActivate", "ks-form-required-for-submit");
 197  
                                     }
 198  0
                                         else if(nextState.equalsIgnoreCase("SUSPENDED") ||
 199  
                                                         nextState.equalsIgnoreCase("RETIRED")){
 200  0
                                                 fieldElement.setRequiredString("reqDeactivate", "ks-form-required-for-submit");
 201  
                                         }
 202  
                                         else {
 203  0
                                                 fieldElement.setRequiredString("requiredOnSubmit", "ks-form-required-for-submit");
 204  
                                         }
 205  
 
 206  
                             }
 207  0
                     } else{
 208  0
                 fieldElement.clearRequiredText();
 209  
             }
 210  
             }
 211  0
     }
 212  
 
 213  
     /**
 214  
      * @see FieldElement#hideLabel()
 215  
      */
 216  
     public void hideLabel(){
 217  0
             fieldElement.hideLabel();
 218  0
     }
 219  
 
 220  
     public boolean isLabelShown(){
 221  0
             return fieldElement.isLabelShown();
 222  
     }
 223  
 
 224  
     public FieldElement getFieldElement(){
 225  0
             return fieldElement;
 226  
     }
 227  
 
 228  
         public String getFieldKey() {
 229  0
         return fieldKey;
 230  
     }
 231  
 
 232  
     public void setFieldKey(String fieldKey) {
 233  0
                 this.fieldKey = fieldKey;
 234  0
         }
 235  
 
 236  
     public String getFieldLabel() {
 237  0
         return fieldElement.getFieldName();
 238  
     }
 239  
 
 240  
     public Widget getFieldWidget(){
 241  0
         if (fieldElement.getFieldWidget() == null){
 242  0
             Widget w = createFieldWidget();
 243  0
             fieldElement.setWidget(w);
 244  
         }
 245  0
         return fieldElement.getFieldWidget();
 246  
     }
 247  
 
 248  
     protected Widget createFieldWidget() {
 249  0
             if (metadata == null) {
 250  
                     // backwards compatibility for old ModelDTO code
 251  
                     // for now, default to textbox if not specified
 252  0
                     Widget result = new KSTextBox();
 253  0
                     addStyleToWidget(result);
 254  0
                     return result;
 255  
             } else {
 256  0
                     Widget result = DefaultWidgetFactory.getInstance().getWidget(this);
 257  0
                     addStyleToWidget(result);
 258  0
                     return result;
 259  
             }
 260  
     }
 261  
 
 262  
     public ModelWidgetBinding<?> getModelWidgetBinding() {
 263  0
         if(modelWidgetBinding == null){
 264  0
             if(fieldElement.getFieldWidget() instanceof RichTextEditor){
 265  0
                     modelWidgetBinding = org.kuali.student.common.ui.client.configurable.mvc.binding.RichTextBinding.INSTANCE;
 266  0
             } else if (fieldElement.getFieldWidget() instanceof KSCheckBox){
 267  0
                     modelWidgetBinding = org.kuali.student.common.ui.client.configurable.mvc.binding.HasValueBinding.INSTANCE;
 268  0
             } else if(fieldElement.getFieldWidget() instanceof MultiplicityComposite){
 269  0
                         modelWidgetBinding = MultiplicityCompositeBinding.INSTANCE;
 270  0
                 } else if (fieldElement.getFieldWidget()instanceof HasText) {
 271  0
                     modelWidgetBinding = org.kuali.student.common.ui.client.configurable.mvc.binding.HasTextBinding.INSTANCE;
 272  0
             } else if (fieldElement.getFieldWidget() instanceof KSSelectItemWidgetAbstract){
 273  0
                 modelWidgetBinding = org.kuali.student.common.ui.client.configurable.mvc.binding.SelectItemWidgetBinding.INSTANCE;
 274  0
             } else if (fieldElement.getFieldWidget() instanceof HasDataValue){
 275  0
                     modelWidgetBinding = org.kuali.student.common.ui.client.configurable.mvc.binding.HasDataValueBinding.INSTANCE;
 276  0
             } else if (fieldElement.getFieldWidget() instanceof HasValue){
 277  0
                     modelWidgetBinding = org.kuali.student.common.ui.client.configurable.mvc.binding.HasValueBinding.INSTANCE;
 278  
             }
 279  
         }
 280  0
         return modelWidgetBinding;
 281  
     }
 282  
 
 283  
     /**
 284  
      * Allows additional processing to happen when a validation check is being processed when the input
 285  
      * widget loses focus defined in the callback
 286  
      * @param callback
 287  
      */
 288  
     public void setValidationCallBack(Callback<Boolean> callback){
 289  0
         validationRequestCallback = callback;
 290  0
     }
 291  
 
 292  
     public Callback<Boolean> getValidationRequestCallback(){
 293  0
         return validationRequestCallback;
 294  
     }
 295  
 
 296  
         /**
 297  
          * Returns true if this field is marked as dirty.  In KS, dirty is when the field has been changed
 298  
          * in some way - however not completely accurate
 299  
          * @return
 300  
          */
 301  
         public boolean isDirty() {
 302  0
                 return dirty;
 303  
         }
 304  
 
 305  
         public void setDirty(boolean dirty) {
 306  0
                 this.dirty = dirty;
 307  0
         }
 308  
 
 309  
         /**
 310  
          * Return true if the field has been touched by the user in some fashion, this is set by the field's section
 311  
          * @return
 312  
          */
 313  
         public boolean hasHadFocus() {
 314  0
                 return hasHadFocus;
 315  
         }
 316  
 
 317  
         public void setHasHadFocus(boolean hasHadFocus) {
 318  0
                 this.hasHadFocus = hasHadFocus;
 319  0
         }
 320  
 
 321  
     public Metadata getMetadata() {
 322  0
                 return metadata;
 323  
         }
 324  
 
 325  
     public void setMetadata(Metadata metadata) {
 326  0
                 this.metadata = metadata;
 327  0
         setupField();
 328  0
         }
 329  
 
 330  
     public void setFieldWidget(Widget fieldWidget) {
 331  0
                 this.fieldElement.setWidget(fieldWidget);
 332  0
         }
 333  
 
 334  
         public String getModelId() {
 335  0
                 return modelId;
 336  
         }
 337  
 
 338  
         public void setModelId(String modelId) {
 339  0
                 this.modelId = modelId;
 340  0
         }
 341  
 
 342  
     /**
 343  
      * Sets the ModelWidgetBinding for this field.  Changing this changes the way data from the server and
 344  
      * passed to the server is processed with the widget.  Set this when some special processing or handling
 345  
      * has to happen with the data in either phase.
 346  
      * @param widgetBinding
 347  
      */
 348  
     public void setWidgetBinding(ModelWidgetBinding widgetBinding) {
 349  0
         this.modelWidgetBinding = widgetBinding;
 350  0
     }
 351  
 
 352  
     public MessageKeyInfo getMessageKey() {
 353  0
         return messageKey;
 354  
     }
 355  
 
 356  
     public void setMessageKey(MessageKeyInfo messageKey) {
 357  0
         this.messageKey = messageKey;
 358  0
     }
 359  
 
 360  
         /**
 361  
          * Sets the optional flag
 362  
          * Fields that are optional should not be displayed if there is no data in some cases,
 363  
          * it is up to the section implementation whether or not to honor this flag
 364  
          * @param optional
 365  
          */
 366  
         public void setOptional(boolean optional){
 367  0
                 this.optional = optional;
 368  0
         }
 369  
 
 370  
         /**
 371  
          * Fields that are optional should not be displayed if there is no data in some cases,
 372  
          * it is up to the section implementation whether or not to honor this flag
 373  
          */
 374  
         public boolean isOptional(){
 375  0
                 return optional;
 376  
         }
 377  
 
 378  
         /**
 379  
          * @return true if this field is visible to the user
 380  
          */
 381  
         public boolean isVisible() {
 382  0
                 if (metadata != null){
 383  0
                         return metadata.isCanView();
 384  
                 } else {
 385  0
                         return true;
 386  
                 }
 387  
         }
 388  
 
 389  
         /**
 390  
          * Reset the requiredness of the field descriptor. Note doing this will also dynamically change
 391  
          * the underlying metadata so ui validation for requiredness works as well.
 392  
          * 
 393  
          */
 394  
         public void setRequired(Boolean isRequired){ 
 395  0
                 fieldElement.setRequiredString("requiredMarker", "ks-form-module-elements-required");
 396  0
                 fieldElement.setRequired(isRequired);
 397  
                 
 398  
                 //FIXME: This could be problematic if minOccurs should be something other than 1
 399  0
                 if (isRequired){
 400  0
                         getMetadata().getConstraints().get(0).setMinOccurs(1);
 401  
                 } else {
 402  0
                         getMetadata().getConstraints().get(0).setMinOccurs(0);
 403  
                 }
 404  0
         }
 405  
 
 406  
     public boolean isIgnoreShowRequired() {
 407  0
         return ignoreShowRequired;
 408  
     }
 409  
 
 410  
     public void setIgnoreShowRequired(boolean ignoreShowRequired) {
 411  0
         this.ignoreShowRequired = ignoreShowRequired;
 412  0
     }
 413  
         
 414  
 }