GWT - Guía rápida

¿Qué es GWT?

Google Web Toolkit (GWT) es un juego de herramientas de desarrollo para crear RICH Internet Applications (RIA). Estas son algunas de sus características notables:

  • GWT ofrece a los desarrolladores la opción de escribir aplicaciones del lado del cliente en JAVA.

  • GWT compila el código escrito en JAVA en código JavaScript.

  • La aplicación escrita en GWT es compatible con varios navegadores. GWT genera automáticamente un código javascript adecuado para cada navegador.

  • GWT es de código abierto, completamente gratuito y utilizado por miles de desarrolladores en todo el mundo. Tiene licencia de Apache License versión 2.0.

En general, GWT es un framework para crear aplicaciones web a gran escala y de alto rendimiento mientras las mantiene tan fáciles de mantener.

¿Por qué utilizar GWT?

Al estar basado en Java, puede utilizar IDE de JAVA como Eclipse para desarrollar una aplicación GWT.

Los desarrolladores pueden utilizar código de autocompletar / refactorización / navegación / gestión de proyectos y todas las funciones de IDEs.GWT, que proporciona una capacidad de depuración completa. Los desarrolladores pueden depurar la aplicación del lado del cliente como una aplicación Java.

  • GWT proporciona una fácil integración con Junit y Maven.

  • Al estar nuevamente basado en Java, GWT tiene una curva de aprendizaje baja para los desarrolladores de Java.

  • GWT genera código javascript optimizado, produce el código javascript específico del navegador por sí mismo.

  • GWT proporciona la biblioteca de widgets que proporciona la mayoría de las tareas necesarias en una aplicación.

  • GWT es extensible y se puede crear un widget personalizado para satisfacer las necesidades de la aplicación.

Además de todo, las aplicaciones GWT pueden ejecutarse en todos los principales navegadores y teléfonos inteligentes, incluidos teléfonos / tabletas basados ​​en Android e iOS.

Desventajas de GWT

Aunque GWT ofrece muchas ventajas, tiene las siguientes desventajas:

  • Not Indexable - Las páginas web generadas por GWT no serían indexadas por los motores de búsqueda porque estas aplicaciones se generan de forma dinámica.

  • Not Degradable - Si el usuario de su aplicación deshabilita Javascript, el usuario solo verá la página básica y nada más.

  • Not Designer's Friendly - GWT no es adecuado para diseñadores web que prefieren usar HTML sin formato con marcadores de posición para insertar contenido dinámico en un momento posterior.

Los componentes de GWT

El marco de GWT se puede dividir en las siguientes tres partes principales:

  • GWT Java to JavaScript compiler- Esta es la parte más importante de GWT, lo que la convierte en una herramienta poderosa para construir RIA. El compilador GWT se utiliza para traducir todo el código de la aplicación escrito en Java a JavaScript.

  • JRE Emulation library- Google Web Toolkit incluye una biblioteca que emula un subconjunto de la biblioteca de tiempo de ejecución de Java. La lista incluye java.lang, java.lang.annotation, java.math, java.io, java.sql, java.util y java.util.logging

  • GWT UI building library - Esta parte de GWT consta de muchas subpartes que incluyen los componentes reales de la interfaz de usuario, la compatibilidad con RPC, la gestión del historial y mucho más.

GWT también proporciona un navegador web alojado GWT que le permite ejecutar y ejecutar sus aplicaciones GWT en modo alojado, donde su código se ejecuta como Java en la máquina virtual Java sin compilar en JavaScript.

Este tutorial lo guiará sobre cómo preparar un entorno de desarrollo para comenzar a trabajar con GWT Framework. Este tutorial también le enseñará cómo configurar JDK, Tomcat y Eclipse en su máquina antes de configurar GWT Framework -

Requisitos del sistema

GWT requiere JDK 1.6 o superior, por lo que el primer requisito es tener JDK instalado en su máquina.

JDK 1.6 o superior.
Memoria ningún requisito mínimo.
Espacio del disco ningún requisito mínimo.
Sistema operativo ningún requisito mínimo.

Siga los pasos dados para configurar su entorno para comenzar con el desarrollo de aplicaciones GWT.

Paso 1: verificar la instalación de Java en su máquina

Ahora abra la consola y ejecute el siguiente comando de Java.

SO Tarea Mando
Windows Abrir consola de comandos c: \> java -version
Linux Terminal de comando abierto $ java -version
Mac Terminal abierta máquina: ~ joseph $ java -version

Verifiquemos la salida para todos los sistemas operativos.

No Señor. SO y salida generada
1

Windows

versión de Java "1.6.0_21"

Entorno de ejecución Java (TM) SE (compilación 1.6.0_21-b07)

VM de cliente Java HotSpot (TM) (compilación 17.0-b17, modo mixto, uso compartido)

2

Linux

versión de Java "1.6.0_21"

Entorno de ejecución Java (TM) SE (compilación 1.6.0_21-b07)

Ava HotSpot (TM) Client VM (compilación 17.0-b17, modo mixto, uso compartido)

3

Mac

versión de Java "1.6.0_21"

Entorno de ejecución Java (TM) SE (compilación 1.6.0_21-b07)

Servidor VM Java HotSpot (TM) de 64 bits (compilación 17.0-b17, modo mixto, uso compartido)

Paso 2: configuración del kit de desarrollo de Java (JDK)

Si no tiene Java instalado, puede instalar Java Software Development Kit (SDK) desde el sitio Java de Oracle: Descargas de Java SE . Encontrará instrucciones para instalar JDK en archivos descargados, siga las instrucciones dadas para instalar y configurar la instalación. Finalmente, configure las variables de entorno PATH y JAVA_HOME para hacer referencia al directorio que contiene java y javac, normalmente java_install_dir / bin y java_install_dir respectivamente.

Selecciona el JAVA_HOMEvariable de entorno para apuntar a la ubicación del directorio base donde está instalado Java en su máquina. Por ejemplo

No Señor. SO y salida
1

Windows

Establezca la variable de entorno JAVA_HOME en C: \ Archivos de programa \ Java \ jdk1.6.0_21

2

Linux

exportar JAVA_HOME = / usr / local / java-current

3

Mac

exportar JAVA_HOME = / Library / Java / Home

Agregue la ubicación del compilador de Java a la ruta del sistema.

No Señor. SO y salida
1

Windows

Agregue la cadena;% JAVA_HOME% \ bin al final de la variable del sistema, Ruta.

2

Linux

export PATH =$PATH:$JAVA_HOME / bin /

3

Mac

no requerido

Alternativamente, si usa un entorno de desarrollo integrado (IDE) como Borland JBuilder, Eclipse, IntelliJ IDEA o Sun ONE Studio, compile y ejecute un programa simple para confirmar que el IDE sabe dónde instaló Java; de lo contrario, realice la configuración adecuada como se indica en el documento. del IDE.

Paso 3: configurar el IDE de Eclipse

Todos los ejemplos de este tutorial se han escrito utilizando Eclipse IDE. Por lo tanto, le sugiero que tenga instalada la última versión de Eclipse en su máquina según su sistema operativo.

Para instalar Eclipse IDE, descargue los últimos binarios de Eclipse desde https://www.eclipse.org. Una vez que descargó la instalación, descomprima la distribución binaria en una ubicación conveniente. Por ejemplo, en C: \ eclipse en Windows, o / usr / local / eclipse en Linux / Unix y finalmente configure la variable PATH de manera apropiada.

Eclipse se puede iniciar ejecutando los siguientes comandos en la máquina con Windows, o simplemente puede hacer doble clic en eclipse.exe

%C:\eclipse\eclipse.exe

Eclipse se puede iniciar ejecutando los siguientes comandos en una máquina Unix (Solaris, Linux, etc.):

$/usr/local/eclipse/eclipse

Después de un inicio exitoso, si todo está bien, debería mostrar el siguiente resultado:

Paso 4: instale el SDK y el complemento de GWT para Eclipse

Siga las instrucciones dadas en el enlace Complemento para Eclipse (incluidos los SDK) para instalar el SDK de GWT y el Complemento para la versión de Eclipse instalada en su máquina.

Después de una configuración exitosa para el complemento GWT, si todo está bien, debería mostrar la siguiente pantalla con Google icon marcado con un rectángulo rojo como se muestra a continuación -

Paso 5: configurar Apache Tomcat

Puede descargar la última versión de Tomcat desde https://tomcat.apache.org/. Una vez que descargó la instalación, descomprima la distribución binaria en una ubicación conveniente. Por ejemplo, en C: \ apache-tomcat-6.0.33 en Windows, o /usr/local/apache-tomcat-6.0.33 en Linux / Unix y establezca la variable de entorno CATALINA_HOME apuntando a las ubicaciones de instalación.

Tomcat se puede iniciar ejecutando los siguientes comandos en la máquina con Windows, o simplemente puede hacer doble clic en startup.bat

%CATALINA_HOME%\bin\startup.bat
or 
/usr/local/apache-tomcat-6.0.33/bin/startup.sh

Después de un inicio exitoso, las aplicaciones web predeterminadas incluidas con Tomcat estarán disponibles visitando http://localhost:8080/. Si todo está bien, debería mostrar el siguiente resultado:

Puede encontrar más información sobre la configuración y ejecución de Tomcat en la documentación incluida aquí, así como en el sitio web de Tomcat: https://tomcat.apache.org/

Tomcat se puede detener ejecutando los siguientes comandos en la máquina con Windows:

%CATALINA_HOME%\bin\shutdown
or
C:\apache-tomcat-5.5.29\bin\shutdown

Tomcat se puede detener ejecutando los siguientes comandos en una máquina Unix (Solaris, Linux, etc.):

$CATALINA_HOME/bin/shutdown.sh
or
/usr/local/apache-tomcat-5.5.29/bin/shutdown.sh

Antes de comenzar con la creación de una aplicación "HelloWorld" real usando GWT, veamos cuáles son las partes reales de una aplicación GWT:

Una solicitud de GWT consta de las siguientes cuatro partes importantes, de las cuales la última parte es opcional, pero las tres primeras son obligatorias.

  • Descriptores de módulo
  • Recursos públicos
  • Código del lado del cliente
  • Código del lado del servidor

Muestras de ubicaciones de diferentes partes de una aplicación típica de gwt HelloWord será como se muestra a continuación -

Nombre Ubicación
Raíz del proyecto Hola Mundo/
Descriptor de módulo src / com / tutorialspoint / HelloWorld.gwt.xml
Recursos públicos src / com / tutorialspoint / war /
Código del lado del cliente src / com / tutorialspoint / client /
Código del lado del servidor src / com / tutorialspoint / server /

Descriptores de módulo

Un descriptor de módulo es el archivo de configuración en forma de XML que se utiliza para configurar una aplicación GWT.

Una extensión de archivo descriptor de módulo es * .gwt.xml, donde * es el nombre de la aplicación y este archivo debe residir en la raíz del proyecto.

A continuación se muestra un descriptor de módulo predeterminado HelloWorld.gwt.xml para una aplicación HelloWorld:

<?xml version = "1.0" encoding = "utf-8"?>
<module rename-to = 'helloworld'>
   <!-- inherit the core web toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.user'/>

   <!-- inherit the default gwt style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>

   <!-- specify the paths for translatable code                    -->
   <source path = '...'/>
   <source path = '...'/>

   <!-- specify the paths for static files like html, css etc.     -->
   <public path = '...'/>
   <public path = '...'/>

   <!-- specify the paths for external javascript files            -->
   <script src = "js-url" />
   <script src = "js-url" />

   <!-- specify the paths for external style sheet files            -->
   <stylesheet  src = "css-url" />
   <stylesheet  src = "css-url" />
</module>

A continuación se muestra un breve detalle sobre las diferentes partes utilizadas en el descriptor del módulo.

No Señor. Nodos y descripción
1

<module rename-to = "helloworld">

Esto proporciona el nombre de la aplicación.

2

<inherits name = "logical-module-name" />

Esto agrega otro módulo gwt en la aplicación al igual que lo hace la importación en las aplicaciones Java. De esta manera se puede heredar cualquier número de módulos.

3

<entry-point class = "classname" />

Esto especifica el nombre de la clase que comenzará a cargar la aplicación GWT. Se puede agregar cualquier número de clases de punto de entrada y se llaman secuencialmente en el orden en que aparecen en el archivo del módulo. Entonces, cuando el onModuleLoad () de su primer punto de entrada termina, se llama inmediatamente al siguiente punto de entrada.

4

<source path = "path" />

Esto especifica los nombres de las carpetas de origen que el compilador de GWT buscará para la compilación de origen.

5

<public path = "path" />

La ruta pública es el lugar de su proyecto donde se almacenan los recursos estáticos a los que hace referencia su módulo GWT, como CSS o imágenes. La ruta pública predeterminada es el subdirectorio público debajo de donde se almacena el archivo XML del módulo.

6

<script src="js-url" />

Inyecta automáticamente el archivo JavaScript externo ubicado en la ubicación especificada por src.

7

<stylesheet src="css-url" />

Inyecta automáticamente el archivo CSS externo ubicado en la ubicación especificada por src.

Recursos públicos

Todos estos son archivos a los que hace referencia su módulo GWT, como la página HTML del host, CSS o imágenes.

La ubicación de estos recursos se puede configurar usando el elemento <public path = "path" /> en el archivo de configuración del módulo. De forma predeterminada, es el subdirectorio público debajo de donde se almacena el archivo XML del módulo.

Cuando compila su aplicación en JavaScript, todos los archivos que se pueden encontrar en su ruta pública se copian en el directorio de salida del módulo.

El recurso público más importante es la página de host, que se utiliza para invocar la aplicación GWT real. Es posible que una página de host HTML típica para una aplicación no incluya ningún contenido de cuerpo HTML visible, pero siempre se espera que incluya la aplicación GWT a través de una etiqueta <script ... /> de la siguiente manera

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>
   
   <body>
      <h1>Hello World</h1>
      <p>Welcome to first GWT application</p>
   </body>
</html>

A continuación se muestra la hoja de estilo de muestra que hemos incluido en nuestra página de host:

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

Código del lado del cliente

Este es el código Java real escrito implementando la lógica comercial de la aplicación y que el compilador GWT traduce a JavaScript, que eventualmente se ejecutará dentro del navegador. La ubicación de estos recursos se puede configurar usando el elemento <source path = "path" /> en el archivo de configuración del módulo.

Por ejemplo Entry Point el código se usará como código del lado del cliente y su ubicación se especificará usando <source path = "path" />.

Un modulo entry-point es cualquier clase que se pueda asignar a EntryPointy que se puede construir sin parámetros. Cuando se carga un módulo, se crea una instancia de cada clase de punto de entrada y suEntryPoint.onModuleLoad()se llama al método. Una clase de ejemplo de punto de entrada HelloWorld será la siguiente:

public class HelloWorld implements EntryPoint {
   public void onModuleLoad() {
      Window.alert("Hello, World!");
   }
}

Código del lado del servidor

