Libro GeneXus
Transcripción
Libro GeneXus
2 GeneXus X – Episodio Uno © Cecilia Fernández – Daniel Márquez Lisboa Diseño de tapa: Fiorella Franzini Aporte artístico: María Inés Carriquiry 3 Prólogo a la 2da. Edición Todo el equipo de Artech ha trabajado mucho en la versión X de GeneXus. Ésta no es, simplemente, una versión más: representa un gran salto adelante en muchos aspectos y, muy especialmente, en su capacidad de integración y de extensión, ya sea por Artech o por la propia Comunidad GeneXus. La Comunidad GeneXus ha participado entusiastamente en las pruebas de las versiones CTP y está adoptando la primera versión liberada del producto, utilizándolo y, en muchos casos, construyendo extensiones. En el XIX Encuentro esperamos la exposición de muchas experiencias importantes. En el año 2007 hemos encomendado a Daniel Márquez Lisboa (autor de “GeneXus, Desarrollo Basado en el Conocimiento – Guía Práctica” editado por Magró en el 2006) y a Cecilia Fernández, con una muy buena experiencia como instructora en nuestros cursos, un libro introductorio sobre la Versión Rocha de GeneXus. Ahora que la versión X es una importante realidad, hemos pedido a Daniel y Cecilia una segunda edición del libro, actualizando aquella historia, contada sobre la marcha del desarrollo de la versión, a la realidad actual donde la versión X está en plena producción. Decía entonces: “No niego que, en una primera instancia, me sorprendió el tono informal del libro. Luego de leerlo cuidadosamente, veo que el “tono informal” es un acierto más de los múltiples que han tenido los autores…”: dos años después ratifico aquellas palabras. Espero una muy buena acogida de esta segunda edición del libro que, acompasándose con los tiempos y la tecnología, se puede obtener libremente en www.genexus.com/genexus/libroepisodio1ed2 y, desde ya, felicito a los autores por su trabajo y les deseo el mayor de los éxitos. Ing. Breogán Gonda Presidente de Artech 4 Estimado lector Para el XVII Evento Internacional de Usuarios GeneXus, realizado en setiembre de 2007, nos pidieron escribir un libro de ágil y amena lectura sobre la versión de GeneXus que se encontraba en plena gestación: la entonces denominada versión Rocha, hoy la X, una verdadera refundación del producto. Desde ese entonces “la X” ha crecido mejorando funcionalidades e incorporando algunas otras. Por esa razón, sentimos la necesidad de adaptar aquella primera edición de nuestro libro “GeneXus Rocha – Episodio Uno” para que siga cumpliendo con su objetivo inicial: mostrar GeneXus a quien, nuevo en esta herramienta, tenga inquietud o deseos de descubrirla, así como a quien, conocedor de versiones anteriores, quiera redescubrirla en esta nueva y fundamental etapa que inicia. He aquí entonces la segunda edición actualizada de aquel libro. Fue elaborada con GeneXus X, upgrade 4. Actualmente, Artech se encuentra trabajando para liberar en unos meses más GeneXus X Evolution 1, que incluye nuevas e importantes funcionalidades, como el permitir generar, además de aplicaciones Web, aplicaciones Win. Esperamos que pueda ir siguiendo las páginas de este libro con el mismo entusiasmo que nos llevó a escribirlas, y que pueda adaptar la lectura a sus ganas y necesidades. Hay capítulos más técnicos que otros, en especial el 3 que trata sobre Patterns. No pudimos ceder ante la tentación de mostrar toda su potencia, lo que nos llevó a desnudar cada vez un poquito más. Quien quiera mucho, lo leerá todo, quien no, lo leerá más superficialmente. Tómese esa libertad, que fue la que nos permitió dar rienda suelta a nuestras ganas de contar. Los últimos capítulos probablemente sean los más ágiles y disfrutables. Para entender más cabalmente lo explicado, sugerimos ir realizando con la versión Trial de GeneXus, de distribución gratuita (www.genexus.com/trial), la aplicación que nuestros personajes irán desarrollando a lo largo del libro. Si bien podrá elegir el generador y DBMS que prefiera, si es usted nuevo en GeneXus, le recomendamos elegir los default que son lo que utilizarán nuestros 5 personajes en el libro. Encontrará en nuestro Download Center, www.gxtechnical.com/genexus/libroepisodio1ed2/zip para ser descargado en forma gratuita, un zip con el contenido final del proyecto desarrollado. Podrá crearse otro proyecto (KB) e importarlo. Le servirá para comparar lo que va haciendo, con la aplicación ya implementada. Encontrará repetidamente en este texto citas a un whitepaper de Breogán Gonda y Nicolás Jodal, “Desarrollo basado en conocimiento. Filosofía y fundamentos teóricos de GeneXus”, publicado en mayo de 2007. Por último, volvemos a agradecer a todos los compañeros que en oportunidad de la primera edición estuvieron de un modo u otro aportando a este trabajo. A todos: ¡gracias! Mary, Diego, Mike y Julia, esperan ansiosos entrar en escena… ¡Buena lectura! Los autores. Montevideo, Mayo de 2009. 6 Sinopsis Capítulo 1: El encuentro. Diego y Mary se reencuentran luego de años sin verse, dando inicio a esta historia de encuentros…con GeneXus. Capítulo 2: Érase una vez…un Proyecto. Julia, Mike y Diego definen e inician el análisis y desarrollo del sistema para una agencia de viajes, a la vez que Diego acompaña a Mary, café mediante, en su primera experiencia de acercamiento al IDE y a la creación de los primeros objetos del proyecto. Capítulo 3: Hecho a máquina, configurado a mano. Mientras Julia va haciendo uso de lo máximo en capacidad de productividad gracias a los patterns, Diego da rienda suelta a su creatividad poniendo de manifiesto que los deseos de los clientes no siempre deben ser un duro escollo a sortear. Capítulo 4: Vuelo al mundo de los Subtipos. A velocidad match 3, Diego enseña a Mary, esta vez en su oficina y no en el bar, por qué los subtipos son la solución ideal al problema de las ambigüedades que se pueden presentar en un modelo de datos, mientras pone el punto final a las últimas transacciones. Capítulo 5: Cuando el “qué” no alcanza. El interés de Mary parecía ir in crescendo. Encuentros frecuentes en la oficina o en el bar. Preguntas todavía sin respuesta. Las transacciones permiten declarar, ¿pero qué pasa con los procesos ‘batch’? Procedimientos GeneXus. Después, más declaraciones: Business Components. Y ¿cómo realizar reportes pdf? Atracciones que invitan. Invitaciones que atraen. Una cena en ciernes ¿Y las consultas dinámicas por ejemplo? Capítulo 6: Más declaraciones. Data Providers para permitir cargar y devolver una salida estructurada, sin programar, sino declarando intenciones. Diego declara sus intenciones. Debe enviar un xml a la aerolínea. Capítulo 7: Soplan nuevos vientos. Su casa (la de ella), vino, música, su nerviosismo (el de él) y más GeneXus. Sabemos que todo tiene un final, pero lo que Diego apenas sospechaba, era que esto podía llegar a ser el principio de algo más. 7 8 Capítulo 1 El encuentro Alguien cuenta que esperando por fin ser atendido en un banco, escuchó casualmente una conversación entre un hombre y una mujer, que también esperaban. Según parecía, se reencontraban luego de años sin verse; ¿antiguos compañeros de estudio? ¿Cómo estás Mary? ¿Cómo van tus cosas? ¿Sigues trabajando donde siempre?–dijo el hombre con voz profunda, impostada, mirada fija, aire seductor. Bueno Diego, luego de tantos años trabajando en la empresa, logré ascender y ahora soy Developer Team Manager desde hace ya dos años – dijo la mujer de rostro sensiblemente cansado. Supongo que debo felicitarte, pero ¿por qué esa cara? ¿tienes problemas en el departamento? He escuchado que los sistemas de ustedes, además de ser de gran porte, son fiables… Lo son, pero no sabes lo que cuesta su mantenimiento. Me estoy enfermando, Diego, no duermo bien; muchas responsabilidades, grandes cargas horarias, poco descanso. Siento en el pecho… …una presión – completó Diego – Entiendo, Mary – dijo al tiempo que depositaba su mano derecha sobre el brazo izquierdo de ella-. ¿Cómo están trabajando? ¿Qué herramientas están usando? Y… de todo un poco, lo usual… al final todo es lo mismo…– contestó Mary ajena a esa cercanía- Tengo un equipo importante de personas trabajando en los desarrollos y mantenimientos, pero aún así no damos abasto. Bueno, no todo es lo mismo… ¿Qué te parece si terminamos este trámite aquí y vamos a tomar un café y me sigues contando? –preguntó Diego a la vez contento y turbado. 9 ¿GeneXus? Ya en el bar, café mediante, Diego escuchaba atentamente a una Mary angustiada. Recordaba su antigua firmeza, su empuje. Sentía un egoísta dejo de alegría de poder ser ahora él quien la ayudara. ¿En cuántos proyectos están? – preguntó Diego para instaurar un clima confesional. Ahora en uno solo. Pero es más que nada en tareas de mantenimiento. Nuestros clientes nos están solicitando cosas nuevas todo el tiempo, pero nos está siendo imposible satisfacer nuevos requerimientos. Además, esta tarea de mantenimiento es realmente costosa, y ya no puedo justificar a la Dirección la incorporación de más recursos humanos; nos saldríamos del punto de equilibrio. ¿No será que los sistemas están un poco anticuados, tecnológicamente hablando? ¿No les convendría hacer una reingeniería de todo a nuevo? – preguntó Diego con liviandad. Sí, eso quisiera, y eso he propuesto. Pero los tiempos realmente no nos dan, y como te dije, puede que estemos ya pasados de costos. No sé qué voy a hacer… estoy agotada Diego, me siento en un callejón sin salida. Mary, no desesperes, ¿sabes cuántos están como tú? Tu caso es uno más entre muchos. Las aplicaciones son cada vez más complejas, y con las herramientas manuales de las que disponíamos, se está tornando imposible desarrollar y peor aún, mantener esos sistemas, pues se están volviendo inmanejables. Pero no todo es negro, Mary. Hoy en día hay soluciones. La alternativa es delegar en procesos automáticos el trabajo que hacemos los humanos, de manera que podamos dedicarnos a las nuevas complejidades y no a tareas como la codificación que pueden ser efectuadas por un programa. Nosotros, sin ir más lejos, estamos llevando adelante varios proyectos importantes en simultáneo, y no tenemos demasiados inconvenientes. ¿Conoces algo de GeneXus? Ah, GeneXus. Sí, algo he escuchado, pero no he tenido tiempo de interiorizarme demasiado… ¿Ustedes trabajan con esa herramienta? He oído 10 que cada vez más gente no técnica puede usarla, lo que me despierta cierta curiosidad. Mira Mary, no sé si GeneXus sea la panacea, pero sí sé que cura varias de las dolencias que tienen empresas como la tuya. Es un programa que hace programas. Automatiza la construcción y mantenimiento de tus aplicaciones y de tu base de datos. Obviamente está en un nivel de abstracción superior a otras herramientas, ¿comprendes? Contéstame a esta pregunta: si tú pudieras disponer de un software que te permitiera lograr hacer esto automáticamente, ¿lo usarías? Absolutamente.–.. Se incorporó en su silla–. Sí, justamente ése es nuestro problema. Tenemos que hacer casi todo a mano… La gran dificultad de la empresa de Mary, pensó Diego, era el de tantas otras: imposibilidad de enfrentar nuevos proyectos porque los tiempos de mantenimiento de los sistemas actuales, fundamentados en metodologías tradicionales, lo impedían. Y eso, para una empresa que vive del desarrollo, es una luz roja de peligro. Bueno, no lo diga en voz alta, pero usted, estimado lector, se estará preguntando: ¿Cómo? ¿Cómo? – preguntó ya más vital. ¿No te parece razonable pensar que dado un conjunto de visiones de datos, - se interrumpió Diego con cierta malicia - exista un único modelo relacional mínimo que lo satisfaga? Hmmm… puede ser, no estoy segura. ¿Pero qué tiene esto que ver con mis problemas de mantenimiento? Y si eso es cierto –continuó Diego, ignorando la vacilación de su compañera- ¿no te parece posible encontrar un procedimiento de ingeniería inversa que partiendo de ese conjunto de visiones dé como resultado el esquema de esa base de datos relacional mínima? Bueno, si me vas a decir “ver para creer”, con gusto acepto una apuesta, como en los viejos tiempos; pero desde ya te adelanto: vas a perder. 11 Una Apuesta “El objetivo de GeneXus es (a través de la descripción de las visiones de los usuarios) conseguir un muy buen tratamiento automático del conocimiento de los sistemas de negocios.”. Breogán Gonda & Nicolás Jodal GeneXus es una herramienta que parte de las visiones de los usuarios; captura su conocimiento y lo sistematiza en una base de conocimiento. A partir de esta última, GeneXus es capaz de diseñar, generar y mantener de manera totalmente automática la estructura de la base de datos y los programas de la aplicación, es decir, los programas necesarios para que los usuarios puedan operar con sus visiones. GeneXus trabaja con conocimiento puro, cuya validez es totalmente independiente de las tecnologías de moda. Mary, GeneXus almacena en una Knowledge Base todos los elementos necesarios para construir la aplicación, y luego la utiliza para el desarrollo del sistema, construyendo automáticamente el modelo de datos en 12 forma normalizada, y también utilizando el lenguaje de programación y base de datos que se le indique. Así que te permite obtener un proyecto con el conocimiento de la empresa… ¡Ah! Entonces eso… Sí –le interrumpió Diego completando la idea-, te permite ganar algo muy importante: la portabilidad futura de la aplicación. Todo ese conocimiento -reglas de negocio, diálogos, controles, listados- que hoy están en un lenguaje de programación determinado, pueden ser convertidos a otros lenguajes sin necesidad de arrancar de cero. O sea que se reutiliza el conocimiento porque la Knowledge Base es conocimiento independiente de la plataforma. Mary parpadeó rápidamente un par de veces y, sin mediar palabra alguna, pidió otra ronda de café. Base de Conocimiento y Modelos Externo y Relacional Nuestros personajes continuaron su conversación en el café. A esas alturas, Mary ya se mostraba particularmente interesada en esa herramienta, y Diego en... Bueno, por un lado está el Modelo Relacional, que como bien sabes, está orientado a obtener una buena representación de los datos en una Base de Datos, obedeciendo a algunas condicionantes muy deseables. Cierto… – aseguró ella –. Eliminar las redundancias e introducir un pequeño conjunto de reglas para evitar las mayores fuentes de inconsistencia de los datos… Eso mismo, reglas que chequean la unicidad y la integridad referencial y además también proveen de un conjunto de operadores para poder manipular los datos a buen nivel – acotó Diego, alegre por el clima de complicidad generado. Sí, claro, por eso todo el mundo las usa. ¿GeneXus no las usa? Bueno, sí, ya voy a llegar a eso. ¿Te acuerdas también del Modelo Externo, aquel donde se representaban las visiones externas? Bueno, GeneXus 13 pone el énfasis en este modelo, porque es donde realmente está el conocimiento genuino, es el realmente importante para los usuarios y los desarrolladores. En él se recoge el conocimiento exterior y todo lo demás, como otros modelos auxiliares que pudieran ayudar, puede inferirse automáticamente a partir de ese Modelo Externo. Ah…el Modelo Externo…no puede contener ningún elemento físico o interno como archivos, tablas, entidades, relaciones entre entidades, índices… ¿verdad? ¡Exacto! O cualquier otro que se pueda inferir automáticamente – exclamó Diego animado, recuperando el entusiasmo que siempre le provocaron las conversaciones con Mary– El Modelo Externo será utilizado para obtener y almacenar el conocimiento. Pretende la representación más directa y objetiva posible de la realidad. Claro, y con eso te independizas de la implementación, porque tienes una descripción del sistema en alto nivel, descripción que sólo cambiará si cambian las visiones sobre el mismo; ¿voy bien? ¡Perfecto! Por ello tomamos las visiones de los diferentes usuarios… Visiones que son almacenadas en el modelo. Luego se captura todo el conocimiento contenido en ellas y se lo sistematiza para maximizar las capacidades de inferencia… Eso es lo que no termino de creer. Me suena a pura teoría… ¿tú intentas decirme que es posible sólo describir y que un programa, con sólo un par de pases de varita mágica te haga realidad la aplicación? – preguntó Mary intentando ser convencida por los razonamientos apasionados de Diego. Bueno, yo no diría que es magia. Es un software al fin de cuentas. Y no digo que no tengas que programar algo, pero sí digo que “casi” nada, lo 14 menos posible. En GeneXus vas a tener algún código que será procedural, pero el lenguaje en que lo escribes, no es el lenguaje al que estás acostumbrada… en él tampoco tienes que nombrar tablas, ni índices, ni nada de lo físico de una base de datos. Pero… no entiendo entonces dónde entra el Modelo Relacional – preguntó Mary confundida. Es el que se utiliza para representar y manipular los datos: es el modelo interno o físico, pero no será creado por ti, será inferido por GeneXus con su procedimiento de ingeniería inversa. Mi apuesta fue tramposa. ¿Y cómo describes las visiones de los usuarios, el Modelo Externo, de manera que pueda aplicarse ingeniería inversa sobre ellas y obtener todo eso que dices? Porque tiene que ser lo bastante formal y rigurosa esa descripción como para que no haya ambigüedades y que un programa pueda “inferir” todo eso que los programadores hacemos todavía a mano. Sí, lo es. Las descripciones se realizan a través de determinados “objetos-tipo” de GeneXus que indican “el qué”. A través de ellos, GeneXus encuentra el “cómo”. Por lo que, para aprender a usar GeneXus, tienes que aprender a describir, a usar esos objetos-tipo. No necesitas grandes conocimientos técnicos; cada vez menos. Diego continuaba hablando, llenando el reducido silencio que su interlocutora había dejado. A pesar de tantos años sin verse, conocía bien a Mary, conocía esa expresión de ojos pensativos, y sabía que su silencio lo invitaba a proseguir. No podía tener mejor suerte - ¡Bendito GeneXus! - pensó, y continuó en tono confidente, manejando las pausas, seguro de lograr los efectos buscados. 15 Para lograr todo esto, GeneXus tiene una base de conocimiento, que inicialmente tiene asociado un conjunto de mecanismos de inferencia y algunas reglas de aplicación general, como las que aseguran la consistencia (las de integridad referencial, por ejemplo). Luego, cuando el analista GeneXus comienza a describir la realidad creando objetos, estas descripciones (el Modelo Externo) son sistematizadas automáticamente y pasan a estar contenidas en la base de conocimiento… Además, sobre ese conocimiento, obtiene un conjunto de resultados que le ayudan a mejorar la eficiencia de las inferencias posteriores. ¡Es una máquina de inferencias! – exclamó Mary. Nada más cierto. Por ejemplo, dada una visión de los datos, puede inferir automáticamente el programa necesario para manipularla. “GeneXus trabaja permanentemente sobre la Knowledge Base. Todo el conocimiento de la Knowledge Base es equivalente al contenido del Modelo Externo (subconjunto de ella), ya que consiste en el propio Modelo Externo más reglas y mecanismos de inferencia independientes de él y un conjunto de otros elementos que son automáticamente inferidos a partir del mismo. El desarrollador puede alterar, modificando objetos de la realidad del usuario, el Modelo Externo y las modificaciones se propagarán automáticamente a todos los elementos que lo necesiten: otros elementos de la Knowledge Base, Base de Datos y programas de la aplicación. De la misma manera, el desarrollador no puede alterar directamente ningún elemento que no pertenezca al Modelo Externo”… TODO el conocimiento está contenido en el Modelo Externo y, por ello, mañana podríamos soportar la Knowledge Base de una manera totalmente diferente y el conocimiento de nuestros clientes seguiría siendo utilizable sin problema alguno”. Breogán Gonda & Nicolás Jodal 16 Diego, debo irme, pero quiero saber más. Me has dejado con la cabeza confusa. Es muy grande el cambio de mentalidad, quisiera palpar esto más de cerca, verlo funcionando. ¿Te parece de encontrarnos la semana próxima, misma hora, mismo lugar? Ciertamente – expresó Diego en tono pretendidamente sereno. 17 18 Capítulo 2 Érase una vez… un Proyecto Travel Agency & Co. es una novel compañía de venta de tickets de viajes y turismo que ha contratado a la software house ACME Business Software Solutions a fin de que le sea desarrollado un sitio en Internet que permita a sus clientes realizar búsquedas de destinos, de vuelos existentes, de servicios brindados, reservar tickets y obtener pasajes y servicios turísticos (front-end). Incluirá también todo el sistema para el mantenimiento de la información relacionada (back-end). ACME, que viene trabajando con GeneXus en sus sucesivas versiones desde hace un tiempo, está integrada por varios técnicos; entre todos ellos, Julia, Mike y Diego fueron los seleccionados para llevar adelante este desarrollo. Diego y Mike son Analistas GeneXus y los que normalmente se encargan de la implementación que les toque en suerte. En particular, Mike se encarga casi siempre de la tarea de testear las aplicaciones, mientras que Diego se encarga del desarrollo. Julia, en cambio, no es técnica en GeneXus, si bien ha trabajado en muchos proyectos relevando requerimientos; su especialidad es documentar y moderar las charlas entre los integrantes del equipo a medida que se van produciendo los avances. El hecho es que los tres conforman un equipo donde todos tienen voz y voto. Las primeras tareas recaen sobre los hombros de Julia y Diego mientras Mike realiza los testings de las otras aplicaciones de ACME en curso para los desarrollos en otros clientes. Desde siempre, Julia documentaba en procesadores de texto, y en proyectos de envergadura le resultaba extremadamente engorroso el mantener decenas de carpetas con cientos de archivos, conjugarlos entre los desarrolladores, coordinar las discusiones del equipo (cuando lograba hacerlo), entre otras tareas. Usted comprenderá el impacto que sintió luego de enterarse que GeneXus X contenía, en forma embebida, un perfil de documentación muy particular que facilitaría su vida de ahora en más. 19 El tiempo es tirano Un breve tiempo atrás, los directores y el equipo de desarrollo se pusieron de acuerdo en el objetivo de un proyecto a corto plazo, tres semanas o algo así. El hecho de que iban a trabajar con GeneXus X, a sabiendas del gran aumento de usabilidad y productividad que ésta traía incorporado, les había permitido reducir a la mitad los plazos que normalmente pactaban con sus clientes. ”Es necesario un dramático aumento de productividad (en el desarrollo de sistemas) pero la productividad de los lenguajes de programación ha llegado hace ya bastante tiempo a una estabilización”… “¿Cómo lograr, entonces, el aumento de productividad que se necesita? Haciendo desarrollo basado en conocimiento y no en programación: ¡la solución es describir en vez de programar!”. Breogán Gonda & Nicolás Jodal Afortunadamente, Travel Agency & Co. ya tenía su esquema interno de funcionamiento bastante sólido, con abundante documentación y esto ya era una ventaja para el equipo de desarrollo. Julia, Diego y Mike acordaron que debían tener reuniones en forma paralela con los involucrados de cada sector e ir recogiendo sus visiones. “Cada usuario tiene una o múltiples visiones de los datos que utiliza cotidianamente. Entre estas visiones, podemos pensar en un primer tipo: el que agrupa aquellas que se utilizan para manipular los datos (introducirlos, modificarlos, eliminarlos y visualizarlos en forma limitada), a estas visiones de usuarios les hemos llamado Transacciones y constituyen el primer objeto-tipo de GeneXus”. Breogán Gonda & Nicolás Jodal Querían rápidamente comenzar a representar la realidad de la empresa; entre otras cosas, detectaron que Travel Agency & Co. apuntaba fundamentalmente al turismo de diversión y esparcimiento con destinos muy 20 variados en el mundo. ¿Qué les decía esto? Antes que nada, que si los destinos eran puntos turísticos de ciudades distribuidas a lo largo y ancho del mundo, debían disponer de un almacenamiento para ello. Es así que el equipo completo se reunió y luego de intercambiar notas e impresiones, concordaron en que Julia comenzara a describir las primeras pautas obtenidas, mientras el resto, en especial Diego, comenzaría a representar en GeneXus aquellas visiones de los usuarios más claramente descriptas. El IDE a primera vista Ahora intentemos adentrarnos un poco en el ambiente de desarrollo de GeneXus X. Si usted lo desea podrá seguir los pasos de Julia para tomar contacto con los elementos que componen este ambiente. Así, creará la base de conocimiento del proyecto y el primer objeto de tipo Transacción. Generalmente conocido por sus siglas (IDE, Integrated Development Environment), en realidad este ambiente de desarrollo supera largamente el verdadero significado de las siglas en su acepción más general. ¿Por qué? Porque el concepto más elemental de IDE es el de un ambiente que ofrece una buena dosis de comodidad, de ventanas en donde seleccionar elementos, de ventanas donde codificar, de cajas de herramientas y de fuentes, y alguna cosa más. Sin embargo, el IDE de GeneXus X no sólo satisface las mínimas especificaciones, sino que pone al alcance del usuario mecanismos de operatividad integral que van más allá. Julia inició la sesión en una computadora con GeneXus X recién instalado, y apareció ante su vista una imagen semejante a la que se ilustra a continuación. 21 Panels Inicialmente, podemos notar dos áreas contenedoras claramente definidas. A la izquierda, tenemos el contenedor Knowledge Base Navigator; su objetivo es ofrecer distintas vistas. Está compuesto por una zona (ahora en blanco) que contendrá, en forma de árbol, los elementos que integren la base de conocimiento en uso, y un grupo de paneles horizontales rotulados con descripciones inherentes a otras vistas. La ventana de la derecha, que podríamos considerar la principal (la mayor parte de nuestra actividad se realizará en ella y así la llamaremos de ahora en adelante), es en principio otro contenedor de múltiples funciones. Note que en su esquina superior izquierda se encuentra un tab (o solapa) rotulado Start Page. Es el primer contenedor abierto dentro de la ventana principal. A medida que usted vaya trabajando, la presencia de nuevos tabs indicará la disposición de nuevos espacios conteniendo elementos del proyecto tales como listados de navegación, diagramas, estructuras de transacciones, etc. Ahora veamos el contenido del único tab que tenemos. Observe que incluye otros contenedores con información. Por ejemplo, Recent Knowledge Bases nos 22 ofrece los nombres de las últimas bases de conocimiento que han sido abiertas, con sus fechas de última modificación; note que también están presentes links que permiten abrir o crear una nueva base de conocimiento. Debajo, un contenedor Extensions presenta herramientas desarrolladas por Artech o terceros, que pueden ser instaladas y utilizadas libremente, y permiten extender en todas direcciones la potencia de la herramienta. El sector rotulado GeneXus Community, contiene sindicaciones de contenidos de sitios Web vinculados a la Comunidad que se actualizan con frecuencia (RSS). Note que debajo de cada noticia aparece More… como link al documento propuesto. A su vez observe la celda Address que le permitirá navegar por la Web o por su sistema de archivos sin salir del IDE. Proyecto nuevo, Base de Conocimiento nueva La primera tarea cuando se comienza a desarrollar una aplicación con GeneXus es crear la base de conocimiento. En el siguiente encuentro que mantuvieron en el café, Mary, con ojos aún graves, preguntó: Me contaste que GeneXus automatiza la creación de base de datos y programas, pero ¿cómo sabe en qué ambiente, en qué lenguaje debe codificar esos programas? Por ejemplo, si tuvieras que desarrollar una aplicación en Java y suponte, otra, o la misma, en .Net…, o en… Bueno, cuando te creas la Knowledge Base (le solemos llamar KB, para abreviar) le puedes decir que quieres generar el proyecto en un ambiente Java. Como ya te comenté, la KB es independiente de la plataforma de ejecución: si tú generaste la aplicación Java, más adelante, con el mismo conocimiento, puedes generarla para .Net, y no tienes que volver a describir los objetos de la realidad que ya habías descrito. Simplemente tienes que indicarle que ahora necesitas asociarle otro ambiente. Cuando le asocias un ambiente, tiene por ahí un lugar donde especificar todo lo referido al mismo. Por ejemplo, 23 en Java tendrás que indicar cosas tales como dónde estarán las clases en la webapp, algo que en .Net no tiene sentido. Es decir, en base al ambiente que hayas elegido, te muestra propiedades necesarias para configurarlo, para que GeneXus pueda luego decirte: “sus deseos son órdenes” y te implemente la aplicación como se lo pediste. Casualmente, mientras Diego y Mary intercambiaban palabras y miradas en el café, Julia realizaba esos mismos pasos que lo invitamos a seguir a usted también. 1. Seleccione File/New/Knowledge Base (o el link New Knowledge Base de la Start Page). A continuación verá en la pantalla el siguiente diálogo: TravelAgency Observe cómo desde aquí se asocia un ambiente de implementación al proyecto. Dejamos la opción por defecto, “C# Environment”. También observe que por defecto se asocia el lenguaje English para la aplicación a ser generada. Todos los mensajes, botones, etc., estarán por defecto en inglés. Pero luego será posible generar la misma aplicación en otro idioma, tanto en forma estática como dinámica (el usuario, en tiempo de ejecución, elige el idioma en que desea utilizar la aplicación) a un costo mínimo. 24 Observe además que la KB será almacenada en una base de datos que por defecto recibirá el nombre GX_KB_nombre en la instalación del SQLServer local que usted tenga. Si no tiene una local, podrá utilizar una de la red. Presionando el botón [Advanced] podrá modificar estos valores. 2. Pulse [Create]. GeneXus creará una base de conocimiento vacía, y usted debería ver una imagen semejante a la que se observa a continuación. Observe que han cambiado algunos contenidos. En el panel Folder View ha aparecido un árbol cuya raíz es la propia base de conocimiento (con el nombre que usted le ha dado) y sus ramas los distintos elementos que la componen: Domains, Objects, Tables, Customization, Documentation, etc. También note que se ha agregado un nuevo sector dentro de la Start Page rotulado Knowledge Navigator conteniendo atajos a las tareas más comunes que probablemente usted querrá ejecutar luego de haber creado o abierto una base de conocimiento. 25 Sobre esta base de conocimiento trabajarán los demás integrantes del equipo; Julia está ahora lista para comenzar a documentar. Documentando desde dentro Documentos, notas, listas de tareas, son sólo ejemplos de la documentación que rodea a cualquier proceso de desarrollo de software. La documentación es una parte importante de cualquier aplicación si se encuentra actualizada y es fácilmente accesible. Esta es la razón por la cuál GeneXus contiene funcionalidades de documentación integradas a la KB. Documentos, archivos, diagramas, objetos pueden estar todos interrelacionados mediante links, fácilmente accesibles por cualquier miembro del equipo. Por tal razón decimos que la documentación es activa e integrada, constituyendo un verdadero wiki. Files Julia había recibido por e-mail varios archivos de documento de los jefes de sección de cada departamento con requerimientos para el sistema, donde listaban las prioridades y características de sus sectores, algunos flujos de procesos, etc. 26 Como estos documentos ya estaban hechos por terceros, decidió que sería importante incorporarlos en su estado original dentro de la base de conocimiento, de forma tal que siempre estuvieran disponibles para su consulta. Eran importantes las “palabras” de los usuarios expuestas por ellos mismos. Así, expandió el nodo Documentation del árbol y seleccionó Files abriéndolo con doble clic. Como resultado, un nuevo tab apareció dentro de la ventana principal, rotulado Files. Allí presionó Add New File y se le abrió una ventana que le permitió explorar su sistema de archivos, y elegir cada archivo a ser insertado en la KB. Así quedó el tab Files luego de esto. Images También le habían enviado por e-mail el logo de la empresa, por lo que accedió al nodo Customization del árbol del Folder View, abrió haciendo doble clic sobre Images el tab correspondiente e importó, almacenándolo en la KB, el archivo Logo.bmp que sería utilizado en el encabezado de las páginas de la aplicación. En ese mismo e-mail recibió decenas de archivos con imágenes sobre atracciones turísticas que luego necesitarían almacenar en la base de datos, por lo que momentáneamente las guardó también aquí, para tenerlas centralizadas en la KB. 27 Main Document Ahora Julia se apresta a empezar a escribir la página principal del wiki de la KB: observe que bajo el nodo Documentation del árbol aparece un objeto Main Document. Cada equipo de desarrollo tiene metodologías diferentes respecto a la documentación. En ACME utilizan esta página principal para describir las generalidades del sistema bajo desarrollo, y para acceder desde aquí a los distintos elementos de documentación. También podría elaborarse aquí mismo el Plan Director del proyecto, en vez de insertarlo como archivo. Pero usted y su equipo de desarrollo utilizará esta página como mejor le parezca. Julia abre Main Document y un nuevo tab aparece en la ventana principal, rotulado Document:Main. Dado que lo que desea es comenzar a escribir, pulsa el selector Edit que se encuentra al pie de la ventana y redacta (dándole formato) lo que se observa en la siguiente imagen. Luego pulsa el selector Preview y tiene la vista normalizada de lo que escribió. Guarda los cambios (Preview se transforma en Browse). Note que le ha dado algún formato al texto. Para ello se valió de la toolbar formatting: 28 Julia ha cambiado el tipo de letra, su tamaño, e incluso puso en negrita el tópico sobre el cual se comenta. Pero además, y bien importante para lograr una documentación integrada, Julia necesitaba agregar links hacia los documentos recibidos e insertados en la KB instantes antes. Para ello se valió de la toolbox1, seleccionó Table de la misma, arrastrándola a la página y luego de insertada la tabla, en cada fila escribió: Queriendo tener una vista previa, presionó el selector Preview y, satisfecha, grabó, luego de lo cual se le mostró lo que aparece en la siguiente figura: 1 El IDE de GeneXus ofrece varios paneles con utilidades que se pueden acoplar en cualquier parte de la ventana y también se pueden ocultar automáticamente cuando pierden el foco para que no ocupen sitio en la ventana principal. Usted puede activarlas mediante View/Other Tool Windows. 29 Observe que como consecuencia de los símbolos que empleó en la segunda columna de la tabla, los documentos se encuentran subrayados: son links a los archivos almacenados. Esto aporta un gran dinamismo a la documentación, y la integra en forma activa en el proyecto a través de la KB. El editor de documentación de GeneXus utiliza dos paréntesis rectos de apertura ([[) y dos de cierre (]]) en el texto para crear links. Si usted está siguiendo el ejemplo en forma práctica, habrá notado que, apenas pulsado el segundo paréntesis de apertura, el editor le propuso una lista de objetos válidos para seleccionar, tal como mostramos en esta imagen: 30 Julia seleccionó File y al digitar ‘.’ se abrió la lista de los archivos disponibles (también podría haber arrastrado desde la toolbox el objeto File y habría obtenido igual resultado que al escribir “[[File.”). Entonces seleccionó el deseado en cada momento, pulsó <Enter> y el texto quedó rematado con los dos paréntesis finales. Creando la primera Transacción: Attraction Diego piensa, mientras conversa con Mary (la mirada perdida en el café humeante), que ya debe dar inicio al volcado de las primeras visiones obtenidas a través de los usuarios que entrevistó en el Departamento de Informaciones Turísticas de Travel Agency & Co. Este departamento tiene a cargo la búsqueda de información que permite agregar o suprimir los destinos e itinerarios que ofrece la empresa a sus clientes. Piensa que ni bien vuelva al trabajo, deberá crear la transacción Attraction, cuando Mary interrumpe su interno discurrir: ¿Sabes Diego? No comprendo muy bien cómo puedes describir rigurosamente las visiones de los usuarios como para que no haya ambigüedades y se pueda inferir todo automáticamente. Bueno. ¿Recuerdas que te comenté que estamos trabajando en un proyecto nuevo? Bien, pues me he estado reuniendo con los usuarios quienes me contaron que manejan atracciones turísticas de cada país y ciudad, para poder ofrecer a los viajeros. Manejan el nombre de la atracción, como podría ser “Centenario Stadium” o “Roman Colisseum” o “Disney World”; manejan el nombre de país y ciudad al que pertenece la atracción, una imagen de la misma. También me contaron que las atracciones están categorizadas. Por ejemplo, 31 “Centenario Stadium” pertenece a la categoría “Great Monuments” o algo así, y “Disney World” a “Entertainment”. Estas son sus visiones de los datos… Ajá. ¿Y cómo describes esas visiones objetivamente? Mediante atributos… y objetos de tipo Transacción. Déjame explicarte un poco. Los atributos son el marco de referencia sobre el cuál hacer las demás descripciones. No es exactamente lo mismo que un atributo en un modelo relacional. El atributo es el elemento semántico fundamental. El significado del mismo vendrá dado por su nombre, por lo que los nombres de los atributos pasan a ser esenciales en GeneXus. Diego tomó una servilleta de papel de la mesa del bar y garabateó algunos nombres: AttractionName, CountryName, CityName, AttractionImage, AttractionCategoryDescription. Y prosiguió: Estos atributos tienen un contenido semántico claro, que cualquier persona podrá entender sin necesidad de contexto alguno. No son sólo “Name” o “Description”. Existe una fuerte relación entre el nombre del atributo y su significado. Ese nombre, a su vez, es una secuencia de caracteres, algo sintáctico, único, con lo que cualquier programa puede trabajar, sin ambigüedad. Si yo coloco AttractionCategoryDescription en algún lado, tú y yo sabemos que estamos hablando de la descripción de la categoría de la atracción. GeneXus lo reconoce, por su sintaxis. Ajá, entonces –interrumpió Mary– claramente no puedes usar el mismo nombre de atributo cuando quieras referirte a otro dato, pues ahí sí habría confusión y ambigüedad. Exacto. Un atributo deberá tener el mismo nombre en todos los lugares donde aparezca y no podrá haber dos atributos diferentes, con significado diferente, que compartan el mismo nombre. En eso se basa. Hay una excepción, pero no te quiero confundir ahora. Estas visiones que me expresaron los usuarios, corresponden a aquellas que se utilizan para manipular los datos (ingresarlos, modificarlos, eliminarlos, visualizarlos), y son las que se representan en GeneXus mediante el objeto de tipo Transacción. Cada transacción tiene un conjunto de elementos: una estructura, reglas, fórmulas, 32 elementos de presentación. Es decir, en un mismo objeto, se matan varios pájaros de un tiro: al tiempo que dices con qué información se va a trabajar, también diseñas la pantalla que se desplegará al usuario final para hacerlo, declaras el comportamiento que deberán tener esos datos cuando se vayan ingresando, y demás. Diego volvió a tomar la servilleta que había dejado sobre la mesa, y esta vez escribió (y luego dijo): - Los datos se presentan de acuerdo a una estructura que debe recogerse con todo rigor. Esta sería la estructura de la transacción Attraction. El asterisco que sigue a AttractionId significa que para cada atracción, existe un único AttractionId, es decir, es lo que le da unicidad a la atracción: un identificador. Toda información debe poder identificarse. Una atracción debe identificarse mediante algún o algunos de sus atributos. Espera, Diego. Me surgen algunas dudas… en definitiva ¿me estás hablando del concepto de clave primaria de una tabla en un modelo relacional? ¿Esos atributos que listas no serán las columnas de una tabla física que almacenará esa información? En definitiva, con la transacción, ¿no estás diseñando una tabla? 33 Sí, indirectamente. Y no: yo no me preocupé de “diseñar” una tabla física del modelo relacional. Solamente me limité a especificar los atributos con los que el usuario final interactuará al ingresar atracciones al sistema. Especifiqué una vista de usuario, no una tabla. Aquí entra lo de la ingeniería inversa y el Modelo Relacional “inferido” por GeneXus. Si no hubiesen más transacciones que ésta en la base de conocimiento, entonces GeneXus “inferiría” una tabla física compuesta exactamente por los atributos listados. Pero la cosa cambia si existen otras transacciones compartiendo algunos de los atributos mencionados. Allí habrá que “normalizar”. Pero no me quiero apurar, espera un poco que ya te contaré de esa parte. Creo que te voy entendiendo, Diego. Pero me llama la atención que pusiste en la estructura de la transacción atributos que no dijiste que hubieran mencionado los usuarios. Todos los que terminan con Id… Por otro lado, ¿por qué no utilizaste el nombre de la atracción como identificador? ¿Y si el nombre de la atracción pudiera repetirse para atracciones de distintos países y ciudades? Suponte: “Disney World” que existe en París, Miami, Las Vegas, Orlando. En este caso, necesitamos un atributo que dé unicidad al conjunto de elementos de información que constituyen una atracción. Toda estructura de transacción debe tener un atributo o atributos que identifiquen a cada instancia. En principio, en esta transacción, coincidirá exactamente con la clave primaria de la tabla “inferida”. Pero no siempre será así, como verás cuando te muestre un ejemplo de transacción con niveles. No te preocupes ahora. En nuestro caso, podremos decirle a GeneXus que AttractionId será numérico y que queremos que se numere automáticamente. ¿Y qué me dices de los atributos AttractionCategoryId, CountryId, CityId? ¿Más identificadores? Sí, serán identificadores en otras transacciones. Me explico: evidentemente los países y ciudades, así como las categorías de atracciones, corresponden en sí mismas a entidades independientes de la atracción turística. Por ejemplo, no hemos entrado a ver eso aún con este cliente, pero evidentemente los vuelos aéreos también tendrán país y ciudad (de origen y de 34 destino). Es decir, resulta evidente que deberemos crear una transacción para representar e ingresar la información de países y sus ciudades, así como de las categorías de atracciones existentes. Esos identificadores de los que me preguntabas, serán, casualmente, los que identifiquen cada dato (instancia) de estos. A ver si entiendo: una vez que definas esas otras transacciones, GeneXus inferirá tablas para almacenar los datos que manipulan, y a partir de allí, se dará cuenta que esos atributos que colocaste en la estructura de la transacción Attraction, (AttractionCategoryId, CountryId y CityId) son los que se traducirían en claves foráneas en la tabla relacional asociada. ¡Sigues siendo la misma luz de siempre, Mary! – exclamó Diego con ojos iluminados. Pero entonces, ¿qué hacen en esa estructura CountryName, CityName, AttractionCategoryDescription? En un modelo relacional normalizado nunca podrían estar en esa tabla. Es que no lo estarán. Por eso te decía que la estructura de una transacción no se corresponde exactamente con la de la tabla física asociada. Ya te lo explicaré, no quiero abrumarte con detalles. Traeré mi notebook al próximo encuentro… porque lo habrá, ¿no? –inquirió Diego vacilante, y prosiguió sin esperar respuesta–, y te mostraré… Ahora el lector verá los pasos que siguió Diego en la creación de la transacción mediante el IDE de GeneXus al día siguiente en su trabajo. Primero que nada, seleccionó el link New Object de la Start Page y apareció a su vista la ventana cuyo diálogo permite la creación de un objeto dentro de la KB. De la caja Select a Type, con la lista de objetos GeneXus disponibles, seleccionó Transaction y luego, en la celda Name escribió el nombre “Attraction”, quedando una imagen semejante a la que se muestra a continuación. 35 A continuación, pulsó [Create], tras lo cual la transacción se abrió. Obsérvese en la imagen siguiente que aparece un nuevo tab en la ventana principal con el nombre de la transacción. ¿Qué se está editando? Una parte de la transacción: su estructura. Los objetos que se abren en la ventana principal del IDE aparecen como tabs, donde uno solo, el activo, se presenta en la ventana central. El objeto abierto puede tener varios elementos o partes, en cuyo caso será uno el activo en cada oportunidad. En el tope inferior de la ventana aparece una barra de acceso que permite, mediante Selectores, seleccionar otra de las partes del objeto activo. La ventana de edición de la estructura contiene una serie de columnas que el desarrollador utiliza para crear un atributo; es decir, darle un nombre, un tipo de dato, una descripción, etc. 36 Selectors Corresponde en GeneXus, a la estructura dibujada por Diego en una servilleta de papel. Diego asignó nombres a los atributos utilizando lo que se conoce como nomenclatura GIK (GeneXus Incremental Knowledge), un estándar creado por Artech y adoptado por la Comunidad, cuyo fin es dotar a un atributo de un nombre único que transmita lo más claramente posible su concepto, su semántica. Usted habrá deducido correctamente que la columna Type pretende dotar al atributo con alguno de los tipos de datos soportados por GeneXus. 37 El tipo de datos Blob posibilita el almacenar una gran diversidad de tipos de información (videos, planillas, documentos de todo tipo, archivos de música, imágenes digitalizadas, etc.) en la propia base de datos. La idea del atributo AttractionImage es justamente almacenar una imagen de la atracción, que podrá ser un archivo bmp, jpg, etc. La columna Description pretende que se asigne una descripción ampliada del atributo. Por ahora dejaremos las sugerencias de GeneXus a partir del nombre del atributo. Para continuar agregando atributos, simplemente pulse <Enter> luego de terminada una definición y repita el mecanismo. Atributos Clave. Como puede usted apreciar, el atributo clave que Diego indicaba con un asterisco en la servilleta, se indica en GeneXus por el ícono de una llave. Cuando Diego estaba asignando el tipo de datos en la columna Type para el atributo AttractionName, consideró oportuno crear un dominio Name para todos los atributos que fueran nombres “de algo”, así como uno Id para todos los que tuvieran la característica de ser identificadores numéricos. Usted estará ya acostumbrado a trabajar con Dominios. Si no es así, lea el siguiente párrafo que le aclarará la cuestión. ¿Dominios? Es común en las aplicaciones tener atributos que comparten las mismas definiciones de tipo de dato, tamaño y algunas otras características. Por ejemplo, en la transacción Attraction hay varios atributos que son Id (AttractionId, AttractionCategoryId, CityId, y CountryId). Diego ha creado un dominio al que llamó Id, con todas las características propias de los identificadores y se lo aplicará a todos aquellos atributos que tengan la funcionalidad de representar un valor inexistente en la realidad pero necesario para lograr la unicidad. 38 Cada dominio es un conjunto de características únicas que pueden ser compartidas por varios atributos. Esto proporciona consistencia y facilita el mantenimiento, ya que cambiando alguna de las características de un dominio se provocará la propagación del cambio a todos los atributos basados en él. Definición de Dominios Diego hizo doble clic sobre el nodo Domains en el árbol del Folder View tras lo cual se abrió en la ventana principal el tab que muestra y permite editar dominios. Ingresó así los dominios Id y Name. El mecanismo es semejante al de crear atributos en una transacción. Existen algunos dominios que son creados automáticamente por GeneXus cuando se crea la KB. Teniendo los primeros dominios definidos, Diego volvió a la transacción, sustituyó por estos dominios y en ese momento se percató de que le faltaba definir un dominio Description para AttractionCategoryDescription. En vez de volver a la ventana Domains, se posicionó en la columna Type del atributo y escribió “Description=Character(20)”2. Al abandonar el campo, ese dominio se crea automáticamente. Luego grabó y pudo ver: 2 En general en GeneXus existen varios caminos que le permiten realizar lo mismo, para que el efectuar su tarea le resulte siempre cómodo. 39 Observe que el asterisco que seguía al nombre de la transacción en el tab desapareció (el asterisco indica que los cambios que se han realizado aún no se han grabado) y que el selector “Structure” cambió de color. Atributo Descriptor. Ahora también apareció un ícono con la imagen de una lupa, precediendo al atributo AttractionName. Es una forma de indicar que si uno tuviera que quedarse con uno solo de todos los atributos, de forma tal que sea el que mejor describa a la entidad, ése será el elegido (el de mayor carga semántica). Ahora no nos preocuparemos de esto, servirá a herramientas que automaticen. La Tabla ATTRACTION Por defecto, cuando GeneXus diseña las tablas les asigna como nombre el mismo que el de la transacción en la que están basadas3. En todo momento puede verse el diseño de la tabla (su estructura, índices, etc.) abriendo el nodo Tables del árbol que se encuentra en el panel Folder View y haciendo doble clic sobre el nombre de la tabla. Para ATTRACTION se verá: 3 En lo que sigue en este material, para diferenciar los nombres de las tablas de los de las transacciones usted verá a las tablas escritas en mayúsculas. 40 Dada la estructura de la transacción, GeneXus diseñará también un form para la misma, que será la pantalla a través de la cuál el usuario ingresará atracciones cuando la transacción sea convertida en programa ejecutable. Puede verse escogiendo el selector WebForm4 de la barra de acceso: 4 Con GeneXus X usted podrá generar exclusivamente aplicaciones Web 2.0. Para generar una aplicación Win, podrá utilizar GeneXus X Evolution 1. Allí la transacción presentará también un WinForm. 41 Este form podrá ser personalizado, moviendo de lugar, insertando, modificando, eliminando controles, cambiando su aspecto, etc. Conforme con el resultado, Diego recordó que no había documentado nada sobre la transacción. Es norma de los integrantes de ACME el ir documentando a medida que se producen los avances, sobre todo en lo que respecta a las entidades, tal como la transacción Attraction. Así que abrió Main Document, insertó una nueva tabla, esta vez titulada “Objects” y agregó en la primera fila un link hacia la transacción recién creada (del mismo modo en que lo hizo Julia varias páginas atrás, al incluir los links a los documentos), junto con una breve descripción. 42 Podemos ver Attraction como link. ¿Qué sucederá cuando Diego haga clic sobre el mismo? Se abrirá un nuevo tab de documentación, esta vez de la transacción, donde podrá editarla, e incluso abrir la propia transacción: Supóngase que necesita abrir la transacción para mirar su estructura antes de documentar. Pues siguiendo el link Open Transaction: Attraction, podrá 43 hacerlo, y luego para efectivamente ir a la ventana de edición de la documentación, podrá directamente elegir el selector Documentation. Como podrá constatar usted mismo, estimado lector, en GeneXus también se aplica aquello de que “varios caminos conducen a Roma”. Creando la segunda Transacción: Country Luego Diego se abocó a crear la transacción que contendría la información inherente a los países y ciudades. Tenía dibujada la estructura en la servilleta de papel que guardó en su bolsillo cuando se despidió de Mary en el encuentro anterior: 44 Con Main Document abierto en edición, agregó un nuevo link en la tabla, ahora a la transacción Country. Pero ¡todavía no la había creado! Observe que aparece un link, sí, pero sobre un símbolo ?: 45 Cuando Diego haga clic sobre ? automáticamente se creará la transacción que aparecerá abierta en la estructura, lista para que él comience a ingresar los atributos, y el link quedará ahora definido en Main Document. (Sin ánimo de ser reiterativos: ¿cómo era aquello de los caminos a Roma?). Country: una Transacción de dos niveles Como todos sabemos, los países contienen ciudades, es decir, la realidad dictamina que se tienen n cantidad de ciudades por país. GeneXus dispone de una forma muy sencilla y práctica de representar esta realidad, y lo hace a través de transacciones de dos niveles. La transacción Country consta entonces de dos niveles: el primer nivel, también conocido como “prólogo” o simplemente “cabezal”, queda de hecho implícito, por lo que no es necesario identificarlo. En este caso, los atributos del prólogo implican que habrá una sola instancia para cada país. Sin embargo, como existen varias ciudades para un mismo país, se recurre a una representación (en el papel: se recurre a un juego de llaves antecedidas de un nombre para el nivel) que determina un grupo de atributos que son repetitivos para el prólogo: se lo conoce como “cuerpo” o segundo nivel. Ahora, note usted que ambos niveles (cabezal y cuerpo) tienen identificadores propios: el atributo CountryId es el identificador del primer nivel, y el atributo CityId es el identificador del segundo nivel, lo que significa que para un país dado (CountryId) no puede repetirse la misma ciudad (CityId). A partir de esta estructura, GeneXus determina que debe crear dos tablas: la tabla COUNTRY y la tabla COUNTRYCITY. En la primera, la clave primaria es CountryId, mientras en que la segunda, la clave primaria está compuesta por CountryId y CityId. La siguiente imagen muestra cómo quedó la estructura luego de que Diego la ingresara5. 5 Obsérvese que aparecen nuevos dominios. Es de suponer que Diego los creó, o bien antes o bien en el mismo momento de ingresar estos atributos, por ejemplo, al ingresar CountryFullName, en la columna Type pudo haber digitado: “LongName = Character(50)” con el mismo resultado. Suponiendo que Diego ya tuviera definido un dominio Details como VarChar(500,200), 46 Como CountryId y CityId ya existen en la KB (recuerde que fueron creados al grabar la transacción Attraction) al comenzar a digitar las primeras letras de cada uno, GeneXus le abrirá una lista conteniendo los atributos de la KB que empiezan igual, de modo tal que usted pueda seleccionar el que corresponda y evitar seguir digitando el nombre completo. Ídem con CityId y CityName. Para insertar un segundo nivel en una transacción, sitúese en el atributo anterior y pulse <Ctrl+L> (en este caso, sitúese sobre el atributo CountryFlag). Luego prosiga con el resto de los atributos según el mecanismo natural. También lo logra haciendo botón derecho sobre el atributo, donde se le desplegará un menú contextual para, por ejemplo, insertar un nuevo nivel, mover el atributo de lugar, etc. Normalización de la Base de Datos Al definir la transacción Country se produjeron algunos cambios importantes. En particular, la tabla ATTRACTION ha variado en su composición por la sencilla razón de que GeneXus la ha normalizado en virtud de las nuevas dependencias funcionales que se introducen. Vea las composiciones de las tablas ahora. también es interesante destacar que cuando inserte el atributo CountryDetails, GeneXus ya inferirá como su tipo de datos, el dominio Details (ya que lo contiene como última parte de su nombre). 47 Recuerde que una dependencia funcional XÆY es una restricción entre dos conjuntos de atributos de la base de datos, que establece que si para 2 registros coinciden los valores de X, entonces también tienen que coincidir los valores de Y. Esto significa que los valores de los atributos en Y en cualquier registro, dependen de los valores de los atributos en X, o están determinados por ellos. ¿Percibe los cambios? Al crear la transacción Country, y al designar a los atributos CountryId y CityId como identificadores de sus respectivos niveles, se está diciendo que: 1. Todo país se identificará (de manera única) por el valor de CountryId y que este valor tendrá asociado (determinará) un CountryName (y sólo uno), CountryFullName, etc. (establece esas dependencias funcionales). En un Modelo Relacional normalizado, esto deriva en una tabla física con clave primaria CountryId para almacenar esta información respetando esta unicidad establecida. 48 2. Toda ciudad se identificará para un país dado (CountryId), por el valor de CityId, y que por tanto, este par (CountryId, CityId) determinará un CityName y CityInformation. De esto se desprende que deberá existir una tabla física con clave primaria {CountryId, CityId} para almacenar la información de cada ciudad6. Pero entonces: ¿qué hace GeneXus con los atributos de igual nombre en diferentes transacciones? Teniendo presentes los dos enunciados mandatarios en la filosofía GeneXus: • Conceptos iguales deben tener el mismo nombre de atributo, y • Conceptos diferentes no deben tener el mismo nombre de atributo. encontrar atributos de igual nombre en distintas transacciones sólo puede querer decir una cosa: ¡son lo mismo!. Entonces ¿qué puede hacer GeneXus para mantener la base de datos inferida normalizada? Transformar en la tabla ATTRACTION los atributos CountryId y CityId en claves foráneas, y quitar, por tanto, a los atributos CountryName y CityName de dicha tabla (pasan a poder “inferirse” a través de los anteriores; si se dejaran se tendría información redundante) Atributos: almacenados e inferidos Aquí viene a cuenta la pregunta que Mary le hiciera a Diego sobre la confusa presencia de los atributos CountryName y CityName referenciados en la estructura de la transacción Attraction, dado que no van a estar presentes en la tabla física asociada… ¿para qué Diego los colocó en Attraction entonces? Si el lector revisa unas páginas atrás (página 34) y observa el form de la transacción, podrá constatar que aparecen esos atributos. ¿Para qué? La transacción se va a convertir en un programa ejecutándose. El usuario final va a acceder a la transacción mediante su Browser, y a través de una 6 Observe que GeneXus nombra la tabla concatenando el nombre de la transacción al nombre del nivel: COUNTRYCITY 49 pantalla cuyo diseño será el del WebForm, podrá, por ejemplo, ingresar una nueva atracción (nuevo registro en la tabla ATTRACTION subyacente). Sería deseable que cuando el usuario final colocara un identificador de país en el campo correspondiente a CountryId, y uno de ciudad en CityId pudiera visualizar automáticamente el nombre de dicho país, CountryName y de dicha ciudad, CityName. Es decir, que CountryName y CityName sean traídos en ejecución de las tablas COUNTRY y COUNTRYCITY a través de la clave foránea {CountryId, CityId}. Para poder colocar en el form atributos a los que se llega a través de claves foráneas, es necesario que estén presentes, inferidos, en la estructura. ¿Qué sucedería si el usuario digitara para CountryId, CityId un par de valores inexistentes? ¿Y si mediante la transacción Country se quisiera eliminar una ciudad para la que existieran atracciones? Si se nos permitieran hacer estas operaciones, se estaría violando una de las condiciones más importantes de los modelos relacionales para mantener la consistencia de los datos: la integridad referencial. De hecho GeneXus agregará automáticamente en las transacciones el código necesario para realizar estos controles sin que el desarrollador se moleste en hacerlo. Integridad Referencial Diego comunicó a Julia que ya había definido las dos primeras transacciones. A los efectos de comprobar que el modelo reflejara la realidad, ella hizo un Diagrama de Bachman (diagrama de relaciones entre tablas) sobre las tablas inferidas por GeneXus a partir de las transacciones. Para ello expandió el nodo Tables del contenedor Folder View, luego, posicionada sobre cualquier tabla, pulsó el botón derecho para abrir el menú contextual y escogió New/Object, tras lo cual se abrió la ventana Create New Object donde seleccionó el objeto Diagram para finalmente pulsar [Create]. Seleccionó las tres tablas simultáneamente y las arrastró hacia la hoja en blanco, obteniendo el siguiente resultado. 50 En este tipo de diagrama, la punta simple de la flecha representa la existencia de una instancia de la tabla apuntada para cada instancia de la otra; es decir: para cada ciudad existe sólo un país; para cada atracción turística existe una sola ciudad. Recíprocamente, la punta doble representa la ocurrencia de varias instancias de la tabla apuntada para cada instancia de la otra, es decir: para cada país existen muchas ciudades relacionadas; para cada ciudad muchas atracciones. Esto permite determinar las relaciones entre ellas. Por ejemplo, la relación entre COUNTRY y COUNTRYCITY es de 1 a N (1 a muchos) y viceversa es N a 1 (muchos a 1). La de COUNTRYCITY con ATTRACTION también es 1 a N. Para asegurar la integridad referencial, deberá entonces controlarse que cuando se inserte o modifique un registro en la tabla ATTRACTION, exista el registro relacionado en la tabla COUNTRYCITY. Y cuando se elimine un registro de la tabla COUNTRYCITY, no existan registros en la tabla ATTRACTION relacionados. GeneXus resuelve esto automáticamente, agregando esta lógica a las transacciones Attraction y Country, en forma transparente para el desarrollador. Análogas consideraciones pueden hacerse 51 respecto a los controles sobre las tablas relacionadas COUNTRY y COUNTRYCITY. Concepto de Tabla Base y Tabla Extendida Mientras Julia comprobaba que estuviera todo bien, al recuerdo de Diego vinieron fragmentos de la última conversación en el bar: Mira, Mary, por una cuestión práctica, le llamamos tabla base, en el contexto que sea, a la tabla en la que estés “parada” en un momento dado, la que sea foco de tu atención en ese momento, en el que estés queriendo hacer algo con sus registros. Así es que en la transacción Attraction, decimos que su tabla base es la del mismo nombre, porque siempre que trabajes con la transacción, vas a trabajar con un registro de esa tabla, ya sea para insertarlo, modificarlo, o eliminarlo. Y luego vas a seguir por otro, y por otro y por otro. Pero además, cuando estás trabajando con un registro individual de una tabla base, podrías acceder también a la información que está unívocamente relacionada con él, de otras tablas. Ajá, siguiendo las relaciones N a 1, desde el registro de la ¿dijiste tabla base? en la que estemos parados, hasta llegar al registro que tiene la información que queremos. Sí, no hay nada nuevo en eso. Es verdad. Solo la terminología: le llamaremos tabla extendida de una tabla base dada, al conjunto de las tablas a las que se llega partiendo de la tabla base, a través de relaciones N a 1. A ver si entiendo… Si no fuera por la necesidad de tener la información desperdigada en varias tablas para evitar redundancias, la tendrías toda en una misma tabla física. En el caso de las atracciones, tendrías una sola gran tabla con todos los atributos de ATTRACTION, más los atributos de COUNTRYCITY, más los de COUNTRY. No puedes, por lo de las posibles inconsistencias que provienen de hacer los datos redundantes, pero de todas formas le das un nombre a la tabla que tendrías si sí pudieras: y ese es el 52 concepto de tabla extendida. Una tabla que físicamente no existe, pero que te la imaginas. Absolutamente cierto. Verás que utilizamos el concepto a lo largo y ancho de GeneXus y por eso se le ha puesto un nombre. Mientras Diego permanecía absorto en sus recuerdos, Julia comprobó que se cumplían correctamente las necesidades de acceso a la información entre las tablas. Mirando un Diagrama de Bachman es fácil determinar cuál es la tabla extendida de cada tabla (base). Todas las tablas a las cuales se pueda llegar siguiendo las flechas de punta simple (relaciones N-1) desde la tabla base, formarán parte de su tabla extendida. La tabla extendida no tiene existencia física, sino sólo conceptual. Nos permitirá referirnos a la información unívocamente relacionada con el registro en el que estemos posicionados. Comenzando a Prototipar Para poder dejarle paso a Mike, y que éste pueda comenzar a jugar su papel de tester del proyecto, Diego pulsó <F5> para dar comienzo a la prototipación. Con GeneXus no hay diferencia entre prototipar e implementar el sistema. La prototipación no es ni más ni menos que la aplicación generada en una plataforma determinada: ¡lo mismo que la aplicación final! La diferencia entre una cosa y otra es solamente el uso que se le da. Recuerde que ya se había seleccionado el ambiente cuando se creó la KB, de modo que varias propiedades ya aparecían configuradas7, entre ellas el DBMS Default (SQLServer). Pero aún falta algo para poder implementar el sistema… Un asistente (wizard) le pedirá a Diego la información del servidor de base de datos (DBMS), y la base de datos, por esta única vez. 7 Si desea modificar alguna de ellas, alcanza con seleccionar el panel Preferences del contenedor Knowledge Base Navigator y allí buscar la preferencia a configurar, por ejemplo el DBMS asociado, o el lenguaje de implementación, etc. y con <F4> sus propiedades (en la ventana que se abre). 53 Su DBMS es Microsoft SQL Server 2005 Express y se encuentra instalado en el servidor DevelopeSrv de la red. El nombre que Diego dará a la base de datos es TravelAgencyTest: Si la base de datos aún no existe, podrá crearla presionando [Edit conection] y luego el link Create database. También puede posponerlo y GeneXus lo hará automáticamente unos pocos pasos más adelante. Luego GeneXus procederá a la generación y ejecución del Developer Menu8, a fin de que los desarrolladores puedan testar inmediatamente las aplicaciones. Una vez que el ambiente está completamente configurado, cada vez que se presione <F5> GeneXus procederá a efectuar todos los pasos pendientes, necesarios para ejecutar la aplicación. Diego esperaba ver lo que se observa en la siguiente imagen: el primer Análisis de Impacto de esta base de conocimiento. 8 Un sencillo menú de prototipación creado por GeneXus (es un xml conteniendo links a los distintos objetos GeneXus creados, para poder invocarlos en forma fácil y rápida como se verá en breve). 54 El Impact Analysis Report (IAR) que estaba observando le decía que se iban a crear las tablas que se mostraban a la izquierda, y cuyas estructuras y demás información como índices, etc. 9, podían verse a la derecha. ¿Qué es el Impact Analysis Report? GeneXus diseña a partir de la base de conocimiento la base de datos. En todo momento pueden hacerse modificaciones a los objetos existentes, o pueden crearse nuevos objetos, modificándose la KB. El IAR es el resultado de un análisis que realiza GeneXus del impacto causado por las definiciones nuevas o cambios del modelo de desarrollo en una KB sobre la base de datos física asociada. Por tanto, este análisis tiene por finalidad informar al desarrollador sobre los 9 Obsérvese que abajo se muestran las sentencias SQL que se utilizarán para reorganizar (en este caso crear) esa tabla. 55 cambios estructurales que GeneXus deberá llevar a cabo sobre la base de datos para dejarla en el nuevo estado, consistente con la KB. De conformidad con el IAR, Diego pulsó [Create] dando paso a la reorganización de la base de datos10. Como en este primer paso ésta no existía, se va a crear. Será en sucesivos cambios que será reorganizada, en tanto reestructurada. Ahora bien, ¿cuándo se generan los programas? ¡Inmediatamente, listos para ser probados! Desde que Diego presionó <F5> e ingresó la información que le faltaba a la plataforma (nombre de BD y servidor), hasta que se le abrió una ventana del Browser con el Developer Menu listo para ejecutar, no pasó más de un minuto. Diego estudió detenidamente unos listados desplegados por GeneXus antes, que le permitieron ver que iba por buen camino… pero no se apure, estimado lector, que sobre este punto volveremos más adelante y Mike está un poco ansioso por entrar en escena. Mike 007… con licencia para matar Era su turno. Debía destrozarlas. A las transacciones, claro. Con su vista fija en el Developer Menu (desplegado en el Browser tras el <F5>) conteniendo links a las transacciones, entrecruzó sus dedos, los hizo sonar, y tras una breve pausa, se dijo “allá vamos”. 10 El término reorganizar refiere a efectuar cambios físicos. 56 La idea era comenzar cuanto antes a probar, cargando información en la base de datos recién creada por GeneXus y verificar el comportamiento de las transacciones. Pulsó sobre Country y apareció ante su vista una imagen semejante a la que se observa a continuación. Esta es la transacción Country en plena ejecución. 57 Inmediatamente se percató de que el atributo CountryId debería ser autonumerado, es decir, que el usuario no debería preocuparse por asignarle un valor específico ya que CountryId no representa nada de la realidad, es sólo un número que otorga unicidad al registro a ser insertado. Asimismo, cuando Mike se enfrentó a CountryWorldRegion, atributo numérico de apenas un dígito, no supo qué valor numérico representaba cada región del mundo (¿cómo indicar que se trata de un país africano, norteamericano, europeo, etc.?) “¡Hay que mostrarle al usuario final los nombres de las regiones y continentes para que escoja el apropiado!”, se dijo. Así que decidió iniciar con Diego una conversación sobre el tema, de forma tal que todos estuvieran de acuerdo sobre lo que proponía. Volvió a la KB, seleccionó Country y luego ingresó a su sección de documentación. Agregando conversación Entre los selectores que se presentan al abrir la documentación, aparece uno: Talk, que tiene por finalidad permitirnos crear un espacio donde establecer un diálogo con otros usuarios sobre el objeto en cuestión. Note que el texto indicador superior está declarando un documento Talk para la transacción y que finaliza con un signo de interrogación. Este signo indica que aún no existe una página Talk para este objeto, y antes de comenzar 58 a escribir, ésta debe ser creada. Cuando Mike salve este espacio de memoria, se generará una página con su contenido. Al cliquear sobre el signo de interrogación, cambió al estado de edición e ingresó el texto, grabó, y pudo ver: Determinando un Atributo Identificador como Autonumber Rato después, en cuanto Diego leyó lo que había escrito Mike, contestó positivamente en el Talk de la transacción Country y se dispuso a realizar los cambios. Quería que se numerara en forma automática el atributo CountryId, identificador numérico, utilizando la funcionalidad Autonumber de los DBMSs que la soportan. Todos los objetos de GeneXus contienen propiedades que afectan tanto su apariencia como su comportamiento. Las propiedades de todo objeto, atributo, etc. pueden editarse en cualquier momento con <F4> y se abrirán en una ventana flotante. 59 Por lo que Diego se posicionó sobre el atributo CountryId y en la estructura de la transacción Country pulsó <F4>11. Primeramente, asignó el valor “True” a la propiedad Autonumber, y configuró las propiedades como mostramos en la figura. 11 Sin necesidad de entrar a la transacción, también puede buscarse el atributo para editar sus propiedades con <F4>, a través de la ventana Attributes, que se accede mediante View/Work With Attributes. Las propiedades que se modifiquen de cualquiera de las dos maneras aplican al atributo en general, no al atributo en una determinada transacción en particular. 60 Cirugía estética para un atributo Para completar el siguiente pedido de Mike en el Talk, Diego se situó en el WebForm de la transacción, se posicionó sobre el control atributo CountryWorldRegion y pulsó <F4> (Diego tiene la ventana de propiedades con Auto Hide, así que ahora la ventana Properties se desplazó hacia afuera). Ya en la ventana de propiedades del control, en la propiedad ControlType seleccionó “Combo Box”; a continuación, seleccionó la propiedad Values para abrir la ventana Values Editor y cargó los valores para el atributo. Eso fue todo. 61 Si en lugar de haber hecho esto específicamente en las propiedades del control CountryWorldRegion del form lo hubiese hecho en las propiedades del atributo CountryWorldRegion en la estructura, habría hincado el bisturí más a fondo: significaría que en todos los forms donde se colocase ese atributo, por defecto se mostrase como Combo Box, con la definición indicada y no como control Edit. Cuando se percata de su distracción, Diego decide hacer esa definición a nivel de la estructura. Aquí queda claro cómo es que el conocimiento se infiere en la base de conocimiento, y es reutilizado. Pero allí reconoce que lo más conveniente, sin embargo, será crear un dominio enumerado para ser utilizado por cualquier atributo que quiera representar regiones del mundo, en particular, CountryWorldRegion. Edita entonces los dominios y crea uno nuevo, WorldRegion de tipo Numeric(1.0), y en la ventana de propiedades (<F4>) se posiciona en la propiedad Enum values y pulsando sobre el botón abre la ventana para ingresar los valores que tomará el dominio enumerado: 62 De esta manera evitará en la aplicación tener que recordar que el valor 8 correspondía en su sistema a South America, y a partir de allí, cuando se quiera referir a ese valor, escribirá WorldRegion.SouthA, que será sin duda más fácil de recordar que 8. Por tanto edita las propiedades del atributo CountryWorldRegion (por ejemplo, posicionándose en ese atributo en la estructura de Country y pulsando <F4>) y modificará su tipo de datos para que ahora sea del dominio WorldRegion. 63 Observará inmediatamente que el valor de la propiedad ControlType del atributo automáticamente pasa a ser “Combo box”, y sus valores los del enumerado. Reorganizando (reorg) Diego cambió propiedades de dos atributos de la transacción Country, ¿esto indirectamente disparará algún cambio en la tabla física? Veamos el siguiente Impact Analysis Report que se observa luego de que pulsara <F5>: 64 Sí, GeneXus encontró que debe realizar una reorganización de la tabla. Note que ahora aparece el texto “Autonumber” a la derecha del tipo de datos en el atributo CountryId. La reorganización deberá “encender” esa propiedad a nivel del DBMS. Pero, ¿qué pasó con el otro cambio, con el de CountryWorldRegion? ¡Nada! No se trata de un cambio a efectuar sobre la base de datos, sino que solamente vale a nivel de la KB: el dominio WorldRegion sigue siendo de tipo Numeric(1.0) por lo que el atributo CountryWorldRegion no cambia su tipo a nivel de la base de datos. El cambio principal será estético: en todos los forms donde se presente ese atributo como control, lo hará como Combo Box con los valores del dominio enumerado. Aquí se evidencia que no todos los cambios que se efectúen en un objeto GeneXus necesariamente provocarán una reorganización. Veremos otro ejemplo de ello a continuación cuando hablemos de las reglas de una transacción. 65 Diego puede decidir pedirle a GeneXus que haga efectivamente los cambios informados ([Reorganize]) o cancelar la reorganización, con lo cual la próxima vez que se presione <F5> volverá a realizarse el IAR. Elige reorganizar. Al releer lo charlado en el Talk, con intenciones de contestar que todo fue hecho, encuentra que Mike proponía que por defecto todos los Ids de transacciones fueran autonumerados. (Su confusión vino a raíz de que Mike, en lugar de haber documentado esto en el Talk del Main Document, lo hizo en el de la transacción Country). Pero entonces, para implementar lo sugerido, ¿tendría que ir una por una a cambiar la propiedad Autonumber de esos atributos? ¡No!, pronto recordó que todos ellos estaban basados en el dominio Id. Alcanzaba con editar el dominio, sus propiedades y allí, de igual manera que se hizo para el atributo CountryId, “encender” la propiedad Autonumber a nivel del dominio. ¡Todos los atributos basados en el presente y en el futuro en ese dominio, automáticamente heredarán la propiedad! Pero…“¡#@!#!@...!”, maldijo en voz alta. “CityId no es clave primaria simple”. Para dotar a un atributo con la característica de autonumerado, éste debe ser clave primaria simple. CityId es “parte” de la clave primaria de COUNTRYCITY12. ¿Cómo hacer, entonces, para que este atributo no tome la propiedad Autonumber como “True” dada a su dominio Id?13 Sencillo. Se editan las propiedades del atributo CityId (como antes se hiciera con CountryId) y se modifica la Autonumber, llevándola a “False”. Se está cambiando, pues, el comportamiento por defecto dado por el dominio. Lo mismo puede decirse del atributo AttractionCategoryId que por el momento es de Attraction y no es clave primaria, aunque pronto lo será cuando Diego cree la nueva transacción AttractionCategory, unos pasos más adelante. Por esta razón, decidió dejar la modificación en la definición del dominio Id 12 Para dar valores consecutivos a este atributo, habrá, pues, que apelar a una regla: Serial. Le proponemos la investigue luego de ver algunas reglas, en la sección siguiente. 13 De todas formas, como GeneXus sabe de esta restricción, si Diego no se hubiera percatado del problema, GeneXus igualmente hubiera actuado con inteligencia y no hubiera definido el atributo en la base de datos con autonumber. A efectos didácticos preferimos hacer trabajar a Diego. 66 (para “encender” la Autonumber) para más adelante, cuando cree la nueva transacción (dejando este “To-Do Diego” que, creado seleccionando el objecto Document desde la ventana New Object, al guardarlo quedó colgado en Orphan Documents). Transacciones: no sólo una cara bonita, también personalidad Diego decidió incorporar a las transacciones el comportamiento que los usuarios habían descrito respecto a la manipulación de los datos. “Nunca debe ingresarse al sistema una atracción a la que no se le haya dado un nombre”. Abrió, por tanto, la transacción Attraction, pulsó sobre el selector Rules y escribió lo siguiente: Error('Attraction must have a name') if AttractionName.IsEmpty(); En ejecución, si el usuario dejara vacío el campo “Attraction Name”, le aparecerá el mensaje indicado y el control quedará en el mismo campo sin dejarle continuar. Tampoco se le permitirá grabar el registro en la tabla correspondiente. “Nunca debe ingresarse al sistema un país sin nombre y nombre completo” “Sí se permite no ingresar detalles del país, pero en este caso se le quiere dar un mensaje al usuario, advirtiéndole por si lo dejó vacío por distracción”. Seleccionó, pues, el selector Rules de la transacción Country y escribió las siguientes tres reglas en un orden cualquiera: Error('Country must have a name') if CountryName.IsEmpty(); Error('Country must be in full') if CountryFullName.IsEmpty(); Msg('No country datails were provided. Do you want to continue?') if CountryDetails.IsEmpty(); 67 La tercera regla también emite un mensaje, pero a diferencia de la Error, no traba el paso del usuario hacia el campo siguiente, y permite grabar el registro. Existen muchas más reglas, que permiten programar el comportamiento específico que tendrá la transacción. Obsérvese que se escriben siguiendo un estilo declarativo: salvo excepciones no importa el orden. En nuestro caso, se dispararán de acuerdo al orden en que se encuentren los atributos CountryName, CountryFullName y CountryDetails en el form. En una transacción de dos niveles, se están manipulando registros de dos tablas. Por tanto, cuando el usuario ingresa un nuevo país y sus n ciudades mediante el form de la transacción Country, se estarán insertando n+1 registros: 1 en la tabla COUNTRY y n en la tabla COUNTRYCITY. Se insertarán en el orden en que se hayan ingresado en el form. Puede querer dispararse una regla inmediatamente antes o inmediatamente después que una grabación de estas se efectúe. Para ello las reglas pueden condicionarse no sólo con cláusulas if como las mostradas, sino también con eventos de disparo como son el BeforeInsert o el AfterInsert. Luego de grabarse los n+1 registros de la transacción (cabezal y sus n líneas) se realiza automáticamente un Commit. Puede ser deshabilitado por el desarrollador mediante una propiedad de la transacción. También existen eventos de disparo de reglas, que capturan el momento inmediatamente anterior a realizarse el Commit, o el inmediatamente posterior. Por ejemplo, si quisiera realizarse un listado del país recién ingresado con sus respectivas ciudades, podría invocarse a otro objeto GeneXus de otro tipo: Procedimiento, asegurándose de invocarlo luego de haberse insertado los n+1 registros, enviándole por parámetro el identificador de país recién ingresado… y si se quiere que sea antes del Commit se escribirá en la sección de reglas: PrintCountry.call( CountryId ) on BeforeComplete;14 14 Obsérvese que se está utilizando un método de invocación corriente en GeneXus: el Call, que está apareciendo aquí como una regla. También obsérvese que lo que precede al call es el nombre del objeto invocado, en nuestro caso será otro objeto GeneXus de tipo Procedimiento, y por último, obsérvese que los eventos de disparo aparecen identificados en la sintaxis de una regla: Rule if condition on Event 68 Y si se quiere que sea luego del Commit se escribirá: PrintCountry.call( CountryId ) on AfterComplete; Diego deja documentadas las modificaciones efectuadas. Piense cuáles fueron los cambios en la KB desde el <F5> anterior. Lo que se esconde tras el <F5> 1. Lo primero que hace GeneXus tras un <F5> es un análisis del impacto que tendrá los cambios efectuados en la KB, sobre la base de datos. En caso de que haya impacto, desplegará el Impact Analysis Report que puede ser confirmado, en cuyo caso se reorganizará físicamente la base de datos, o cancelado, como ya vimos. Sabiendo que lo único que modificó desde el anterior <F5> fue agregar reglas a ambas transacciones, anticipa que cuando ahora vuelva a pulsar <F5> no habrá reorganización, pero sí los pasos que siguen. 2. GeneXus da paso a la etapa de especificación, que es un análisis de todo lo definido en cada uno de los elementos que componen los objetos que usted haya cambiado, como por ejemplo, las estructuras, los forms, las reglas incorporadas, y más. Esta tarea da por resultado un listado de navegación en el cual GeneXus informa de la lógica que ha interpretado para cada objeto que haya sufrido cambios desde el <F5> anterior, junto con advertencias y errores (en caso de haberlos). En el caso de nuestros amigos, dos objetos han cambiado: Country y Attraction, por lo que esto será lo que Diego verá tras su <F5>: …donde condition es una condición booleana y Event es uno de los eventos de disparo válidos, que invitamos al lector interesado a buscar en las distintas fuentes de documentación del producto. Listamos algunos: AfterValidate, BeforeInsert, BeforeUpdate, BeforeDelete, AfterInsert, AfterUpdate, AfterDelete, BeforeComplete, AfterComplete, etc. 69 No hubo advertencias ni errores. Observe el lector cómo al elegir Country a la izquierda, le informa las tablas en las que realizarán las grabaciones y la integridad referencial que controlará cuando se pretenda eliminar un país, o una ciudad a través de esta transacción. 3. Luego de especificar, se pasa a la etapa de generar el o los objetos que GeneXus entienda necesarios para que la ejecución pueda realizarse en forma 70 completa (si hay objetos que no han variado, y fueron generados en un <F5> anterior, ¿para qué hacerlo nuevamente?). La generación implica la escritura de líneas de código que implementen la programación de los objetos en el lenguaje elegido por usted para el ambiente de desarrollo (por cada una de las dos transacciones de Diego, se generará entre otras cosas un archivo .cs. No olvide que él eligió C# como su lenguaje de desarrollo). Además mientras el desarrollador no cree su propio objeto principal, se creará un xml: el Developer Menu, que contendrá links a todos los programas (objetos) generados. 4. Por último, GeneXus compila los programas y ejecuta (hasta que no se cree un objeto principal de la ejecución) el Developer Menu en el Browser instalado en esa máquina. Todo este proceso es enteramente automático. Usted nunca tendrá que especificar y/o generar un objeto en forma explícita, pues GeneXus es lo suficientemente inteligente como para determinar qué acciones disparar en cada momento. Así fue que, avisado por Diego, previo paso por el Developer Menu, Mike ejecutó Country, ingresó unos cuantos registros, modificó algunos, eliminó otros, y documentó los resultados en la Documentation de la transacción otorgándole el status de okay. Observe lo que sucedió cuando accidentalmente, teniendo al país 1, Argentina, en edición, borró su nombre y salió del campo. ¿Recuerda que Diego programó una regla Error con exactamente ese mensaje un poco antes? 71 Le faltaba aún probar Attraction (habiendo encontrado cosas a modificar en Country, había decidido esperar un poco), cuando vio en el Talk de la misma que Diego le indicaba que postergara la prueba hasta que él creara la siguiente transacción relacionada. Creando la Transacción AttractionCategory En la transacción Attraction, Diego nombró al par de atributos como AttractionCategoryId y AttractionCategoryDescription sabiendo que más tarde crearía una transacción AttractionCategory, que los contendría. 72 La finalidad de esta transacción es categorizar los variados puntos de atracciones de una ciudad. Ejemplos de esto pueden ser las categorías “Buildings/Structures”, “Nightlife”, “Relaxation”, “Adventure”, “Safaris”, “Gastronomy”, “Business”, etc. Con las estructuras de las transacciones Attraction y AttractionCategory estamos diciendo implícitamente que: una “atracción” pertenecerá a una y solo una “categoría”, mientras que una “categoría” podrá contener n “atracciones”. El siguiente Diagrama muestra la nueva relación existente ahora entre las tablas involucradas hasta el momento en el proyecto. GeneXus deberá reorganizar la base de datos en el próximo <F5>, creando la nueva tabla ATTRACTIONCATEGORY y quitando de ATTRACTION al atributo AttractionCategoryDescription, que pasará ahora a ser inferido, a partir de la nueva clave foránea, AttractionCategoryId. Antes de dar <F5>, Diego recordó modificar el dominio Id para que fuera Autonumber (como había dejado anotado en la documentación para hacer cuando creara la transacción AttractionCategory). Ahora sí, <F5>. Analizó los Impact Analysis Report correspondientes: 73 Observe cómo informa sobre los cambios habidos en Attraction. Y para AttractionCategory informa que la tabla es nueva, deberá ser creada. Además presenta una advertencia15: 15 ¿Y si Mike hubiese testado anteriormente la transacción Attraction y hubiese insertado registros antes de esta reorganización? Supóngase que hubiera ingresado una atracción con Id de categoría 1 y descripción “Nightlife”… ¿qué pasaría entonces en la reorganización, cuando se cree la tabla ATTRACTIONCATEGORY y AttractionCategoryId pase a ser clave foránea en ATTRACTION y AttractionCategoryDescription inferido? ¿Qué pasará con el registro pre-existente en ATTRACTION? Para que la base de datos quede en estado consistente, tendrá que crearse un registro en ATTRACTIONCATEGORY, con Id 1… y con descripción “Nightlife”. Esto lo hará el propio programa de reorganización. ¿Qué pasará si existían dos o más atracciones con el mismo Id de categoría y distinta descripción? Es lo que informa la advertencia. Investíguelo. 74 Tras el [Reorganize] confirmado por Diego, obtuvo la nueva base de datos y los nuevos programas generados. Cerró el Browser (pues la tarea de testar todo era de Mike) y finalmente documentó la nueva transacción y los cambios de Autonumber. Mike leyó lo escrito por Diego y se dispuso a prototipar todas las transacciones. <F5> nuevamente… Para que el usuario final no tenga que recordar códigos para las claves foráneas, GeneXus crea objetos automáticamente que se conocen como Prompts y que se invocan mediante la imagen que aparece en ejecución al lado de todo campo clave foránea. Se trata de listas de selección que muestran toda la información de la tabla referida, para que el usuario seleccione la deseada. Así, cuando Mike probó la transacción Attraction, al momento de ingresar la categoría de la atracción “Tour Eiffel”, no recordaba el código que correspondía a “Buildings/Structures” que debía asociarle. Entonces presionó en la imagen que sucede al campo para ingresar el Id de la categoría, , y se le abrió la lista de selección de categorías creada automáticamente por GeneXus, como aparece en la siguiente imagen: 75 Con esa lista abierta, Mike elegirá la primera opción, “Buildings/Structures”, y el control volverá a la transacción Attraction, donde aparecerá en el campo AttractionCategoryId el valor 1, correspondiente; el foco estará allí, y el usuario podrá pasar a ingresar el siguiente campo (en nuestro caso, el Id de país, para lo que el procedimiento podría repetirse). También probó que estuvieran funcionando bien los controles de integridad referencial, por lo que colocó en AttractionCategoryId un valor inexistente de categoría, y vio cómo en ejecución aparecía un mensaje de error indicándoselo16. 16 Segundos más tarde, probaría intentar eliminar en la transacción AttractionCategory una categoría con atracciones asociadas, esperando ver que fallara la eliminación, asegurándose así la integridad referencial. Amigo lector: pruebe y verá. 76 Para que el usuario final no tenga que estar abriendo estas listas de selección cada vez que tenga que ingresar el valor de una clave foránea, GeneXus provee de la facilidad de “disfrazar” en ejecución un atributo por otro. Mike propuso esto último; Diego lo realizó cambiando dos propiedades del atributo AttractionCategoryId, y como resultado, ahora para ingresar la categoría “Buildings/Structures” Mike sólo debe recordar con qué letras empieza, digitarlas y tendrá autocomplete: El atributo en el que el usuario final estará ingresando la descripción, es en realidad la clave foránea AttractionCategoryId, que está “disfrazada” de AttractionCategoryDescription, de manera que el usuario final ingresa descripciones, pero lo que en realidad queda almacenado es ni más ni menos que el Id correspondiente, que se fue a buscar a la tabla, en forma transparente. 77 Ahora quedaba mejorar un poco la usabilidad y la apariencia de lo que iba del proyecto, y dejaron por escrito que habría que comenzar a hacer uso de algunas de las otras herramientas de GeneXus, como los GXpatterns. Y tal vez ya podrían reunirse todos para discutir si poner en producción alguna parte de la aplicación en el sector Tourism de la agencia, de tal forma que pudieran ir probando, cargando países y ciudades y enviando de retorno las sugerencias que encontraran. “La prototipación es un gran recurso, y utilizándola adecuadamente, se evita una exagerada dependencia de la prueba final o, aún, de la puesta en marcha (el “Día D”, donde todo ocurre y donde la ley de Murphy se aplica especialmente)”. Breogán Gonda & Nicolás Jodal Por las características de GeneXus, todo se puede probar en cualquier momento y, en particular, se puede prototipar cada objeto en el momento en que se considere oportuno hacerlo. Esta posibilidad es siempre muy importante y es consecuencia directa de la teoría de GeneXus y, en particular, de su propagación automática de los cambios. 78 Capítulo 3 Hecho a máquina, terminado a mano Planes Inmediatos El proyecto de Travel Agency & Co. ya ha avanzado lo suficiente como para que varias de sus partes sean puestas en preproducción. Es momento de pruebas por parte de los usuarios y a la vez de seguir incrementando el sistema. El problema: pobreza de la aplicación Julia asiste con los usuarios finales para tener un feedback temprano y corregir posibles desvíos. Cuando éstos comienzan a probar el ingreso de datos mediante las transacciones, manifiestan estar satisfechos respecto a los controles y la forma de ingresar los datos individuales, pero quieren consultas con vistas más completas y con algún aumento de funcionalidad. Por ejemplo, quieren poder trabajar con los países de una forma más vistosa y amigable. Quieren poder visualizar en un grid todos los países existentes, seleccionar uno de ellos para poder modificarlo o eliminarlo, o incluso ver la información completa de ese país, incluyendo sus ciudades y toda otra información que pueda tener relacionada. La solución: aplicación vistosa en sólo tres clics Sin titubeos Julia se dijo “Patterns, allá vamos…” Con la transacción Country en edición, tres clics y un <F5>, preguntó a los usuarios finales “¿Es esto lo que desean?”. Los usuarios finales ven en el Developer Menu un nuevo link, Work With Country, que al ser cliqueado abre un nuevo objeto en el browser, WWCountry, que no sólo muestra en un grid los países existentes en el sistema, ordenados 79 por su atributo más significativo, el nombre17, sino que además permite filtrar en ese grid también por nombre (a medida que el usuario final va digitando en la variable de filtro, automáticamente se va filtrando en el grid), así como pagina automáticamente el grid18: Filtro sobre el grid Para desplazamiento entre las páginas del grid 17 Ver página 32. Cuando se ingresan los atributos en la estructura de una transacción, el primero de tipo Character o VarChar ingresado, se marca como el descriptor. Se puede cambiar de descriptor posicionándose en la estructura sobre el nuevo atributo descriptor, y –vía botón derecho- eligiéndolo mediante “Toogle description attribute” en el menú contextual desplegado. Aunque como veremos apenas más adelante, Julia tendrá otra alternativa para hacerlo… 18 El valor por defecto es de 10 registros por página (es decir, carga 10 líneas del grid). Lo hemos modificado a 3 en un lugar centralizado del Pattern Work With del que luego hablaremos. 80 Y no sólo eso, también permite realizar acciones tales como: • insertar un nuevo país, cliqueando en la imagen que aparece arriba y a la derecha del grid, , lo que llamará a la transacción Country en modo Insert19 (para insertar un país y sus ciudades) y retornar. • modificar información de uno de los países, cliqueando en la imagen que aparece en la línea del grid correspondiente a ese país (lo que también llamará a la transacción Country pero en modo Update, para actualizar ese país y retornar). • eliminar un país del sistema (cliqueando ahora la imagen ) lo que también llamará a la transacción, en modo Delete y luego retornará. • visualizar toda la información de un país determinado: observar el link que aparece sobre el nombre del país; cliqueando aquí se abrirá un nuevo objeto (creado automáticamente), ViewCountry, que mostrará toda la información relacionada con el país (en tabs): 19 El objeto Transacción tiene una variable predefinida, &Mode, que identifica el modo en que se encuentra en un momento dado (modo Insert, modo Update, modo Delete o modo Display) de acuerdo a la operación que se vaya a realizar. Si se le envía por parámetro el modo y se recibe en esta variable, entonces la transacción sabrá qué operación se quiere realizar. 81 Es decir, se creó un verdadero “Trabajar con Países” (Work With Countries). Lo mismo podrían pedir los usuarios para trabajar con las otras entidades: atracciones y categorías de las atracciones. ¿Patrones? En líneas generales usted ya sabe de qué se habla cuando se dice patrón: conocimiento común que puede ser aplicado a situaciones distintas. Un pattern en GeneXus es muy potente; evita que usted deba crear “manualmente” objetos de gran versatilidad funcional; permite el reuso del conocimiento; y es un utilitario abierto: llegado el caso, usted mismo puede crear sus propios patrones. Introducen un alto nivel de productividad y generan aplicaciones de calidad, con interfaces más uniformes, y una lógica consistente. Es un generador de conocimiento a partir de conocimiento. 82 Todo lo que Julia hizo fue aplicar el patrón Work With (uno de los existentes) a la transacción Country, obteniendo un trozo de aplicación que implementa todo lo necesario para poder “Trabajar con…” la información asociada, en nuestro caso: “países”. Podemos pensar al patrón Work With (así como muchos otros) como una máquina, cuya entrada es una transacción (y la KB) y cuya salida es un conjunto de nuevos objetos GeneXus (algunos de los cuales mostramos antes en ejecución), más cambios en algunas propiedades de la KB, más la propia transacción modificada20… Personalización: siempre hay un “pero…” Los usuarios respondieron a la pregunta de Julia: “Sí, es esto lo que deseamos, pero… 1. En el grid sólo queremos ver el nombre del país y la región del mundo (el resto de la información no nos interesa). 2. Además quisiéramos poder ordenar los países mostrados no sólo por nombre de país, sino también por región del mundo, así como también por el par {región del mundo-nombre de país}, y por último queremos poder… 20 En particular la transacción deberá declarar que recibe parámetros, dado que ahora se la quiere poder invocar desde el objeto Work With Countries (WWCountry). 83 3. …filtrar el grid por región del mundo (además de por nombre)”. Todo lo que Julia había hecho para aplicar el pattern Work With fue abrir la transacción Country, pulsar sobre el selector Work With que se halla al pie (1er. clic), marcar el check box “Apply this pattern on save” (2do. clic) como podrá observar en la imagen de abajo y grabar (3er. clic). Luego <F5> (tras lo cual los usuarios pudieron ver la aplicación funcionando). Observe los nuevos objetos creados y desplegados en el listado de navegación: Sin embargo, en la siguiente imagen usted apreciará que el selector Work With no sólo permite marcar el check box, sino que edita una estructura arbórea que Julia ni miró al realizar los pasos anteriores. Aquí es donde tendrá que trabajar ahora para realizar las modificaciones pedidas. 84 Configura el web panel WWCountry: Configura el web panel ViewCountry Esta imagen muestra una “instancia” del pattern Work With, la instancia para Country, que permite personalizarlo21. Se puede observar una estructura jerárquica conteniendo nodos y un conjunto de propiedades cuyos valores determinarán la apariencia y el comportamiento de lo generado por el pattern para esta instancia. El pattern Work With generará, entre otras cosas, dos nuevos objetos GeneXus, que pudimos ver en ejecución, y que hemos incluido para usted sobre la imagen de la instancia, de un tipo del que aún no habíamos hablado: Web 21 No hemos definido “instancia”. Puede pensarse como la declaración de cómo se aplicará el patrón al caso particular de este objeto. 85 Panel. Permite, grosso modo, hacer consultas interactivas a la base de datos, de modo amigable22. En la imagen hemos indicado qué nodos de la instancia del pattern permiten configurar las características de qué web panel. Los usuarios le han pedido a Julia modificaciones sobre el primer web panel. Nada han dicho por el momento del segundo (pero tal vez lo hagan…). Observe el lector lo primero que aparece bajo el nodo “Level (Country)”. Recuerde que CountryName era el atributo descriptor de la transacción (el de mayor carga semántica). ¿Qué repercusión tiene el atributo descriptor en lo generado por el pattern? Mucha. Por ejemplo, define cuál será el atributo del grid del WWCountry que contendrá un link hacia el web panel ViewCountry creado automáticamente para ver la información de ese país. Pero este es sólo un ejemplo, como veremos, se utiliza en varios lugares. No obstante, el atributo descriptor, que es heredado de la transacción, si bien puede ser cambiado por otro en el árbol, no implicará que sea también cambiado en la estructura; es sólo el descriptor para el pattern. Web panel WWCountry: nodo Selection (Work With Countries) Obsérvese que el nodo Selection contiene como subnodos: Attributes, Orders y Filter (además de acciones: Ins, Upd, Del, Dis). El subnodo Attributes muestra por defecto todos los atributos del primer nivel de la transacción Country, y son los listados (¿casualmente?) en el grid del web panel WWCountry (“Work With Country”)23 generado… Los usuarios pidieron a Julia solamente mostrar CountryName y CountryWorldRegion, por lo que lo único que ésta debe hacer es eliminar de 22 En un caso tenemos un web panel que nos permite entre otras cosas listar los países, filtrando por nombre, y en el otro tenemos un web panel que nos muestra toda la información de un país determinado y sus ciudades (éstas en un grid dentro de un tab). 23 Este web panel, WWCountry tiene su descripción en singular “Work With Country”, sin embargo, en su pantalla se presenta como “Work With Countries”. Por lo que para referirnos al mismo, utilizaremos indistintamente el singular y el plural, de acuerdo a lo más natural al contexto en el que lo estemos referenciando. 86 ese nodo, con <Supr>, el resto de los atributos, con la excepción de CountryId. Este último no puede ser eliminado tan alegremente. Aunque no se le muestre al usuario final en el grid, deberá estar presente, invisible, como justificaremos en breve. Por lo tanto, lo que hará Julia en ese caso será posicionarse sobre el atributo en ese nodo, y, editando sus propiedades con <F4>, modificar la propiedad Visible pasándola a “False”. Así quedará la instancia; vea la repercusión en ejecución. ¡Requerimiento 1 listo! El subnodo Orders indica el orden por el que en forma predeterminada se desean recuperar los datos que se cargarán en el grid. El orden elegido por defecto es (¿casualmente?) el de CountryName, el atributo descriptor. Los usuarios pidieron poder elegir entre ese orden y uno por CountryWorldRegion, así como uno compuesto por ambos. ¿Cómo puede el usuario decidir en ejecución por cuál criterio quiere ordenar la información a recuperar? Mediante un combo box que le muestre las posibilidades. Julia necesitará agregar los nuevos ordenamientos pedidos bajo el subnodo Orders, a los que les dará un nombre (World Region, Both) que 87 serán las opciones que el usuario final verá en ejecución en el combo insertado automáticamente por el pattern. Así quedará la instancia y luego el web panel generado (en ejecución): Observe un detalle: puede también ordenar en ejecución lo que haya cargado en el grid en un momento dado, por alguna de sus columnas, simplemente cliqueando sobre la misma. ¡Pero cuidado! Si está trabajando con paginado, aquí estará ordenando sólo la página actualmente cargada. ¡Requerimiento 2 listo! Sólo queda el 3. Los usuarios quieren también poder filtrar los países a cargar en el grid por región del mundo. Evidentemente: subnodo Filter de la instancia. Obsérvese lo que aparece en forma predefinida bajo este nodo (vea página 77). Otra vez, ¿casualmente CountryName? Tómese un tiempo, estimado lector, para deducir qué significan los dos subnodos Attributes y Conditions, mirando el web panel 88 WWCountry que GeneXus implementó y la instancia en la página 77. ¿Y si ahora le mostramos lo que hizo Julia en la instancia y la repercusión que esto tuvo en el web panel? Al agregar el atributo CountryWorldRegion bajo el subnodo Attributes, se creará automáticamente una variable del mismo nombre, 24 &CountryWorldRegion , que se colocará sobre el grid del web panel y que permitirá que el usuario final seleccione una región del mundo para que en el grid se le muestren sólo los países de esa región (¿por qué aparece como combo box25 ?). ¿Cómo sabe que debe hacer ese filtro? Pues por lo que Julia escribió bajo el subnodo Conditions. Observe algunos detalles: hay dos filtros definidos (por nombre de país y región). ¿Cuál aplica? Ambos, a menos que alguno no deba ser aplicado (observe las cláusulas “when” de cada filtro. Por ejemplo, cuando &CountryWorldRegion tenga el valor del dominio correspondiente a None. En tal caso, no aplicará ese filtro). ¡Requerimiento 3 listo! Obsérvese un hecho interesante y nada casual: cuando Julia mostró en ejecución a los usuarios cómo modificar un país o eliminarlo, se abrió la 24 Ya habrá deducido que las variables aparecen identificadas en todo el código GeneXus por un ‘&’ que es colocado como prefijo del nombre de la variable. 25 Recuerde que el atributo CountryWorldRegion está basado en el dominio WorldRegion de tipo enumerado. De ese modo cuando se presenta como control en un form, será un combo box. Aquí se trata de una variable que se basa en este atributo y observe cómo hereda el mismo comportamiento, sin tener que decirle nada. 89 transacción Country, editándose el país seleccionado del grid. Para ello el pattern WorkWith tuvo que modificar las reglas de la transacción, agregando algunas que se suman a las que ya habíamos declarado. ¿Para qué? Para poder instanciar, editar el país, recibirlo por parámetro. Echemos un vistazo: La forma que tiene todo objeto GeneXus de declarar los parámetros que recibe y devuelve, es mediante la regla Parm. Cuando desde el “Work With Countries” se elija un país del grid para modificarlo, se invocará a esta transacción enviándole el modo ‘UPD’ que es recibido en la variable &Mode y se enviará además el CountryId correspondiente (es por este motivo que tuvimos que dejarlo en el grid, oculto). Cuando se quiere insertar, se envía ‘INS’ y el valor vacío como segundo parámetro, y así… 90 Web panel ViewCountry: Nodo View (Country Information) Cuando hacemos clic en ejecución sobre el nombre del país, se abre el web panel ViewCountry que nos muestra la información del país correspondiente (ver página 74). Observemos qué datos nos muestra: el nombre del país recibido por parámetro (¿otra vez casualmente CountryName?) y luego en un tab General todos los atributos del primer nivel de la transacción Country para ese país, y dos botones para poder llamar a la transacción pasándole ese país por parámetro (para modificar el país y eliminarlo respectivamente) y en un tab City, nos muestra en un grid todas las ciudades del país. Observe detenidamente el nodo “View (Country Information)” de la instancia del Work With de Country. Si el usuario pidiera que además del nombre del país, apareciera el nombre completo en la parte fija, luego en el tab General sólo se mostraran los atributos CountryWorldRegion, CountryDetails y CountryFlag, y si además quisiera que en el tab City, en el grid de las ciudades sólo se mostrara el nombre de las mismas, ¿qué cambios debería hacer sobre estos nodos para lograrlo? ¿Y si no se deseara el tab City en absoluto? En la instancia aparecerá un “Tab” por cada 26 tabla subordinada (muchos a 1) … pero Julia podría agregar otros Tabs… con otras cosas… No nos apuremos… 26 COUNTRY sólo tiene a COUNTRYCITY en esta relación (muchas ciudades por cada país) y por ello sólo aparece un Tab. Obsérvese que el nombre que eligió GeneXus para nombrarlo fue la descripción del nivel 2 de la transacción Country (la descripción de las líneas, que era City). 91 Concluyendo: Cuando Julia marca “Apply this pattern on save” en el selector Work With de Country, al grabar, se generarán objetos GeneXus como los que estuvimos viendo en ejecución, pudiendo acceder a ellos en el Folder View, bajo la transacción. Asimismo, cada vez que modifique la instancia, al grabar, se regenerarán los objetos que corresponda, en forma automática. ¿Más? Los usuarios, que por momentos parecen tener como especial misión en el mundo encontrar las flaquezas de los desarrolladores y volver sus vidas miserables, pidieron a Julia que luego de modificar un país, no se volviera al llamador, WWCountry, sino que les mostrara toda la información del país recién modificado (es decir, ir al “View Country”). Julia, inmutable, se posicionó en el nodo raíz de la instancia y cambió el valor de la propiedad AfterUpdate para “Go to View”… Obsérvese que inicialmente toma un valor “<default>”. En algún lado tiene que decir cuál es este (y los demás) “<default>”. Este valor especifica cuál es el comportamiento predeterminado para todas las instancias (aquí tenemos la de Country, pero también podríamos querer un Work With Attractions y un Work 92 With AttractionCategories). Si en la mayoría o todos los Work With de la aplicación queremos que el comportamiento por defecto sea que luego de modificar se vaya al View, entonces en lugar de cambiar a mano una por una esta propiedad en la instancia de cada transacción, podemos cambiar en un lugar centralizado el “<default>”. ¿Cuál es este lugar centralizado? Se encuentra bajo el grupo Preferences de la ventana Knowledge Base Navigator: nodo Patterns. Observe en la imagen que sigue que al momento sólo se encuentran los patterns Work With y Category. Pero ya habrá otros de usted, de Artech y de terceros… (“Extensibilidad”… “integrabilidad”… piezas claves en la ideología X) Obsérvese que aquí aparecen todos los comportamientos predeterminados que heredarán las instancias particulares del Work With para cada transacción. Por ejemplo, si queremos cambiar para todos los Work With la cantidad de 93 registros por página de los grids (cosa que ya habíamos hecho sin mostrarle, pasándola a 3), tenemos el nodo Grid27. Invitamos al lector interesado a investigar sobre las configuraciones generales que puede realizar desde aquí. Por ejemplo, puede cambiar el aspecto general de las páginas, los nombres de las mismas, las imágenes para las acciones de Insert, Update, Delete y mucho más. Reflexione el lector sobre lo que significa “trabajar con”. ¿Qué otras cosas podría pedir el infame usuario? (perdón lector, es que también somos humanos). Pregúntele a Julia y ella le podrá contar que en esa misma reunión, se le ocurrió a esa especie particular de homo sapiens que quería: A. Exportar la información de los países con los que estaba trabajando, a una planilla Excel. ¿Puede usted creer? Además dijeron que no querían: B. Que se pudiera eliminar países desde el Work With Countries. Y por si esto no fuera poco, que querían: y C. Agregar otra acción más sobre los países (además de las de Insert Update ), cuyo efecto fuera mostrar en otra pantalla cada país con sus ciudades, ¡todos juntos!, y en un formato más agradable a la vista. (Un poco ambiguo para nuestro gusto… y el de Julia) Julia disfrutó con sorna: ante cada requerimiento que el usuario pedía con ojos maliciosos, hacía un par de clics y listo. Salvo para el último, donde el usuario logró su aparente objetivo: complicarle la vida. Pero ni tanto. No olvidemos que Julia no es técnica, luego en la oficina pediría ayuda a Diego. 27 Si observa en su KB la propiedad Page del nodo Grid, encontrará que tiene el valor “Page.Rows”. ¿Qué es Page? Pues un dominio enumerado creado automáticamente, que contiene un solo valor, Rows, cuyo valor por defecto es 10 y que nosotros hemos cambiado a 3. 94 Acciones en el Work With Countries Aquí se muestran las acciones que el pattern Work With tiene disponibles para implementar sobre los datos del WWCountry Exportación con los que trabaja. En particular obsérvese que existe la opción Export. ¡Otra vez los valores “default”! Otra vez, sí, estos estarán configurados en el objeto WorkWith que vimos antes que centraliza todo esto. En nuestro caso, los valores “default” para Delete y para Export, eran “true” y “false”, respectivamente. Para deshabilitar la acción de eliminación de un país y habilitar la acción de exportación, Julia sólo tuvo que modificar los combos que mostramos, pasándolos a “false” y “true” respectivamente. ¡Requerimientos A y B listos! Ahora queda el último requerimiento, el C: agregar una acción que no es una de las predefinidas. Pero el objeto al que hay que invocar como resultado de la acción, no es creado por el pattern Work With; se quiere hacer “a medida”, para estos datos en particular. Se aparta de lo que es común a 95 cualquier “trabajar con”. Aquí es donde deberá intervenir Diego, implementando en GeneXus ese objeto. Julia escribe en el selector Documentation del WorkWithCountry que esto queda pendiente (¿qué selector Documentation? Observe el tab que se abre cuando usted presiona doble clic sobre el nodo WorkWithCountry del Folder View, bajo la transacción Country -vea página 84 y la siguiente página- … todos los caminos conducen…). Lo hizo bajo un título “To-Do Diego” para que sea implementado por él cuando ponga manos a la obra28. Los insaciables de siempre: ¡quiero más! Tras volver de la reunión con los usuarios, Julia se encuentra con un e-mail de… ¡los usuarios!... solicitándole que en la lista de países del “Trabajar con” se pudiera ver aunque más no fuere un trozo de los detalles del país, CountryDetails, cuestión de poder comentar a sus clientes algo muy breve de cada uno. También querían poder ver para cada país su cantidad de atracciones. Necesitaba agregar dos nuevas columnas al grid del Work With Countries. Una para la descripción corta, y otra para la cantidad de atracciones. Pero no sabía cómo lograrlo, ya que esa información no estaba almacenada en atributos de la base de datos, sino que se tenía que obtener mediante algún procesamiento. Ya tenía en mente llamar a Diego para que le enseñara cómo implementar el requerimiento que había quedado pendiente, el C, por lo que tomó el teléfono inmediatamente. Diego creó el dominio ShortDetails de tipo Character(70). Luego abrió el Work With Pattern Instance de Country y mediante el comando “Add” agregó una variable al grid. 28 Como verá un poco más adelante, a Diego no le resultará nada complicado ubicar esta nota de Julia entre toda la documentación que se supone “desperdigada” entre tantos objetos de la KB, y será gracias a las Categorías Dinámicas, de las que él mismo hablará en el capítulo 6. 96 Obsérvese que en el diálogo de propiedades se le debe necesariamente dar un nombre y asignar un dominio a la variable que se está insertando (es lo que ). Diego le asigna como nombre ShortDetails y como indica el ícono dominio el recién creado de igual nombre. Solamente resta decirle el valor que deberá tomar para cada línea del grid. Como bien intuyes, Julia, cuando ponías los atributos bajo este nodo no era preciso decir más: GeneXus recorrería la tabla COUNTRY, y para cada registro que cumpliera con las condiciones de los filtros especificados (en este caso veo que son por CountryName y CountryWorldRegion), cargaría una línea nueva en el grid del “Trabajar con” con los valores de esos atributos en cada columna. Si los usuarios quisieran la descripción completa del país, bastaría con incluir el atributo CountryDetails, pero lo que quieren es acortar la descripción a 70 caracteres. Por lo que, lo único que tendremos que hacer es especificar en la propiedad LoadCode de la variable que ingresamos, el valor que tomará para cada línea: será la operación de quedarnos con los primeros 70 caracteres del string29, es decir: &ShortDetails = Substr(CountryDetails,1,70). Sí, CountryDetails es el atributo. Así de sencillo. Ya puedes hacerlo sola. Entiendo, y entonces para el otro requerimiento, el de mostrar el número de atracciones que tiene cada país… 29 GeneXus cuenta con múltiples funciones para trabajar con los distintos tipos de datos. En particular, tiene la función substr, que permite recuperar un substring de un string dado. 97 Bueno, en principio ahí el procesamiento que debes hacer para cargar la variable que contendrá esa información es un poco más complejo. Sólo en principio. Fíjate que para el país que se esté cargando en una línea del grid en cada oportunidad, se deberán recorrer todas sus atracciones para contarlas. Tienes tres caminos: uno es programar ese código que te permite recorrer información30, otra vez en el LoadCode; otro, es agregar un atributo a la transacción Country, que contenga esa información, suponte un CountryTotalAttractions. De esta manera sólo tienes que agregar este atributo al nodo Attributes de la instancia, y ¡listo! Pero, espera, no entiendo… En ese caso, cuando el usuario ingrese la información de un país mediante la transacción, ¿va a digitar el número de atracciones que contendrá? ¿Y si después resulta que no coincide con las atracciones reales que ingresó? Eso no les va a gustar a los usuarios. ¡Julia, en cualquier momento me dejas en la calle! Exactamente, no tiene sentido que el usuario tenga que digitar una cantidad que es resultado de un cálculo entre valores que están en la base de datos. Este atributo, CountryTotalAttractions será de un tipo especial, que en GeneXus se denomina fórmula. Verás, ingresaré ese atributo nuevo en la estructura de la transacción Country, y le diré a GeneXus que tengo más información, que sé cómo debe calcularlo, y que por tanto no lo almacene en la base de datos, sino que siempre que se le pida su valor, la calcule como yo le digo: 30 Será el comando For Each, del que hablaremos más adelante, en el Capítulo 5. 98 ¿Count(AttractionName)? Ajá, con eso le dices a GeneXus que quieres que para ese país cuente la cantidad de valores de AttractionName asociados31, ¿no? Ahora, ¿cómo sabe que tiene que contar las atracciones de “ese” país y no de “todos” los países? Bueno, recuerda que GeneXus conoce las relaciones entre los datos y sabe que una atracción pertenece a un país y ciudad determinado. Y cada paísciudad pertenece en particular a un país. Con eso basta. ¿Y qué ventaja tiene esta solución para nosotros, frente a la otra, la de agregar una variable en la instancia del pattern y decirle cómo hacer el cálculo con ese comando for each que dijiste -cosa que, por otra parte, no sé hacer-? Pues que aquí has implementado una solución general. Si en otros lados, en otros objetos, cuando recuperas un país necesitas saber su cantidad de atracciones, sólo con nombrar el atributo fórmula ya se dispara el cálculo, y no tienes que programarlo a mano. Ahora sólo tienes que agregar al nodo Attributes de la instancia este atributo fórmula. ¡Ya están listos los dos requerimientos del e-mail! 31 Para usted que sí es informático: para cada país contar la cantidad de registros asociados de la tabla donde se encuentra AttractionName, que en este momento es ATTRACTION. 99 Pero espera… habías dicho que teníamos tres caminos posibles para implementar esto último y sólo me has explicado dos. Qué suerte, Julia. Estaba esperando que te dieras cuenta. De hecho, el tercer camino es el que yo hubiera utilizado desde un principio en este caso. Si no lo hice fue para aprovechar a explicarte esto de las fórmulas globales. Lo que hicimos sería lo más recomendable si creyéramos que muchas veces, y en lugares distintos, vamos a tener que realizar esa misma cuenta de las atracciones. Pero si por el contrario, será en pocos lugares, tal vez lo mejor hubiese sido directamente definir la variable como antes en el nodo Attributes de la instancia del Work With, por ejemplo &CountryTotalAttractions, y en la propiedad LoadCode de la variable, especificar &CountryTotalAttractions = count( AttractionName ). ¿Pero no es la misma fórmula que antes? Sí y no. Aquí no has definido un atributo fórmula, sino que estás utilizando la fórmula localmente, como si fuera una función. Por el contexto en el que está especificada -en nuestro caso estará en un grid que recorre la tabla de países- sabrá que deberá contar solamente las atracciones del país que se está cargando en cada caso. Existen diversos tipos de fórmulas en GeneXus (para sumar, contar, buscar, maximizar, minimizar, sacar promedios). Es importante destacar que se pueden definir tanto en forma global, como atributos, así como localmente donde se necesite. Cuando se definen globales serán atributos virtuales: no estarán almacenados en la base de datos. Así mostró a Julia, tras el <F5> cómo luce ahora el “trabajar con países”: 100 Quod Pattern non dat, GeneXus praestat32 En lo que Julia demoró en ir a servirse un café, saludar a un compañero y volver con la taza humeante, Diego ya había implementado el último requerimiento (el C pendiente). Como no estaban seguros si implementaba lo que deseaban los clientes, capturaron la siguiente pantalla y se la enviaron por e-mail: 32 N. de A. Frase latina, “Quod Natura non dat, Salamantica non praestat” para referir a que si la naturaleza no da inteligencia, ni la prestigiosa Universidad de Salamanca puede darla. 101 ¿Cómo lo hiciste tan rápido, Diego? ¿Me enseñas? No puede ser complicado. No lo es; lo que hice fue crear un nuevo objeto en GeneXus, de tipo Web Panel. Es el que estás viendo en la pantalla en ejecución. Fíjate que lo que deseas es mostrar información de países y de ciudades (las suyas). Olvídate por un momento de los países, y supón que sólo quieres listar información de las ciudades, en un grid. Lo único que debes hacer es insertar un control de este tipo en el form del web panel y decir qué atributos compondrán sus columnas. Sólo con hacer esto, cuando presiones <F5> verás en ejecución listadas todas las ciudades existentes. GeneXus observa los atributos presentes y de ellos infiere la tabla que deseas navegar (le llama tabla base). Fíjate –creó un web 102 panel “Cities” en un segundo, insertó el grid mediante la toolbox que ofrece todos los controles que pueden insertarse en un form, eligió el atributo CityName, y quedó el form del objeto como puede ver a continuación- en nuestro caso: los atributos que colocamos en el grid (CityName) son de la tabla COUNTRYCITY y esa será la que navegará GeneXus. La grilla que hemos insertado se conoce como Grid, a secas, grid estándar, ya que permite mostrar la información de un registro de una tabla -y de su extendida-, con un elemento de información por columna -observa que nuestro grid tiene una sola columna, corresponde al atributo CityName, pero si tuviéramos otro, lo colocaría en otra columna-. Ahora olvídate por un instante de las ciudades, y piensa que algo muy parecido quieres hacer con los países, sólo que quieres mostrar la información de cada país no en columnas rígidas, sino de forma más flexible, como puedes ver en la pantalla que le enviamos a los clientes. Observa el web panel que hice mientras ibas por tu taza de café: - 103 - Observa que ahora aparece el grid de ciudades, junto con los atributos CountryFlag y CountryName dentro de un rectángulo. ¿Qué es? Otro tipo de grid, uno llamado Free Style, pues como ves, la información del país que quieres listar se muestra no por columna, sino toda junta, con el diseño que tú decidas. Pero también es un grid. Mostrará por tanto, tantos países como existan en la tabla de países. Observa que en sus propiedades aparece una Columns, a la que modifiqué el valor de “1” a “3”, y es por ello que puedes ver en la imagen que le enviamos a los usuarios tres países por fila. Rows “0” significa que no realice paginado, sino que muestre todos los países existentes. Luego observa que dentro del Free Style, podemos colocar cualquier control, incluyendo otro grid. Y al hacerlo, GeneXus encuentra las 104 relaciones entre las tablas, y es por esta razón que está listando por cada país, sus ciudades en el grid, sin tener que indicarle nada, lo hace solo. Si no te convence, míralo tú misma en el listado de navegación que aparece luego de la especificación, tras el <F5>. A ver si lo entiendes… El país que se está cargando en el id - Lo que sí entiendo es por qué lo implementaste tan rápido… no es que “Natura te dio”, es que “GeneXus te prestó”. ¡No tuviste que hacer casi nada!, Diego… Cambia esa cara, estaba bromeando. ¿No vendrá tu amiga hoy a verte? - … además podríamos hacer otro web panel, invocado vía un Tab del View de países, que dado el país muestre las imágenes de las atracciones de 105 ese país, como una galería de imágenes que se deslizan. Quisiera hacer algo que impresione… – quedó pensativo. - ¿A los usuarios?… Bueno, Diego, veo que hoy no estás para bromas. Antes de que sigas, quedó pendiente poder llamar desde el Work With Countries a este web panel que acabas de implementar. - Es verdad, pero eso sabes hacerlo. Simplemente debes agregar una nueva acción a las ya predeterminadas (insert, update, etc.) en la instancia del Work With de Country: la asociaremos a un botón fuera del grid y le indicaremos que en esa acción queremos invocar al web panel “CitiesPerCountry” que acabo de crear. Sobre el nodo Selection (Work With Countries), con botón derecho, seleccionar “Add actions” y luego posicionarse sobre el nuevo nodo Action y otra vez repetir la operación. Vea lo que hicieron nuestros amigos, y la repercusión en ejecución: 106 el web panel creado… ¿Botón dentro del grid? ¿Se quiere imagen para la acción en lugar de botón? ¡Requerimiento C listo! Nuevo Tab en el View Country Diego quería impresionar a alguien, no sabemos a quién (quizás a usted, estimado lector) con un nuevo requerimiento: en el ViewCountry, agregar un nuevo control Tab que muestre en una galería de imágenes las atracciones de ese país. Es decir, algo que en ejecución se verá: 107 En todo momento se mostrarán 3 imágenes, siendo la del centro la activa. Cuando se pulsa sobre la de la derecha, la del medio pasa a la izquierda, la de la derecha al centro (pasa a ser la activa) y aparece una nueva a la derecha. Para lograr esto, Diego agrega un nodo Tab en la instancia del pattern Work With correspondiente al View, que se cargará con el contenido de otro objeto GeneXus que luego deberá programar. Podría ser un web panel, pero en realidad tiene que ser un objeto que se pueda ejecutar dentro de otro (en nuestro caso del ViewCountry). Se tratará de un Web Component. A los demás efectos será igual que un web panel. Le llamará CountryAttractions, y deberá indicar en el nuevo Tab que se cargará con el mismo: 108 Se le deberá pasar el CountryId como parámetro al Web Component que implementa la galería de imágenes. ¿Cómo se implementa? Otra cara de la extensibilidad: User controls Además de los controles estándares que usted suele insertar en sus paneles (attribute, grid, text block, etc.), GeneXus X le permite crear sus propios controles personalizados: user controls. Una vez que un control de usuario es instalado, puede ser utilizado como cualquier control estándar. GeneXus X viene con algunos controles de usuario ya instalados (vea la toolbox en el IDE; ellos se encuentran agrupados bajo el rótulo User Constrols): uno de ellos es el ImageGallery. Observemos en la foto composición que sigue que Diego insertó un control de ese tipo en el form del web component, y luego lo único que tuvo que hacer fue configurarle sus propiedades. Al insertar este control en el form, se crea automáticamente un tipo de datos estructurado, ImagesData, más conocido como SDT por su sigla en inglés, que en forma flexible permite construir listas de largo variable de records. En este caso, una lista de elementos en el que cada uno tendrá la información de la atracción a mostrar; esto es: su id, el lugar donde se encuentra la imagen (en general se almacena la 109 imagen en tamaño grande, y la misma imagen en tamaño pequeño) y la descripción de la misma. Junto con el SDT se crea automáticamente la variable &imagesData basada en el mismo, justamente para cargar las imágenes a mostrar. Por tanto, lo único que Diego debió hacer fue programar la carga de esa variable. ¿Cómo lo hizo? En el evento Start del Web Component escribió la invocación a un objeto que creó inmediatamente, que justamente tiene por objeto cargar estructuras jerárquicas de datos, como SDTs. Ese objeto es de un tipo que veremos más adelante (el Data Provider). No importa que ahora no entienda este código. Sólo queríamos mostrarle lo fácil que es lograr este requerimiento. 110 ¿Look & Feel consistente, sin transpirar? “Tener un look&feel consistente es hoy en día un deber de toda aplicación Web. Crear y mantener cada página de una aplicación Web asegurando la consistencia con el resto del sitio toma gran tiempo de programación” leyó Julia en voz alta de una revista técnica que estaba sobre la mesa del escritorio de Diego. 111 No con las Master Pages –acotó Diego-. Como te conté, un web panel puede implementar una página web común y corriente, como la que hicimos, CitiesPerCountry -con los grids Free Style y estándar-, es decir, una página web independiente que será invocada por otra -en nuestro caso por WWCountry-; o puede implementar una página web que será parte componente de otra, como la de la galería de imágenes que era un Web Component parte del ViewCountry; o por último, puede ser una página maestra, es decir, una página que contiene centralizado allí lo que será común a todas las páginas del sitio, como el encabezado y el pie de página que has visto por todos lados repetido; incluso suelen tener un menú vertical a la izquierda, de manera tal que sólo esté en un lugar, y que si se quiere modificar, se lo haga en un solo sitio. ¡Con razón! Cuando empezamos con el proyecto vi a Mike ejecutando las transacciones y no entendía cómo era que todas salían con el mismo encabezado y pie de página (que también veo ahora que tienen el Work With Countries, y el View Country) siendo que en el form de esos objetos esas partes no aparecían. Existe una master page que programa ese look común, ¿verdad? Efectivamente. Mira esto- dijo Diego a la vez que seleccionaba la transacción Country y accedía a sus propiedades-. Verás lo mismo en cada transacción y en los web panels creados por patterns: AppMasterPage es una master page creada automáticamente con la KB -observa el contenido de la carpeta GeneralWeb- que es la que contiene ese encabezado y pie de página que se repite por 112 doquier. Si abrimos este web panel, mira sus propiedades. ¿Qué te dice esto? Y aquí no sólo se programa el form común: también se le puede programar el comportamiento común. Es más que habitual que las páginas sean restringidas a usuarios autorizados, que deberán loguearse al sistema antes de comenzar a navegar por las distintas páginas del sitio. Antiguamente la validación de que el usuario estuviera autorizado debía repetirse en cada página web a la que se accediera. Sin embargo, con las master pages, alcanza con escribir el código de validación solamente una vez, aquí, y ¡listo! Nos aseguramos que todas las páginas asociadas realizarán esa validación, y su código no estará desperdigado por todos lados. ¡Qué bueno! Digo, para ustedes, que tienen que trabajar menos para que les quede todo consistente. Me voy a almorzar y luego a impactar el pattern al resto de las transacciones que tenemos… ¿Ya almorzaste? 113 114 Capítulo 4 Vuelo al mundo de los subtipos El proyecto avanzaba según los tiempos estipulados. Cada uno en lo suyo. Dada la filosofía incremental de GeneXus, no temían a los cambios ni a lo nuevo. El modelo amparaba todo lo necesario. Sólo lo necesario. Restantes Transacciones al vuelo Diego no había almorzado, no tenía apetito. Debía comenzar a crear el resto de las transacciones, pero decidió esperar un rato. Quería lucirse frente a Mary, a quien había ofrecido mostrar más de GeneXus. En lugar de llevar su notebook al bar, le había propuesto encontrarse, esa misma tarde, en su oficina en ACME. Mary, te ves bien. ¿Cómo estás? Mejor. Te agradezco que me hubieras invitado a vernos de nuevo, aún cuando estemos en empresas competidoras. Es un lindo gesto de tu parte. - Mmm… bueno, no es tan así – dijo pensativo, mientras en la pizarra a sus espaldas escribía: “la Comunidad” –. No, por favor, no me malinterpretes, no quise decir… Me refería a esto de la rivalidad, de la competencia. Mira, entre los miles de desarrolladores GeneXus existe en forma “etérea” algo que se conoce como “la Comunidad”. Es muy estimulante. “Sinérgico”, escucharás decir. Tú me preguntarás si es un club; pues no desde el punto de vista de actividades sociales o deportivas, al menos por ahora –agregó sonriendo, intentando disimular el temblor en la voz –, pero sí respecto a la interacción. Es como un tejido del que todos formamos parte, compartiendo conocimiento y soluciones a través de GeneXus y de las demás herramientas que se han desarrollado alrededor. Existen muchas extensiones creadas por la Comunidad para satisfacer distintas necesidades. Hay muchos sitios Web de donde descargarlas, y a donde subirlas también. La Comunidad es muy activa. Existen decenas de foros donde se intercambian soluciones de todo 115 tipo. Yo estoy en las listas de varios de ellos, y me considero un miembro activo. Suena lindo. Pero, ¿funciona? O sea, ¿te dan una mano desinteresada cuando la necesitas, o es puro cuento? Es que es hoy por mí y mañana por ti. Todos necesitamos una mano de vez en cuando. Si en vez de vernos como competidores nos sentimos cooperadores, trabajamos mejor, más animados, llegamos a mejores soluciones, para todos. Tiramos para un mismo lado: hacer crecer a la Comunidad, en beneficio de todos. Puede resultarte un poco ingenuo, pero la prueba está a la vista: funciona. Como evidencia de ello, el propio GeneXus está compuesto por extensiones de GeneXus. Es decir, la Comunidad puede seguir haciendo crecer a GeneXus con sus propias extensiones. De hecho, GeneXus puede considerarse en sí mismo una extensión. Tras reparar en esa luz antigua, conocida, que adquiría su mirada, continuó más seguro: Mira – invitó Diego al tiempo que abría la ventana Extensions Manager a través de la opción Tools del menú de GeneXus –. Esta es la lista de extensiones que componen GeneXus X en este momento. La mayoría han sido desarrolladas por Artech y las otras por personas de la Comunidad que las publican para que diferentes miembros puedan hacer uso de ellas. Veo que en la Start Page aparece una sección que dice “Extensions”, con un abstract escrito por el autor, y un link para instalarla. Sí, y si te fijas, aparecen unas cuantas que no dicen By Artech. Son extensiones realizadas por terceros, miembros de la Comunidad. En cuanto al proyecto en sí y al punto en que habían llegado, la realidad reclamaba otras transacciones. Tras una breve ojeada a lo ya hecho, Diego pasó a crear Airline. Su finalidad: 116 almacenar el nombre de las compañías aéreas involucradas con Travel Agency & Co. Transacción Flight: vuelo al mundo de los Subtipos En cuanto Diego miró sus papeles, agradeció por la complejidad asociada a la próxima transacción. A Mary solían gustarle los desafíos. Mary, vamos a crear la siguiente transacción, pero va a hacer falta un poco de concentración de tu parte –advirtió tratando de predisponerla – Mira este caso. La transacción contendrá información relativa a los vuelos; cosas tales como la descripción, la aerolínea, la hora de despegue, la cantidad máxima de pasajeros, fáciles de entender y de inferir; y por otro lado, tendremos la ciudad de donde se sale y la ciudad a donde se llega33, información que debemos conocer y almacenar porque está comprendida en la realidad. ¿Recuerdas que te comenté en el bar que dos atributos con igual concepto deben tener el mismo nombre? Sí, fue cuando me explicaste cómo crear la transacción Country. Recuerdo perfectamente que mencionaste que había una excepción… Llegó la hora de aclarar eso. Hay momentos en los que no hay más remedio que llamar distinto a dos atributos que son básicamente lo mismo. Mira, en la transacción Flight se nos presenta el problema que da lugar a la excepción que lo resuelve: en cada vuelo existen dos países y dos ciudades involucradas, que cumplen roles diferentes: un país y una ciudad son de partida y otro país y ciudad son de destino. Es decir, tenemos dos referencias a una misma tabla, COUNTRYCITY. Observa, si sólo escribiéramos esto que dibujo en el papel: 33 Recuerde que estamos simplificando la realidad: supondremos que no habrá nunca más de un aeropuerto por ciudad. 117 …podríamos decir que los atributos CountryId y CityId en la transacción Flight representan una ciudad de un país, pues se llaman igual que los identificadores de la transacción Country. Es decir, serán una clave foránea a la tabla COUNTRYCITY. Hasta allí nada nuevo bajo el sol, pero ahora debemos agregar otra ciudad34, en otro rol, el de “ciudad de destino”. Es decir, tiene que aparecer una doble referencia a una ciudad: en un caso en un rol de ciudad origen, y en el otro de ciudad destino. ¿Cómo hacemos esto? GeneXus no nos permitirá agregar la misma pareja de atributos en la misma transacción, eso es obvio porque ¡¿cómo se discriminaría entre un atributo y otro?! Pero podríamos hacer algo así –tomó el lápiz y escribió dos atributos nuevos, luego de lo cual antecedió a los dos primeros con la D de “departure” y a los dos últimos con A de “arrival”. Hmm… pero si GeneXus entiende que si se llaman distinto son otra cosa… ya no habría conexión entre vuelos y ciudades, ¿verdad?... Verdad. ¿Solución? Poder decirle a GeneXus que si bien se les ha cambiado el nombre a los atributos, por la necesidad de diferenciarlos, de todos modos corresponden al mismo concepto. Es decir, que no se confunda: que por más que el atributo DCountryId no se llame 34 Cuando hablamos de “ciudad”, nos referimos a una ciudad real, como París. Para identificar una ciudad, necesariamente debemos dar el par {CountryId, CityId}. Está sobreentendido. Por el diseño de las transacciones, una ciudad no tiene existencia independiente del país al que pertenece. Así, existe ciudad Rosario de Argentina y ciudad Rosario de Uruguay. Pero no existe Rosario, independiente del país del que se trate. 118 CountryId, a todos los efectos es como si se llamase así, de modo que siga estableciendo las relaciones de integridad referencial igual que antes. Decirle esto a GeneXus es decirle que DCountryId es un subtipo de CountryId35. La solución completa, por tanto, es agrupar estas parejas en dos grupos de subtipos, ¿ves? – inquiría mientras rearmaba el croquis. - ¡Ah, comprendo! Gracias a ello, DCountryId y DCityId serán una clave foránea en FLIGHT a la tabla COUNTRYCITY, al igual que ACountryId y ACityId, ¿verdad? Tienes dos claves foráneas a la misma tabla. Así es, y se deben realizar todos los controles de integridad referencial automáticamente –contestó al tiempo que descubría un entusiasmo nuevo en Mary. ¿Podría luego quitarla del teclado? De todos modos el problema no se terminó aquí – continuó Diego –. En esta transacción, además de representar que un vuelo tiene un país-ciudad de origen y uno de destino, cosa que ya hicimos, queremos poder inferir el nombre de cada país y ciudad, para verlos en el form. Es decir, queremos inferir para mostrar atributos que pertenecen a la tabla extendida de FLIGHT, pero ¿cómo los inferimos si tenemos una doble referencia a la tabla?, es decir, no podemos poner directamente CityName, porque ¿de cuál se trataría? ¿Del atributo que se debe inferir de la clave foránea {DCountryId, DCityId}, o del que se debe inferir de la otra clave foránea {ACountryId, ACityId}? ¡Hay una ambigüedad! “Y bueno, le tendrás que cambiar el nombre también a esos atributos que necesitas inferir (CountryName, CityName) y colocarlos en el grupo de 35 Recíprocamente, decimos que CountryId es supertipo de DCountryId 119 subtipos adecuado” – pensó Mary, pero no emitió palabra –. Muéstrame cómo se definen estos grupos en GeneXus, anda; así lo termino de entender. Diego seleccionó el icono Subtype Group de la ventana New Object, y le asignó por nombre “Departure”. Mary mantenía un silencio cómplice… Te había dicho grupos, y realmente es así. Creas uno, le das un nombre y le incluyes los “nuevos atributos”36 apuntando a sus respectivos supertipos. Entonces, expresado así, GeneXus sabrá que el atributo FlightDepartureCountryName será inferido a través del atributo FlightDepartureCountryId y no otro. Por pertenecer ambos al mismo grupo, “Departure”. ¿Comprendes la conexión? Comprendo –respondió ella–. Esto quiere decir que cuando el usuario digite un valor sobre la clave foránea {FlightDepartureCountryId, FlightDepartureCityId} no sólo se va a disparar el control de integridad referencial sobre COUNTRYCITY sino que se va a inferir en el atributo FlightDepartureCityName el nombre correspondiente a ese código de paísciudad y también, por tabla extendida, se inferirá el CountryName de ese país 36 Obsérvese cómo Diego ha establecido el nombre de los subtipos respetando la nomenclatura GIK y comienzan con “Flight”, el nombre de la transacción, y continúan con el nombre del grupo, “Departure”. De esta manera se asegura una semántica clara, estableciendo el rol que juegan en la transacción. 120 en FlightDepartureCountryName. Ahora habría que crear el otro grupo de subtipos, el de los arribos, ¿verdad? Bueno, no necesariamente, pues ya están claramente identificados los “dos” grupos; uno es el explícito, el que acabo de crear. Y el otro está implícito si yo coloco los atributos CountryId, CityId en la estructura de la transacción37. Pero por claridad, solemos en casos de este tipo mantener todas las ocurrencias en grupos, porque si no uno de los roles, en este caso el de ciudad de arribo, no quedaría claramente identificado. Te invito a que tú sola armes el otro grupo, ¿te animas? –preguntó Diego, señalando con su mano el laptop. Bueno, veamos… – dijo ella controlando su impulso de abalanzarse sobre el teclado. Tipeó, tipeó y tipeó, hasta que… ¡Perfecto! Diego ya tenía todo para crear la transacción Flight en GeneXus. Esta transacción será de dos niveles dado que para cada vuelo existen tres clases (“Business”, “Standard” y “Executive”) con tres precios diferentes, lo que en realidad la transforma en una lista de precios (relación 1-N). Todos los vuelos tienen hasta tres tarifas, no más. Otra cosa importante: el identificador de vuelo no deberá numerarse automáticamente, dado que ese número lo determina un estándar internacional, que considera ruta y hora. Por ejemplo, el vuelo 0174, Montevideo-Buenos 37 Sólo con cambiarle el nombre a una de las dos ocurrencias de CountryId y de CityId alcanzaría para eliminar la ambigüedad. 121 Aires, corresponde a un vuelo de la aerolínea Pluna de los sábados de mañana. Si ese vuelo cambiara para la tarde, también cambiaría su número de vuelo. La siguiente imagen muestra la estructura final. Observe usted el ícono que indica que el atributo es un subtipo y además parte de clave foránea; y el ícono que indica que es un subtipo, por un lado, e inferido, por el otro. Capacidad máxima de pasajeros en el vuelo Viendo que el atributo FlightClassType sólo puede tener hasta tres posibles valores fijos: “Business”, “Standard” y “Executive”, Diego creó un dominio enumerado para él. El atributo FlightDepartureTime almacena la hora del vuelo (no nos interesa la fecha). Obsérvese su tipo de datos DateTime. De él, sólo se utilizará la parte que representa Time, por lo que en sus propiedades, modificaremos bajo el grupo Picture la de nombre Date format. La pasaremos a ‘None’. Antes y después Para mostrarle a Mary en ejecución la nueva transacción, hizo <F5> tras lo cual tuvo que reorganizar la base de datos (tres nuevas tablas serán creadas). 122 Mary no pudo ocultar su cara de decepción. Diego, que estaba pendiente de todas sus reacciones y que también percibió un form un poco pesado de ver, agregó inmediatamente: “vamos a arreglar este form que está muy poco amigable”. Para hacer el ingreso de país de partida y ciudad de partida más amigable para el usuario, podría utilizar combo boxes dinámicos, o disfrazar al Id del Nombre (recuerde lo que hizo en la página 69) y lo mismo con el país y ciudad de llegada, así como con la aerolínea. Comencemos con los países y las ciudades; podría editar las propiedades del atributo FlightDepartureCountryId en la estructura y modificar su propiedad ControlType (y hacer lo mismo para los demás) a Dynamic Combo Box. Ajá -se apuró Mary-,y en la propiedad ItemDescriptions tendrías que poner FlightDepartureCountryName, que es donde quieres inferir el nombre del país, así saldrán como descripciones en el combo, los nombres de los países de la tabla COUNTRY… 123 Bueno, sí, eso es lo que quieres, pero por eso mismo, allí deberás colocar el atributo CountryName y no el subtipo, porque en verdad es así como se llama el atributo en la tabla COUNTRY y no FlightDepartureCountryName. Así, Diego escribió CountryName en la propiedad ItemDescriptions y también modificó el valor de la propiedad ContextualTitle del atributo FlightDepartureCountryName que ahora dice “Departure country”. Hizo análogas acciones para FlightDepartureCityId, para FlightArrivalCountryId y para FlightArrivalCityId. Observe cómo luce ahora el form, sin que Diego tocara nada en él. Hizo lo mismo para Airline, y además editó las propiedades del grid en el form, modificando el valor de Rows, pasándolo de 5 a 0. El valor de esta propiedad especifica la cantidad de líneas vacías que se presentan en ejecución para que el usuario ingrese los datos allí. Obsérvese que no es necesario tener ninguna línea vacía, dado que abajo del grid aparece [New Row] para agregar una línea vacía al grid y poder llenarla. Nuestro amigo hizo <F5> para mostrarle a Mary en ejecución el “después”: 124 ...cuando ésta sorpresivamente anunció que debía irse. Mil gracias, Diego. Ahora discúlpame pero debo irme, te llamo en estos días”. Un vuelo real: transacción FlightInstance La transacción Flight pretende manejar datos generales de un vuelo, por ejemplo, el 0174 de la aerolínea Pluna, Montevideo-Buenos Aires, que sale todos los días a las 11:15 AM. Pero por cada vuelo 0174, tendremos un avión real, que realizará en un día determinado, ese viaje, con determinados pasajeros. Lo que consigue la siguiente transacción es representar el vuelo real, con información como la fecha y hora de partida, la cantidad de pasajeros, sus 125 reservas con sus nombres y números de asiento. Aquí el identificador sí será numérico y autonumerado. count(FlightInstancePassengerSeatNumber) Con este diseño, FlightNumber será una clave foránea. ¿Cuántos FlightInstance podrán haber para el mismo vuelo, FlightNumber? ¿Corresponde a lo deseado? Existen algunos atributos aquí que merecen especial atención. Por ejemplo, FlightDepartureTime38 es la hora de partida estipulada del vuelo (es inferido de FlightNumber), mientras que FlightInstanceDate y FlightInstanceTime son la fecha y hora reales, del vuelo real. La hora del vuelo real, se inicializa en forma predeterminada con el valor de FlightDepartureTime, pudiendo ser cambiada por el usuario, para lo que Diego declaró en las Rules: default(FlightInstanceTime, FlightDepartureTime); Esta regla sólo se ejecuta cuando el modo es Insert. Por otro lado, observe la otra regla que ha agregado Diego: error('Number of passengers exceeded') if FlightInstanceNumberOfPassengers > FlightTotalCapacity; 38 Si bien los atributos FlightDepartureTime y FlightInstanceTime son de tipo DateTime (permiten almacenar una fecha+hora), sólo nos interesa la representación de la hora. 126 El atributo FlightTotalCapacity es inferido a través de FlightNumber. ¿Y FlightInstanceNumberOfPassengers? Es una fórmula que cuenta la cantidad de líneas que se han ingresado en el subnivel. Cada vez que se agregue un nuevo pasajero en el grid del form (que no hemos mostrado pero que usted imaginará sin problemas), se estará agregando un nuevo número de asiento para ese vuelo real, FlightInstancePassengerSeatNumber, por lo que la fórmula se actualizará, sumando 1, y si con ese pasajero se ha superado la capacidad, se disparará el error que impedirá que se inserte en la base de datos el pasajero para ese vuelo. Por último, Diego declara la regla: error('The flight departure cannot be advanced') if update and (FlightInstanceDate < FlightInstanceDate.GetOldValue() or (FlightInstanceDate = FlightInstanceDate.GetOldValue() and FlightInstanceTime < FlightInstanceTime.GetOldValue())); Es decir, si se está actualizando un vuelo real, sólo se permitirá modificar su día y/u hora de partida, siempre y cuando no se lo adelante. Obsérvese que se utiliza el método GetOldValue() de un atributo, que devuelve el valor que está almacenado en la base de datos, mientras que el atributo tiene ahora en memoria el valor que el usuario colocó en el form en este momento (aún no se ha grabado). Por supuesto, se necesitará almacenar información de los pasajeros, por lo que Diego creará la última transacción de esta etapa y el atributo PassengerId pasará a ser clave foránea en FlightInstance, así como PassengerName inferido. De carne y hueso: Transacción Passenger 127 El siguiente Diagrama representa las relaciones entre las nuevas tablas y las viejas. Diego pulsó <F5>. Todo parecía bien, así que reorganizó la base de datos. Luego accedió al Wiki, documentó la terminación de esta etapa y dejó a Mike la tarea de probar lo hecho y a Julia la tarea de aplicar los Work With al resto de las transacciones. 128 Capítulo 5 Cuando el “qué” no alcanza ¿Sólo declarar?... ¿y las descripciones procedurales? Fue lo que Mary cuestionó un par de citas después, en el bar de siempre. Se habían estado viendo con mayor frecuencia, en su oficina o en el bar, lo que había dado oportunidad a intercambiar sobre varios asuntos, como patterns, web panels y demás. Llegó a este nuevo encuentro, reflexiva. Decía entender que las transacciones permiten de manera declarativa describir las visiones de datos de los usuarios. Pero, ¿es esto suficiente? ¿Qué pasa con los procesos “batch” o similares, que en toda aplicación se necesitan? Estos procesos no se pueden inferir a partir del conocimiento aportado por las visiones de datos. ¿Entonces? Sí, faltaría un lenguaje procedural, estamos de acuerdo –ratificó Diego-. Si piensas en la filosofía de GeneXus, evidentemente deberá ser un lenguaje de muy alto nivel. Más aún: deberá ser un lenguaje que no utilice jamás referencias al Modelo Relacional, sino sólo al Externo. Te mostraré un ejemplo tonto: suponte que necesitas, por la razón que fuere, por ejemplo por malas condiciones en la pista de un determinado aeropuerto en un determinado momento de un determinado día, posponer la hora de partida prevista de todos los vuelos grandes (que sobrepasan determinada capacidad de pasajeros). Suponte que deberán atrasarse. En definitiva, deberás acceder a todos los registros de FLIGHTINSTANCE, tales que el valor de FlightInstanceDate y FlightInstanceTime coincidan con la fecha y hora en la que no deben despegar los aviones, y tales que correspondan a un vuelo que parta del país y ciudad determinado (aeropuerto) y que tenga una capacidad de pasajeros mayor que un valor determinado. Para esos registros debes modificar el valor de FlightInstanceTime, y eventualmente también el de FlightInstanceDate por los nuevos. Podrías hacerlo filtrando en el “Work With FlightInstance” por esos 129 valores, y para cada línea (un vuelo real) desplegada en el grid, llamar a la transacción FlightInstance en modo Update y modificar los valores manualmente, lo que seguramente será inaceptable para el usuario final. La otra alternativa: desde algún web panel en variables pedirle al usuario los valores de fecha y hora a modificar, país y ciudad, y capacidad, y el valor de la nueva fecha y hora de esos vuelos, y luego llamar mediante una acción (botón o imagen) a un procedimiento que recorra los registros y realice el cambio – al tiempo que lo decía, abría su notebook e implementaba el web panel. ¿Un procedimiento común y corriente? Sí, será un nuevo tipo de objeto GeneXus. Queremos que cuando el usuario presione sobre el botón “Change flights”, que está asociado al evento Enter del web panel, se invoque a este nuevo objeto. Event Enter ChangeDepartureTime.Call( &FlightDate, &FlightTime, &DepartureCountry,&DepartureCity, &NewDate, &NewTime, &capacity ) EndEvent - Vamos a crearlo entonces. Segundos después, Diego tenía creado el procedimiento que implementaba la lógica anterior: 130 donde las variables que aparecen fueron declaradas por Diego como parámetros de entrada del procedimiento, en el selector Rules: parm( in: &CurrentInstanceDate, in: &CurrentInstanceTime, in: &CountryId, in: &CityId, in: &NewFlightInstanceDate, in: &NewFlightInstanceTime, in: &FlightTotalCapacity); Espera, Diego, que me he perdido. Deduzco que el For each es un comando que permite acceder a la base de datos recorriendo una tabla. Pero, ¿dónde le dices que es FLIGHTINSTANCE la tabla que debe recorrer? Además estás utilizando atributos de otra tabla también. ¿No deberías recorrer primero FLIGHT, y para los registros que cumplan con las condiciones, hacer un join con la tabla FLIGHTINSTANCE? No es necesario –observó Diego-. Para empezar, la tabla no se la dices directamente. Ahí está la gracia. Mira lo que le estás diciendo con el For each que hemos escrito:39 para cada registro de “la tabla donde se encuentren FlightInstanceDate, FlightInstanceTime, FlightDepartureCountryId, FlightDepartureCityId, y FlightTotalCapacity mencionados, tales que los 39 Respire profundo, amigo lector, no se nos vaya a quedar sin aire a la mitad… 131 valores de los primeros cuatro coincidan con los de las variables recibidas por parámetro, y el valor de FlightTotalCapacity sea mayor o igual que el de la variable también recibida por parámetro”, cámbiale el valor que contengan FlightInstanceDate y a FlightInstanceTime por los indicados. Sigo sin entender. Esos atributos están en tablas distintas. FlightInstanceDate y FlightInstanceTime están en FLIGHTINSTANCE, sí, pero los demás están todos en FLIGHT… ¿No deberías recorrer…? Cierto –interrumpió Diego sin dejarla terminar-. Pero FLIGHT está en la tabla extendida de FLIGHTINSTANCE. Es decir, por cada vuelo real, tienes un FlightNumber (clave foránea) que corresponde a un y solo un registro de FLIGHT, que tiene un FlightDepartureCountryId, un FlightDepartureCityId, y un FlightTotalCapacity. Hay una relación N a 1 entre FLIGHTINSTANCE y FLIGHT. Con eso no necesita más: sabrá que debe recorrer la tabla FLIGHTINSTANCE, accediendo para cada registro a su correspondiente en FLIGHT, es decir, haciendo un join automático para saber el valor de los atributos FlightDepartureCountryId, FlightDepartureCityId y FlightTotalCapacity y con eso decidir si se queda o no con el registro para procesarlo (por supuesto, siempre y cuando el propio registro de FLIGHTINSTANCE cumpla con ser del día y hora indicados por las variables &CurrentInstanceDate y &CurrentInstanceTime). La condición que impone sobre un registro para quedarse con él y procesarlo, es la concatenación con And de todas las condiciones lógicas que ves en las cláusulas Where del For each. Como antes hablamos de tabla base, aquí también: para el For each es la tabla que recorre. Observa, entonces, cómo conociendo las relaciones entre las tablas y dónde están los atributos en el momento del análisis, GeneXus puede inferir el acceso a los datos sin que le tengas que nombrar tabla alguna. Esto tiene una consecuencia importante: permite que se puedan cambiar de tabla esos atributos, sin que la descripción del procedimiento pierda validez40. 40 Por supuesto: siempre y cuando los atributos pertenezcan a una misma tabla extendida. En caso contrario GeneXus arrojará un error, informando que no fue posible relacionar los atributos. 132 ¿Por qué? Hmmm… parece un poco mágico, no me sentiría segura programando así. No Mary, créeme, nada es mágico. Miremos el listado de navegación para aclarar tus dudas (y las mías; siempre antes de ejecutar un procedimiento para probarlo, el primer test ya está hecho: es el listado de navegación, que informa sobre gran parte de lo que te da inseguridad. Por eso, nunca ejecuto sin antes mirar ese listado. Bueno, está bien, a ti no te voy a mentir. A veces no lo miro, de apurado nomás, pero luego me arrepiento). Mirémoslo juntos… (si sólo quiere verse el listado de navegación, en lugar de hacer <F5> que además construirá todos los programas, se puede hacer botón derecho sobre el tab del procedimiento, y elegir la opción View Navigation) Tabla base ¿Ves? Te informa la tabla base del For each, el orden por el que procesará la información41 y el índice que propone para satisfacer ese orden (si es que existe). Además te informa los filtros de navegación, es decir, desde dónde empezará a recorrer la tabla y hasta dónde (en nuestro caso, recorrerá 41 El orden puede indicarse mediante la cláusula “Order” del comando For each. Como no se ha indicado, y no existe ningún índice en la base de datos que permita optimizar la búsqueda de los registros que cumplan estas condiciones, asume clave primaria de la tabla a recorrer. 133 toda la tabla42). Luego se observan las Constraints (filtros sobre los registros a recorrer). Más abajo te informa que por cada registro de FLIGHTINSTANCE deberá hacerse un join con la tabla FLIGHT de su extendida (para realizar los filtros por FlightDepartureCountryId, FlightDepartureCityId y FlightTotalCapacity) y que se actualizarán de la tabla FLIGHTINSTANCE los atributos FlightInstanceDate y FlightInstanceTime. Sé franca, Mary, y dime cuánto más te costará implementar esto mismo utilizando las herramientas que usas. Habrías implementado tú misma este join. Lo mejor de todo, en verdad, es el hecho de seguir hablando en alto nivel. Piensa qué pasaría en tu implementación, que seguramente requerirá nombrar las tablas, si mueves por ejemplo, el atributo FlightTotalCapacity de FLIGHT a FLIGHTINSTANCE. Aquí no tendrás que tocar absolutamente nada. Simplemente volver a generar el objeto, para que GeneXus infiera la nueva navegación y ¡listo! Sí, comprendo. ¿Y cómo insertas y eliminas registros de la base de datos? Bueno, el comando For each te permite recorrer una tabla de la base de datos, y acceder a su extendida, tanto para consultar solamente, como para actualizar atributos. Por este motivo, es un comando importantísimo, que podrá ser utilizado no sólo en procedimientos sino en todo objeto GeneXus donde se admita código procedural (como los eventos de transacciones y web panels, por ejemplo). Para las inserciones y eliminaciones se han creado dos comandos específicos: New y Delete43. 42 Si existiera un índice compuesto por {FlightInstanceDate, FlightInstanceTime, FlightDepartureCountryId, FlightDepartureCityId, FlightTotalCapacity}, lo propondría sin dudar, optimizando la consulta, de modo que ya no tendría que recorrer toda la tabla y esto lo informaría en los “Navigation filters” del listado de Navegación. Aun si no existiera este índice, pero sí uno compuesto por un subconjunto de esos atributos, lo propondría. Es decir, siempre que exista un índice que le permita optimizar la consulta, lo propondrá. (No lo dijimos antes, pero en cualquier momento usted puede pedirle a GeneXus la creación de un índice de usuario) ¿Por qué decimos “lo propondrá”? Pues para los generadores SQL si bien esta información es útil e influyente, en realidad es el propio DBMS quien resuelve el plan de acceso más apropiado. Para los no SQL (RPG, COBOL, VB-Access, VFP-DBF) en cambio, es vital. Existe mucho para conversar sobre este tema, estimado lector, pero no viene a cuento en esta historia. 43 En ninguno de los casos se hacen controles de integridad referencial, permitiendo así insertar registros que apuntan a un registro inexistente, o eliminar registros que son apuntados por otros, dejando a estos últimos huérfanos. El único control que se realiza es el de unicidad. Hablamos de controles programáticos, ya que en GeneXus se puede indicar si se desea activar la integridad referencial a nivel del DBMS, en cuyo caso siempre se controlará. 134 Ahora que pienso, no estás controlando en el procedimiento lo que sí controlas en la transacción FlightInstance: que no intenten cambiar la fecha y hora del vuelo, adelantándolo. Si mal no recuerdo, allí tenías una regla de error que controlaba eso. ¡Tienes razón! -. Pensativo, Diego rumiaba algo. Mary lo miraba entusiasmada. Tras unos segundos de meditación, rompió el silencio-. Tendría que hacer ese control en el procedimiento, preguntando con un comando If antes de apresurarme a actualizar así sin más. Toda vez que quiera modificar fecha y/u hora de un vuelo real, en cualquier objeto, tendré que volver a hacer el mismo control que hacía mediante la regla en la transacción. Repitiendo el código. Una y otra, y otra vez. Quizás en alguna me olvido…. o me equivoco. A lo seguro y sin transpirar: el “qué” de los Business Components Ya que sabemos –continuó con una renovada vitalidad- que las transacciones son uno de los objetos más importantes, porque en ellas se recogen las visiones de los usuarios, las reglas del negocio, las fórmulas, y se ejecutan los controles de unicidad y de integridad referencial, ¿qué me dices sobre reutilizarlas desde cualquier otro objeto GeneXus? ¡¿Ehh?! –la expresión de su rostro provocó una sonrisa en nuestro amigo. Sí, reutilizarlas, aprovechando todo ese conocimiento que ya tienen incorporado. Por ejemplo, si yo tuviera una forma de encapsular toda la lógica de la transacción FlightInstance, dejando de lado la interfaz… es decir, si pudiera tener algún buffer o parecido, alguna estructura temporal, por ejemplo una variable compuesta, estructurada, donde cargar los datos de un vuelo real (cabezal y líneas)… podría ingresarlo en la base de datos desde cualquier objeto… Si realmente tuviera encapsulada la lógica de la transacción de esa forma, entonces se permitirá la inserción solamente si se cumple todo lo que se debe cumplir para ingresar esa misma información vía la transacción… Quiero decir: si quisiera insertar una instancia de vuelo asociándole un vuelo 135 (FlightNumber) inexistente, o pretendiendo agregar algún pasajero inexistente, fallará y no me dejará hacer la inserción, tal como ocurre con la transacción. O, por ejemplo, si quiero agregar un pasajero existente pero con el que se excede la capacidad del vuelo, no me dejará realizar esa inserción, tal como sucede cuando quiero hacerlo vía la transacción, dado que tengo una regla de error en ella que lo impide –. Diego abrió la transacción FlightInstance y mostró a Mary la sección Rules. Usted, estimado lector, puede buscarla en el capítulo anterior-. Encapsular la lógica de la transacción, quedándose con la estructura pero dejando de lado el form es posible: se llama Business Component. Es un tipo de datos generado por GeneXus a partir de una transacción44. ¿Y cómo haces para encapsular todo eso? ¿y cómo lo usas? A la transacción que quieres utilizar de esta otra forma “silenciosa”, alcanza con cambiarle el valor de su propiedad Business Component a “True”. A partir del momento en que guardas este cambio, GeneXus creará un nuevo tipo de datos Business Component, cuyo nombre será el de la transacción45. ¿Y cómo lo utilizas? –volvió a preguntar, intrigada. A través de variables basadas en ese tipo de datos. Siempre que quieras actualizar, insertar o eliminar una instancia de vuelo, en vez de utilizar For each, New y Delete, podrás utilizar toda la potencia de la transacción FlightInstance a través de las propiedades y los métodos de una variable basada en el tipo de datos FlightInstance, business component, que GeneXus generó. Modifiquemos el procedimiento anterior para hacer la actualización mediante un Business Component – prosiguió Diego su monólogo; Mary observaba expectante sin hablar-. Necesitamos una variable, &FlightInstance de este tipo de datos (generado 44 Los Business Components se obtienen de objetos transacción, pero pueden ser utilizados desde cualquier otro objeto GeneXus, para hacer inserciones, eliminaciones y modificaciones en sus tablas. 45 Si la transacción tiene 2 niveles, creará un business component por nivel. 136 cuando pasamos a “True” la propiedad Business Component de la transacción). En ella alojaremos todo el contenido de la instancia de vuelo a cambiar. Esta variable será, lógicamente, estructurada, será el “buffer” donde temporalmente tener los datos del vuelo que queremos manipular en un momento dado. El código nos quedaría más o menos así…- y sustituyó en el objeto las dos asignaciones directas por las cuatro sentencias que se pueden observar en la siguiente imagen-. Observa que aquí sólo utilizamos el For each para recorrer la tabla base y filtrar, pero no actualizamos directamente: ¿Ves? Lo primero que hago aquí es cargar desde la base de datos la instancia de vuelo a modificar, dentro de la variable &FlightInstance, según el valor de FlightInstanceId que corresponde al registro que cumplió las condiciones de filtro dentro del For each (es el registro de esa iteración). Luego cambio solamente los datos que debemos modificar, y luego grabo, con el método Save. Observa la analogía: para editar un vuelo mediante la transacción, hubieses colocado el identificador en el campo FlightInstanceId del form, y salido del campo. Para hacer eso con el Business Component, utilizamos el método Load. En la transacción hubieses cambiado en el form los valores de los campos FlightInstanceDate 137 y FlightInstanceTime y luego hubieras presionado [Confirm]. ¿Y aquí? Cambias lo valores en la variable, y luego ejecutas el método Save. Lo que pasará como consecuencia de esas acciones, es prácticamente lo mismo. Quiero decir, ¿qué pasaría si en la transacción modificaras fecha y hora de forma tal que adelantaras el vuelo? Se dispararía el error ese que habías programado como regla, y no nos dejaría grabar, o sea, no se realizaría ese cambio en el registro. Lo dejaría como estaba antes. Exactamente. ¿Y qué piensas que sucederá aquí, en el procedimiento? ¿Lo… mismo? –preguntó y afirmó al mismo tiempo-. ¡Lo mismo! Disparará la regla, y como su condición se satisfará, no te dejará grabar el cambio. ¿Y cómo me entero? Bueno, en el caso de la transacción, como es un objeto interactivo, allí no hay duda, te despliega mensaje. Sin embargo, un Business Component no lo es, razón por la cuál los posibles mensajes y errores producto del procesamiento (ejecución de métodos Load, Save o también Delete, el utilizado para eliminar) deben quedarte almacenados en algún lado. GeneXus los inserta en una colección (lista) de mensajes asociados a la variable Business Component de que se trate, y la puedes recuperar con un método (GetMessages). Luego tienes la posibilidad de recorrer esa lista de mensajes… Pero si sólo te interesa saber si se produjo algún error, tienes el método Fail que te devuelve “True” en caso de error: así, en nuestro caso, programarías “if &FlightInstance.Fail()…” Entiendo. ¿Y qué otros errores podrían producirse cuando manipulas la información con el Business Component? Los mismos que cuando intentas insertar mediante la transacción FlightInstance. Es decir: los de integridad referencial (si quisieras cambiar por ejemplo el vuelo, FlightNumber, que es clave foránea, por uno inexistente), el de unicidad (por ejemplo, si quisieras insertar una instancia de vuelo nueva, y le asignas un FlightInstanceId que ya existe para otro registro), las validaciones 138 de tipos de datos, las reglas de error declaradas en la transacción, y cuyas condiciones de disparo se satisficieran…Observa cómo en ese sentido es más seguro hacer un procedimiento utilizando Business Component donde sabes que se van a disparar todas las reglas del negocio, sin tener que repetirlas. ¿Todas? Si no te quedas con la interfaz visual, ¿qué ocurre si en la transacción hubiera una regla que disparara una llamada a un web panel, por ejemplo? Bueno, no todas… GeneXus ignora en el Business Component las reglas que hacen uso de interfaces de usuario. Esa no será incluida. Interesante… -dijo pensativa. ¿A dónde quieres llegar? Atracciones que invitan… en pdf El lector observador habrá notado lo que Mary en su momento (y que prefirió dejar en el tintero esperando la oportunidad propicia): Entre los selectores del objeto de tipo Procedimiento aparece uno “Layout”. ¿Para qué? ¿Acaso un procedimiento te permite efectuar una salida visual? Pues sí. Casualmente los usuarios nos han pedido como requerimiento que el sitio de la agencia de viajes tenga una página que muestre las atracciones por ciudad, y un link del estilo “Ver como pdf”. Para implementar esto último, creé un procedimiento al que llamé AttractionsPerCity. Mira –dijo Diego mientras mostraba el objeto en su notebook, seleccionando “Layout”-: 139 Le tendré que decir a este procedimiento que tendrá una salida como pdf. La idea del Layout es declarar qué es lo que quiero mostrar en la salida. Como resultado de la ejecución, quiero mostrar para cada ciudad, todas sus atracciones. Evidentemente, el usuario final tendrá que tener Acrobat Reader instalado para que desde el Browser se pueda abrir correctamente el archivo pdf generado. Para indicar qué es lo que se quiere mostrar en la salida y su formato, el Layout contiene áreas de impresión, que son controles llamados printblock. Coloreados en la imagen, puedes ver donde empiezan dos de esos controles: City_Block y Attraction_Block. Esos nombres se los di yo luego de insertar cada control. Dentro de cada printblock insertas la información (atributos, variables, imágenes, líneas, recuadros, etc.) que deseas desplegar en esa área de datos, cuando sea invocada desde el código. Por ejemplo, observa que dentro de City_Block he insertado los controles atributo CityId y CityName, porque esa es la información de cada ciudad que querré mostrar en el listado. ¿Y sólo con eso GeneXus ya sabe qué listar y cómo? Mary, espera. El amigo no es mago, sólo es inteligente. Con eso solo no puede saber cómo quieres listar la información. Falta el código –dijo enarcando las cejas, con expresión risueña, viendo aparecer un rubor... 140 Bueno, lo presentas como tan maravilloso… bien podría adivinar tus intenciones, ¿no? –sonrieron ambos. ¿Tú las adivinas? –avanzó Diego. ¿Y si las explicitas? –contestó Mary sin retroceder, mirándolo fijamente a los ojos, con una expresión que a él se le antojó encantadora. ¡Sus deseos son órdenes!...–. Y tras un breve silencio que a ambos pareció eterno, él continuó-. Quiero recorrer las ciudades, y para cada ciudad, sus atracciones. Esto implica la utilización de dos For eachs anidados. Mary aproximó su silla a la de él, tomó el mouse sin preguntar, siempre sonriendo, y se posicionó en el selector Source. La ventana se transformó, mostrando el siguiente código: Diego continuó: Observa que no necesitas establecer el filtro por ciudad para el segundo For each: es que existe una relación 1-N directa entre las tablas que se recorren, siendo uno de los casos más comunes de For eachs anidados. La tabla base del For each externo es COUNTRYCITY, y la del interno es ATTRACTION (por los atributos de los printblocks involucrados), y como GeneXus conoce las relaciones entre las tablas, él implícitamente aplica la 141 restricción sobre los registros a recuperar. Pero es más potente que eso, encuentra también relaciones 1-N indirectas. Por ejemplo, si quisieras listar para cada aerolínea, la cantidad de instancias de vuelo46. Mary, muda, no emitía palabra. Miraba la pantalla. Una sonrisa seguía estampada en su rostro. Posicionó el mouse sobre el Tab del procedimiento donde se encontraba su nombre, y con botón derecho seleccionó “View Navigation” del menú contextual. Diego la miraba. Apareció un nuevo Tab en la pantalla, conteniendo la siguiente ventana: ¿Ves lo que digo? Está restringiendo la búsqueda- explicó a esta mujer que tenía tan cerca y que sólo observaba y sonreía-. Te lo indican los filtros en la navegación de la tabla ATTRACTION en el segundo For each. Observa el “@”. Está indicando que toma esos valores del contexto. ¿Cuáles son @CountryId y @CityId? Los del registro del primer For each con el que se encuentra trabajando en esa iteración. Esas intenciones GeneXus sí las descubre. Diego continuó, llenando el silencio que Mary no llenaba: Para convertir este listado en ejecutable, deberás cambiar 2 propiedades: Main Program = “True”, Call Protocol = “HTTP” para que se 46 Entre las tablas AIRLINE y FLIGHTINSTANCE hay una relación 1-N indirecta (está la tabla FLIGHT de intermediara). GeneXus encuentra en muchos otros casos relación 1-N indirecta y hace el join. 142 ejecute en el Browser. Además deberás incluir una regla en el selector Rules “output_file("AttractionsPerCity", "pdf");” que es donde le dices que la salida será un archivo pdf, con el nombre que especifiques. También puedes decirle que quieres que envíe la salida a alguna impresora de las que estén bajo el alcance de la computadora desde la que se esté ejecutando el navegador con la aplicación. Para ejecutarlo, Diego se posicionó sobre el tab con el nombre del procedimiento, y con botón derecho seleccionó la opción “Run with this only”, para que GeneXus construya y ejecute ese procedimiento únicamente. La cara de asco de Mary no se hizo esperar. Es que el listado sólo mostraba datos, sin ningún formato. Diego no esperó. Grabó el procedimiento con otro nombre, y le mejoró el diseño y la información mostrada… …obteniendo ahora en ejecución: 143 Por fin Mary rompió el silencio. Me atrae París… ¿Por qué no vienes esta noche a cenar a casa? Y me sigues contando… 144 ¿Y las consultas? Otro de los grandes alcances de GeneXus X es el objeto Query que permite crear consultas a la base de datos, y mejorar la salida de la información recuperada. Ésta puede ser visualizada en diferentes formatos (barras, tortas, 3D o planas, área, etc.) lo que ayudará a enriquecer sus aplicaciones. Además permitirá al usuario realizar consultas dinámicas (qué measure ver) a través de pivot tables, etc. Una consulta de este tipo es definida como una estructura. Supongamos que deseamos obtener una consulta de cantidad de despegues en una fecha dada (FlightInstanceDate) por aerolínea (AirlineName) para cada ciudad (CityName); en la siguiente imagen se resume esta intención: Al objeto Query, creado desde la ventana New Object, le hemos asignado el nombre QSoldSeatsByArrivalCities. En el nodo Attributes hemos insertado los atributos AirlineName y FlightArrivalCountryName, que serán los que queramos ver en tiempo de ejecución. Inmediatamente debajo, hemos 145 declarado la función Count, pero según las necesidades, se pueden declarar también a Sum y Average47 Podemos también incluir parámetros, y en este caso en particular hemos incluido una variable basada en un atributo en particular. Luego tenemos el nodo Filters donde, si fuera necesario, estableceremos las condiciones para recuperar los registros a mostrar; en este caso, todos aquellos cuyo valor del atributo FlightInstanceDate (incluido en FLIGHTINSTANCE) sea igual al valor del parámetro que hemos nombrado como &DepartureDate. Por último, tenemos el nodo OrderBy donde, de ser necesario, deberemos establecer el o los atributos que deberán ser considerados; en este caso, los datos se mostrarán en orden alfabético según el contenido del atributo AirlineName. Ahora, ¿cómo ejecutamos esta consulta? Deberemos crear un web panel con el aspecto que deseemos, como cualquier otro, pero al cual le deberemos insertar el User Control Query Viewer que se encuentra en la Toolbox del IDE. 47 No todas las funciones de agregación aplican para todos los tipos de atributos. Sum y Average están solo disponibles para atributos numéricos. 146 Luego accedemos a sus propiedades y seteamos aquellas que sean necesarias. En este caso se ha modificado la propiedad Query, a la cual se le asignó el nombre del objeto Query que definimos al inicio, QSoldSeatsByArrivalCities, y también se ha modificado la propiedad Type; esta tiene por finalidad establecer alguno de los tres tipos posibles de salidas: Pivot Table, Table, y Chart. Le recomendamos que pruebe para ver las diferencias entre ellos. Considere que para el valor Chart dispondrá además de la propiedad ChartType donde podrá escoger un estilo entre doce posibles. Y eso es todo para lograr una construcción básica de un Query. En el selector SQL statement se puede observar el código que GeneXus ha generado y que será ejecutado sobre la base de datos para obtener la vista deseada en cuanto se ejecute el web panel diseñado para tal fin. 147 Ahora <F5> y, ¡voilà! 148 Capítulo 6 Más declaraciones Nuevos protagonistas del conocimiento: Data Providers La meta de GeneXus: que usted declare, declare y declare. Cuando uno declara dice el qué, y no se preocupa por el cómo, pues éste varía con el tiempo. Con frecuencia se encuentran nuevas formas de implementación, mejores, más eficientes. El qué es estable, la esencia. Conocimiento que permite automatizar. Las transacciones y los business components eran, así, las estrellas de GeneXus: declaraban la lógica del negocio. Declaraban conocimiento. GeneXus encontraba, en cada etapa de su evolución, un cómo. Ahora, un nuevo protagonista irrumpe en escena: el Data Provider. Para una gran variedad de casos, pasaremos de programar procedimientos, a declarar data providers. ¿Quedó usted intrigado? Mary también. Vea lo que pasó. ¿Declarándose? Sin responder a la pregunta que ella dejara sonando en el aire, Diego abre torpemente su inbox de correo y encuentra un e-mail de la agencia donde le solicitaban un nuevo requerimiento: debía generar un archivo de interfase con formato XML para enviársela a cierta aerolínea donde figurara la cantidad de plazas vendidas por vuelo de esa aerolínea, en cierta fecha. Mira esto –susurró a Mary girando la notebook para que ella leyera. Fácil –contestó ella demorando en leer. Su vista se había detenido antes en otros mensajes de correo, y Diego lo percibió, pero no dijo nada–. Implementas un procedimiento parecido al que vimos. Pero ¿cómo se hace en GeneXus para generar la salida en formato XML? –si bien parecía estar 149 pensando en otra cosa, su conversación era completamente coherente. ¿En qué pensaría sin decirlo? No es complicado, ya veremos. ¿Pero sabes una cosa? Esta vez no lo haremos con un procedimiento. Antes de la X no nos quedaba otra opción, pero ahora disponemos de algo más potente para este tipo de casos. ¿Más potente? Sí, nada de codificación procedural. Fíjate que aquí necesitas consignar información estructurada: imagínate cómo debería ser el XML resultante, por ejemplo… –. Creó un documento xml, escribió y mostró: Los ojos de Mary, entrecerrados por el ceño contraído, habilitaron a Diego a continuar la explicación: Para esto usaremos un Data Provider, que es un objeto ideado específicamente para devolver datos estructurados. Te voy a mostrar una cosa: si quisieras implementar este requerimiento como un procedimiento en GeneXus, programarías algo así: 150 ¿Ves cómo en este código están mezclados los elementos de la entrada, con los elementos de la salida? Dime la verdad, sin importar si entiendes o no el código, ¿no te resulta complicado aquí encontrar cuál será la estructura resultante y de qué modo se carga? Esto es porque los procedimientos privilegian el lenguaje de transformación del input para obtener el output. Por lo que el output queda oscurecido y no es fácil identificarlo. Sin embargo, en los Data Providers lo privilegiado será el Output. De hecho, declaras tu intención, mediante simples ecuaciones del estilo x = y para cada elemento de la estructura jerárquica que quieres devolver cargada. Muy interesante, Diego, pero se me hace un poco tarde, y veo que esto puede llevar un tiempo… ¡Cinco minutos! – acotó inmediatamente, ocupado en demorar el fin del encuentro. Mary dudó unos segundos y por fin continuó: Bueno, entonces anda, “declara”. Diego creó un nuevo objeto, al que llamó LoadNumberOfPassengers, de tipo Data Provider. Mary observó la imagen. 151 Ajá. Tiene casi los mismos selectores que los procedimientos. Sí, lo que varía es lo que se escribe en el selector Source, pues no será código procedural, sino declarativo. Pero vamos por partes. Según el requerimiento, algún otro objeto capturará del usuario el número de aerolínea y la fecha, y a ambos los recibiremos como parámetros. Así que definamos entonces las variables y declarémoslas en las Rules –tras lo cual lo hizo. parm(in:&AirlineId, in:&DepartureDate); Bien, ahora vamos al Source –dijo Mary con confianza. Bueno, todavía no. Debemos definir un tipo de datos estructurado que será el tipo de datos de la información devuelta por el Data Provider. ¿Tipo de datos estructurado? ¿Como si fuera un struct en C, o .Net, o un record de Pascal? Eso. Una estructura que puede contener tanto datos simples como otras estructuras, así como colecciones (listas) de datos. Incluso puede ser él mismo una colección. En GeneXus defines esos tipos de datos, no en la ventana de Dominios, sino como objetos (algo parecido a como se hace en Java definiendo clases que luego se instancian). Mira, en nuestro caso definiremos un objeto de tipo “Structured Data Type”, o simplificando, SDT, y allí ingresaremos su estructura: 152 ¿Ves? –se apresuró a proseguir-. Este tipo de datos se compone de tres miembros, de nombres Name, DepartureDate y Flight. Los dos primeros, de tipos de datos simples, y el tercero, será una colección de ítems (observa el check box Is Collection), cada uno de los cuales es estructurado; les hemos llamado FlightItem. Los miembros de éste último estructurado también son tres. Observa que como he llamado a los dos primeros miembros igual que atributos de mi KB, infirió como su tipo de datos el mismo que el de los atributos homónimos. Les puse el mismo nombre que a los atributos porque en realidad voy a cargar este SDT con los valores de esos atributos, luego. Pero no nos detengamos en ello. Ante la atenta mirada de Mary, Diego picó sobre el nombre del SDT que ahora aparecía en el árbol del Folder View, lo arrastró hacia el área Source del Data Provider y soltó el botón, tras lo cual apareció: 153 De forma automática, GeneXus inicializó en el Source la estructura que será devuelta por el Data Provider; es más: dio valor a la propiedad Output que indica cuál será la estructura devuelta por el Data Provider. Ahora Diego sólo debe sustituir los comentarios del Source por los valores que correspondan a cada miembro. 154 Vio cómo los rojos labios de Mary se unían para pronunciar lo que adivinó ser un “pero…”, aunque no llegó a saberlo, pues continuó: Espera, no he terminado. Aún no estoy filtrando por los parámetros recibidos. Mira cómo completo el Data Provider: ¿Cláusulas where, como las de los for eachs? –preguntó Mary pensativa, haciendo la asociación clave. 155 Efectivamente. En el lenguaje de Data Providers tendrás grupos, estos son los que contienen elementos. En nuestro caso: SeatsSoldByFlight, Flight y FlightItem son grupos. Los grupos contienen elementos, variables y/u otros grupos. En cualquier caso, un grupo podrá presentarse en la salida una sola vez, o múltiples veces. GeneXus tiene la inteligencia para determinarlo. ¿Cómo? Por lo que decías tú. La presencia de atributos a la derecha de las asignaciones en los elementos, así como en las cláusulas where del grupo están determinando que esos grupos son como for eachs, que acceden a una tabla base. Si no hubiera cláusulas where, tal como estaba el Source antes de que las agregaras, y si en lugar de atributos a la derecha de las asignaciones tuvieras variables o literales… …en ese caso, en la salida tendrías cargado el SDT de forma tal que la colección Flight estaría compuesta únicamente por un ítem. En cambio, en nuestro caso, el grupo Flight contendrá muchos ítems, tantos como instancias de vuelos haya para esa fecha y aerolínea, y en cambio el grupo raíz SeatsSoldByFlight será único, puesto que estamos filtrando por AirlineId. Ah, entiendo –exclamó Mary–. Es como si tuvieras un par de For Each anidados. El primero, por los atributos que aparecen, recorrería la tabla AIRLINE, y el segundo la tabla FLIGHTINSTANCE. Perfectísimamente. Mira el listado de navegación: 156 Observa además cómo en el lado derecho de los elementos puedes poner tanto atributos como cualquier función o fórmula que devuelva un resultado del mismo tipo de datos que el que definiste para ese elemento en el SDT. Por eso utilizamos una fórmula count para SoldSeats. En realidad, como esa fórmula la tenemos también global en el atributo FlightNumberOfPassengers, podríamos haber colocado directamente este atributo. No lo hice para mostrarte que es posible escribir cualquier cálculo. Entiendo ahora por qué dices que los Data Providers son declarativos. Lo que tienes que hacer es declarar cuál será la estructura resultante, y luego cargarla a través de ecuaciones x = y donde en y declaras cómo se calcula x. Ahora, ¿cómo devuelves el XML que necesitas? Bueno, en realidad deberás invocar a este Data Provider como si fuera un procedimiento común y corriente desde cualquier objeto, y hacer lo que quieras con la salida, por ejemplo, convertirla a XML –contestó Diego-. Creo que lo mejor será que lo haga, así lo vas a entender mejor. 157 Creó un Web Panel y tres variables: &AirlineId, &DepartureDate y &SeatsSoldXML, esta última de tipo de datos LongVarChar y de 2 MB (será la que contendrá la información en formato xml): A continuación se posicionó en el selector Events y escribió: Espera, espera… ¿y la variable &SeatsSold a la que le estás asignando lo devuelto por el DataProvider? Olvidé definirla –tras lo cual lo hizo. Su tipo de datos será, lógicamente, el SDT devuelto, es decir: SeatsSoldByFlight. Ajá, e infiero que el método ToXML aplicado a una variable SDT tiene por efecto convertir el contenido del formato SDT al formato XML. ¿Cierto? 158 Cierto. El valor “true” del parámetro solamente está indicando que le agregue el header al xml; ahora no importa… entonces luego de ejecutado el método, la información está contenida en memoria dentro de esa variable &SeatsSold. Ahora lo único que te queda es generar, por ejemplo, un archivo con ese contenido y listo –agregó Mary mirando su reloj pulsera. Sí, y todavía quizá mejor: puedo enviarlo directamente por mail a la compañía aérea. Ajá… Diego, escúchame, no sigas. Te agradezco la explicación, no te enfades conmigo, pero me esperan, debo marcharme. Te espero esta noche en casa, y seguimos con las declaraciones… ¿te parece? Sí, pero omití decirte que un Data Provider también puede cargar y devolver un business component, porque la estructura de un business component es como la de un SDT. Luego puede ser insertado en la base de datos desde el objeto llamador, con el método Save, como siempre. Me lo cuentas luego. Nos vemos esta noche, ¿sí? Diego permaneció en el bar varios minutos más, con la vista perdida por donde ella había partido minutos antes, dejando tras de sí su estela de perfume. 159 160 Capítulo 7 Soplan nuevos vientos Poniendo la casa en orden: categorización Esa noche Diego llevó el vino. Y la notebook. La música de fondo creaba un ambiente acogedor. Debes cumplir lo prometido –reclamó Mary sonriendo, mientras llenaba su copa-. Dime más. ¿Sobre GeneXus? Bueno, no sé qué más decirte…- Parecía nervioso, aunque pronto salió del paso-. Hoy por ejemplo estuve categorizando los objetos de la KB. No sé si recuerdas, pero el IDE tiene un panel rotulado Category View en el contenedor Knowledge Base Navigator que despliega, en árbol, todos los objetos que hayas clasificado dentro de sus correspondientes categorías. Espera que te muestro –dijo depositando la notebook en su falda y la copa sobre la mesita próxima al sofá. No veo clara la finalidad –respondió Mary sentándose a su lado. Bueno -mientras la notebook terminaba de iniciar, Diego llenaba el incómodo silencio-, por ejemplo, en mi compañía solemos tener categorías para identificar el estado de los objetos: “Operating”, “Under Review”, “Under Construction”. Así, puedes saber en un simple pestañeo, qué objetos están prontos para ser testeados, simplemente abriendo la categoría “Under Review”. En este caso, separamos nuestro universo de objetos en categorías disjuntas. De hecho, hasta puedes darle un nombre a este criterio de clasificación de tu universo, justamente mediante una categoría, “Status” por ejemplo, que contenga a estas tres 161 subcategorías. Pero simultáneamente, podemos querer categorizar el mismo universo, de acuerdo a otros criterios. Lo que quiero decir: puedes incluir un mismo objeto en varias categorías distintas. Por ejemplo, suponte que quieres organizar además por sistema, subsistema, etc. Nada impide que un objeto esté en la categoría correspondiente a un subsistema, y además en la “Under Review” de la “Status”. O puedes categorizar los objetos según si se utilizarán en el Back-end de la aplicación, o en el Front-end. Las categorías nos permiten organizar y visualizar la información de nuestra KB de acuerdo con nuestras necesidades puntuales. Multicategorizar…–. A ella le brillaban los ojos. Al menos eso le pareció a él. Cuando seleccionas un objeto del Folder View, debajo de la ventana de propiedades, tienes un contenedor rotulado Categories que comienza con una pequeña toolbar en donde puedes tanto agregarle categorías al objeto como eliminárselas. Pero, como siempre, tienes varios caminos para llegar a Roma… Hoy estuviste categorizando… eso significa que tuviste que ir uno por uno a todos los objetos y asignarles las categorías correspondientes, ¿no? ¡Mucho trabajo! Sí –contestó un poco avergonzado-. Es que en esta versión aparecieron las categorías, y no me he acostumbrado a usarlas, si bien fue decisión de la empresa el uso de categorías de estado. A veces es difícil cambiar las costumbres. Tendríamos que haber creado estas categorías desde el vamos, y los objetos ya con la categoría “Under Construction” y luego ir moviéndolos de categoría, en el cambio de estado. Hoy tuve que poner la casa en orden, ya eran muchos los objetos de la KB. Pero sólo tuve que hacerlo para 162 las subcategorías de “Status” porque la categoría “To-Do Diego” es dinámica, se mantiene sola. ¿Sola? ¿Cómo es eso? –. Mary estiró el brazo hasta la botella de vino. En este caso, aprovechando que tenía que poner la casa en orden, puse comentarios que contenían el texto “To-Do Diego” en algunos objetos que tenía pendientes de cerrar; también otros compañeros lo habían hecho. Luego creé una categoría dinámica. Las categorías dinámicas tienen asociada una condición de búsqueda (que podrá ser por palabras o por propiedades, como ya te mostraré). Todos los objetos resultantes de la búsqueda, pasan a formar parte de la categoría. Puse en la condición de búsqueda que tuviera el texto “To-Do Diego” y allí aparecieron todas las cosas que tengo por hacer, varias por cierto… Veamos cómo lo hizo… El que busca encuentra GeneXus X tiene la capacidad de encontrar “cualquier cosa” que se halle en la KB. Esta facilidad se presenta de la misma forma que experimentamos cuando hacemos uso de los motores de búsqueda en la Web. Basado en complejos algoritmos, el motor full-text search de GeneXus X localiza lo buscado entre lo que podrían ser toneladas de palabras contenidas en los objetos que componen la KB. Y al igual que la mayoría de los buscadores, nos permite utilizar sintaxis intuitivas para acotar las búsquedas. En la siguiente imagen puede ver la búsqueda por “To-Do Diego”… 163 El motor full-text search exploró la KB y reportó todos los lugares donde estaba presente. Obsérvese además cómo desde esta consulta puede crearse la categoría dinámica de igual nombre, “To-Do Diego”, que Diego mostró antes a Mary. Para ello alcanza con hacer [Save as Category]. Tal vez sea lo que hizo Diego ese día… Note que debajo se propone también una búsqueda, pero en este caso sólo de propiedades. Por ejemplo, si se quisieran ubicar todos los objetos de tipo procedimiento, agregaríamos una propiedad: 164 Base, rimel, sombras… Maquillando la aplicación Esa misma tarde, unas cuantas horas antes, Julia terminaba la documentación en la KB cuando Mike, que finalizaba los últimos testings, comentó: Me enteré por Diego que quieren cambiar el estilo de las páginas. Él estaba muy atareado, terminando el desarrollo de unos objetos, y me derivó este tema a mí. Me dijo que hablara contigo para pedirte los detalles. Sí, me enviaron un e-mail que describe más o menos el aspecto que desean y lo copié como Talk del Main Document; luego lo documento bien. Dejaron a nuestro gusto la composición de colores y demás. Yo ya había ingresado hace tiempo el logo de ellos a la KB, en el nodo Images. Quieren que quede situado arriba, en todas las páginas. Eso lo hago ahora mismo. Sólo tengo que abrir la master page AppMasterPage creada con la KB y utilizada por defecto por todos los objetos. Es donde se encuentra centralizado el encabezado y pie que tendrán todas las páginas del sitio; luego sustituyo el encabezado “Application Header” que sale por defecto, y listo. Para todo lo otro que piden, llamemos a Paul, el diseñador gráfico… Pero Paul no sabe nada de GeneXus, ¡¿y tú le vas a permitir que abra los objetos y modifique los controles en los forms, cambiándoles el color, el tamaño, y demás?! Mike no pudo reprimir la carcajada. No Julia, cada uno en lo suyo. ¡Ni loco le dejo tocar nuestros objetos! Pero necesito que configure para cada control de los forms, su diseño. Por ejemplo, necesito que especifique para los botones de las transacciones su tamaño, color de fondo, tipo de letra, etc. Lo mismo para los grids y demás controles. Quisiera centralizar eso y que él no tenga que tocar los controles “en” los objetos. Para eso es que existe el objeto de tipo Theme. Observa que bajo el nodo Themes bajo el Customer del Folder View, aparece uno de nombre GeneXusX. Es el que asoció GeneXus por defecto a nuestros objetos. Si lo 165 abro, verás que se trata de una lista, en árbol, de clases de controles (aunque no sólo). Aquí se configuran los aspectos de diseño para esta clase. Cada tipo de control tiene varias clases posibles. Dentro de la clase Button hay una subclase que es BtnDelete, que por defecto hereda sus propiedades pero que te permite cambiarlas. Si ahora vamos a cualquier transacción y editamos el form, podremos ver en las propiedades del botón [Delete] que tiene asociada esta clase. Puedes cambiársela por otra. Con esto te queda perfectamente separado en tu aplicación el comportamiento del diseño. Te digo más: ahora lo que haré será exportar48 de nuestra KB el theme GeneXusX, enviárselo a Paul para que él configure los aspectos de las clases y me lo envíe nuevamente, momento en que 48 No habíamos hablado hasta el momento de exportación/importación en GeneXus. En este libro solamente diremos que se pueden exportar objetos GeneXus y otro tipo de conocimiento de una KB en archivos de extensión xpz, que luego pueden ser importados en otra o la misma KB. Para ello está la opción del menú Knowledge Manager. 166 lo importaré y ¡listo! ¡Diseño aplicado! Paul no necesita tener GeneXus, sino solamente el Theme Editor, un utilitario que sólo permite trabajar con estos objetos. Es específico para el diseñador gráfico. Si no, también lo puedes hacer desde dentro de GeneXus. “De hecho, ahora que veo, el logo de la empresa está en la línea de los naranjas. Así que mejor utilicemos el Theme de nombre Orange (para ello lo marcamos en el check box) y hagámosle un Save as… TravelAgencyTheme. Así usamos por entero uno propio. Enviémosle este a Paul. Para probar y mostrarte, mira, modifiquemos los colores de algunas clases del theme –tras lo cual, con el nuevo theme TravelAgencyTheme abierto, se puso a cambiar las propiedades de las clases Grid, FreestyleGrid y Table, y grabó-. Ahora lo que tengo que hacer es decirle a la KB49 que utilice en todos los objetos ese tema:” Y habiendo cambiado también el web panel AppMasterPage, tras pedirle a GeneXus que reconstruya todos los programas (Build/Rebuild All) para que 49 Estrictamente hablando a la versión de la KB que esté activa. Sobre versionado de KB, busque próximos episodios o en las distintas fuentes de documentación de GeneXus. 167 tome en cuenta que los objetos tendrán ahora este theme, observe cómo luce en ejecución. Por ejemplo, veamos el Work With Airline, y luego la transacción para modificar una aerolínea: Los tipos de objeto Master Page y Theme son los que más importan a la hora de diseñar la estética de la aplicación. Espejito, espejito…dime cuándo y te diré cómo eras Otra interesante posibilidad que ofrece GeneXus X es la de poder consultar la evolución de los objetos dentro de la KB a través del tiempo. Es decir, cada objeto tendrá asociada una historia que estará registrada con su fecha y hora y el usuario que hizo las modificaciones. Con esto se podrá seguir la evolución de un objeto desde su inicio mismo. Y lo que es más: usted podrá volver a una versión anterior del objeto, seleccionando esa versión como activa, a través de botón derecho. En la siguiente imagen… 168 …se observa, a la izquierda, el contenido del panel Latest Changes View y a la derecha la historia de la transacción Airline. Note que existieron seis modificaciones desde su fecha de creación, por lo tanto hay siete números de versión. También podrá comparar dos versiones entre sí y obtener sus diferencias. Por ejemplo, puede seleccionar dos versiones y luego, mediante botón derecho del mouse, pulsar sobre “Compare Selected Revisions”. Como resultado de comparar las revisiones 12 versus 14 de la transacción Country, GeneXus mostrará las diferencias en dos ventanas adyacentes, como se muestra a continuación para el ejemplo (en la versión 12 no habían reglas definidas). 169 ¿Versionado de la KB? El tiempo hacía lo único que sabe hacer; Mary y Diego parecían no percatarse de ese avance inexorable. Diego… ¿Sí? ¿No crees que llegó el momento de… –se alisó la falda con las manos, y carraspeó antes de continuar, segundos eternos para él-…hacer una copia de la KB? ¡Ah, copias de respaldo! Bueno, en realidad en GeneXus lo puedes hacer muy muy muy fácilmente. De hecho, dentro de la misma KB puedes contener “fotografías” de diversos estados de la misma en el tiempo. Y no solo eso, puedes hacer desarrollos paralelos de distintas versiones. Por ejemplo, quieres poner en producción una versión, y continuar desarrollando nuevos requerimientos, de tal manera que puedes ir luego haciendo correcciones a la versión de producción a medida que el usuario lo va requiriendo, que no estorben el desarrollo principal. Interesante, ¿me lo cuentas luego? –. Se levantó del sofá y parada delante de él le ofreció extendida la palma de su mano-. ¿Vienes? 170 Situaciones que se repiten: para no decir siempre lo mismo, Data Selectors Es común encontrarnos codificando las mismas “situaciones” (filtros y órdenes de acceso a la base de datos) en distintos lugares de nuestra aplicación donde recuperamos datos, desperdigando por variados lugares la misma lógica (o parte de ella). Observando los siguientes trozos de código: …y comparando entre ellos, es evidente que obtienen resultados distintos, aunque con un mismo trozo de lógica: ambos quieren filtrar por ciudades africanas (CountryWorldRegion=WorldRegion.Africa), ordenándolas por nombre. El primero despliega en la salida exactamente eso (supongamos que City_Block contiene atributos de ciudades), y el segundo las atracciones de categoría ‘Safaris’ de ciudades africanas. Obsérvese el bloque de código en común, compuesto por un orden y un filtro. Podría haber decenas de objetos GeneXus donde se necesitaran las ciudades africanas ordenadas por nombre. Un Data Selector permite la centralización y el reuso de navegaciones, evitando la duplicación de código. Pero tiene más ventajas. Es muy común que la gente incorporada al staff de desarrollo en un momento dado desconozca los criterios o filtros comunes de 171 una aplicación. Suponiendo que se quiere promocionar fuertemente el turismo africano, suele pasar, por ejemplo, que se le pida un listado de ciudades y salgan todas las ciudades en lugar de sólo las africanas. La centralización de estos criterios reduce enormemente la curva de aprendizaje haciendo productivo al personal rápidamente y reduciendo el número de errores. En el ejemplo anterior, si le diéramos un nombre a esas cláusulas comunes como si fuera una macro que luego pudiera expandirse… ¡obtendríamos un Data Selector! Un nombre para un criterio de búsqueda de datos. En nuestro caso: AfricanCities. Los procedimientos anteriores nos quedarán: 172 A la derecha del comando For each se ha escrito la cláusula Using la cual dice que “se utilice el contenido” del Data Selector AfricanCities; y eso es todo… si luego modificamos el criterio, lo hacemos dentro del objeto Data Selector y ¡se cambia en todos lados automáticamente! El camino de los procesos Los humanos somos buenos para muchas cosas, pero en tareas como buscar un documento entre cientos de otros, tener presente sus vencimientos, y asegurarse de que el trabajo terminado pase de un lugar a otro respetando una secuencia previamente definida... hmmm ¿lo somos? GXflow, conocida herramienta de Artech que se relaciona con la automatización de los procesos en las empresas, determinando flujos de tareas, está también integrado a GeneXus X. ¿Por qué? Las aplicaciones cada vez más contienen flujos de procesos, que requieren de componentes de Workflow. En efecto, usted podrá crear sus propios diagramas dentro del IDE de GeneXus, simplificando mucho las tareas de integración y haciendo sencilla la vinculación de las actividades con los objetos de la KB. En la imagen se ha diseñado un sencillo ejemplo que muestra el flujo del proceso de reserva de un vuelo. 173 La reserva en sí misma, expresada por el primer bloque del esquema, no es ni más ni menos que la puesta en ejecución de la transacción a la que representa. Luego se evalúa determinada condición, como podría ser que los datos están correctos y completos. A los efectos de exponer un ejemplo, vea las siguientes condiciones: Print Tickets IF FlightStatus = ‘AUTHORIZED’ Get in contact with Client IF FlightStatus = ‘CORRECT PERSONAL INFORMATION’ Estas condiciones son expresadas mediante un editor de condiciones. La idea es que se puedan expresar reglas, donde primero se indica el nombre de la tarea sucesora e inmediatamente la condición que se tiene que cumplir para que siga por ese camino. Sencillo, práctico y potente. En caso de que así sea, se imprime el ticket de reserva y se envía a su destinatario. De lo contrario, el flujo se deriva hacia la puesta en contacto con el cliente, se actualiza la información (seguramente mediante la transacción correspondiente a los clientes), y se vuelve a retornar el flujo hacia la reserva, reiniciándose el ciclo. Back y Front End - Intra e Inter Net En este libro nos hemos permitido inmiscuir en algunos momentos de la vida de nuestros personajes, lo que nos permitió observar cómo se han implementado algunas partes de la aplicación. El objetivo: ver un poco de cada cosa. Pero no hemos visto el todo. Por momentos los autores sentimos un poco de vergüenza de nuestro voyeurismo y quitamos la mirada, para luego, tentados, volver nuestros ojos a la historia. Por esta razón, hemos visto entremezclados objetos que serán utilizados en el back-end de la aplicación (por ejemplo los Work With, dado que la actualización de la información de 174 aerolíneas, países, ciudades, vuelos, pasajeros, etc., sólo podrá ser efectuado por empleados de la empresa, con permisos, a través de la intranet), con otros que pueden utilizarse en el front-end, es decir, en el sitio Web que será consultado por personas de todo el mundo vía Internet. Sin embargo, Diego, Mike y Julia han trajinado mucho más de lo que hemos podido apreciar. Han hecho la página principal de la Intranet, modificando un poco la página Home creada por el pattern Work With. Han agregado una sección de login a la master page… web panels con consultas variadas y distintos menús. Esa misma tarde, sin ir más lejos, antes de la cita con Mary, mientras Paul recibía un theme para modificar, Julia se contactaba con los usuarios de Travel Agency y Mike terminaba los testeos. Diego observaba detenidamente cómo había quedado la página principal del sitio web de la empresa: 175 Obsérvese que se brinda información de hoteles, de rentadoras de autos y del estado del clima. Estas son aplicaciones de terceros, publicadas como Web Services que se pueden integrar (consumir) fácilmente dentro de GeneXus, con el nuevo tipo de objeto External Object. A esta altura el lector tiene todos los elementos para hacer la radiografía de la página y sospechar qué puede haber detrás de la misma: un web panel, web components, master page, theme, imágenes, contoles… Final… ¿o principio? Hasta aquí hemos obtenido distintas piezas de un rompecabezas que usted irá armando, valiéndose de la más importante: su imaginación. Con ella, las piezas faltantes se le irán descubriendo en una búsqueda continua. Inacabable. El puzzle está siempre próximo a… pero nunca se completa. Allí radica su secreto: mantenernos eternamente en vilo, buscando, pensando, imaginando, cambiando. “Amigos, con fe, con alegría, con generosidad, con la sinergia de esta Comunidad, vamos adelante, y preservemos la magia.” Breogán Gonda en el cierre del XVI Encuentro Internacional GeneXus 20 de Setiembre de 2006 ¿Qué pasó con nuestros personajes? ¿Fue exitoso el proyecto de desarrollo? ¿Cómo terminó esa cena con música? ¿Se habrá convencido Mary de ingresar a la Comunidad? Piezas que faltan, y que cada uno encontrará… como los secretos que aún gesta la X… …¿O es que esta historia recién empieza? 176 177