Creating a component extension

In this tutorial we create a simple extension that can be attached to a PasswordField, displaying a floating notification if the user’s Caps Lock seems to be enabled. We assume the reader is already familiar with the Creating a UI extension tutorial.

This extension has almost no server-side functionality; the whole Extension class is as follows:

Java

  1. public class CapsLockWarning extends AbstractExtension {
  2. protected CapsLockWarning(PasswordField field) {
  3. // Non-public constructor to discourage direct instantiation
  4. extend(field);
  5. }
  6. public static CapsLockWarning warnFor(PasswordField field) {
  7. return new CapsLockWarning(field);
  8. }
  9. }

When there’s nothing to configure for the extension, users just want to enable it for some component and be done with it. By defining a static factory method, the user only needs to do something like CapsLockWarning.warnFor(myPasswordField); to make myPasswordField get the new functionality.

The client side is not overly complicated, either. We override the extend method, called by the framework when the client-side extension connector is attached to its target the client-side counterpart of the connector to which the server-side extension instance is attached in this case, PasswordFieldConnector.

We add a key press handler to the password widget, checking if the input looks like Caps Lock might be enabled. The Caps Lock state cannot be directly queried in GWT/JavaScript, so we use a trick: check if either

  • the shift key was not held but the entered character was uppercase, or

  • the shift key was held but the entered character was lowercase.

If this is the case, we show a warning in the form of a floating widget (VOverlay). This demonstrates how an extension may make use of UI elements even though it is not a part of the layout hierarchy. A frequent use case for extensions is showing different types of floating overlay elements that are temporary in character.

Java

  1. @Connect(CapsLockWarning.class)
  2. public class CapsLockWarningConnector extends AbstractExtensionConnector {
  3. @Override
  4. protected void extend(ServerConnector target) {
  5. final Widget passwordWidget = ((ComponentConnector) target).getWidget();
  6. final VOverlay warning = new VOverlay();
  7. warning.setOwner(passwordWidget);
  8. warning.add(new HTML("Caps Lock is enabled!"));
  9. passwordWidget.addDomHandler(new KeyPressHandler() {
  10. @Override
  11. public void onKeyPress(KeyPressEvent event) {
  12. if (isEnabled() && isCapsLockOn(event)) {
  13. warning.showRelativeTo(passwordWidget);
  14. } else {
  15. warning.hide();
  16. }
  17. }
  18. }, KeyPressEvent.getType());
  19. }
  20. private boolean isCapsLockOn(KeyPressEvent e) {
  21. return e.isShiftKeyDown() ^ Character.isUpperCase(e.getCharCode());
  22. }
  23. }

To use the Caps Lock warning, compile your widgetset and extend a PasswordField with something like this

Java

  1. PasswordField field = new PasswordField("Enter your password");
  2. CapsLockWarning.warnFor(field);
  3. addComponent(field);