Esta es la parte del lado del servidor de su aplicación y es muy opcional. Si no está realizando ningún procesamiento de backend dentro de su aplicación, entonces no necesita esta parte, pero si se requiere algún procesamiento en el backend y su aplicación del lado del cliente interactúa con el servidor, entonces tendrá que desarrollar estos componentes.

El próximo capítulo hará uso de todos los conceptos mencionados anteriormente para crear la aplicación HelloWorld usando Eclipse IDE.

Como el poder de GWT reside en Write in Java, Run in JavaScript, usaremos Java IDE Eclipse para demostrar nuestros ejemplos.

Comencemos con una sencilla aplicación HelloWorld :

Paso 1: crear proyecto

El primer paso es crear un proyecto de aplicación web simple utilizando Eclipse IDE. Iniciar el asistente de proyectos usando la opciónGoogle Icon

> New Web Application Project.... Ahora nombre su proyecto como HelloWorld usando la ventana del asistente de la siguiente manera:

Deseleccionar Use Google App Engine porque no lo estamos usando en este proyecto y dejamos otros valores predeterminados (mantenga Generate Sample project code opción marcada) como tal y haga clic en el botón Finalizar.

Una vez que su proyecto se haya creado con éxito, tendrá el siguiente contenido en su Explorador de proyectos:

Aquí hay una breve descripción de todas las carpetas importantes

No Señor. Carpeta y ubicación
1

src

Archivos de código fuente (clases de Java).

Carpeta del cliente que contiene las clases java específicas del lado del cliente responsables de la visualización de la interfaz de usuario del cliente.

Carpeta del servidor que contiene las clases de Java del lado del servidor responsables del procesamiento del lado del servidor.

Carpeta compartida que contiene la clase del modelo Java para transferir datos del servidor al cliente y viceversa.

HelloWorld.gwt.xml, un archivo descriptor de módulo necesario para que el compilador de GWT compile el proyecto HelloWorld.

2

test

Archivos fuente de código de prueba (clases java).

Carpeta del cliente que contiene las clases de Java responsables de probar el código del lado del cliente de gwt.

3

war

Esta es la parte más importante, representa la aplicación web implementable real.

WEB-INF que contiene clases compiladas, bibliotecas gwt, bibliotecas de servlets.

HelloWorld.css, hoja de estilo del proyecto.

HelloWorld.html, HTML activo que invocará la aplicación GWT UI.

Paso 2: modificar el descriptor del módulo: HelloWorld.gwt.xml

El complemento GWT creará un archivo descriptor de módulo predeterminado src / com.tutorialspoint / HelloWorld.gwt.xml que se proporciona a continuación. Para este ejemplo, no lo estamos modificando, pero puede modificarlo según sus necesidades.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.  You can change       -->
   <!-- the theme of your GWT application by uncommenting          -->
   <!-- any one of the following lines.                            -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>
   <!-- <inherits name = 'com.google.gwt.user.theme.chrome.Chrome'/> -->
   <!-- <inherits name = 'com.google.gwt.user.theme.dark.Dark'/>     -->

   <!-- Other module inherits                                      -->

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>

   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

Paso 3: modificar la hoja de estilo: HelloWorld.css

El complemento GWT creará un archivo de hoja de estilo predeterminado war / HelloWorld.css . Modifiquemos este archivo para mantener nuestro ejemplo en el nivel más simple de comprensión:

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

Paso 4: modificar el archivo de host: HelloWorld.html

El complemento GWT creará un archivo host HTML predeterminado war / HelloWorld.html . Modifiquemos este archivo para mantener nuestro ejemplo en el nivel más simple de comprensión:

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>Hello World</h1>
      <p>Welcome to first GWT application</p>
   </body>
</html>

Puede crear más archivos estáticos como HTML, CSS o imágenes en el mismo directorio de origen o puede crear más subdirectorios y mover archivos en esos subdirectorios y configurar esos subdirectorios en el descriptor de módulo de la aplicación.

Paso 5: modificar el punto de entrada: HelloWorld.java

El complemento GWT creará un archivo Java predeterminado src / com.tutorialspoint / HelloWorld.java , que mantiene un punto de entrada para la aplicación.

Modifiquemos este archivo para mostrar "¡Hola, mundo!"

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.user.client.Window;

public class HelloWorld implements EntryPoint {
   public void onModuleLoad() {
      Window.alert("Hello, World!");
   }
}

Puede crear más archivos Java en el mismo directorio de origen para definir puntos de entrada o para definir rutinas auxiliares.

Paso 6: compilar la aplicación

Una vez que esté listo con todos los cambios realizados, es hora de compilar el proyecto. Usa la opciónGoogle Icon

> GWT Compile Project... para iniciar el cuadro de diálogo Compilar GWT como se muestra a continuación:

Mantenga intactos los valores predeterminados y haga clic en el botón Compilar. Si todo va bien, verá el siguiente resultado en la consola de Eclipse

Compiling module com.tutorialspoint.HelloWorld
   Compiling 6 permutations
      Compiling permutation 0...
      Compiling permutation 1...
      Compiling permutation 2...
      Compiling permutation 3...
      Compiling permutation 4...
      Compiling permutation 5...
   Compile of permutations succeeded
Linking into C:\workspace\HelloWorld\war\helloworld
   Link succeeded
   Compilation succeeded -- 33.029s

Paso 7: ejecutar la aplicación

Ahora haga clic en el

menú Ejecutar aplicación y seleccioneHelloWorld aplicación para ejecutar la aplicación.

Si todo está bien, debe ver el Modo de desarrollo de GWT activo en Eclipse que contiene una URL como se muestra a continuación. Haga doble clic en la URL para abrir la aplicación GWT.

Debido a que está ejecutando su aplicación en modo de desarrollo, deberá instalar el complemento GWT para su navegador. Simplemente siga las instrucciones en pantalla para instalar el complemento.

Si ya tiene el complemento GWT configurado para su navegador, debería poder ver el siguiente resultado

¡Felicidades! ha implementado su primera aplicación utilizando Google Web Toolkit (GWT).

Este tutorial le explicará cómo crear una aplicación. "war" y cómo implementarlo en la raíz de Apache Tomcat Websever.

Si entendió este simple ejemplo, también podrá implementar una aplicación GWT compleja siguiendo los mismos pasos.

Tengamos Eclipse IDE funcionando junto con el complemento GWT en su lugar y sigamos los siguientes pasos para crear una aplicación GWT:

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modify HelloWorld.gwt.xml, HelloWorld.css, HelloWorld.html and HelloWorld.java as explained below. Keep rest of the files unchanged.
3 Compile and run the application to make sure business logic is working as per the requirements.
4 Finally, zip the content of the war folder of the application in the form of war file and deploy it in Apache Tomcat Webserver.
5 Launch your web application using appropriate URL as explained below in the last step.

Following is the content of the modified module descriptor src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>

   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

Following is the content of the modified Style Sheet file war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

Following is the content of the modified HTML host file war/HelloWorld.html.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>Hello World</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

I modified HTML a little bit from previous example. Here I created a placeholder <div>...</div> where we will insert some content using our entry point java class. So let us have following content of Java file src/com.tutorialspoint/HelloWorld.java.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.user.client.ui.HTML;
import com.google.gwt.user.client.ui.RootPanel;

public class HelloWorld implements EntryPoint {
   public void onModuleLoad() {
      HTML html = new HTML("<p>Welcome to GWT application</p>");
      
      RootPanel.get("gwtContainer").add(html);
   }
}

Here we created on basic widgest HTML and added it inside the div tag having id="gwtContainer". We will study different GWT widgets in coming chapters.

Once you are ready with all the changes done, let us compile and run the application in development mode as we did in GWT - Create Application chapter. If everything is fine with your application, this will produce following result −

Create WAR File

Now our applictaion is working fine and we are ready to export it as a war file.

Follow the following steps −

  • Go into your project's war directory C:\workspace\HelloWorld\war

  • Select all the files & folders available inside war directory.

  • Zip all the selected files & folders in a file called HelloWorld.zip.

  • Rename HelloWorld.zip to HelloWorld.war.

Deploy WAR file

  • Stop the tomcat server.

  • Copy the HelloWorld.war file to tomcat installation directory > webapps folder.

  • Start the tomcat server.

  • Look inside webapps directory, there should be a folder helloworld got created.

  • Now HelloWorld.war is successfully deployed in Tomcat Webserver root.

Run Application

Enter a url in web browser: http://localhost:8080/HelloWorld to launch the application

Server name (localhost) and port (8080) may vary as per your tomcat configuration.

GWT widgets rely on cascading style sheets (CSS) for visual styling. By default, the class name for each component is gwt-<classname>.

For example, the Button widget has a default style of gwt-Button and similar way TextBox widgest has a default style of gwt-TextBox.

In order to give all buttons and text boxes a larger font, you could put the following rule in your application's CSS file

.gwt-Button  { font-size: 150%; }

.gwt-TextBox { font-size: 150%; }

By default, neither the browser nor GWT creates default id attributes for widgets. You must explicitly create a unique id for the elements which you can use in CSS. In order to give a particular button with id my-button-id a larger font, you could put the following rule in your application's CSS file −

#my-button-id { font-size: 150%; }

To set the id for a GWT widget, retrieve its DOM Element and then set the id attribute as follows −

Button b = new Button();
DOM.setElementAttribute(b.getElement(), "id", "my-button-id")

CSS Styling APIs

There are many APIs available to hangle CSS setting for any GWT widget. Following are few important APIs which will help you in your day to day web programming using GWT −

Sr.No. API & Description
1

public void setStyleName(java.lang.String style)

This method will clear any existing styles and set the widget style to the new CSS class provided using style.

2

public void addStyleName(java.lang.String style)

This method will add a secondary or dependent style name to the widget. A secondary style name is an additional style name that is,so if there were any previous style names applied they are kept.

3

public void removeStyleName(java.lang.String style)

This method will remove given style from the widget and leaves any others associated with the widget.

4

public java.lang.String getStyleName()

This method gets all of the object's style names, as a space-separated list.

5

public void setStylePrimaryName(java.lang.String style)

This method sets the object's primary style name and updates all dependent style names.

For example, let's define two new styles which we will apply to a text −

.gwt-Big-Text { 
   font-size:150%;
}

.gwt-Small-Text { 
   font-size:75%;
}

.gwt-Red-Text { 
   color:red;
}

Now you can use setStyleName(Style) to change the default setting to new setting. After applying the below rule, a text's font will become large

txtWidget.setStyleName("gwt-Big-Text");

We can apply a secondary CSS rule on the same widget to change its color as follows −

txtWidget.addStyleName("gwt-Red-Text");

Using above method you can add as many styles as you like to apply on a widget. If you remove first style from the button widget then second style will still remain with the text.

txtWidget.removeStyleName("gwt-Big-Text");

Primary & Secondary Styles

By default, the primary style name of a widget will be the default style name for its widget class for example gwt-Button for Button widgets. When we add and remove style names using AddStyleName() method, those styles are called secondary styles.

The final appearance of a widget is determined by the sum of all the secondary styles added to it, plus its primary style. You set the primary style of a widget with the setStylePrimaryName(String) method. To illustrate, let's say we have a Label widget. In our CSS file, we have the following rules defined −

.MyText {
   color: blue;
}

.BigText {
   font-size: large;
}

.LoudText {
   font-weight:  bold;
}

Let's suppose we want a particular label widget to always display blue text, and in some cases, use a larger, bold font for added emphasis.

We could do something like this −

// set up our primary style
Label someText = new Label();
someText.setStylePrimaryName("MyText");
...

// later on, to really grab the user's attention
someText.addStyleName("BigText");
someText.addStyleName("LoudText");
...

// after the crisis is over
someText.removeStyleName("BigText");
someText.removeStyleName("LoudText");

Associating CSS Files

There are multiple approaches for associating CSS files with your module. Modern GWT applications typically use a combination of CssResource and UiBinder. We are using only first approach in our examples.

  • Using a <link> tag in the host HTML page.

  • Using the <stylesheet> element in the module XML file.

  • Using a CssResource contained within a ClientBundle.

  • Using an inline <ui:style> element in a UiBinder template.

GWT CSS Example

This example will take you through simple steps to apply different CSS rules on your GWT widgest. Let us have working Eclipse IDE along with GWT plug in place and follow the following steps to create a GWT application −

Step Description
1 Create a project with a name HelloWorld under a package com.tutorialspoint as explained in the GWT - Create Application chapter.
2 Modify HelloWorld.gwt.xml, HelloWorld.css, HelloWorld.html and HelloWorld.java as explained below. Keep rest of the files unchanged.
3 Compile and run the application to verify the result of the implemented logic.

Following is the content of the modified module descriptor src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>

   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

Following is the content of the modified Style Sheet file war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

.gwt-Button { 
   font-size: 150%; 
   font-weight: bold;
   width:100px;
   height:100px;
}

.gwt-Big-Text { 
   font-size:150%;
}

.gwt-Small-Text { 
   font-size:75%;
}

Following is the content of the modified HTML host file war/HelloWorld.html to accomodate two buttons.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <div id = "mytext"><h1>Hello, World!</h1></div>
      <div id = "gwtGreenButton"></div>
      <div id = "gwtRedButton"></div>
   </body>
</html>

Let us have following content of Java file src/com.tutorialspoint/HelloWorld.java which will take care of adding two buttons in HTML and will apply custom CSS style.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.user.client.ui.Button;
import com.google.gwt.user.client.ui.HTML;
import com.google.gwt.user.client.ui.RootPanel;

public class HelloWorld implements EntryPoint {
   public void onModuleLoad() {
     
   // add button to change font to big when clicked.
   Button Btn1 = new Button("Big Text");
   Btn1.addClickHandler(new ClickHandler() {
      public void onClick(ClickEvent event) {
         RootPanel.get("mytext").setStyleName("gwt-Big-Text");
      }
   });

   // add button to change font to small when clicked.
   Button Btn2 = new Button("Small Text");
   Btn2.addClickHandler(new ClickHandler() {
      public void onClick(ClickEvent event) {
         RootPanel.get("mytext").setStyleName("gwt-Small-Text");
      }
   });

   RootPanel.get("gwtGreenButton").add(Btn1);
   RootPanel.get("gwtRedButton").add(Btn2);
   }
}

Once you are ready with all the changes done, let us compile and run the application in development mode as we did in GWT - Create Application chapter. If everything is fine with your application, this will produce following result −

Now try clicking on the two buttons displayed and observe "Hello, World!" text which keeps changing its font upon clicking on the two buttons.

Every user interface considers the following three main aspects −

  • UI elements − Thes are the core visual elements the user eventually sees and interacts with. GWT provides a huge list of widely used and common elements varying from basic to complex which we will cover in this tutorial.

  • Layouts − They define how UI elements should be organized on the screen and provide a final look and feel to the GUI (Graphical User Interface). This part will be covered in Layout chapter.

  • Behavior − These are events which occur when the user interacts with UI elements. This part will be covered in Event Handling chapter.

GWT UI Elements

