Tuesday, September 14, 2010

Apache Click 2.2.0: Dynamic Form Validation

In this three part series I'll blog about some of the new features added in 2.2.0, namely:

  1. DataProviders (Japanese version)
  2. Explicit binding (Japanese version)
  3. Dynamic Form Validation ( Japanese version)


    In this third and final installment I'll cover conditional form validation. This feature simplifies dynamic Form behavior by allowing developers to control when form validation should be applied. Generally form validation should be applied when the user submits the completed form. However when working with dynamic forms, the form is often submitted (using JavaScript) to the server in an incomplete state in order to add dynamic Fields and components. I'll refer to this as "incomplete submissions" because the form was not completely filled out by the user when the submission occurs.

    Ideally form validation should not be applied to incomplete submissions as the user will be presented any validation errors that occurred. Seeing the errors will be unexpected for the user since they are still busy filling out the form and did not intend to perform a submission.

    Let's look at a simple example. Below we have a page with a required name field and a checkbox that indicates whether the user is married or not. If the married checkbox is checked, a field is displayed for capturing the spouse's name. The spouse's name field is also required.

    public class ValidationDemo extends BorderPage {
    
        Form form = new Form("form");
    
        // The submit button is used to submit the completed form. Note, do not
        // name the Submit button "submit". That will cause a JavaScript error when
    // the button is clicked.
        Submit ok = new Submit("ok");
    
        @Override
        public void onInit() {
            super.onInit();
            form.add(new TextField("fullName", true));
            addControl(form);
            Checkbox married = new Checkbox("married", "Married?");
    
            // We use JavaScrcipt to submit the "incomplete" form when the user
            // clicks the checkbox.
            married.setAttribute("onclick", "form.submit();");
            form.add(married);
    
            form.add(ok);
    
            // Explicitly bind the checkbox
            ClickUtils.bind(married);
            if (married.isChecked()) {
                form.add(new TextField("spouseName", true));
            }
        }
    }

    Note the highlighted line above. We are using JavaScript to submit the form when the married checkbox is checked.

    Here is a screenshot of the page:

    When the checkbox is checked, the form is submitted and the spouse field is added. Next the form is validated and the required name and spouse error messages is displayed, as seen in the screenshot below:


    We don't want to show the validation messages yet. Particularly problematic is the error message for the Spouse Name field since the user has not seen this field until now.

    One solution is to clear the error messages if the form was submitted but the "ok" Submit button was not clicked, meaning the form was submitted through some other means ie. JavaScript. We can add this logic to the onPost page event:

    ...
    
    @Override
    public void onPost() {
        // Clear form errors if the "ok" submit button was not clicked
        if (!ok.isClicked()) {
            form.clearErrors();
        }
    } 

    This produces the screenshot below:

    This looks better. The Spouse Name has been added and no error messages are shown.

    However this solution is not ideal because the validation step is still performed and we have to manually clear the errors. So is there a better way?

    Indeed there is. By leveraging Explicit binding, one can conditionally switch off form and field validation as follows:

    if(!submit.isClicked()) { 
        form.setValidate(false); 
    }

    Let's revisit our earlier example and update it to use conditional validation:

    public class ValidationDemo extends BorderPage {
    
        Form form = new Form("form");
    
        // The submit button is used to submit the completed form. Note, do not
        // name the Submit button "submit". That will cause a JavaScript error when
    // the button is clicked.
        Submit ok = new Submit("ok");
    
        @Override
        public void onInit() {
            super.onInit();
            form.add(new TextField("fullName", true));
            addControl(form);
            Checkbox married = new Checkbox("married", "Married?");
    
            // We use JavaScript to submit the "incomplete" form when the user
            // clicks the checkbox.
            married.setAttribute("onclick", "form.submit();");
            form.add(married);
    
            form.add(ok);
    
            // Bind the submit button. If it wasn't clicked it means the Form was
            // submitted using JavaScript and we don't want to validate yet
            ClickUtils.bind(submit);
    
            // If the Form was submitted but the submit was not clicked, don't validate
            if(form.isFormSubmission() && !submit.isClicked()) {
                form.setValidate(false);
            }
    
            if (married.isChecked()) {
                form.add(new TextField("spouseName", true));
            }
        }
    }

    The screenshot below shows the form after checking the checkbox:



    As you can see, no errors are displayed since the form was not validated.

    Conditional form validation together with explicit binding provides a good combination for adding dynamic behavior to your Click pages and forms.

    This concludes the three part series covering the new features of Click 2.2.0.

    Apache Click 2.2.0: Explicit Binding

    In this three part series I'll blog about some of the new features added in 2.2.0, namely:

    1. DataProviders (Japanese version)
    2. Explicit binding (Japanese version)
    3. Dynamic Form Validation ( Japanese version)

      In this second installment I'll cover explicit binding which allows developers to bind, or set, a control's value to incoming request parameters. Not all controls support binding though. The most common bindable controls include Field, Form and Abstractlink. Explicit binding simplifies dynamic Page and Form behavior as it allows you to bind and query control values whenever you need to, for example in the onInit event or the Page constructor.

      When talking about explicit binding it is also worth mentioning implicit binding.
      Implicit binding occurs automatically every request as part of the onProcess event. So Control values will automatically be set once the onProcess event has occurred.Click Controls uses a convention of handling bind logic in a method called bindRequestValue().

      A typical bindRequestValue() implementation looks like this:

      public void bindRequestValue() {
          Context context = getContext();
          String controlName = getName();
          String value = context.getRequestParameter(controlName);
          if(value != null) {
              setValue(value);
          }
      }
      
      // For completeness sake we show the onProcess implementation as well.
      // onProcess delegates the binding logic to bindRequestValue
      public void onProcess() {
          bindRequestValue();
          ...
      }
      
      

      Generally the control name is used to lookup the incoming request parameter, but
      other conventions can be used as well.

      Under most circumstances, implicit binding is all you need. But sometimes you might want to know the value of a control before the onProcess event occurs. For example, within the onInit()
      event, you might want to check which drop-down value a user selected in order to
      add another Field to the Form. Remember, only Controls that are created and
      attached to their parent page/container partakes in the onProcess event. That's why it's important to create and attach Controls to their parents prior to the onProcess event.

      As you probably know already, the onInit event occurs before onProcess, and because implicit binding only occurs during onProcess, the Control value is not available for querying from onInit.

      So how can you find out the value of a Control during the onInit event? This is
      where explicit binding comes into play. Explicit binding allows developers to
      dictate when the Control value is bound. So how can one explicitly bind the Control value?
      By invoking the Control method, bindRequestValue(). (Recall that not all Controls
      support binding and might not provide a bindRequestValue() method.)

      There are a couple of caveats to be aware of when invoking bindRequestValue
      directly.

      • Fields should only be bound if their parent Form has been submitted, otherwise if you have multiple forms on the page, you might end up in a situation where form1 fields could be bound to form2 fields.
      • Forwarded requests have already been processed and should not be used for binding purposes.

      To make things easier, Click provides a couple of helper methods that takes care
      of the caveats mentioned above. These helper methods are available from the
      ClickUtils class which sports a variety of bind() and bindAndValidate() methods.
      You can even pass in Containers such as Form to these methods and all child
      controls will be bound.

      The bindAndValidate() method will both bind and validate a Field or Form.



      It's worth mentioning that it's also possible to lookup request parameters directly through the Context object instead of having to bind the control value.

      Let's look at some examples next.

      In this first example we will look at a dynamic Form where a Checkbox determines
      whether a TextField should be added to the Form or not.


      public class DynamicFormDemo extends Page {
      
          @Override
          public void onInit() {
              super.onInit();
              Form form = new Form("form");
              addControl(form);
              Checkbox chk = new Checkbox("chk");
              form.add(chk);
      
              Submit ok = new Submit("ok");
              form.add(ok);
      
              // Explicitly bind the checkbox in the onInit event which allows us to query
              // whether the Checkbox was checked or not.
              ClickUtils.bind(chk);
              if(chk.isChecked()) {
                  form.add(new TextField("name"));
              }
          }
      }
      

      We use the ClickUtils.bind() method to explicitly bind the Checkbox value so we
      can query whether it is checked or not.

      In this second example we expand the first by adding a Select field as well.



      public class DynamicFormDemo extends Page {
      
          @Override
          public void onInit() {
              super.onInit();
              Form form = new Form("form");
              addControl(form);
              Checkbox chk = new Checkbox("chk");
              form.add(chk);
              Select countries = new CountrySelect("countries");
              countries.getOptionList().add(Option.EMPTY_OPTION);
              form.add(countries);
      
              Submit ok = new Submit("ok");
              form.add(ok);
      
              // Explicitly bind the Form (and all it's child controls) in the onInit
              // event, allowing us to query whether he user checked the Checkbox and
              // which country was selected.
              ClickUtils.bind(form);
              if (chk.isChecked()) {
                  form.add(new TextField("name"));
              }
      
              if (StringUtils.isNotBlank(countries.getValue())) {
                  form.add(new TextField("location"));
              }
          }
      }
      

      Note, instead of binding the Checkbox and Select separately we pass the Form to
      the ClickUtils.bind() method. When passing a container such as a Form to the
      ClickUtils.bind() methods, all bindable controls in the container will have their
      values bound. This provides an easy shortcut to quickly bind multiple controls.

      In this third and final example we show how to both bind and validate a form.

      public class DynamicFormDemo extends BorderPage {
      
          @Override
          public void onInit() {
              super.onInit();
              Form form = new Form("form");
              addControl(form);
              Checkbox chk = new Checkbox("chk");
              form.add(chk);
              Select countries = new CountrySelect("countries", true);
              countries.getOptionList().add(Option.EMPTY_OPTION);
              form.add(countries);
      
              Submit ok = new Submit("ok");
              form.add(ok);
      
              // Explicitly bind and check that the Form (and all it's child
              // controls) is valid in the onInit event, allowing us to safely
              // query whether he user checked the Checkbox and
              // which country was selected.
              if (ClickUtils.bindAndValidate(form)) {
                  if (chk.isChecked()) {
                      form.add(new TextField("name"));
                  }
      
                  // The form validation passed and since the countries field is required
                  // we can safely assume that a valid country has been selected
                  form.add(new TextField("location"));
              }
          }
      }
      

      The ClickUtils.bindAndValidate() methods will bind and validate the field/s and return true if the field/s are valid, false otherwise.

      Friday, September 10, 2010

      ClickIDE 2.2.0.0 has been released!

      ClickIDE 2.2.0.0 is now available! ClickIDE is an Eclipse plug-in for the developing Click web applications.

      This version supports Apache Click 2.2.0 and fixes some minor issues.

      For details about ClickIDE see:
      http://click.apache.org/docs/click-ide.html

      Download is available here:
      http://click.apache.org/docs/click-ide-downloads.html
       

      Enjoy Click