The GWT library provides classes in a well-defined class hierarchy to create complex web-based user interfaces. All classes in this component hierarchy has been derived from the UIObject base class as shown below −

Every Basic UI widget inherits properties from Widget class which in turn inherits properties from UIObject. Tree and Menu will be covered in complex widgets tutorial.

Sr.No. Widget & Description
1 GWT UIObject Class

This widget contains text, not interpreted as HTML using a <div>element, causing it to be displayed with block layout.

2 GWT Widget Class

This widget can contain HTML text and displays the html content using a <div> element, causing it to be displayed with block layout.

Basic Widgets

Following are few important Basic Widgets

Sr.No. Widget & Description
1 Label

This widget contains text, not interpreted as HTML using a <div>element, causing it to be displayed with block layout.

2 HTML

This widget can contain HTML text and displays the html content using a <div> element, causing it to be displayed with block layout.

3 Image

This widget displays an image at a given URL.

4 Anchor

This widget represents a simple <a> element.

Form widgets allows users to input data and provides them interaction capability with the application. Every Form widget inherits properties from Widget class which in turn inherits properties from UIObject and Wigdet classes.

Sr.No. Widget & Description
1 GWT UIObject Class

This widget contains text, not interpreted as HTML using a <div>element, causing it to be displayed with block layout.

2 GWT Widget Class

This widget can contain HTML text and displays the html content using a <div> element, causing it to be displayed with block layout.

Form Widgets

Following are few important Form Widgets

Sr.No. Widget & Description
1 Button

This widget represents a standard push button.

2 PushButton

This widget represents a normal push button with custom styling.

3 ToggleButton

This widget represents a stylish stateful button which allows the user to toggle between up and down states.

4 CheckBox

This widget represents a standard check box widget. This class also serves as a base class for RadioButton.

5 RadioButton

This widget represents a mutually-exclusive selection radio button widget.

6 ListBox

This widget represents a list of choices to the user, either as a list box or as a drop-down list.

7 SuggestBox

This widget represents a text box or text area which displays a pre-configured set of selections that match the user's input. Each SuggestBox is associated with a single SuggestOracle. The SuggestOracle is used to provide a set of selections given a specific query string.

8 TextBox

This widget represents a single line text box.

9 PasswordTextBox

This widget represents a text box that visually masks its input to prevent eavesdropping..

10 TextArea

This widget represents a text box that allows multiple lines of text to be entered.

11 RichTextArea

This widget represents a rich text editor that allows complex styling and formatting.

12 FileUpload

This widget wraps the HTML <input type='file'> element.

13 Hidden

This widget represets a hidden field in an HTML form.

"Complex widgets" allows users to advanced interaction capability with the application. Every Complex widget inherits properties from Widget class which in turn inherits properties from UIObject.

Sr.No. Widget & Description
1 GWT UIObject Class

This widget contains text, not interpreted as HTML using a <div>element, causing it to be displayed with block layout.

2 GWT Widget Class

This widget can contain HTML text and displays the html content using a <div> element, causing it to be displayed with block layout.

Complex Widgets

Following are few important Complex Widgets −

Sr.No. Widget & Description
1 Tree

This widget represents a standard hierarchical tree widget. The tree contains a hierarchy of TreeItems that the user can open, close, and select.

2 MenuBar

This widget represents a standard menu bar widget. A menu bar can contain any number of menu items, each of which can either fire a Command or open a cascaded menu bar.

3 DatePicker

This widget represents a standard GWT date picker.

4 CellTree

This widget represents a view of a tree. This widget will only work in standards mode, which requires that the HTML page in which it is run have an explicit <!DOCTYPE> declaration.

5 CellList

This widget represents a single column list of cells.

6 CellTable

This widget represents a tabular view that supports paging and columns.

7 CellBrowser

This widget represents a browsable view of a tree in which only a single node per level may be open at one time. This widget will only work in standards mode, which requires that the HTML page in which it is run have an explicit <!DOCTYPE> declaration.

Layout Panels can contain other widgets. These panels controls the way widgets to be shown on User Interface. Every Panel widget inherits properties from Panel class which in turn inherits properties from Widget class and which in turn inherits properties from UIObject class.

Sr.No. Widget & Description
1 GWT UIObject Class

This widget contains text, not interpreted as HTML using a <div>element, causing it to be displayed with block layout.

2 GWT Widget Class

This widget can contain HTML text and displays the html content using a <div> element, causing it to be displayed with block layout.

3 GWT Panel Class

This is an is the abstract base class for all panels, which are widgets that can contain other widgets.

Paneles de diseño

A continuación se muestran algunos paneles de diseño importantes :

No Señor. Widget y descripción
1 FlowPanel

Este widget representa un panel que formatea sus widgets secundarios utilizando el comportamiento de diseño HTML predeterminado.

2 HorizontalPanel

Este widget representa un panel que coloca todos sus widgets en una sola columna horizontal.

3 VerticalPanel

Este widget representa un panel que coloca todos sus widgets en una sola columna vertical.

4 HorizontalSplitPanel

Este widget representa un panel que organiza dos widgets en una sola fila horizontal y permite al usuario cambiar interactivamente la proporción del ancho dedicado a cada uno de los dos widgets. Los widgets contenidos en HorizontalSplitPanel se decorarán automáticamente con barras de desplazamiento cuando sea necesario.

5 VerticalSplitPanel

Este widget representa un panel A que organiza dos widgets en una sola columna vertical y permite al usuario cambiar de forma interactiva la proporción de la altura dedicada a cada uno de los dos widgets. Los widgets contenidos en VertialSplitPanel se decorarán automáticamente con barras de desplazamiento cuando sea necesario.

6 FlexTable

Este widget representa una tabla flexible que crea celdas a pedido. Puede ser irregular (es decir, cada fila puede contener un número diferente de celdas) y las celdas individuales se pueden configurar para abarcar varias filas o columnas.

7 Cuadrícula

Este widget representa una cuadrícula rectangular que puede contener texto, html o un widget secundario dentro de sus celdas. Debe redimensionarse explícitamente al número deseado de filas y columnas.

8 DeckPanel

panel que muestra todos sus widgets secundarios en una 'plataforma', donde solo uno puede ser visible a la vez. Es utilizado por TabPanel.

9 DockPanel

Este widget representa un panel que coloca sus widgets secundarios "acoplados" en sus bordes exteriores y permite que su último widget ocupe el espacio restante en su centro.

10 HTMLPanel

Este widget representa un panel que contiene HTML y que puede adjuntar widgets secundarios a elementos identificados dentro de ese HTML.

11 TabPanel

Este widget representa un panel que representa un conjunto de páginas con pestañas, cada una de las cuales contiene otro widget. Sus widgets secundarios se muestran cuando el usuario selecciona las diversas pestañas asociadas a ellos. Las pestañas pueden contener HTML arbitrario.

12 Compuesto

Este widget representa un tipo de widget que puede envolver otro widget, ocultando los métodos del widget envuelto. Cuando se agrega a un panel, un compuesto se comporta exactamente como si se hubiera agregado el widget que envuelve.

13 SimplePanel

Este widget representa una clase Base para paneles que contienen solo un widget.

14 ScrollPanel

Este widget representa un panel simple que envuelve su contenido en un área desplazable

15 FocusPanel

Este widget representa un panel simple que hace que su contenido sea enfocable y agrega la capacidad de capturar eventos de mouse y teclado.

dieciséis FormPanel

Este widget representa un panel que envuelve su contenido en un elemento HTML <FORM>.

17 PopupPanel

Este widget representa un panel que puede pop upsobre otros widgets. Se superpone al área del cliente del navegador (y cualquier ventana emergente creada previamente).

18 Caja de diálogo

Este widget representa una forma de ventana emergente que tiene un área de título en la parte superior y puede ser arrastrada por el usuario. A diferencia de PopupPanel, las llamadas a PopupPanel.setWidth (String) y PopupPanel.setHeight (String) establecerán el ancho y el alto del cuadro de diálogo, incluso si aún no se ha agregado un widget.

GWT proporciona un modelo de controlador de eventos similar a los marcos de interfaz de usuario Java AWT o SWING.

  • Una interfaz de escucha define uno o más métodos que el widget llama para anunciar un evento. GWT proporciona una lista de interfaces correspondientes a varios eventos posibles.

  • Una clase que desea recibir eventos de un tipo particular implementa la interfaz de controlador asociada y luego pasa una referencia a sí misma al widget para suscribirse a un conjunto de eventos.

Por ejemplo, el Button la clase publica click eventspor lo que tendrá que escribir una clase para implementar ClickHandler para manejarclick evento.

Interfaces del controlador de eventos

Todos los controladores de eventos de GWT se han ampliado desde la interfaz EventHandler y cada controlador tiene un único método con un único argumento. Este argumento es siempre un objeto de tipo de evento asociado. Cadaeventobject tiene varios métodos para manipular el objeto de evento pasado. Por ejemplo, para el evento de clic, tendrá que escribir su controlador de la siguiente manera:

/**
 * create a custom click handler which will call 
 * onClick method when button is clicked.
 */
public class MyClickHandler implements ClickHandler {
   @Override
   public void onClick(ClickEvent event) {
      Window.alert("Hello World!");
   }
}

Ahora cualquier clase que desee recibir eventos de clic llamará addClickHandler() para registrar un controlador de eventos de la siguiente manera:

/**
 * create button and attach click handler
 */
Button button = new Button("Click Me!");
button.addClickHandler(new MyClickHandler());

Cada widget que admita un tipo de evento tendrá un método de la forma HandlerRegistration addFooManipulador(FooEvento) donde Foo es el evento real como Click, Error, KeyPress, etc.

A continuación se muestra la lista de controladores de eventos GWT importantes y eventos asociados y métodos de registro de controladores:

No Señor. Interfaz de eventos Método y descripción del evento
1 Antes del controlador de selección <I>

void on Before Selection (Before Selection Event<I> event);

Se llama cuando se activa BeforeSelectionEvent.

2 BlurHandler

void on Blur(Blur Event event);

Se llama cuando se activa el evento Blur.

3 ChangeHandler

void on Change(ChangeEvent event);

Se llama cuando se activa un evento de cambio.

4 ClickHandler

void on Click(ClickEvent event);

Se llama cuando se activa un evento de clic nativo.

5 CloseHandler <T>

void on Close(CloseEvent<T> event);

Se llama cuando se activa CloseEvent.

6 Controlador de menú contextual

void on Context Menu(Context Menu Event event);

Se llama cuando se activa un evento de menú contextual nativo.

7 Controlador de doble clic

void on Double Click(Double Click Event event);

Se llama cuando se activa un evento de doble clic.

8 Manejador de errores

void on Error(Error Event event);

Se llama cuando se activa un evento de error.

9 Controlador de enfoque

void on Focus(Focus Event event);

Llamado cuando se dispara Focus Event.

10 Panel de formulario Enviar controlador completo

void on Submit Complete(Form Panel.Submit Complete Event event);

Se activa cuando un formulario se ha enviado correctamente.

11 FormPanel.SubmitHandler

void on Submit(Form Panel.Submit Event event);

Despedido cuando se envía el formulario.

12 Controlador de llave abajo

void on Key Down(Key Down Event event);

Se llama cuando se activa KeyDownEvent.

13 KeyPressHandler

void on KeyPress(KeyPressEvent event);

Se llama cuando se activa KeyPressEvent.

14 KeyUpHandler

void on KeyUp(KeyUpEvent event);

Se llama cuando se activa KeyUpEvent.

15 LoadHandler

void on Load(LoadEvent event);

Se llama cuando se activa LoadEvent.

dieciséis MouseDownHandler

void on MouseDown(MouseDownEvent event);

Se llama cuando se activa MouseDown.

17 MouseMoveHandler

void on MouseMove(MouseMoveEvent event);

Se llama cuando se activa MouseMoveEvent.

18 MouseOutHandler

void on MouseOut(MouseOutEvent event);

Se llama cuando se activa MouseOutEvent.

19 MouseOverHandler

void on MouseOver(MouseOverEvent event);

Se llama cuando se activa MouseOverEvent.

20 MouseUpHandler

void on MouseUp(MouseUpEvent event);

Se llama cuando se activa MouseUpEvent.

21 MouseWheelHandler

void on MouseWheel(MouseWheelEvent event);

Se llama cuando se activa MouseWheelEvent.

22 ResizeHandler

void on Resize(ResizeEvent event);

Se activa cuando se cambia el tamaño del widget.

23 ScrollHandler

void on Scroll(ScrollEvent event);

Se llama cuando se activa ScrollEvent.

24 SelectionHandler <I>

void on Selection(SelectionEvent<I> event);

Se llama cuando se activa SelectionEvent.

25 ValueChangeHandler <I>

void on ValueChange(ValueChangeEvent<I> event);

Se llama cuando se activa ValueChangeEvent.

26 Window.CierreHandler

void on WindowClosing(Window.ClosingEvent event);

Se activa justo antes de que se cierre la ventana del navegador o navegue a un sitio diferente.

27 Window.ScrollHandler

void on WindowScroll(Window.ScrollEvent event);

Se activa cuando se desplaza por la ventana del navegador.

Métodos de eventos

Como se mencionó anteriormente, cada controlador tiene un único método con un único argumento que contiene el objeto de evento, por ejemplo, void onClick (evento ClickEvent) o void onKeyDown (evento KeyDownEvent) . Los objetos de eventos como ClickEvent y KeyDownEvent tienen algunos métodos comunes que se enumeran a continuación:

No Señor. Método y descripción
1

protected void dispatch(ClickHandler handler) Este método solo debe ser llamado por HandlerManager

2

DomEvent.Type <FooHandler> getAssociatedType() Este método devuelve el tipo utilizado para registrarse. Foo evento.

3

static DomEvent.Type<FooHandler> getType() Este método obtiene el tipo de evento asociado con Foo eventos.

4

public java.lang.Object getSource() Este método devuelve la fuente que disparó este evento por última vez.

5

protected final boolean isLive() Este método devuelve si el evento está en vivo.

6

protected void kill() Este método mata el evento

Ejemplo

Este ejemplo le llevará a través de sencillos pasos para mostrar el uso de un Click Evento y KeyDownManejo de eventos en GWT. Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>

   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>Event Handling Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java que demostrará el uso de Event Handling en GWT.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.event.dom.client.KeyCodes;
import com.google.gwt.event.dom.client.KeyDownEvent;
import com.google.gwt.event.dom.client.KeyDownHandler;
import com.google.gwt.user.client.Window;
import com.google.gwt.user.client.ui.Button;
import com.google.gwt.user.client.ui.DecoratorPanel;
import com.google.gwt.user.client.ui.HasHorizontalAlignment;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TextBox;
import com.google.gwt.user.client.ui.VerticalPanel;

public class HelloWorld implements EntryPoint {
   public void onModuleLoad() {
      /**
       * create textbox and attach key down handler
       */
      TextBox textBox = new TextBox(); 
      textBox.addKeyDownHandler(new MyKeyDownHandler());

      /*
       * create button and attach click handler
       */
      Button button = new Button("Click Me!");
      button.addClickHandler(new MyClickHandler());

      VerticalPanel panel = new VerticalPanel();
      panel.setSpacing(10);
      panel.setHorizontalAlignment(HasHorizontalAlignment.ALIGN_CENTER);
      panel.setSize("300", "100");
      panel.add(textBox);
      panel.add(button);

      DecoratorPanel decoratorPanel = new DecoratorPanel();
      decoratorPanel.add(panel);
      RootPanel.get("gwtContainer").add(decoratorPanel);
   }

   /** 
    * create a custom click handler which will call 
    * onClick method when button is clicked.
    */
   private class MyClickHandler implements ClickHandler {
      @Override
      public void onClick(ClickEvent event) {
         Window.alert("Hello World!");
      }
   }

   /**
    * create a custom key down handler which will call 
    * onKeyDown method when a key is down in textbox.
    */
   private class MyKeyDownHandler implements KeyDownHandler {
      @Override
      public void onKeyDown(KeyDownEvent event) {
         if(event.getNativeKeyCode() == KeyCodes.KEY_ENTER){
            Window.alert(((TextBox)event.getSource()).getValue());
         }
      }
   }
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

GWT ofrece tres formas de crear elementos de interfaz de usuario personalizados. Hay tres estrategias generales a seguir:

  • Create a widget by extending Composite Class- Esta es la forma más común y sencilla de crear widgets personalizados. Aquí puede utilizar los widgets existentes para crear una vista compuesta con propiedades personalizadas.

  • Create a widget using GWT DOM API in JAVA- Los widgets básicos de GWT se crean de esta manera. Aún así, es una forma muy complicada de crear un widget personalizado y debe usarse con precaución.

  • Use JavaScript and wrap it in a widget using JSNI- Por lo general, esto solo debe hacerse como último recurso. Teniendo en cuenta las implicaciones de los métodos nativos en varios navegadores, se vuelve muy complicado y también más difícil de depurar.

Crear widget personalizado con clase compuesta

Este ejemplo lo llevará a través de sencillos pasos para mostrar la creación de un widget personalizado en GWT. Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo de Widgets básicos -

Aquí vamos a crear un widget personalizado ampliando la clase Composite, que es la forma más sencilla de crear widgets personalizados.

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>

   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>Custom Widget Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java que demostrará la creación de un widget personalizado.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.user.client.ui.CheckBox;
import com.google.gwt.user.client.ui.Composite;
import com.google.gwt.user.client.ui.HorizontalPanel;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TextBox;

public class HelloWorld implements EntryPoint {

   /**
    * A composite of a TextBox and a CheckBox that optionally enables it.
    */
   private static class OptionalTextBox extends Composite implements
   ClickHandler {

      private TextBox textBox = new TextBox();
      private CheckBox checkBox = new CheckBox();
      private boolean enabled = true;

      public boolean isEnabled() {
         return enabled;
      }

      public void setEnabled(boolean enabled) {
         this.enabled = enabled;
      }

      /**
       * Style this widget using .optionalTextWidget CSS class.<br/>
       * Style textbox using .optionalTextBox CSS class.<br/>
       * Style checkbox using .optionalCheckBox CSS class.<br/>
       * Constructs an OptionalTextBox with the given caption 
       * on the check.
       * @param caption the caption to be displayed with the check box
       */
      public OptionalTextBox(String caption) {
         // place the check above the text box using a vertical panel.
         HorizontalPanel panel = new HorizontalPanel();
         // panel.setBorderWidth(1);
         panel.setSpacing(10);
         panel.add(checkBox);
         panel.add(textBox);

         // all composites must call initWidget() in their constructors.
         initWidget(panel);
         
         //set style name for entire widget
         setStyleName("optionalTextWidget");
         
         //set style name for text box
         textBox.setStyleName("optionalTextBox");
         
         //set style name for check box
         checkBox.setStyleName("optionalCheckBox");
         textBox.setWidth("200");
         
         // Set the check box's caption, and check it by default.
         checkBox.setText(caption);
         checkBox.setValue(enabled);
         checkBox.addClickHandler(this);
         enableTextBox(enabled,checkBox.getValue());
      }

      public void onClick(ClickEvent event) {
         if (event.getSource() == checkBox) {
            // When the check box is clicked,
            //update the text box's enabled state.
            enableTextBox(enabled,checkBox.getValue());
         }
      }

      private void enableTextBox(boolean enable,boolean isChecked){
         enable = (enable && isChecked) || (!enable && !isChecked);
         textBox.setStyleDependentName("disabled", !enable);
         textBox.setEnabled(enable);	  
      }
   }

   public void onModuleLoad() {
      // Create an optional text box and add it to the root panel.
      OptionalTextBox otb = new OptionalTextBox(
         "Want to explain the solution?");
      otb.setEnabled(true);
      RootPanel.get().add(otb);
   }    
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

You can notice following points

  • La creación de un widget personalizado ampliando el widget compuesto es bastante fácil.

  • Hemos creado un widget con widgets incorporados de GWT, TextBox y CheckBox, utilizando así el concepto de reutilización.

  • TextBox se deshabilita / habilita según el estado de la casilla de verificación. Hemos proporcionado una API para habilitar / deshabilitar el control.

  • Hemos expuesto estilos de widgets internos a través de estilos CSS documentados.

Introducción

UiBinder es un marco diseñado para separar la funcionalidad y la vista de la interfaz de usuario.

  • El marco UiBinder permite a los desarrolladores crear aplicaciones gwt como páginas HTML con widgets GWT configurados en todas ellas.

  • El marco UiBinder facilita la colaboración con los diseñadores de UI que se sienten más cómodos con XML, HTML y CSS que con el código fuente de Java.

  • UIBinder proporciona una forma declarativa de definir la interfaz de usuario.

  • UIBinder separa la lógica programática de la interfaz de usuario.

  • El UIBinder es similar a lo que JSP es para Servlets.

Flujo de trabajo de UiBinder

Paso 1: crear un archivo XML de declaración de IU

Cree un archivo de declaración de interfaz de usuario basado en XML / HTML. Hemos creado unLogin.ui.xml archivo en nuestro ejemplo.

<ui:UiBinder xmlns:ui = 'urn:ui:com.google.gwt.uibinder'
   xmlns:gwt = 'urn:import:com.google.gwt.user.client.ui' 
   xmlns:res = 'urn:with:com.tutorialspoint.client.LoginResources'>
   <ui:with type = "com.tutorialspoint.client.LoginResources" field = "res">
   </ui:with>
   <gwt:HTMLPanel>
   ...  
   </gwt:HTMLPanel>
</ui:UiBinder>

Paso 2: use el campo ui: para el enlace posterior

Utilice el atributo ui: field en el elemento XML / HTML para relacionar el campo UI en XML con el campo UI en el archivo JAVA para un enlace posterior.

<gwt:Label ui:field = "completionLabel1" />
<gwt:Label ui:field = "completionLabel2" />

Paso 3: crear la contraparte de Java de UI XML

Cree una contraparte basada en Java del diseño basado en XML ampliando el widget compuesto. Hemos creado unLogin.java archivo en nuestro ejemplo.

package com.tutorialspoint.client;
   ...
public class Login extends Composite {
   ...
}

Paso 4: vincular los campos de la interfaz de usuario de Java con la anotación UiField

use la anotación @UiField en Login.java para designar miembros de clase de contraparte para enlazar a campos basados ​​en XML en Login.ui.xml

public class Login extends Composite {
   ...
   @UiField
   Label completionLabel1;

   @UiField
   Label completionLabel2;  
   ...
}

Paso 5: vincular la interfaz de usuario de Java con la interfaz de usuario XML con la anotación de UiTemplate

Indique a GWT que vincule el componente basado en java Login.java y diseño basado en XML Login.ui.xml usando la anotación @UiTemplate

public class Login extends Composite {

   private static LoginUiBinder uiBinder = GWT.create(LoginUiBinder.class);

   /*
    * @UiTemplate is not mandatory but allows multiple XML templates
    * to be used for the same widget. 
    * Default file loaded will be <class-name>.ui.xml
    */
   
   @UiTemplate("Login.ui.xml")
   interface LoginUiBinder extends UiBinder<Widget, Login> {
   }
   ...
}

Paso 6: crear archivo CSS

Crea un archivo CSS externoLogin.css y recurso basado en Java LoginResources.java archivo equivalente a estilos CSS

.blackText {
   font-family: Arial, Sans-serif;
   color: #000000;
   font-size: 11px;
   text-align: left;
}
...

Paso 7: crear un archivo de recursos basado en Java para un archivo CSS

package com.tutorialspoint.client;
...
public interface LoginResources extends ClientBundle {
   public interface MyCss extends CssResource {
      String blackText();

      ...
   }

   @Source("Login.css")
   MyCss style();
}

Paso 8: adjunte el recurso CSS en el archivo de código de la interfaz de usuario de Java.

Adjuntar un archivo CSS externoLogin.css usando Contructor de la clase de widget basada en Java Login.java

public Login() {
   this.res = GWT.create(LoginResources.class);
   res.style().ensureInjected();
   initWidget(uiBinder.createAndBindUi(this));
}

Ejemplo completo de UIBinder

Este ejemplo lo llevará a través de sencillos pasos para mostrar el uso de un UIBinder en GWT. Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>
   <!-- Inherit the UiBinder module.                               -->
   <inherits name = "com.google.gwt.uibinder.UiBinder"/>
   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>
  
   <!-- Specify the paths for translatable code                    -->
   <source path ='client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>UiBinder Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Ahora cree una nueva plantilla de UiBinder y una clase de propietario (Archivo → Nuevo → UiBinder).

Elija el paquete de cliente para el proyecto y luego asígnele el nombre Iniciar sesión. Deje todos los demás valores predeterminados. Haga clic en el botón Finalizar y el complemento creará una nueva plantilla de UiBinder y una clase de propietario.

Ahora cree el archivo Login.css en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

.blackText {
   font-family: Arial, Sans-serif;
   color: #000000;
   font-size: 11px;
   text-align: left;
}

.redText {
   font-family: Arial, Sans-serif;
   color: #ff0000;
   font-size: 11px;
   text-align: left;
}

.loginButton {
   border: 1px solid #3399DD;
   color: #FFFFFF;
   background: #555555;
   font-size: 11px;
   font-weight: bold;
   margin: 0 5px 0 0;
   padding: 4px 10px 5px;
   text-shadow: 0 -1px 0 #3399DD;
}

.box {
   border: 1px solid #AACCEE;
   display: block;
   font-size: 12px;
   margin: 0 0 5px;
   padding: 3px;
   width: 203px;
}

.background {
   background-color: #999999;
   border: 1px none transparent;
   color: #000000;
   font-size: 11px;
   margin-left: -8px;
   margin-top: 5px;
   padding: 6px;
}

Ahora cree el archivo LoginResources.java en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

package com.tutorialspoint.client;

import com.google.gwt.resources.client.ClientBundle;
import com.google.gwt.resources.client.CssResource;

public interface LoginResources extends ClientBundle {
   /**
    * Sample CssResource.
    */
   public interface MyCss extends CssResource {
      String blackText();

      String redText();

      String loginButton();

      String box();

      String background();
   }

   @Source("Login.css")
   MyCss style();
}

Reemplace el contenido de Login.ui.xml en src/com.tutorialspoint/client paquete con lo siguiente

<ui:UiBinder xmlns:ui = 'urn:ui:com.google.gwt.uibinder'
   xmlns:gwt = 'urn:import:com.google.gwt.user.client.ui' 
   xmlns:res = 'urn:with:com.tutorialspoint.client.LoginResources'>
   
   <ui:with type = "com.tutorialspoint.client.LoginResources" field = "res">
   </ui:with>
   
   <gwt:HTMLPanel>
      <div align = "center">
         
         <gwt:VerticalPanel res:styleName = "style.background">
            <gwt:Label text = "Login" res:styleName = "style.blackText" />
            <gwt:TextBox ui:field="loginBox" res:styleName = "style.box" />
            <gwt:Label text = "Password" res:styleName = "style.blackText" />
            <gwt:PasswordTextBox ui:field = "passwordBox" res:styleName = "style.box" />
            
            <gwt:HorizontalPanel verticalAlignment = "middle">
               <gwt:Button ui:field = "buttonSubmit" text="Submit"
                  res:styleName = "style.loginButton" />
               <gwt:CheckBox ui:field = "myCheckBox" />
               <gwt:Label ui:field = "myLabel" text = "Remember me"
                  res:styleName = "style.blackText" />
            </gwt:HorizontalPanel>
            
            <gwt:Label ui:field = "completionLabel1" res:styleName = "style.blackText" />
            <gwt:Label ui:field = "completionLabel2" res:styleName = "style.blackText" />
         </gwt:VerticalPanel>
         
      </div>
   </gwt:HTMLPanel>
   
</ui:UiBinder>

Reemplace el contenido de Login.java en src/com.tutorialspoint/client paquete con lo siguiente

package com.tutorialspoint.client;

import com.google.gwt.core.client.GWT;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.logical.shared.ValueChangeEvent;
import com.google.gwt.uibinder.client.UiBinder;
import com.google.gwt.uibinder.client.UiField;
import com.google.gwt.uibinder.client.UiHandler;
import com.google.gwt.uibinder.client.UiTemplate;
import com.google.gwt.user.client.Window;
import com.google.gwt.user.client.ui.Composite;
import com.google.gwt.user.client.ui.Label;
import com.google.gwt.user.client.ui.TextBox;
import com.google.gwt.user.client.ui.Widget;

public class Login extends Composite {

   private static LoginUiBinder uiBinder = GWT.create(LoginUiBinder.class);

   /*
    * @UiTemplate is not mandatory but allows multiple XML templates
    * to be used for the same widget. 
    * Default file loaded will be <class-name>.ui.xml
    */
   @UiTemplate("Login.ui.xml")
   interface LoginUiBinder extends UiBinder<Widget, Login> {
   }

   @UiField(provided = true)
   final LoginResources res;

   public Login() {
      this.res = GWT.create(LoginResources.class);
      res.style().ensureInjected();
      initWidget(uiBinder.createAndBindUi(this));
   }

   @UiField
   TextBox loginBox;

   @UiField
   TextBox passwordBox;

   @UiField
   Label completionLabel1;

   @UiField
   Label completionLabel2;

   private Boolean tooShort = false;

   /*
    * Method name is not relevant, the binding is done according to the class
    * of the parameter.
    */
   @UiHandler("buttonSubmit")
   void doClickSubmit(ClickEvent event) {
      if (!tooShort) {
         Window.alert("Login Successful!");
      } else {
         Window.alert("Login or Password is too short!");
      }
   }

   @UiHandler("loginBox")
   void handleLoginChange(ValueChangeEvent<String> event) {
      if (event.getValue().length() < 6) {
         completionLabel1.setText("Login too short (Size must be > 6)");
         tooShort = true;
      } else {
         tooShort = false;
         completionLabel1.setText("");
      }
   }

   @UiHandler("passwordBox")
   void handlePasswordChange(ValueChangeEvent<String> event) {
      if (event.getValue().length() < 6) {
         tooShort = true;
         completionLabel2.setText("Password too short (Size must be > 6)");
      } else {
         tooShort = false;
         completionLabel2.setText("");
      }
   }
}

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java que demostrará el uso de UiBinder.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.user.client.ui.RootPanel;

public class HelloWorld implements EntryPoint {
   public void onModuleLoad() {
      RootPanel.get().add(new Login());   
   }    
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

Una aplicación basada en GWT generalmente consta de un módulo del lado del cliente y un módulo del lado del servidor. El código del lado del cliente se ejecuta en el navegador y el código del lado del servidor se ejecuta en el servidor web. El código del lado del cliente tiene que realizar una solicitud HTTP a través de la red para acceder a los datos del lado del servidor.

RPC, Remote Procedure Call es el mecanismo utilizado por GWT en el que el código del cliente puede ejecutar directamente los métodos del lado del servidor.

  • GWT RPC está basado en servlets.

  • GWT RPC es asincrónico y el cliente nunca se bloquea durante la comunicación.

  • Usando GWT RPC, los objetos Java se pueden enviar directamente entre el cliente y el servidor (que son serializados automáticamente por el marco de GWT).

  • El servlet del lado del servidor se denomina service.

  • La llamada a procedimiento remoto que llama a métodos de servlets del lado del servidor desde el código del lado del cliente se denomina invoking a service.

Componentes de GWT RPC

A continuación se muestran los tres componentes utilizados en el mecanismo de comunicación GWT RPC

  • Un servicio remoto (servlet del lado del servidor) que se ejecuta en el servidor.
  • Código de cliente para invocar ese servicio.
  • Objetos de datos Java que se pasarán entre el cliente y el servidor.

El cliente y el servidor de GWT serializan y deserializan datos automáticamente para que los desarrolladores no estén obligados a serializar / deserializar objetos y los objetos de datos puedan viajar a través de HTTP.

El siguiente diagrama muestra la arquitectura RPC.

Para comenzar a usar RPC, debemos seguir las convenciones de GWT.

Flujo de trabajo de comunicación RPC

Paso 1: crear una clase de modelo serializable

Defina un objeto de modelo Java en el lado del cliente que debería ser serializable.

public class Message implements Serializable {
   ...
   private String message;
   public Message(){};

   public void setMessage(String message) {
      this.message = message;
   }
   ...
}

Paso 2: crear una interfaz de servicio

Defina una interfaz para el servicio en el lado del cliente que amplíe RemoteService y enumere todos los métodos de servicio.

Utilice la anotación @RemoteServiceRelativePath para asignar el servicio con una ruta predeterminada del servlet remoto en relación con la URL base del módulo.

@RemoteServiceRelativePath("message")
public interface MessageService extends RemoteService {
   Message getMessage(String input);
}

Paso 3: crear una interfaz de servicio asincrónica

Defina una interfaz asincrónica para el servicio en el lado del cliente (en la misma ubicación que el servicio mencionado anteriormente) que se utilizará en el código de cliente de GWT.

public interface MessageServiceAsync {
   void getMessage(String input, AsyncCallback<Message> callback);
}

Paso 4: crear una clase de servlet de implementación de servicios

Implemente la interfaz en el lado del servidor y esa clase debería extender la clase RemoteServiceServlet.

public class MessageServiceImpl extends RemoteServiceServlet
   implements MessageService{
   ...
   public Message getMessage(String input) {
      String messageString = "Hello " + input + "!";
      Message message = new Message();
      message.setMessage(messageString);
      return message;
   }
}

Paso 5: actualice Web.xml para incluir la declaración de servlet

Edite el descriptor de implementación de la aplicación web (web.xml) para incluir la declaración MessageServiceImpl Servlet.

<web-app>
   ...
   <servlet>
      <servlet-name>messageServiceImpl</servlet-name>
      <servlet-class>com.tutorialspoint.server.MessageServiceImpl
      </servlet-class>
   </servlet>
   
   <servlet-mapping>
      <servlet-name>messageServiceImpl</servlet-name>
      <url-pattern>/helloworld/message</url-pattern>
   </servlet-mapping>
</web-app>

Paso 6: realice la llamada al procedimiento remoto en el código de la aplicación

Cree la clase de proxy de servicio.

MessageServiceAsync messageService = GWT.create(MessageService.class);

Cree el controlador AsyncCallback para manejar la devolución de llamada RPC en la que el servidor devuelve el mensaje al cliente

class MessageCallBack implements AsyncCallback<Message> {

   @Override
   public void onFailure(Throwable caught) {
      Window.alert("Unable to obtain server response: "
      + caught.getMessage());	
   }

   @Override
   public void onSuccess(Message result) {
      Window.alert(result.getMessage()); 
   }	   
}

Llamar al servicio remoto cuando el usuario interactúa con la interfaz de usuario

public class HelloWorld implements EntryPoint {
   ... 
   public void onModuleLoad() {
   ...
      buttonMessage.addClickHandler(new ClickHandler() {			
         @Override
         public void onClick(ClickEvent event) {
            messageService.getMessage(txtName.getValue(), 
            new MessageCallBack());
         }
      });
   ...
   }
}

Ejemplo completo de comunicación RPC

Este ejemplo lo llevará a través de sencillos pasos para mostrar un ejemplo de una comunicación RPC en GWT. Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>
   <!-- Inherit the UiBinder module.                               -->
   <inherits name = "com.google.gwt.uibinder.UiBinder"/>
   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>
  
   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>RPC Communication Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Ahora cree el archivo Message.java en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

package com.tutorialspoint.client;

import java.io.Serializable;

public class Message implements Serializable {
 
   private static final long serialVersionUID = 1L;
   private String message;
   public Message(){};

   public void setMessage(String message) {
      this.message = message;
   }

   public String getMessage() {
      return message;
   }
}

Ahora cree el archivo MessageService.java en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

package com.tutorialspoint.client;

import com.google.gwt.user.client.rpc.RemoteService;
import com.google.gwt.user.client.rpc.RemoteServiceRelativePath;

@RemoteServiceRelativePath("message")
public interface MessageService extends RemoteService {
   Message getMessage(String input);
}

Ahora cree el archivo MessageServiceAsync.java en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

package com.tutorialspoint.client;

import com.google.gwt.user.client.rpc.AsyncCallback;

public interface MessageServiceAsync {
   void getMessage(String input, AsyncCallback<Message> callback);
}

Ahora cree el archivo MessageServiceImpl.java en el src/com.tutorialspoint/server empaque y coloque el siguiente contenido en él

package com.tutorialspoint.server;

import com.google.gwt.user.server.rpc.RemoteServiceServlet;
import com.tutorialspoint.client.Message;
import com.tutorialspoint.client.MessageService;

public class MessageServiceImpl extends RemoteServiceServlet 
   implements MessageService{

   private static final long serialVersionUID = 1L;

   public Message getMessage(String input) {
      String messageString = "Hello " + input + "!";
      Message message = new Message();
      message.setMessage(messageString);
      return message;
   }   
}

Actualizar el contenido del descriptor de implementación de la aplicación web modificado war/WEB-INF/web.xml para incluir la declaración MessageServiceImpl Servlet.

<?xml version = "1.0" encoding = "UTF-8"?>
<!DOCTYPE web-app
   PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
   "http://java.sun.com/dtd/web-app_2_3.dtd">

<web-app>
   <!-- Default page to serve -->
   <welcome-file-list>
      <welcome-file>HelloWorld.html</welcome-file>
   </welcome-file-list>
   
   <servlet>
      <servlet-name>messageServiceImpl</servlet-name>
      <servlet-class>com.tutorialspoint.server.MessageServiceImpl
      </servlet-class>
   </servlet>

   <servlet-mapping>
      <servlet-name>messageServiceImpl</servlet-name>
      <url-pattern>/helloworld/message</url-pattern>
   </servlet-mapping>
</web-app>

Reemplace el contenido de HelloWorld.java en src/com.tutorialspoint/client paquete con lo siguiente

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.core.client.GWT;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.event.dom.client.KeyCodes;
import com.google.gwt.event.dom.client.KeyUpEvent;
import com.google.gwt.event.dom.client.KeyUpHandler;
import com.google.gwt.user.client.Window;
import com.google.gwt.user.client.rpc.AsyncCallback;
import com.google.gwt.user.client.ui.Button;
import com.google.gwt.user.client.ui.DecoratorPanel;
import com.google.gwt.user.client.ui.HasHorizontalAlignment;
import com.google.gwt.user.client.ui.HorizontalPanel;
import com.google.gwt.user.client.ui.Label;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TextBox;
import com.google.gwt.user.client.ui.VerticalPanel;

public class HelloWorld implements EntryPoint {
	
   private MessageServiceAsync messageService = 
   GWT.create(MessageService.class);

   private class MessageCallBack implements AsyncCallback<Message> {
      @Override
      public void onFailure(Throwable caught) {
         /* server side error occured */
         Window.alert("Unable to obtain server response: " + caught.getMessage());	
      }
      @Override
      public void onSuccess(Message result) {
          /* server returned result, show user the message */
         Window.alert(result.getMessage());
      }	   
   }

   public void onModuleLoad() {
      /*create UI */
      final TextBox txtName = new TextBox(); 
      txtName.setWidth("200");
      txtName.addKeyUpHandler(new KeyUpHandler() {
         @Override
         public void onKeyUp(KeyUpEvent event) {
            if(event.getNativeKeyCode() == KeyCodes.KEY_ENTER){
               /* make remote call to server to get the message */
               messageService.getMessage(txtName.getValue(), 
               new MessageCallBack());
            }				
         }
      });
      Label lblName = new Label("Enter your name: ");

      Button buttonMessage = new Button("Click Me!");

      buttonMessage.addClickHandler(new ClickHandler() {			
         @Override
         public void onClick(ClickEvent event) {
            /* make remote call to server to get the message */
            messageService.getMessage(txtName.getValue(), 
            new MessageCallBack());
         }
      });

      HorizontalPanel hPanel = new HorizontalPanel();	
      hPanel.add(lblName);
      hPanel.add(txtName);
      hPanel.setCellWidth(lblName, "130");

      VerticalPanel vPanel = new VerticalPanel();
      vPanel.setSpacing(10);
      vPanel.add(hPanel);
      vPanel.add(buttonMessage);
      vPanel.setCellHorizontalAlignment(buttonMessage, 
      HasHorizontalAlignment.ALIGN_RIGHT);

      DecoratorPanel panel = new DecoratorPanel();
      panel.add(vPanel);

      // Add widgets to the root panel.
      RootPanel.get("gwtContainer").add(panel);
   }    
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

GWT proporciona un soporte excelente para las pruebas automatizadas del código del lado del cliente utilizando el marco de prueba JUnit. En este artículo demostraremos la integración de GWT y JUNIT.

Descargar archivo Junit

Sitio oficial JUnit - https://www.junit.org

Descargar Junit-4.10.jar

SO Nombre de archivo
Ventanas junit4.10.jar
Linux junit4.10.jar
Mac junit4.10.jar

Guarde el archivo jar descargado en alguna ubicación de su computadora. Lo hemos almacenado enC:/ > JUNIT

Busque la carpeta de instalación de GWT

SO Carpeta de instalación de GWT
Ventanas C: \ GWT \ gwt-2.1.0
Linux /usr/local/GWT/gwt-2.1.0
Mac /Library/GWT/gwt-2.1.0

Clase GWTTestCase

GWT proporciona GWTTestCaseclase base que proporciona integración JUnit. La ejecución de una clase compilada que extiende GWTTestCase bajo JUnit inicia el navegador HtmlUnit que sirve para emular el comportamiento de su aplicación durante la ejecución de la prueba.

GWTTestCase es una clase derivada de TestCase de JUnit y se puede ejecutar usando JUnit TestRunner.

Usando webAppCreator

GWT proporciona una herramienta de línea de comandos especial webAppCreator que puede generar un caso de prueba de inicio para nosotros, además de objetivos de hormigas y configuraciones de lanzamiento de eclipse para probar tanto en modo de desarrollo como en modo de producción.

Abra el símbolo del sistema y vaya a C:\ > GWT_WORKSPACE > donde desea crear un nuevo proyecto con soporte de prueba. Ejecute el siguiente comando

C:\GWT_WORKSPACE>C:\GWT\gwt-2.1.0\webAppCreator 
   -out HelloWorld 
   -junit C:\JUNIT\junit-4.10.jar 
   com.tutorialspoint.HelloWorld

Puntos destacados

  • Estamos ejecutando la utilidad de línea de comandos webAppCreator.
  • HelloWorld es el nombre del proyecto que se creará
  • La opción -junit indica a webAppCreator que agregue soporte de junit al proyecto
  • com.tutorialspoint.HelloWorld es el nombre del módulo

Verifique la salida.

Created directory HelloWorld\src
Created directory HelloWorld\war
Created directory HelloWorld\war\WEB-INF
Created directory HelloWorld\war\WEB-INF\lib
Created directory HelloWorld\src\com\tutorialspoint
Created directory HelloWorld\src\com\tutorialspoint\client
Created directory HelloWorld\src\com\tutorialspoint\server
Created directory HelloWorld\src\com\tutorialspoint\shared
Created directory HelloWorld\test\com\tutorialspoint
Created directory HelloWorld\test\com\tutorialspoint\client
Created file HelloWorld\src\com\tutorialspoint\HelloWorld.gwt.xml
Created file HelloWorld\war\HelloWorld.html
Created file HelloWorld\war\HelloWorld.css
Created file HelloWorld\war\WEB-INF\web.xml
Created file HelloWorld\src\com\tutorialspoint\client\HelloWorld.java
Created file 
HelloWorld\src\com\tutorialspoint\client\GreetingService.java
Created file 
HelloWorld\src\com\tutorialspoint\client\GreetingServiceAsync.java
Created file 
HelloWorld\src\com\tutorialspoint\server\GreetingServiceImpl.java
Created file HelloWorld\src\com\tutorialspoint\shared\FieldVerifier.java
Created file HelloWorld\build.xml
Created file HelloWorld\README.txt
Created file HelloWorld\test\com\tutorialspoint\HelloWorldJUnit.gwt.xml
Created file HelloWorld\test\com\tutorialspoint\client\HelloWorldTest.java
Created file HelloWorld\.project
Created file HelloWorld\.classpath
Created file HelloWorld\HelloWorld.launch
Created file HelloWorld\HelloWorldTest-dev.launch
Created file HelloWorld\HelloWorldTest-prod.launch

Comprender la clase de prueba: HelloWorldTest.java

package com.tutorialspoint.client;

import com.tutorialspoint.shared.FieldVerifier;
import com.google.gwt.core.client.GWT;
import com.google.gwt.junit.client.GWTTestCase;
import com.google.gwt.user.client.rpc.AsyncCallback;
import com.google.gwt.user.client.rpc.ServiceDefTarget;

/**
 * GWT JUnit tests must extend GWTTestCase.
 */
public class HelloWorldTest extends GWTTestCase {

   /**
    * must refer to a valid module that sources this class.
    */
   public String getModuleName() {
      return "com.tutorialspoint.HelloWorldJUnit";
   }

   /**
    * tests the FieldVerifier.
    */
   public void testFieldVerifier() {
      assertFalse(FieldVerifier.isValidName(null));
      assertFalse(FieldVerifier.isValidName(""));
      assertFalse(FieldVerifier.isValidName("a"));
      assertFalse(FieldVerifier.isValidName("ab"));
      assertFalse(FieldVerifier.isValidName("abc"));
      assertTrue(FieldVerifier.isValidName("abcd"));
   }

   /**
    * this test will send a request to the server using the greetServer
    *  method in GreetingService and verify the response.
    */
   public void testGreetingService() {
      /* create the service that we will test. */
      GreetingServiceAsync greetingService = 
      GWT.create(GreetingService.class);
      ServiceDefTarget target = (ServiceDefTarget) greetingService;
      target.setServiceEntryPoint(GWT.getModuleBaseURL() 
      + "helloworld/greet");

      /* since RPC calls are asynchronous, we will need to wait 
       for a response after this test method returns. This line 
       tells the test runner to wait up to 10 seconds 
       before timing out. */
      delayTestFinish(10000);

      /* send a request to the server. */
      greetingService.greetServer("GWT User", 
         new AsyncCallback<String>() {
         public void onFailure(Throwable caught) {
            /* The request resulted in an unexpected error. */
            fail("Request failure: " + caught.getMessage());
         }

         public void onSuccess(String result) {
            /* verify that the response is correct. */
            assertTrue(result.startsWith("Hello, GWT User!"));

            /* now that we have received a response, we need to 
             tell the test runner that the test is complete. 
             You must call finishTest() after an asynchronous test 
             finishes successfully, or the test will time out.*/
            finishTest();
         }
      });
   }
}

Puntos destacados

No Señor. Nota
1 La clase HelloWorldTest se generó en el paquete com.tutorialspoint.client en el directorio HelloWorld / test.
2 La clase HelloWorldTest contendrá casos de prueba unitaria para HelloWorld.
3 La clase HelloWorldTest extiende la clase GWTTestCase en el paquete com.google.gwt.junit.client.
4 La clase HelloWorldTest tiene un método abstracto (getModuleName) que debe devolver el nombre del módulo GWT. Para HelloWorld, esto es com.tutorialspoint.HelloWorldJUnit.
5 La clase HelloWorldTest se genera con dos casos de prueba de muestra testFieldVerifier, testSimple. Hemos agregado testGreetingService.
6 Estos métodos utilizan una de las muchas funciones assert * que hereda de la clase JUnit Assert, que es un antepasado de GWTTestCase.
7 La función assertTrue (booleano) afirma que el argumento booleano pasado se evalúa como verdadero. De lo contrario, la prueba fallará cuando se ejecute en JUnit.

GWT - Ejemplo completo de integración de JUnit

Este ejemplo lo llevará a través de sencillos pasos para mostrar un ejemplo de integración JUnit en GWT.

Siga los siguientes pasos para actualizar la aplicación GWT que creamos anteriormente:

Paso Descripción
1 Importe el proyecto con un nombre HelloWorld en eclipse utilizando el asistente de importación de proyectos existentes (Archivo → Importar → General → Proyectos existentes en el espacio de trabajo).
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra la estructura del proyecto en eclipse.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>
   <!-- Inherit the UiBinder module.                               -->
   <inherits name = "com.google.gwt.uibinder.UiBinder"/>
   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>
  
   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>JUnit Integration Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Reemplace el contenido de HelloWorld.java en src/com.tutorialspoint/client paquete con lo siguiente

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;

import com.google.gwt.core.client.GWT;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.event.dom.client.KeyCodes;
import com.google.gwt.event.dom.client.KeyUpEvent;
import com.google.gwt.event.dom.client.KeyUpHandler;

import com.google.gwt.user.client.Window;
import com.google.gwt.user.client.rpc.AsyncCallback;
import com.google.gwt.user.client.ui.Button;
import com.google.gwt.user.client.ui.DecoratorPanel;
import com.google.gwt.user.client.ui.HasHorizontalAlignment;
import com.google.gwt.user.client.ui.HorizontalPanel;
import com.google.gwt.user.client.ui.Label;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TextBox;
import com.google.gwt.user.client.ui.VerticalPanel;

public class HelloWorld implements EntryPoint {
	
   public void onModuleLoad() {
      /*create UI */
      final TextBox txtName = new TextBox(); 
      txtName.setWidth("200");
      txtName.addKeyUpHandler(new KeyUpHandler() {
         @Override
         public void onKeyUp(KeyUpEvent event) {
            if(event.getNativeKeyCode() == KeyCodes.KEY_ENTER){
               Window.alert(getGreeting(txtName.getValue()));
            }				
         }
      });
      Label lblName = new Label("Enter your name: ");

      Button buttonMessage = new Button("Click Me!");

      buttonMessage.addClickHandler(new ClickHandler() {			
         @Override
         public void onClick(ClickEvent event) {
            Window.alert(getGreeting(txtName.getValue()));
         }
      });

      HorizontalPanel hPanel = new HorizontalPanel();	
      hPanel.add(lblName);
      hPanel.add(txtName);
      hPanel.setCellWidth(lblName, "130");

      VerticalPanel vPanel = new VerticalPanel();
      vPanel.setSpacing(10);
      vPanel.add(hPanel);
      vPanel.add(buttonMessage);
      vPanel.setCellHorizontalAlignment(buttonMessage, 
      HasHorizontalAlignment.ALIGN_RIGHT);

      DecoratorPanel panel = new DecoratorPanel();
      panel.add(vPanel);

      // Add widgets to the root panel.
      RootPanel.get("gwtContainer").add(panel);
   }  
   
   public String getGreeting(String name){
      return "Hello "+name+"!";
   }
}

Reemplace el contenido de HelloWorldTest.java en test/com.tutorialspoint/client paquete con lo siguiente

package com.tutorialspoint.client;

import com.tutorialspoint.shared.FieldVerifier;
import com.google.gwt.core.client.GWT;
import com.google.gwt.junit.client.GWTTestCase;
import com.google.gwt.user.client.rpc.AsyncCallback;
import com.google.gwt.user.client.rpc.ServiceDefTarget;

/**
 * GWT JUnit tests must extend GWTTestCase.
 */
public class HelloWorldTest extends GWTTestCase {

   /**
    * must refer to a valid module that sources this class.
    */
   public String getModuleName() {
      return "com.tutorialspoint.HelloWorldJUnit";
   }

   /**
    * tests the FieldVerifier.
    */
   public void testFieldVerifier() {
      assertFalse(FieldVerifier.isValidName(null));
      assertFalse(FieldVerifier.isValidName(""));
      assertFalse(FieldVerifier.isValidName("a"));
      assertFalse(FieldVerifier.isValidName("ab"));
      assertFalse(FieldVerifier.isValidName("abc"));
      assertTrue(FieldVerifier.isValidName("abcd"));
   }

   /**
      * this test will send a request to the server using the greetServer
      *  method in GreetingService and verify the response.
   */
   public void testGreetingService() {
      /* create the service that we will test. */
      GreetingServiceAsync greetingService = 
      GWT.create(GreetingService.class);
      ServiceDefTarget target = (ServiceDefTarget) greetingService;
      target.setServiceEntryPoint(GWT.getModuleBaseURL() 
      + "helloworld/greet");

      /* since RPC calls are asynchronous, we will need to wait 
       for a response after this test method returns. This line 
       tells the test runner to wait up to 10 seconds 
       before timing out. */
      delayTestFinish(10000);

      /* send a request to the server. */
      greetingService.greetServer("GWT User", 
         new AsyncCallback<String>() {
         public void onFailure(Throwable caught) {
            /* The request resulted in an unexpected error. */
            fail("Request failure: " + caught.getMessage());
         }

         public void onSuccess(String result) {
            /* verify that the response is correct. */
            assertTrue(result.startsWith("Hello, GWT User!"));

            /* now that we have received a response, we need to 
             tell the test runner that the test is complete. 
             You must call finishTest() after an asynchronous test 
             finishes successfully, or the test will time out.*/
            finishTest();
         }
      });
	
      /**
         * tests the getGreeting method.
      */
      public void testGetGreeting() {
         HelloWorld helloWorld = new HelloWorld();
         String name = "Robert";
         String expectedGreeting = "Hello "+name+"!";
         assertEquals(expectedGreeting,helloWorld.getGreeting(name));
      }
   }
}

Ejecute casos de prueba en Eclipse utilizando configuraciones de lanzamiento generadas

Ejecutaremos pruebas unitarias en Eclipse utilizando las configuraciones de lanzamiento generadas por webAppCreator tanto para el modo de desarrollo como para el modo de producción.

Ejecute la prueba JUnit en modo de desarrollo

  • En la barra de menú de Eclipse, seleccione Ejecutar → Ejecutar configuraciones ...
  • En la sección JUnit, seleccione HelloWorldTest-dev
  • Para guardar los cambios en los argumentos, presione Aplicar
  • Para ejecutar la prueba, presione Ejecutar

Si todo está bien con su aplicación, esto producirá el siguiente resultado:

Ejecute la prueba JUnit en modo de producción

  • En la barra de menú de Eclipse, seleccione Ejecutar → Ejecutar configuraciones ...
  • En la sección JUnit, seleccione HelloWorldTest-prod
  • Para guardar los cambios en los argumentos, presione Aplicar
  • Para ejecutar la prueba, presione Ejecutar

Si todo está bien con su aplicación, esto producirá el siguiente resultado:

GWT proporciona una excelente capacidad para depurar el código del lado del cliente y del lado del servidor.

Durante el modo de desarrollo, la aplicación GWT se basa en código Java y no se traduce a JavaScript.

Cuando una aplicación se ejecuta en modo de desarrollo, la máquina virtual Java (JVM) en realidad está ejecutando el código de la aplicación como código de bytes Java compilado, utilizando la capacidad de GWT para conectarse a una ventana del navegador.

GWT utiliza un complemento basado en navegador para conectarse a JVM.

Por lo tanto, los desarrolladores pueden usar cualquier IDE basado en Java para depurar tanto el código GWT del lado del cliente como el código del lado del servidor.

En este artículo, demostraremos el uso de la depuración del código del cliente GWT utilizando Eclipse. Haremos las siguientes tareas:

  • Establezca puntos de interrupción en el código y véalos en BreakPoint Explorer.
  • Recorra el código línea por línea durante la depuración.
  • Ver los valores de la variable.
  • Inspeccione los valores de todas las variables.
  • Inspecciona el valor de una expresión.
  • Muestre el marco de la pila para hilos suspendidos.

Ejemplo de depuración

Este ejemplo lo llevará a través de sencillos pasos para demostrar la depuración de una aplicación GWT. Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>

   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

.gwt-Label{ 
   font-size: 150%; 
   font-weight: bold;
   color:red;
   padding:5px;
   margin:5px;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html para acomodar dos botones.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>

   <body>
      <h1>Debugging Application Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java con el cual demostraremos la capacidad de depuración del código GWT.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.event.dom.client.KeyCodes;
import com.google.gwt.event.dom.client.KeyUpEvent;
import com.google.gwt.event.dom.client.KeyUpHandler;
import com.google.gwt.user.client.Window;
import com.google.gwt.user.client.ui.Button;
import com.google.gwt.user.client.ui.DecoratorPanel;
import com.google.gwt.user.client.ui.HasHorizontalAlignment;
import com.google.gwt.user.client.ui.HorizontalPanel;
import com.google.gwt.user.client.ui.Label;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TextBox;
import com.google.gwt.user.client.ui.VerticalPanel;

public class HelloWorld implements EntryPoint {
	
   public void onModuleLoad() {
      /*create UI */
      final TextBox txtName = new TextBox(); 
      txtName.setWidth("200");
      txtName.addKeyUpHandler(new KeyUpHandler() {
         @Override
         public void onKeyUp(KeyUpEvent event) {
            if(event.getNativeKeyCode() == KeyCodes.KEY_ENTER){
               Window.alert(getGreeting(txtName.getValue()));
            }				
         }
      });
      Label lblName = new Label("Enter your name: ");

      Button buttonMessage = new Button("Click Me!");

      buttonMessage.addClickHandler(new ClickHandler() {			
      @Override
      public void onClick(ClickEvent event) {
         Window.alert(getGreeting(txtName.getValue()));
      }});

      HorizontalPanel hPanel = new HorizontalPanel();	
      hPanel.add(lblName);
      hPanel.add(txtName);
      hPanel.setCellWidth(lblName, "130");

      VerticalPanel vPanel = new VerticalPanel();
      vPanel.setSpacing(10);
      vPanel.add(hPanel);
      vPanel.add(buttonMessage);
      vPanel.setCellHorizontalAlignment(buttonMessage, 
      HasHorizontalAlignment.ALIGN_RIGHT);

      DecoratorPanel panel = new DecoratorPanel();
      panel.add(vPanel);

      // Add widgets to the root panel.
      RootPanel.get("gwtContainer").add(panel);
   }  
   
   public String getGreeting(String name){
      return "Hello "+name+"!";
   }
}

Paso 1 - Coloque BreakPoints

Coloque un punto de interrupción en la primera línea de onModuleLoad() de HelloWorld.java

Paso 2: Aplicación de depuración

Ahora haga clic en el

menú de la aplicación Depurar y seleccioneHelloWorld aplicación para depurar la aplicación.

Si todo está bien, debe ver el Modo de desarrollo de GWT activo en Eclipse que contiene una URL como se muestra a continuación. Haga doble clic en la URL para abrir la aplicación GWT.

Tan pronto como se inicie la aplicación, verá el enfoque en el punto de interrupción de Eclipse, ya que hemos colocado el punto de interrupción en la primera línea del método de punto de entrada.

Puede ver el seguimiento de pila para subprocesos suspendidos.

Puede ver los valores de las expresiones.

Puede ver la lista de puntos de interrupción colocados.

Ahora siga presionando F6 hasta llegar a la última línea del método onModuleLoad (). Como referencia para las teclas de función, F6 inspecciona el código línea por línea, F5 avanza hacia adentro y F8 reanuda la aplicación. Ahora puede ver la lista de valores de todas las variables del método onModuleLoad ().

El código de cliente GWT se puede depurar de la misma forma que se puede depurar una aplicación Java. Coloque puntos de interrupción en cualquier línea y juegue con las capacidades de depuración de GWT.

GWT proporciona tres formas de internacionalizar una aplicación GWT. Demostraremos el uso de la internacionalización de cadenas estáticas que se usa más comúnmente entre los proyectos.

No Señor. Técnica y descripción
1

Static String Internationalization

Esta técnica es la más frecuente y requiere muy poca sobrecarga en tiempo de ejecución; es una técnica muy eficaz para traducir tanto cadenas constantes como parametrizadas; la más sencilla de implementar.

La internacionalización de cadenas estáticas utiliza archivos de propiedades Java estándar para almacenar cadenas traducidas y mensajes parametrizados, y se crean interfaces Java fuertemente tipadas para recuperar sus valores.

2

Dynamic String Internationalization

Esta técnica es muy flexible pero más lenta que la internacionalización de cadenas estáticas. La página del host contiene las cadenas localizadas, por lo tanto, no es necesario volver a compilar las aplicaciones cuando agregamos una nueva configuración regional. Si la aplicación GWT se va a integrar con un sistema de localización del lado del servidor existente, entonces se debe utilizar esta técnica.

3

Localizable Interface

Esta técnica es la más poderosa entre las tres técnicas. La implementación de Localizable nos permite crear versiones localizadas de tipos personalizados. Es una técnica de internacionalización avanzada.

Flujo de trabajo de internacionalización de una aplicación GWT

Paso 1: crear archivos de propiedades

Cree un archivo de propiedades que contenga los mensajes que se utilizarán en la aplicación. Hemos creado unHelloWorldMessages.properties archivo en nuestro ejemplo.

enterName = Enter your name
clickMe = Click Me
applicationTitle = Application Internationalization Demonstration
greeting = Hello {0}

Cree archivos de propiedades que contengan valores traducidos específicos de la configuración regional. Hemos creado unHelloWorldMessages_de.propertiesarchivo en nuestro ejemplo. Este archivo contiene traducciones en idioma alemán. _de especifica la configuración regional alemana y vamos a admitir el idioma alemán en nuestra aplicación.

Si está creando un archivo de propiedades usando Eclipse, cambie la codificación del archivo a UTF-8. Seleccione el archivo y luego haga clic derecho en él para abrir su ventana de propiedades. Seleccione Codificación del archivo de texto como Other UTF-8. Aplicar y guardar el cambio.

enterName = Geben Sie Ihren Namen
clickMe = Klick mich
applicationTitle = Anwendung Internationalisierung Demonstration
greeting = Hallo {0}

Paso 2: agregue el módulo i18n al archivo XML del descriptor del módulo

Actualizar archivo de módulo HelloWorld.gwt.xml para incluir soporte para la configuración regional alemana

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   ...
   <extend-property name = "locale" values="de" />
   ...
</module>

Paso 3: crear una interfaz equivalente al archivo de propiedades

Cree la interfaz HelloWorldMessages.java ampliando la interfaz de Mensajes de GWT para incluir soporte para la internalización. Debe contener los mismos nombres de método que las claves en el archivo de propiedades. El marcador de posición se reemplazaría con un argumento de cadena.

public interface HelloWorldMessages extends Messages {
	
   @DefaultMessage("Enter your name")
   String enterName();
 
   @DefaultMessage("Click Me")
   String clickMe();
 
   @DefaultMessage("Application Internalization Demonstration")
   String applicationTitle();

   @DefaultMessage("Hello {0}")
   String greeting(String name);
}

Paso 4: use la interfaz de mensajes en el componente UI.

Usa el objeto de HelloWorldMessages en HelloWorld para recibir los mensajes.

public class HelloWorld implements EntryPoint {
   
   /* create an object of HelloWorldMessages interface 
      using GWT.create() method */
   private HelloWorldMessages messages = 
   GWT.create(HelloWorldMessages.class);
   
   public void onModuleLoad() {
   ...
      Label titleLabel = new Label(messages.applicationTitle());
      //Add title to the application
      RootPanel.get("gwtAppTitle").add(titleLabel);
   ...
   }
}

Internacionalización - Ejemplo completo

Este ejemplo lo llevará a través de sencillos pasos para demostrar la capacidad de internacionalización de una aplicación GWT.

Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
  <!-- Inherit the core Web Toolkit stuff.                        -->
  <inherits name = 'com.google.gwt.user.User'/>

  <!-- Inherit the default GWT style sheet.                       -->
  <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

  <!-- Specify the app entry point class.                         -->
  <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>
  <extend-property name = "locale" values="de" />
  <!-- Specify the paths for translatable code                    -->
  <source path = 'client'/>
  <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html.

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>
   <body>

      <h1 id = "gwtAppTitle"></h1>
      <div id = "gwtContainer"></div>

   </body>
</html>

Ahora cree el archivo HelloWorldMessages.properties en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

enterName = Enter your name
clickMe = Click Me
applicationTitle = Application Internationalization Demonstration
greeting = Hello {0}

Ahora cree el archivo HelloWorldMessages_de.properties en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

enterName = Geben Sie Ihren Namen
clickMe = Klick mich
applicationTitle = Anwendung Internationalisierung Demonstration
greeting = Hallo {0}

Ahora cree la clase HelloWorldMessages.java en el src/com.tutorialspoint/client empaque y coloque el siguiente contenido en él

package com.tutorialspoint.client;
import com.google.gwt.i18n.client.Messages;

public interface HelloWorldMessages extends Messages {	
   @DefaultMessage("Enter your name")
   String enterName();
 
   @DefaultMessage("Click Me")
   String clickMe();
 
   @DefaultMessage("Application Internationalization Demonstration")
   String applicationTitle();

   @DefaultMessage("Hello {0}")
   String greeting(String name);
}

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java mediante el cual demostraremos la capacidad de internacionalización del código GWT.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;
import com.google.gwt.core.client.GWT;

import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.event.dom.client.KeyCodes;
import com.google.gwt.event.dom.client.KeyUpEvent;
import com.google.gwt.event.dom.client.KeyUpHandler;

import com.google.gwt.user.client.Window;
import com.google.gwt.user.client.ui.Button;
import com.google.gwt.user.client.ui.DecoratorPanel;
import com.google.gwt.user.client.ui.HasHorizontalAlignment;
import com.google.gwt.user.client.ui.HorizontalPanel;
import com.google.gwt.user.client.ui.Label;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TextBox;
import com.google.gwt.user.client.ui.VerticalPanel;

public class HelloWorld implements EntryPoint {

   /* create an object of HelloWorldMessages interface 
      using GWT.create() method */
   private HelloWorldMessages messages = 
   GWT.create(HelloWorldMessages.class);
   
   public void onModuleLoad() {
      /*create UI */
      final TextBox txtName = new TextBox(); 
      txtName.setWidth("200");
      txtName.addKeyUpHandler(new KeyUpHandler() {
         @Override
         public void onKeyUp(KeyUpEvent event) {
            if(event.getNativeKeyCode() == KeyCodes.KEY_ENTER){
               Window.alert(getGreeting(txtName.getValue()));
            }				
         }
      });
      Label lblName = new Label(messages.enterName() + ": ");

      Button buttonMessage = new Button(messages.clickMe() + "!");

      buttonMessage.addClickHandler(new ClickHandler() {			
         @Override
         public void onClick(ClickEvent event) {
            Window.alert(getGreeting(txtName.getValue()));
         }
      });

      HorizontalPanel hPanel = new HorizontalPanel();	
      hPanel.add(lblName);
      hPanel.add(txtName);      

      VerticalPanel vPanel = new VerticalPanel();
      vPanel.setSpacing(10);
      vPanel.add(hPanel);
      vPanel.add(buttonMessage);
      vPanel.setCellHorizontalAlignment(buttonMessage, 
      HasHorizontalAlignment.ALIGN_RIGHT);

      DecoratorPanel panel = new DecoratorPanel();
      panel.add(vPanel);
      Label titleLabel = new Label(messages.applicationTitle());
      //Add title to the application
      RootPanel.get("gwtAppTitle").add(titleLabel);
      // Add widgets to the root panel.
      RootPanel.get("gwtContainer").add(panel);
   }  
   
   public String getGreeting(String name){
      return messages.greeting(name + "!");
   }
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

Ahora actualice la URL para que contenga locale = de.Set URL - http://127.0.0.1:8888/HelloWorld.html?gwt.codesvr=127.0.0.1:9997&locale=de. Si todo está bien con su aplicación, esto producirá el siguiente resultado:

Las aplicaciones GWT son normalmente aplicaciones de una sola página que ejecutan JavaScripts y no contienen muchas páginas, por lo que el navegador no realiza un seguimiento de la interacción del usuario con la aplicación. Para utilizar la funcionalidad del historial del navegador, la aplicación debe generar un fragmento de URL único para cada página navegable.

GWT proporciona History Mechanism para manejar esta situación.

GWT usa un término tokenque es simplemente una cadena que la aplicación puede analizar para volver a un estado particular. La aplicación guardará este token en el historial del navegador como un fragmento de URL.

Por ejemplo, un token de historial llamado "pageIndex1" se agregaría a una URL de la siguiente manera:

http://www.tutorialspoint.com/HelloWorld.html#pageIndex0

Flujo de trabajo de gestión de historial

Paso 1: habilitar la compatibilidad con el historial

Para utilizar la compatibilidad con el historial de GWT, primero debemos incrustar el siguiente iframe en nuestra página HTML de host.

<iframe src = "javascript:''"
   id = "__gwt_historyFrame"
   style = "width:0;height:0;border:0"></iframe>

Paso 2: agregar token al historial

Siguiendo las estadísticas de ejemplo sobre cómo agregar token al historial del navegador

int index = 0;
History.newItem("pageIndex" + index);

Paso 3: recuperar el token del historial

Cuando el usuario usa el botón Atrás / Adelante del navegador, recuperaremos el token y actualizaremos el estado de nuestra aplicación en consecuencia.

History.addValueChangeHandler(new ValueChangeHandler<String>() {
   @Override
   public void onValueChange(ValueChangeEvent<String> event) {
      String historyToken = event.getValue();
      /* parse the history token */
      try {
         if (historyToken.substring(0, 9).equals("pageIndex")) {
            String tabIndexToken = historyToken.substring(9, 10);
            int tabIndex = Integer.parseInt(tabIndexToken);
            /* select the specified tab panel */
            tabPanel.selectTab(tabIndex);
         } else {
            tabPanel.selectTab(0);
         }
      } catch (IndexOutOfBoundsException e) {
         tabPanel.selectTab(0);
      }
   }
});

Ahora veamos la clase de historia en acción.

Clase de historia - Ejemplo completo

Este ejemplo le llevará a través de sencillos pasos para demostrar la gestión del historial de una aplicación GWT. Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>  
   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>
   
   <body>
      <iframe src = "javascript:''"id = "__gwt_historyFrame"
         style = "width:0;height:0;border:0"></iframe>
      <h1> History Class Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java mediante el cual demostraremos la gestión de historial en código GWT.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;

import com.google.gwt.event.logical.shared.SelectionEvent;
import com.google.gwt.event.logical.shared.SelectionHandler;
import com.google.gwt.event.logical.shared.ValueChangeEvent;
import com.google.gwt.event.logical.shared.ValueChangeHandler;

import com.google.gwt.user.client.History;
import com.google.gwt.user.client.ui.HTML;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TabPanel;

public class HelloWorld implements EntryPoint {

   /**
    * This is the entry point method.
    */
   public void onModuleLoad() {
      /* create a tab panel to carry multiple pages */  
      final TabPanel tabPanel = new TabPanel();

      /* create pages */
      HTML firstPage = new HTML("<h1>We are on first Page.</h1>");
      HTML secondPage = new HTML("<h1>We are on second Page.</h1>");
      HTML thirdPage = new HTML("<h1>We are on third Page.</h1>");

      String firstPageTitle = "First Page";
      String secondPageTitle = "Second Page";
      String thirdPageTitle = "Third Page";
      tabPanel.setWidth("400");
      
	  /* add pages to tabPanel*/
      tabPanel.add(firstPage, firstPageTitle);
      tabPanel.add(secondPage,secondPageTitle);
      tabPanel.add(thirdPage, thirdPageTitle);

      /* add tab selection handler */
      tabPanel.addSelectionHandler(new SelectionHandler<Integer>() {
         @Override
         public void onSelection(SelectionEvent<Integer> event) {
            /* add a token to history containing pageIndex 
             History class will change the URL of application
             by appending the token to it.
            */
            History.newItem("pageIndex" + event.getSelectedItem());
         }
      });
      
      /* add value change handler to History 
       this method will be called, when browser's 
       Back button or Forward button are clicked 
       and URL of application changes.
       */
      History.addValueChangeHandler(new ValueChangeHandler<String>() {
         @Override
         public void onValueChange(ValueChangeEvent<String> event) {
            String historyToken = event.getValue();
            /* parse the history token */
            try {
               if (historyToken.substring(0, 9).equals("pageIndex")) {
                  String tabIndexToken = historyToken.substring(9, 10);
                  int tabIndex = Integer.parseInt(tabIndexToken);
                  /* select the specified tab panel */
                  tabPanel.selectTab(tabIndex);
               } else {
                  tabPanel.selectTab(0);
               }
            } catch (IndexOutOfBoundsException e) {
               tabPanel.selectTab(0);
            }
         }
      });

      /* select the first tab by default */
      tabPanel.selectTab(0);

      /* add controls to RootPanel */
      RootPanel.get().add(tabPanel);
   }
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

  • Ahora haga clic en cada pestaña para seleccionar diferentes páginas.

  • Debería notar que, cuando se selecciona cada pestaña, se cambia la URL de la aplicación y se agrega #pageIndex a la URL.

  • También puede ver que los botones de avance y retroceso del navegador están habilitados ahora.

  • Use el botón Atrás y Adelante del navegador y verá que las diferentes pestañas se seleccionan en consecuencia.

GWT admite la gestión del historial del navegador mediante una clase de historial para la que puede consultar el capítulo GWT - Clase de historial .

GWT usa un término tokenque es simplemente una cadena que la aplicación puede analizar para volver a un estado particular. La aplicación guardará este token en el historial del navegador como un fragmento de URL.

En el capítulo GWT - History Class , manejamos la creación del token y la configuración en el historial escribiendo código.

En este artículo, discutiremos un hipervínculo de widget especial que realiza la creación de tokens y la administración del historial por nosotros automáticamente y brinda a la aplicación la capacidad de marcar.

Ejemplo de marcadores

Este ejemplo lo llevará a través de sencillos pasos para demostrar la creación de marcadores de una aplicación GWT.

Los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>

   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>  
   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>

</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>
   
   <body>
      <iframe src = "javascript:''"id = "__gwt_historyFrame"
         style = "width:0;height:0;border:0"></iframe>
      <h1> Bookmarking Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java mediante el cual demostraremos el uso de marcadores en código GWT.

package com.tutorialspoint.client;

import com.google.gwt.core.client.EntryPoint;

import com.google.gwt.event.logical.shared.ValueChangeEvent;
import com.google.gwt.event.logical.shared.ValueChangeHandler;

import com.google.gwt.user.client.History;
import com.google.gwt.user.client.ui.HTML;
import com.google.gwt.user.client.ui.HorizontalPanel;
import com.google.gwt.user.client.ui.Hyperlink;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TabPanel;
import com.google.gwt.user.client.ui.VerticalPanel;

public class HelloWorld implements EntryPoint {
   
   private TabPanel tabPanel;

   private void selectTab(String historyToken){
      /*  parse the history token */
      try {
         if (historyToken.substring(0, 9).equals("pageIndex")) {
            String tabIndexToken = historyToken.substring(9, 10);
            int tabIndex = Integer.parseInt(tabIndexToken);
            /* Select the specified tab panel */
            tabPanel.selectTab(tabIndex);
         } else {
            tabPanel.selectTab(0);
         }
      } catch (IndexOutOfBoundsException e) {
         tabPanel.selectTab(0);
      }
   }

   /**
    * This is the entry point method.
    */
   public void onModuleLoad() {
      /* create a tab panel to carry multiple pages */  
      tabPanel = new TabPanel();

      /* create pages */
      HTML firstPage = new HTML("<h1>We are on first Page.</h1>");
      HTML secondPage = new HTML("<h1>We are on second Page.</h1>");
      HTML thirdPage = new HTML("<h1>We are on third Page.</h1>");

      String firstPageTitle = "First Page";
      String secondPageTitle = "Second Page";
      String thirdPageTitle = "Third Page";

      Hyperlink firstPageLink = new Hyperlink("1", "pageIndex0");
      Hyperlink secondPageLink = new Hyperlink("2", "pageIndex1");
      Hyperlink thirdPageLink = new Hyperlink("3", "pageIndex2");

      HorizontalPanel linksHPanel = new HorizontalPanel();
      linksHPanel.setSpacing(10);
      linksHPanel.add(firstPageLink);
      linksHPanel.add(secondPageLink);
      linksHPanel.add(thirdPageLink);		

      /* If the application starts with no history token, 
         redirect to a pageIndex0 */
      String initToken = History.getToken();

      if (initToken.length() == 0) {
         History.newItem("pageIndex0");
         initToken = "pageIndex0";
      }		

      tabPanel.setWidth("400");
      /* add pages to tabPanel*/
      tabPanel.add(firstPage, firstPageTitle);
      tabPanel.add(secondPage,secondPageTitle);
      tabPanel.add(thirdPage, thirdPageTitle);

      /* add value change handler to History 
       * this method will be called, when browser's Back button 
       * or Forward button are clicked.
       * and URL of application changes.
       * */
      History.addValueChangeHandler(new ValueChangeHandler<String>() {
         @Override
         public void onValueChange(ValueChangeEvent<String> event) {
            selectTab(event.getValue());				
         }
      });

      selectTab(initToken);

      VerticalPanel vPanel = new VerticalPanel();

      vPanel.setSpacing(10);
      vPanel.add(tabPanel);
      vPanel.add(linksHPanel);

      /* add controls to RootPanel */
      RootPanel.get().add(vPanel);
   }
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

  • Ahora haga clic en 1, 2 o 3. Puede notar que la pestaña cambia con los índices.

  • Debería notar que, cuando hace clic en 1,2 o 3, la URL de la aplicación cambia y #pageIndex se agrega a la URL

  • También puede ver que los botones de avance y retroceso del navegador están habilitados ahora.

  • Use el botón Atrás y Adelante del navegador y verá que las diferentes pestañas se seleccionan en consecuencia.

  • Haga clic derecho en 1, 2 o 3. Puede ver opciones como abrir, abrir en una nueva ventana, abrir en una nueva pestaña, agregar a favoritos, etc.

  • Haga clic derecho en 3. Elija agregar a favoritos. Guardar el marcador como página 3.

  • Abra favoritos y elija la página 3. Verá la tercera pestaña seleccionada.

El marco de registro emula java.util.logging, por lo que usa la misma sintaxis y tiene el mismo comportamiento que el código de registro del lado del servidor

El registro de GWT se configura mediante archivos .gwt.xml.

Podemos configurar el registro para que esté habilitado / deshabilitado; podemos habilitar / deshabilitar manejadores particulares y cambiar el nivel de registro predeterminado.

Tipos de registrador

Los registradores están organizados en una estructura de árbol, con el registrador raíz en la raíz del árbol.

El nombre del registrador determina las relaciones padre / hijo usando . para separar secciones del nombre.

Como ejemplo, si tenemos dos registradores Hospital.room1 y Hospital.room2, entonces son hermanos, y su padre es el registrador llamado Hospital. El registrador del hospital (y cualquier registrador con un nombre que no contenga un punto ".") Tiene el registrador raíz como padre.

private static Logger room1Logger = Logger.getLogger("Hospital.room1");
private static Logger room2Logger = Logger.getLogger("Hospital.room2");
private static Logger hospitalLogger = Logger.getLogger("Hospital");
private static Logger rootLogger = Logger.getLogger("");

Manejadores de troncos

GWT proporciona controladores predeterminados que mostrarán las entradas de registro realizadas mediante registradores.

Manipulador Registros a Descripción
SystemLogHandler stdout Estos mensajes solo se pueden ver en el modo de desarrollo en la ventana DevMode.
DevelopmentModeLogHandler Ventana DevMode Registros llamando al método GWT.log. Estos mensajes solo se pueden ver en el modo de desarrollo en la ventana DevMode.
ConsoleLogHandler consola javascript Se registra en la consola de JavaScript, que es utilizada por Firebug Lite (para IE), Safari y Chrome.
FirebugLogHandler Firebug Inicia sesión en la consola de Firebug.
PopupLogHandler surgir Inicia sesión en la ventana emergente que se encuentra en la esquina superior izquierda de la aplicación cuando este controlador está habilitado.
SimpleRemoteLogHandler servidor Este controlador envía mensajes de registro al servidor, donde se registrarán mediante el mecanismo de registro del lado del servidor.

Configurar el registro en la aplicación GWT

El archivo HelloWorld.gwt.xml debe configurarse para habilitar el registro de GWT de la siguiente manera:

# add logging module
   <inherits name = "com.google.gwt.logging.Logging"/>                
# To change the default logLevel 
   <set-property name = "gwt.logging.logLevel" value = "SEVERE"/>  
# To enable logging   
   <set-property name = "gwt.logging.enabled" value = "TRUE"/>       
# To disable a popup Handler   
   <set-property name = "gwt.logging.popupHandler" value = "DISABLED" />

Utilice el registrador para registrar las acciones del usuario

/* Create Root Logger */
private static Logger rootLogger = Logger.getLogger("");
...
rootLogger.log(Level.SEVERE, "pageIndex selected: " + event.getValue());
...

Ejemplo de marco de registro

Este ejemplo lo llevará a través de sencillos pasos para demostrar la capacidad de registro de una aplicación GWT. Siga los siguientes pasos para actualizar la aplicación GWT que creamos en GWT - Capítulo Crear aplicación -

Paso Descripción
1 Cree un proyecto con un nombre HelloWorld en un paquete com.tutorialspoint como se explica en el capítulo GWT - Crear aplicación .
2 Modifique HelloWorld.gwt.xml , HelloWorld.css , HelloWorld.html y HelloWorld.java como se explica a continuación. Mantenga el resto de los archivos sin cambios.
3 Compile y ejecute la aplicación para verificar el resultado de la lógica implementada.

A continuación se muestra el contenido del descriptor de módulo modificado src/com.tutorialspoint/HelloWorld.gwt.xml.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
   <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>
   <inherits name = "com.google.gwt.logging.Logging"/>
   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>  
   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>
   <set-property name = "gwt.logging.logLevel" value="SEVERE"/>          
   <set-property name = "gwt.logging.enabled" value = "TRUE"/>            
   <set-property name = "gwt.logging.popupHandler" value=  "DISABLED" />
</module>

A continuación se muestra el contenido del archivo de hoja de estilo modificado war/HelloWorld.css.

body {
   text-align: center;
   font-family: verdana, sans-serif;
}

h1 {
   font-size: 2em;
   font-weight: bold;
   color: #777777;
   margin: 40px 0px 70px;
   text-align: center;
}

A continuación se muestra el contenido del archivo de host HTML modificado war/HelloWorld.html

<html>
   <head>
      <title>Hello World</title>
      <link rel = "stylesheet" href = "HelloWorld.css"/>
      <script language = "javascript" src = "helloworld/helloworld.nocache.js">
      </script>
   </head>
   
   <body>
      <iframe src = "javascript:''"id = "__gwt_historyFrame"
         style = "width:0;height:0;border:0"></iframe>
      <h1> Logging Demonstration</h1>
      <div id = "gwtContainer"></div>
   </body>
</html>

Tengamos el siguiente contenido del archivo Java src/com.tutorialspoint/HelloWorld.java mediante el cual demostraremos el uso de marcadores en código GWT.

package com.tutorialspoint.client;

import java.util.logging.Level;
import java.util.logging.Logger;

import com.google.gwt.core.client.EntryPoint;

import com.google.gwt.event.logical.shared.ValueChangeEvent;
import com.google.gwt.event.logical.shared.ValueChangeHandler;

import com.google.gwt.logging.client.HasWidgetsLogHandler;

import com.google.gwt.user.client.History;
import com.google.gwt.user.client.ui.HTML;
import com.google.gwt.user.client.ui.HorizontalPanel;
import com.google.gwt.user.client.ui.Hyperlink;
import com.google.gwt.user.client.ui.RootPanel;
import com.google.gwt.user.client.ui.TabPanel;
import com.google.gwt.user.client.ui.VerticalPanel;

public class HelloWorld implements EntryPoint {
   
   private TabPanel tabPanel;
   /* Create Root Logger */
   private static Logger rootLogger = Logger.getLogger("");
   private VerticalPanel customLogArea;

   private void selectTab(String historyToken){
      /*  parse the history token */
      try {
         if (historyToken.substring(0, 9).equals("pageIndex")) {
            String tabIndexToken = historyToken.substring(9, 10);
            int tabIndex = Integer.parseInt(tabIndexToken);
            /* Select the specified tab panel */
            tabPanel.selectTab(tabIndex);
         } else {
            tabPanel.selectTab(0);
         }
      } catch (IndexOutOfBoundsException e) {
         tabPanel.selectTab(0);
      }
   }

   /**
    * This is the entry point method.
    */
   public void onModuleLoad() {
      /* create a tab panel to carry multiple pages */  
      tabPanel = new TabPanel();

      /* create pages */
      HTML firstPage = new HTML("<h1>We are on first Page.</h1>");
      HTML secondPage = new HTML("<h1>We are on second Page.</h1>");
      HTML thirdPage = new HTML("<h1>We are on third Page.</h1>");

      String firstPageTitle = "First Page";
      String secondPageTitle = "Second Page";
      String thirdPageTitle = "Third Page";

      Hyperlink firstPageLink = new Hyperlink("1", "pageIndex0");
      Hyperlink secondPageLink = new Hyperlink("2", "pageIndex1");
      Hyperlink thirdPageLink = new Hyperlink("3", "pageIndex2");

      HorizontalPanel linksHPanel = new HorizontalPanel();
      linksHPanel.setSpacing(10);
      linksHPanel.add(firstPageLink);
      linksHPanel.add(secondPageLink);
      linksHPanel.add(thirdPageLink);		

      /* If the application starts with no history token, 
         redirect to a pageIndex0 */
      String initToken = History.getToken();

      if (initToken.length() == 0) {
         History.newItem("pageIndex0");
         initToken = "pageIndex0";
      }		

      tabPanel.setWidth("400");
      /* add pages to tabPanel*/
      tabPanel.add(firstPage, firstPageTitle);
      tabPanel.add(secondPage,secondPageTitle);
      tabPanel.add(thirdPage, thirdPageTitle);

      /* add value change handler to History 
       * this method will be called, when browser's Back button 
       * or Forward button are clicked.
       * and URL of application changes.
       * */
      History.addValueChangeHandler(new ValueChangeHandler<String>() {
         @Override
         public void onValueChange(ValueChangeEvent<String> event) {
            selectTab(event.getValue());	
            rootLogger.log(Level.SEVERE, "pageIndex selected: " 
            + event.getValue());			
         }
      });

      selectTab(initToken);

      VerticalPanel vPanel = new VerticalPanel();

      vPanel.setSpacing(10);
      vPanel.add(tabPanel);
      vPanel.add(linksHPanel);
	  
      customLogArea = new VerticalPanel();	   
      vPanel.add(customLogArea);

      /* an example of using own custom logging area. */
      rootLogger.addHandler(new HasWidgetsLogHandler(customLogArea));

      /* add controls to RootPanel */
      RootPanel.get().add(vPanel);
   }
}

Una vez que esté listo con todos los cambios realizados, compilemos y ejecutemos la aplicación en modo de desarrollo como hicimos en el capítulo GWT - Crear aplicación . Si todo está bien con su aplicación, esto producirá el siguiente resultado:

Ahora haga clic en 1, 2 o 3. Puede notar que cuando hace clic en 1,2 o 3, puede ver que el registro se imprime mostrando el índice de página. Verifique la salida de la consola en Eclipse. También puede ver que el registro se imprime en la consola de Eclipse.

Fri Aug 31 11:42:35 IST 2012 
SEVERE: pageIndex selected: pageIndex0
Fri Aug 31 11:42:37 IST 2012 
SEVERE: pageIndex selected: pageIndex1
Fri Aug 31 11:42:38 IST 2012 
SEVERE: pageIndex selected: pageIndex2
Fri Aug 31 11:42:40 IST 2012 
SEVERE: pageIndex selected: pageIndex0
Fri Aug 31 11:42:41 IST 2012 
SEVERE: pageIndex selected: pageIndex1
Fri Aug 31 11:42:41 IST 2012 
SEVERE: pageIndex selected: pageIndex2

Ahora actualice el descriptor del módulo src/com.tutorialspoint/HelloWorld.gwt.xml para habilitar popupHandler.

<?xml version = "1.0" encoding = "UTF-8"?>
<module rename-to = 'helloworld'>
  <!-- Inherit the core Web Toolkit stuff.                        -->
   <inherits name = 'com.google.gwt.user.User'/>

   <!-- Inherit the default GWT style sheet.                       -->
   <inherits name = 'com.google.gwt.user.theme.clean.Clean'/>
   <inherits name = "com.google.gwt.logging.Logging"/>
   <!-- Specify the app entry point class.                         -->
   <entry-point class = 'com.tutorialspoint.client.HelloWorld'/>  
   <!-- Specify the paths for translatable code                    -->
   <source path = 'client'/>
   <source path = 'shared'/>
   <set-property name = "gwt.logging.logLevel" value = "SEVERE"/>          
   <set-property name = "gwt.logging.enabled" value = "TRUE"/>            
   <set-property name="gwt.logging.popupHandler" value = "ENABLED" />
</module>

Una vez que esté listo con todos los cambios realizados, vuelva a cargar la aplicación actualizando la ventana del navegador (presione F5 / botón de recarga del navegador). Observe que ahora hay una ventana emergente en la esquina superior izquierda de la aplicación.

Ahora haga clic en 1, 2 o 3. Puede notar, cuando hace clic en 1,2 o 3, puede ver que el registro se imprime mostrando el pageIndex en la ventana emergente.