Manual GTK Comunidad
Manual GTK Comunidad
Manual GTK Comunidad
com
Mexicali, B.C.
JULIO DE 2007.
2007 is
granted
under the terms of the GNU Free Documentation License, Version 1.2 or any later version published by the Free Software Foundation; with no Invariant Sections, no Front-Cover Texts, and no Back-Cover Texts. "GNU Free Documentation License". A copy of the license is included in the section entitled
ndice de contenido
Contenido Pgina
ndice de contenido................................................................................................................................... i ndice de tablas........................................................................................................................................iv ndice de figuras....................................................................................................................................... v Introduccin.......................................................................................................................................... viii Problemas a resolver...............................................................................................................................ix Estructura general del documento......................................................................................................... x GNU Free Documentation License........................................................................................................xi Captulo I Antecedentes Introduccin...........................................................................................................................................2 1.1 Glib..................................................................................................................................................2 1.2 GTK.................................................................................................................................................4 Captulo II Glib Introduccin...........................................................................................................................................7 2.1 Tipos de datos..................................................................................................................................7 2.1.1 Lmites de tipos de datos......................................................................................................... 8 2.2 Macros y definiciones tiles............................................................................................................9 2.2.1 Macros bsicas.......................................................................................................................10 2.2.2 Macros referentes al SO.........................................................................................................10 2.2.3 Macros de operaciones matemticas sencillas.......................................................................11 2.2.4 Macros para verificacin de errores excepciones y depurado............................................... 12 2.2.5 Macros para manejo de memoria...........................................................................................14 2.2.6 Macros de conversin de tipos.............................................................................................. 16 2.3 Tratamiento de mensajes............................................................................................................... 18 2.3.1 Despliegue de informacin variada....................................................................................... 20 2.3.1.1 Ejemplo.......................................................................................................................... 21 2.3.2 Registro de mensajes y advertencias..................................................................................... 24 2.3.2.1 Ejemplo.......................................................................................................................... 24 2.3.3 Registro y despliegue de errores............................................................................................26 2.4 Tratamiento de cadenas................................................................................................................. 27 2.4.1 Perspectiva procedimental..................................................................................................... 28 2.4.2 Perspectiva Orientada a Objetos: Gstring..............................................................................30 2.4.2.1 Propiedades.................................................................................................................... 30 2.4.2.2 Constructor de clase de GString.................................................................................... 31 2.4.2.3 Mtodos de clase............................................................................................................31 i
2.4.2.4 Destructor de clase.........................................................................................................34 2.4.2.5 Ciclo de vida.................................................................................................................. 35 2.4.3 Ejemplo..................................................................................................................................36 2.5 Estructuras de datos: Listas enlazadas simples............................................................................. 38 2.5.1 Propiedades............................................................................................................................38 2.5.2 Constructor de clase...............................................................................................................39 2.5.3 Funciones asociadas o Mtodos de clase...............................................................................39 2.5.4 Destructor de clase.................................................................................................................46 2.5.5 Ciclo de vida de una lista enlazada simple............................................................................ 47 2.5.6 Ejemplo..................................................................................................................................48 Captulo III Introduccin a GTK+ Introduccin.........................................................................................................................................56 3.1 El concepto de widget................................................................................................................... 56 3.2 Nomenclatura GTK....................................................................................................................... 57 3.3 Proceso de creacin de un widget................................................................................................. 59 3.4 Teora de seales y retrollamadas................................................................................................. 62 3.4.1 Rutinas de tratamiento de seales y eventos..........................................................................63 3.4.2 Eventos.................................................................................................................................. 65 3.4.3 Bucle de ejecucin y eventos.................................................................................................67 3.4.4 Ejemplo..................................................................................................................................69 3.5 Widgets contenedores....................................................................................................................73 3.5.1 Mtodos de la clase GtkContainer.........................................................................................74 3.6 Cajas.............................................................................................................................................. 76 3.6.1 Descripcin............................................................................................................................ 76 3.6.2 Constructor de clase...............................................................................................................77 3.6.3 Mtodos de clase bsicos.......................................................................................................78 3.6.4 Mtodos de clase avanzados..................................................................................................79 3.7 Tablas............................................................................................................................................ 82 3.7.1 Descripcin............................................................................................................................ 82 3.7.2 Constructor de clase...............................................................................................................84 3.7.3 Mtodos de clase....................................................................................................................85 3.8 Etiquetas........................................................................................................................................ 89 3.8.1 Descripcin............................................................................................................................ 89 3.8.2 Constructor de clase...............................................................................................................89 3.8.3 Mtodos de clase bsicos.......................................................................................................90 3.8.4 Mtodos de clase avanzados..................................................................................................91 3.8.5 Ejemplos................................................................................................................................ 93 3.9 Botones........................................................................................................................................102 3.9.1 Descripcin.......................................................................................................................... 102 3.9.2 Constructores de clase......................................................................................................... 102 3.9.3 Mtodos de clase..................................................................................................................104 3.9.4 Seales y eventos.................................................................................................................106 3.9.4.1 La seal "clicked"........................................................................................................ 106 3.9.5 Ejemplos.............................................................................................................................. 107 3.10 Cajas de texto............................................................................................................................ 114 3.10.1 Constructor de clase...........................................................................................................116 ii
3.10.2 Mtodos de clase................................................................................................................116 3.10.3 Seales............................................................................................................................... 118 3.10.4 Ejemplos............................................................................................................................ 120 Captulo IV Glade y libglade Introduccin.......................................................................................................................................127 4.1 Conociendo Glade....................................................................................................................... 127 4.2 Trabajando con un proyecto........................................................................................................ 131 4.3 Introduccin a libglade................................................................................................................135 4.3.1 Proceso de creacin de una aplicacin con libglade............................................................135 4.4 Constructor de clase.................................................................................................................... 137 4.5 Mtodos de clase......................................................................................................................... 138 4.6 Ejemplos......................................................................................................................................140 4.6.1 Ejemplo 1 Ciclo de vida de una aplicacin con libglade..................................................140
iii
ndice de tablas
Tabla 1: Tipos de datos de Glib................................................................................................................. 7 Tabla 2: Otros tipos de datos definidos en Glib.........................................................................................8 Tabla 3: Lmites de tipos de datos..............................................................................................................9 Tabla 4: Constantes matemticas predefinidas en Glib........................................................................... 12 Tabla 5: Etiquetas vlidas para el lenguaje de marcado de Pango...........................................................93 Tabla 6: Atajos de teclado disponible en cajas de texto.........................................................................115 Tabla 7: Seales de GtkEntry que usan el mismo prototipo de ret8rollamada...................................... 119
iv
ndice de figuras
Figura 2.3.1.............................................................................................................................................. 25 Figura 2.3.2.............................................................................................................................................. 26 Figura 2.4.1.............................................................................................................................................. 37 Figura 2.5.1.............................................................................................................................................. 38 Figura 2.5.2.............................................................................................................................................. 50 Figura 2.5.3.............................................................................................................................................. 54 Figura 3.1.1.............................................................................................................................................. 56 Figura 3.1.2.............................................................................................................................................. 57 Figura 3.3.1.............................................................................................................................................. 61 Figura 3.4.1.............................................................................................................................................. 72 Figura 3.5.1.............................................................................................................................................. 73 Figura 3.6.1.............................................................................................................................................. 76 Figura 3.7.1.............................................................................................................................................. 82 Figura 3.7.2.............................................................................................................................................. 83 Figura 3.8.1.............................................................................................................................................. 89 Figura 3.8.2.............................................................................................................................................. 89 Figura 3.8.3.............................................................................................................................................. 95 Figura 3.8.4.............................................................................................................................................. 98 Figura 3.8.5............................................................................................................................................ 100 Figura 3.8.6............................................................................................................................................ 101 Figura 3.9.1............................................................................................................................................ 102 Figura 3.9.2............................................................................................................................................ 102 Figura 3.9.3............................................................................................................................................ 103 Figura 3.9.4............................................................................................................................................ 109 Figura 3.9.5............................................................................................................................................ 112 Figura 3.9.6............................................................................................................................................ 113 Figura 3.9.7............................................................................................................................................ 113 Figura 3.9.8............................................................................................................................................ 114 Figura 3.10.1.......................................................................................................................................... 114 Figura 3.10.2.......................................................................................................................................... 115 Figura 3.10.3.......................................................................................................................................... 122 Figura 3.10.4.......................................................................................................................................... 125 Figura 3.10.5.......................................................................................................................................... 125 Figura 4.1.1............................................................................................................................................ 128 Figura 4.1.2............................................................................................................................................ 129 Figura 4.2.1............................................................................................................................................ 131 Figura 4.2.2............................................................................................................................................ 131 Figura 4.2.3............................................................................................................................................ 132 Figura 4.2.4............................................................................................................................................ 132 Figura 4.2.5............................................................................................................................................ 133 Figura 4.2.6............................................................................................................................................ 133 Figura 4.2.7............................................................................................................................................ 134 Figura 4.3.1............................................................................................................................................ 136 Figura 4.6.1............................................................................................................................................ 141 v
vi
Anexos
ANEXO 4.6.1.1 : El COMPILADOR GCC ANEXO 4.6.1.2 : MAKE ANEXO 4.6.1.3 : STOCK ITEMS.
vii
Introduccin
Las interfases grficas de usuario(GUI, Graphical User Interfase) son interfases Humano-Mquina cuyo uso es altamente popular en estos dias. Cuando llega la necesidad de crear una GUI se puede recurrir a herramientas de alto nivel como Visual Basic, Delphi o Java, de gran popularidad en ambientes Windows, sin embargo si se necesita interactuar en ambientes en tiempo real o interactuar directamente con el hardware, ah es cuando se hace un salto hacia el pasado, remontndose a los aos 90 y se utilizan sistemas MS-DOS para tales tareas. Esto funciona efectivamente si los requerimientos no son grandes, sin embargo cuando se hace inminente crear grficos que representen la informacin necesaria acerca de un proceso, echar mano de comunicaciones por TCP/IP mientras se ejecutan otra gran variedad de tareas, una de las mejores solucin en costo/beneficio ser el sistema operativo Linux. El objetivo de este documento es dejar una gua actualizada de una de las mejores herramientas de creacin de interfases grficas de usuario disponibles en Linux: Gtk+.
viii
Problemas a resolver
GTK+ es una librera que evoluciona da a da, haciendo correcciones y mejoras y constantemente adaptndose a las necesidades de los usuarios. GTK+ es un proyecto de Software Libre y es la base del ambiente de escritorio GNOMEII donde gran cantidad de aplicaciones aparecen da a da. No obstante el gran esfuerzo de la comunidad de Software Libre por documentar el uso y aplicacin de GTK+ de una manera coherente, an existen partes sin escribir o simplemente sin integrar en un solo documento. Su carcter de proyecto de colaboracin internacional sobrepasa la oferta de guas actualizadas en nuestro idioma: la documentacin que ofrecen los escritores y colaboradores de GTK+ se mantiene a un nivel altamente tcnico; grande es el desafo al que se enfrentan cualquier posible colaborador del proyecto Micro Laboratorio Virtualsin un manual que allane el camino por recorrer. Este documento condensar de forma coherente y ordenada la informacin disponible en diferentes fuentes bibliogrficas, en Internet y principalmente de los manuales altamente tcnicos que crean los escritores y colaboradores de GTK+. El resultado final de esta tesis ser un texto de referencia til a los desarrolladores que busquen alguna caja de herramientas grficas para presentar la informacin generada de algn proceso industrial o cientfico.
I
El trmino Software Libre viene del termino en ingls Free Software. En el idioma espaol no se da la dualidad contextual con la palabra Free, pues este homnimo puede tomar el significado de gratis y libre. En el espaol podemos decir Libre sin riesgo a dualidad de interpretaciones.. II El proyecto GNOME busca brindar un ambiente grfico, amigable y fcil de entender a los usuarios del Sistema Operativo Linux. http://www.gnome.org
ix
El captulo I describe el cmo y porqu de la creacin de librera GTK+. El captulo II presenta una breve resea de los componentes (libreras) que conforman el entorno de programacin de GTK+
En el captulo III se discuten los conceptos primordiales de GTK+ como el widget, la nomenclatura de libreras objetos y mtodos de GTK+, la teora de seales y retrollamadas y widgets contenedores como cajas y tablas.
El captulo II aborda de lleno y con detalles la librera base de GTK+, Glib, la cual contiene tipos de datos, macros, tratamiento de mensajes, cadenas y estructuras de datos.
El captulo IV examina la librera libglade se utiliza para permitir que los programadores tengan la posibilidad de crear GUI's de manera visual.
Una vez reseada la librera libglade, cabe la posibilidad de usar widgets ms complejos en el captulo V.
0. PREAMBLE
The purpose of this License is to make a manual, textbook, or other functional and useful document "free" in the sense of freedom: to assure everyone the effective freedom to copy and redistribute it, with or without modifying it, either commercially or noncommercially. Secondarily, this License preserves for the author and publisher a way to get credit for their work, while not being considered responsible for modifications made by others.
This License is a kind of "copyleft", which means that derivative works of the document must themselves be free in the same sense. It complements the GNU General Public License, which is a copyleft license designed for free software.
We have designed this License in order to use it for manuals for free xi
software, because free software needs free documentation: a free program should come with manuals providing the same freedoms that the software does. But this License is not limited to software manuals; it can be used for any textual work, regardless of subject matter or whether it is published as a printed book. We recommend this License principally for works whose purpose is instruction or reference.
This License applies to any manual or other work, in any medium, that contains a notice placed by the copyright holder saying it can be distributed under the terms of this License. Such a notice grants a world-wide, royalty-free license, unlimited in duration, to use that work under the conditions stated herein. The "Document", below, refers to any such manual or work. Any member of the public is a licensee, and is addressed as "you". You accept the license if you copy, modify or distribute the work in a way requiring permission under copyright law.
A "Modified Version" of the Document means any work containing the Document or a portion of it, either copied verbatim, or with modifications and/or translated into another language.
xii
A "Secondary Section" is a named appendix or a front-matter section of the Document that deals exclusively with the relationship of the publishers or authors of the Document to the Document's overall subject (or to related matters) and contains nothing that could fall directly within that overall subject. (Thus, if the Document is in part a textbook of mathematics, a Secondary Section may not explain any mathematics.) The relationship could be a matter of historical connection with the subject or with related matters, or of legal, commercial, philosophical, ethical or political position regarding them.
The "Invariant Sections" are certain Secondary Sections whose titles are designated, as being those of Invariant Sections, in the notice that says that the Document is released under this License. If a section does not fit the above definition of Secondary then it is not allowed to be designated as Invariant. The Document may contain zero Invariant Sections. If the Document does not identify any Invariant Sections then there are none.
The "Cover Texts" are certain short passages of text that are listed, as Front-Cover Texts or Back-Cover Texts, in the notice that says that the Document is released under this License. A Front-Cover Text may be at most 5 words, and a Back-Cover Text may be at most 25 words.
xiii
A "Transparent" copy of the Document means a machine-readable copy, represented in a format whose specification is available to the general public, that is suitable for revising the document straightforwardly with generic text editors or (for images composed of pixels) generic paint programs or (for drawings) some widely available drawing editor, and that is suitable for input to text formatters or for automatic translation to a variety of formats suitable for input to text formatters. A copy made in an otherwise Transparent file format whose markup, or absence of markup, has been arranged to thwart or discourage subsequent modification by readers is not Transparent. An image format is not Transparent if used for any substantial amount of text. A copy that is not "Transparent" is called "Opaque".
Examples of suitable formats for Transparent copies include plain ASCII without markup, Texinfo input format, LaTeX input format, SGML or XML using a publicly available DTD, and standard-conforming simple HTML, PostScript or PDF designed for human modification. Examples of transparent image formats include PNG, XCF and JPG. Opaque formats include proprietary formats that can be read and edited only by proprietary word processors, SGML or XML for which the DTD and/or processing tools are not generally available, and the machine-generated HTML, PostScript or PDF produced by some word processors for output purposes only.
xiv
The "Title Page" means, for a printed book, the title page itself, plus such following pages as are needed to hold, legibly, the material this License requires to appear in the title page. For works in formats which do not have any title page as such, "Title Page" means the text near the most prominent appearance of the work's title, preceding the beginning of the body of the text.
A section "Entitled XYZ" means a named subunit of the Document whose title either is precisely XYZ or contains XYZ in parentheses following text that translates XYZ in another language. (Here XYZ stands for a specific section name mentioned below, such as "Acknowledgements", "Dedications", "Endorsements", or "History".) To "Preserve the Title" of such a section when you modify the Document means that it remains a section "Entitled XYZ" according to this definition.
The Document may include Warranty Disclaimers next to the notice which states that this License applies to the Document. These Warranty Disclaimers are considered to be included by reference in this License, but only as regards disclaiming warranties: any other implication that these Warranty Disclaimers may have is void and has no effect on the meaning of this License.
2. VERBATIM COPYING xv
You may copy and distribute the Document in any medium, either commercially or noncommercially, provided that this License, the copyright notices, and the license notice saying this License applies to the Document are reproduced in all copies, and that you add no other conditions whatsoever to those of this License. You may not use technical measures to obstruct or control the reading or further copying of the copies you make or distribute. However, you may accept compensation in exchange for copies. If you distribute a large enough number of copies you must also follow the conditions in section 3.
You may also lend copies, under the same conditions stated above, and you may publicly display copies.
3. COPYING IN QUANTITY
If you publish printed copies (or copies in media that commonly have printed covers) of the Document, numbering more than 100, and the Document's license notice requires Cover Texts, you must enclose the copies in covers that carry, clearly and legibly, all these Cover Texts: Front-Cover Texts on the front cover, and Back-Cover Texts on the back cover. Both covers must also clearly and legibly identify you as the publisher of these copies. The front cover must present xvi
the full title with all words of the title equally prominent and visible. You may add other material on the covers in addition. Copying with changes limited to the covers, as long as they preserve the title of the Document and satisfy these conditions, can be treated as verbatim copying in other respects.
If the required texts for either cover are too voluminous to fit legibly, you should put the first ones listed (as many as fit reasonably) on the actual cover, and continue the rest onto adjacent pages.
If you publish or distribute Opaque copies of the Document numbering more than 100, you must either include a machine-readable Transparent copy along with each Opaque copy, or state in or with each Opaque copy a computer-network location from which the general network-using public has access to download using public-standard network protocols a complete Transparent copy of the Document, free of added material. If you use the latter option, you must take reasonably prudent steps, when you begin distribution of Opaque copies in quantity, to ensure that this Transparent copy will remain thus accessible at the stated location until at least one year after the last time you distribute an Opaque copy (directly or through your agents or retailers) of that edition to the public.
xvii
It is requested, but not required, that you contact the authors of the Document well before redistributing any large number of copies, to give them a chance to provide you with an updated version of the Document.
4. MODIFICATIONS
You may copy and distribute a Modified Version of the Document under the conditions of sections 2 and 3 above, provided that you release the Modified Version under precisely this License, with the Modified Version filling the role of the Document, thus licensing distribution and modification of the Modified Version to whoever possesses a copy of it. In addition, you must do these things in the Modified Version:
A. Use in the Title Page (and on the covers, if any) a title distinct from that of the Document, and from those of previous versions (which should, if there were any, be listed in the History section of the Document). You may use the same title as a previous version if the original publisher of that version gives permission. B. List on the Title Page, as authors, one or more persons or entities responsible for authorship of the modifications in the Modified Version, together with at least five of the principal authors of the Document (all of its principal authors, if it has fewer than five), unless they release you from this requirement. xviii
C. State on the Title page the name of the publisher of the Modified Version, as the publisher. D. Preserve all the copyright notices of the Document. E. Add an appropriate copyright notice for your modifications adjacent to the other copyright notices. F. Include, immediately after the copyright notices, a license notice giving the public permission to use the Modified Version under the terms of this License, in the form shown in the Addendum below. G. Preserve in that license notice the full lists of Invariant Sections and required Cover Texts given in the Document's license notice. H. Include an unaltered copy of this License. I. Preserve the section Entitled "History", Preserve its Title, and add to it an item stating at least the title, year, new authors, and publisher of the Modified Version as given on the Title Page. If there is no section Entitled "History" in the Document, create one stating the title, year, authors, and publisher of the Document as given on its Title Page, then add an item describing the Modified Version as stated in the previous sentence. J. Preserve the network location, if any, given in the Document for public access to a Transparent copy of the Document, and likewise the network locations given in the Document for previous versions it was based on. These may be placed in the "History" section. You may omit a network location for a work that was published at least four years before the Document itself, or if the original xix
publisher of the version it refers to gives permission. K. For any section Entitled "Acknowledgements" or "Dedications", Preserve the Title of the section, and preserve in the section all the substance and tone of each of the contributor acknowledgements and/or dedications given therein. L. Preserve all the Invariant Sections of the Document, unaltered in their text and in their titles. Section numbers or the equivalent are not considered part of the section titles. M. Delete any section Entitled "Endorsements". Such a section may not be included in the Modified Version. N. Do not retitle any existing section to be Entitled "Endorsements" or to conflict in title with any Invariant Section. O. Preserve any Warranty Disclaimers.
If the Modified Version includes new front-matter sections or appendices that qualify as Secondary Sections and contain no material copied from the Document, you may at your option designate some or all of these sections as invariant. To do this, add their titles to the list of Invariant Sections in the Modified Version's license notice. These titles must be distinct from any other section titles.
You may add a section Entitled "Endorsements", provided it contains nothing but endorsements of your Modified Version by various parties--for example, statements of peer review or that the text has xx
You may add a passage of up to five words as a Front-Cover Text, and a passage of up to 25 words as a Back-Cover Text, to the end of the list of Cover Texts in the Modified Version. Only one passage of Front-Cover Text and one of Back-Cover Text may be added by (or through arrangements made by) any one entity. If the Document already includes a cover text for the same cover, previously added by you or by arrangement made by the same entity you are acting on behalf of, you may not add another; but you may replace the old one, on explicit permission from the previous publisher that added the old one.
The author(s) and publisher(s) of the Document do not by this License give permission to use their names for publicity for or to assert or imply endorsement of any Modified Version.
5. COMBINING DOCUMENTS
You may combine the Document with other documents released under this License, under the terms defined in section 4 above for modified versions, provided that you include in the combination all of the Invariant Sections of all of the original documents, unmodified, and xxi
list them all as Invariant Sections of your combined work in its license notice, and that you preserve all their Warranty Disclaimers.
The combined work need only contain one copy of this License, and multiple identical Invariant Sections may be replaced with a single copy. If there are multiple Invariant Sections with the same name but different contents, make the title of each such section unique by adding at the end of it, in parentheses, the name of the original author or publisher of that section if known, or else a unique number. Make the same adjustment to the section titles in the list of Invariant Sections in the license notice of the combined work.
In the combination, you must combine any sections Entitled "History" in the various original documents, forming one section Entitled "History"; likewise combine any sections Entitled "Acknowledgements", and any sections Entitled "Dedications". You must delete all sections Entitled "Endorsements".
6. COLLECTIONS OF DOCUMENTS
You may make a collection consisting of the Document and other documents released under this License, and replace the individual copies of this License in the various documents with a single copy that is included in xxii
the collection, provided that you follow the rules of this License for verbatim copying of each of the documents in all other respects.
You may extract a single document from such a collection, and distribute it individually under this License, provided you insert a copy of this License into the extracted document, and follow this License in all other respects regarding verbatim copying of that document.
A compilation of the Document or its derivatives with other separate and independent documents or works, in or on a volume of a storage or distribution medium, is called an "aggregate" if the copyright resulting from the compilation is not used to limit the legal rights of the compilation's users beyond what the individual works permit. When the Document is included in an aggregate, this License does not apply to the other works in the aggregate which are not themselves derivative works of the Document.
If the Cover Text requirement of section 3 is applicable to these copies of the Document, then if the Document is less than one half of the entire aggregate, the Document's Cover Texts may be placed on covers that bracket the Document within the aggregate, or the xxiii
electronic equivalent of covers if the Document is in electronic form. Otherwise they must appear on printed covers that bracket the whole aggregate.
8. TRANSLATION
Translation is considered a kind of modification, so you may distribute translations of the Document under the terms of section 4. Replacing Invariant Sections with translations requires special permission from their copyright holders, but you may include translations of some or all Invariant Sections in addition to the original versions of these Invariant Sections. You may include a translation of this License, and all the license notices in the Document, and any Warranty Disclaimers, provided that you also include the original English version of this License and the original versions of those notices and disclaimers. In case of a disagreement between the translation and the original version of this License or a notice or disclaimer, the original version will prevail.
If a section in the Document is Entitled "Acknowledgements", "Dedications", or "History", the requirement (section 4) to Preserve its Title (section 1) will typically require changing the actual title. xxiv
9. TERMINATION
You may not copy, modify, sublicense, or distribute the Document except as expressly provided for under this License. Any other attempt to copy, modify, sublicense or distribute the Document is void, and will automatically terminate your rights under this License. However, parties who have received copies, or rights, from you under this License will not have their licenses terminated so long as such parties remain in full compliance.
The Free Software Foundation may publish new, revised versions of the GNU Free Documentation License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. See http://www.gnu.org/copyleft/.
Each version of the License is given a distinguishing version number. If the Document specifies that a particular numbered version of this License "or any later version" applies to it, you have the option of xxv
following the terms and conditions either of that specified version or of any later version that has been published (not as a draft) by the Free Software Foundation. If the Document does not specify a version number of this License, you may choose any version ever published (not as a draft) by the Free Software Foundation.
To use this License in a document you have written, include a copy of the License in the document and put the following copyright and license notices just after the title page:
Copyright (c) YEAR YOUR NAME. Permission is granted to copy, distribute and/or modify this document under the terms of the GNU Free Documentation License, Version 1.2 or any later version published by the Free Software Foundation; with no Invariant Sections, no Front-Cover Texts, and no Back-Cover Texts. A copy of the license is included in the section entitled "GNU Free Documentation License".
If you have Invariant Sections, Front-Cover Texts and Back-Cover Texts, replace the "with...Texts." line with this:
xxvi
with the Invariant Sections being LIST THEIR TITLES, with the Front-Cover Texts being LIST, and with the Back-Cover Texts being LIST.
If you have Invariant Sections without Cover Texts, or some other combination of the three, merge those two alternatives to suit the situation.
If your document contains nontrivial examples of program code, we recommend releasing these examples in parallel under your choice of free software license, such as the GNU General Public License, to permit their use in free software.
xxvii
CAPTULO I 1 I Antecedentes
Introduccin
GTK+1 es una librera para la creacin de interfases grficas de usuario. Su licencia permite desarrollar programas libres as como aplicaciones comerciales. Todo esto sin tener que pagar licencias o regalas para su uso. GTK es el acrnimo de GIMP2 Toolkit debido a que originalmente fue escrito para desarrollar el "Programa de manipulacin de imgenes de GNU" (GNU Image Manipulation Program, GIMP), sin embargo GTK+ se usa en gran cantidad de proyectos de software incluyendo al proyecto GNOME3. GTK+ se ha construido en base a tres libreras: GDK (GIMP Drawing Kit) para interactuar con el sistema grfico, gdk-pixbuf para manipular imgenes y Glib para integracin con el Sistema Operativo(SO). GTK+ se construye mediante un concepto de objetos, eventos y retro llamadas. Su diseo, en constante mejora, le permite lograr una gran velocidad de ejecucin, interactuar con diferentes lenguajes de programacin como C, C++, Python, C#, Perl o Ada, y ser un ambiente de desarrollo multi-plataforma (UNIX, Linux, WindowsTM, MacOS XTM).
1.1 Glib
Glib es una librera de utilidades de propsito general, la cual provee tipos de datos tiles, as como macros, conversiones de tipo, utilidades de cadenas y muchas ms. Glib se encarga de interactuar con el SO anfitrin sin distraer al programador de las diferencias ms significativas entre las distintas variedades de estos. La siguiente lista rene las principales caractersticas de Glib:
Tipos de datos: Los tipos de datos (char, int, float, double), pueden diferir entre los diferentes SO arquitecturas de hardware. Glib libera al programador de prestar atencin a estos detalles y en su lugar ofrece un conjunto de tipos propios (gchar, gint, gint32, guint, guint32, guchar, etc).
Gestin de memoria: La gestin de memoria es una tarea especialmente delicada, por lo que Glib la administra automticamente. Si el programador necesita mayor control de la memoria entonces tendr a su disposicin gran cantidad de mtodos que le permitirn un control ms exhaustivo de sta.
Estructuras de datos: Las estructuras de datos ms comunes como listas enlazadas, arreglos dinmicos, tablas de tablas de claves, pilas y colas, ya estn disponibles en Glib.
Sistema de objetos: El sistema de Objetos de Glib se ha diseado con miras a flexibilidad y capacidad de adaptarse a las necesidades del usuario. Se ha creado con la intencin de integrarse fcilmente con otros lenguajes de programacin diferentes de C. Todos los objetos de GTK+ se derivan de la clase fundamental de Glib: Gobject.
Bucle de ejecucin: En lugar de crear un bucle de eventos nosotros mismos, podemos dejarle la tarea a Glib para centrarnos en el diseo y desarrollo de nuestras aplicaciones asncronas. Glib se encarga de la distribucin de seales y mensajes a las diferentes partes de nuestro programa. Tambin se encarga de administrar alarmas (temporizadores para aplicaciones sncronas), momentos de inactividad de la aplicacin, eventos de entrada y salida en tuberas, sockets, o descriptores de archivos, as como hilos.
1.2 GTK
GTK es la parte grfica que interacta con el usuario. GTK nos ofrece todo lo necesario para el desarrollo de interfaces grficas de usuario como botones, cajas de texto, mens, ventanas, etc.. Otras formas o widgets mucho ms complejos pueden ser creados para satisfacer nuestras necesidades. Los componentes de GTK+ se interrelacionan de la siguiente manera:
GLIB es la base del proyecto GTK+ y GNOME. GLIB provee las estructuras de datos en el lenguaje C. Contiene la infraestructura necesaria para construir envoltorios e interfases para tales funciones como un bucle de ejecucin de eventos, hilos, carga dinmica de mdulos y un sistema de objetos.
GObject es la parte de Glib que implementa las caractersticas de lenguajes orientados a objetos que no estn disponibles intrnsecamente en el lenguaje C. De esta manera GTK+ puede ser una caja de herramientas orientada a objetos sin tener que implementarse en C++ u otro lenguaje.
GDK interacta con las primitivas grficas de cada SO y las prepara para su uso con GTK+. En los sistemas UNIX soportados como Linux y Solaris, GDK interacta con el sistema X-Win dow[1], mientras que en sistemas WindowsTM lo har con GDI[2] y en MacOS XTM con QuartzTM[3]. El programador de aplicaciones no notar esto y, salvo raras excepciones, un programa escrito para GTK+ en Linux, por ejemplo, puede compilarse en Windows sin mayor modificacin.
GdkPixbuf contiene convenientes rutinas para el manejo de imgenes(png, gif, jpeg, etc), tales como abrir y guardar, trabajar con mapas de imgenes en memoria, escalado y animaciones entre otras caractersticas.
Pango es un entorno de trabajo para el procesamiento y despliegue de texto internacional utilizando UTF8 como cdigo base[4]; est integrado por completo en GTK+ y permite desplegar caracteres en distintos alfabetos y estilos de escritura (occidental, cirlico, rabe, chino, etc), texto bidireccional y con atributos como color, tamao y estilo o rotacin.
ATK proporciona un conjunto de interfases de accesibilidad. Si uno soporta interfases ATK en un programa, tendremos la posibilidad de utilizar diferentes herramientas como magnificadores de pantalla, texto a dilogo o diferentes mtodos de entrada para personas con capacidades diferentes.
CAPTULO II 2 II Glib
Introduccin
Algunos tipos no tan conocidos como void *, son realmente tiles para pasar diferentes estructuras de datos a funciones. Otros tipos definidos en Glib se muestran en la siguiente tabla.
Tabla 2: Otros tipos de datos definidos en Glib. Tipo definido en Glib gint8 gint16 gint32 gint64 guint8 guint16 guint32 guint64 guchar gushort gulong gsize gssize gconstcopointer Equivalencia en ANSI C typedef signed char gint8; typedef signed short gint16; typedef signed int gint32; typedef signed long long gint64; typedef unsigned char guint8; typedef unsigned short guint16; typedef unsigned int guint32; typedef unsigned long long guint64; typedef unsigned char typedef unsigned short typedef unsigned long guchar; gushort; gulong;
typedef unsigned int gsize; typedef signed int gssize; typedef const void *gconstpointer;
En lugar de incrementar la complejidad a las aplicaciones, los tipos de datos de Glib compactan complicadas definiciones de tipos en identificadores de tipos de datos que se reconocen ms fcilmente, pues son auto descriptivos. Esto elimina gran cantidad de trabajo a la hora de mantener programas muy grandes y el cdigo se hace ms legible.
2.1.1
Por cada tipo de dato definido en la tablas 1 y 2 Glib ha definido valores indicando los valores mnimos y mximos que pueden contener, exceptuando los tipos de datos gpointer, gconstpointer,gssize y gboolean. En lugar de memorizar nombres complicados podemos adivinar los lmites de un tipo de datos mediante su nombre. Por ejemplo, gint est definido entre los lmites G_MAXINT y G_MININT
mientras que un gdouble se encuentra acotado entre G_MAXDOUBLE y G_MINDOUBLE. Por conveniencia la tabla 3 agrupa los tipos de datos y sus lmites definidos. Tabla 3: Lmites de tipos de datos Tipo de dato gint gshort glong guint gushort gulong gint8 gint16 gint32 gint64 guint8 guint16 guint32 guint64 gfloat gdouble gsize Lmites G_MININT a G_MAXINT G_MINSHORT a G_MAXSHORT G_MINLONG a G_MAXLONG 0 a G_MAXUINT 0 a G_MAXUSHORT 0 a G_MAXULONG G_MININT8 a G_MAXINT8 G_MININT16 a G_MAXINT16 G_MININT32 a G_MAXINT32 G_MININT64 a G_MAXINT64 0 a G_MAXUINT8 0 a G_MAXUINT16 0 a G_MAXUINT32 0 a G_MAXUINT64 G_MINFLOAT a G_MAXFLOAT G_MINDOUBLE a G_MAXDOUBLE 0 a G_MAXSIZE
2.2.1
Macros bsicas
Las dos macros ms sencillas, sin embargo tiles son TRUE, FALSE. Solamente definen un cero para un valor de tipo gboolean falso y diferente de cero para un valor positivo. Sin embargo es mucho mas intuitivo y explicativo trabajar con TRUE y FALSE. Veamos una comparacin. #include <glib.h> /*...*/ if (encender_motor == TRUE) encender_motor(); abrir_cerradura(); /*...*/ /*...*/ if (encender_motor > 1) encender_motor(); abrir_cerradura(); /*...*/
2.2.2
Este conjunto de macros son tiles al escribir programas que funcionarn en diferentes SO. G_OS_WIN32, G_OS_BEOS, G_OS_UNIX solo estn definidas en su respectivo SO y deben utilizase entre bloques de directivas del compilador #ifdef ... #elif ...#endif, lo que causar que el compilador construya bloques diferentes en cada plataforma. Veamos el siguiente ejemplo. #include <glib.h> /* ... */ #ifdef G_OS_WIN32 const gchar *dispositivo = COM1; #elif G_OS_BE_OS const gchar *dispositivo = /dev/usb0; #else // G_OS_UNIX const gchar *dispositivo = /dev/ttyS0; 10
#endif /* ... */ El ejemplo anterior definir una compilacin condicional en la cual, dependiendo del SO donde se compile el programa, la cadena dispositivo tendr diferente valor y apuntar al canal de comunicaciones correcto en cada SO. Las macros G_DIR_SEPARATOR y G_DIR_SEPARATOR_S contienen el carcter separador de directorios. Su valor es '/' en sistemas UNIX y BeOS y '\' en sistemas Windows. La segunda macro contiene la misma informacin que la primera pero en formato de cadena: '/' y '\'. G_IS_DIR_SEPARATOR(c) acepta un carcter c y determina si es el carcter separador de directorios. Esta macro devuelve TRUE si el carcter es '/' en sistemas UNIX o '\' en Sistemas Windows. G_SEARCHPATH_SEPARATOR y G_SEARCHPATH_SEPARATOR_S devuelven el carcter
separador de rutas en forma de carcter o cadena respectivamente. Este carcter es ':' para sistemas UNIX y ';' para Windows.
2.2.3
Existen ciertas operaciones matemticas comunes que no se encuentran disponibles en la biblioteca estndar de C. MIN(a,b) y MAX(a,b) calculan el valor mnimo y mximo de entre dos nmeros a y b, mientras que ABS(n) calcula el valor absoluto de un nmero n. CLAMP(x,a,b) se asegura de que el nmero x se encuentre dentro de los lmites a y b. Si x se encuentra dentro de estos lmites, CLAMP() regresar el nmero x, si esto no se cumple y x es mayor que el lmite superior b, CLAMP() regresar este valor, de lo contrario (x es menor que el lmite
11
inferior a), CLAMP() regresar el valor de lmite inferior a. Esta macro resulta confusa, pero es til al posicionar objetos grficos en la pantalla y simular cierta resistencia al movimiento. La siguiente tabla muestra constantes matemticas predefinidas en Glib. En la documentacin de Glib existen uniones para acceder al signo, la mantisa y el exponente de nmeros de tipo coma flotante que cumplan con el estndar IEEE 754. Tabla 4: Constantes matemticas predefinidas en Glib. Smbolo matemtico Definicin en Glib G_PI G_PI2 G_PI4 G_SQRT2 G_E G_LN2 G_LN10 Valor 3.1415926535897932384626433832795028841971 1.5707963267948966192313216916397514420985 0.7853981633974483096156608458198757210492 1.4142135623730950488016887242096980785696 2.7182818284590452353602874713526624977572 0.6931471805599453094172321214581765680755 2.3025850929940456840179914546843642076011
2 4
2
e ln 2 ln 10 log 10 2
G_LOG2_BASE10 2.3025850929940456840179914546843642076011
2.2.4
Un buen diseo de software no viene de la noche a la maana. Parte importante del tiempo de desarrollo de un programa se consume en la depuracin de errores. Tambin es cierto que parte importante del total del cdigo fuente escrito de un programa robusto se dedica a la validacin y correccin de posibles errores, es decir, que las cosas que deban estar en orden realmente lo estn. Los desarrolladores de Glib nos ofrecen diferentes herramientas 7 macros para ayudarnos a mejorar nuestros programas.
12
g_assert() recibe como parmetro una expresin, tal y como se usa en el condicional if ... then ... else ... Si la condicin especificada falla o es FALSE, el programa termina especificando un mensaje de error. Por ejemplo la siguiente instruccin...
#include <glib.h> /* ... */ g_assert(puerto == ABIERTO); /* ... */
...terminar el programa con un mensaje de error Assertion failed, si la variable puerto es falsa.
puerto==ABIERTO
El complemento de g_assert() es g_assert_not_reached(). Esta macro termina el programa y despliega un mensaje de error si alguna vez se llega a ella. Un buen ejemplo de aplicacin de estas macros se dara en un hipottica funcin que transforma cadenas provenientes, por ejemplo, de una comunicacin serial:
void transformo_cadenas (gchar *cadena) { /* ... */ g_assert (cadena != NULL); /* ... */ }
g_assert() comprobar si el contenido de la variable cadena est vaco, de ser as interrumpir el programa impidiendo que ms errores se propaguen. Estas macros puede desactivarse en compilaciones finales mediante la definicin de G_DISABLE_ASSERT al momento de compilar la aplicacin. g_return_if_fail() toma una expresin y regresa de la funcin si tal expresin no resulta verdadera o TRUE. De lo contrario registra un mensaje de aviso y regresa de la funcin. g_return_if_fail() slo se puede utilizar en funciones que no regresan ningn valor. Para aquellas funciones que debe regresar un valor, est g_return_val_if_fail(expr, val), que regresa el valor val en funcin del la expresin expr al igual que g_return_if_fail().
13
Parecido al par anterior, g_return_if_reached() y g_return_val_if_reched() regresan de la funcin si alguna vez son ejecutadas. La primera macro no toma ninguna expresin mientras que la segunda espera como parmetro el valor que ha de regresar la funcin. Por ltimo G_BREAKPOINT inserta una instruccin de punto de rompimiento con el objeto de depurar el programa. Esta macro solo est disponible en la arquitectura x86.
2.2.5
Como hemos discutido previamente, Glib maneja la memoria de de los objetos que nosotros creamos, pero tambin nos ofrece la posibilidad de tomar el control de la memoria en nuestras manos. Esto es conveniente si trabajamos con vectores o matrices que cambian de tamao o estamos implementando un nuevo objeto. Gran parte de las funciones de Glib se basan en la implementacin disponibles en la librera estndar de C de UNIX. Una regin de memoria tiene un ciclo de vida simple, como el mostrado en la Figura . Comencemos con la macro que define un puntero nulo: NULL. Est definida en prcticamente cualquier implementacin de C. Esta macro es til para inicializar punteros a memoria o estructuras vacas, por ende, un objeto que no est inicializado contiene un puntero nulo. Kernighan y Ritchie4 establecieron tres funciones para manejar memoria de manera dinmica: malloc(), calloc() y free(). Estas pueden cubrir por completo el proceso mostrado en la figura . El primer paso del ciclo de vida de un bloque de memoria es la funcin estndar de C malloc(): void *malloc(size_t n);
14
malloc() toma como nico parmetro el nmero de bytes de memoria a reservar. Si tal peticin no pudo completarse regresar entonces el puntero NULL. Por otro lado se encuentra calloc(), cuyo prototipo es: void *calloc(size_t n, size_t size); calloc() reservar memoria para un arreglo de n estructuras de tamao size. Como malloc() y calloc() regresan punteros de tipo void, se hace necesario hacer un casting o moldeado al tipo deseado. Consideremos el siguiente ejemplo. int *ip; ip = (int *) calloc(n, sizeof(int)); Con el objetivo de no recibir quejas del compilador de C, debemos moldear correctamente el puntero a la memoria reservada que nos entrega calloc(). Cerrando el ciclo de vida de una regin de memoria creada dinmicamente, se encuentra free(), el cual libera la memoria asignada a un puntero en especial. Glib ofrece g_malloc()y g_free(); sendas funciones operan de igual manera que sus homlogas en la librera estndar de C, slo que trabajan con el tipo gpointer. Adems de las dos funciones anteriores, existe un abanico de posibilidades que ahorran gran cantidad de trabajo al crear una regin de memoria. Comencemos con dos macros. #define #define g_new() g_new(struct_type, n_structs) g_new0(struct_type, n_structs) y g_new0() reservan memoria para n_structs estructuras de tipo
struct_type. La segunda macro adems inicializar a cero la regin de memoria. El puntero a la memoria reservada ya estar moldeado a struct_type. Si cualquiera de las dos macros falla al reservar el nmero indicado de estructuras en memoria, abortarn el programa con un mensaje de error.
15
La versin segura de las macros anteriores se encuentran en g_try_new() y g_try_new0() quienes regresarn un puntero NULL moldeado a struct_type, en lugar de abortar el programa. El ciclo de memoria dinmica incluye cambiar el tamao de sta, para ello tendremos dos macros: #define #define g_renew(struct_type, mem, n_structs) g_try_renew(struct_type, mem, n_structs)
Ambas cambian el tamao de una regin de memoria a la que apunta mem. La nueva regin de memoria contendr n_structs de tipo struct_type. g_renew() aborta el programa en cualquier error mientras que g_try_renew() regresa un puntero NULL moldeado a struct_type, Ambas regresan un puntero moldeado a struct_type que apunta a la nueva memoria reservada. Existen otras macros como g_memove() o g_newa(), empero su discusin escapa ya el alcance de este manual
2.2.6
Las aplicaciones escritas en GTK+ usualmente necesitan pasar datos entre las diferentes partes del programa. Glib define seis macros bsicas de conversin de tipos, sin embargo, conforme avancemos veremos que habr macros de conversin de tipo para casi cualquier objeto o widget que usemos. Como veremos ms tarde, ser comn convertir un tipo de dato en otro. Generalmente referido como casting o moldeado en C, esta tcnica permite que GTK+ se comporte como una librera orientada a Objetos. La manera de pasar datos de una parte de la aplicacin a otra generalmente se hace utilizando gpointer,el cual es lo suficientemente general como para pasar cualquier estructura de datos, sin embargo existe una limitante al querer pasar nmeros en lugar de estructuras de datos. Si, por 16
ejemplo, deseramos pasar un nmero entero en lugar de una estructura de datos, deberamos de hacer algo como esto: gint *ip = g_new (int, 1); *ip = 42; Ahora tenemos un puntero a una constante de tipo gint. La desventaja de esto es que ahora nosotros tendremos que hacernos cargo de liberar la memoria del nmero entero. Los punteros siempre tienen un tamao de al menos 32 bits (en las plataformas que Glib est portada). en base a esto podramos tratar de asignar el valor que queremos pasar a un puntero: gpointer p; int i; p = (void*) (long) 42; i = (int) (long) p; Pero esto es incorrecto en ciertas plataformas y en tal caso habra que hacer lo que sigue: gpointer p; int i; p = (void*) (long) 42; i = (int) (long) p; Esto se vuelve demasiado complicado como para llevarlo a la prctica, por eso los desarrolladores de Glib han creado las macros GINT_TO_POINTER(), GUINT_TO_POINTER() y
GSIZE_TO_POINTER() para empacar un gint, guint o gsize en un puntero de 32 bits. Anlogamente GPOINTER_TO_GINT(), G_POINTER_TO_GUINT() y
GPOINTER_TO_GSIZE() sirven para obtener el nmero que se ha empacado en el puntero de 32 bits. El ejemplo anterior se cambia a: #include <glib.h> gpointer p;
17
gint i; p = GINT_TO_GPOINTER(42); i = GPOINTER_TO_GINT(p); No es buena idea tratar de empacar en un puntero otro tipo de dato que no sea gint o guint, la razn de esto es que estas macros solo preservan los 32 bits del entero, cualquier valor fuera de estos lmites ser truncado. As mismo es incorrecto guardar punteros en un entero, por las mismas razones expuestas arriba, el puntero ser truncado y conducir a gran cantidad de fallos en el programa.
18
memoria, fallas en el hardware y problemas de seguridad. El resultado de desplegar un mensaje de error es la terminacin definitiva del programa.
19
2.3.1
void
Descripcin: g_print() funciona exactamente igual que printf() de la librera estndar de C. Parmetros:
format: El formato del mensaje, consulte la documentacin de printf(). ...: Los parmetros que se insertarn en el mensaje.
A diferencia de printf(), que manda cualquier mensaje directamente a la salida estndar de C (stdout), g_print() lo hace a travs de un manejador. Este manejador, que usualmente es printf(), puede ser cambiado a conveniencia. Este manejador puede, en lugar de sacar mensajes a stdout, hacerlo a un archivo o a una terminal en un puerto serial. El explicar como registrar el manejador de g_print() allanar el camino para el siguiente captulo. Un manejador (handler, en el idioma anglosajn), es el puntero a una funcin escrita por el programador. El prototipo de la funcin que servir como manejador de g_print() es el siguiente:
void
mi_manejador
El puntero de esta funcin es simplemente su nombre. Este puntero se provee como parmetro de otra funcin que lo registra como manejador de g_print().
20
GPrintFunc
g_set_print_handler
(GPrintFunc func);
Descripcin: Establece el manejador de la funcin g_print(). Cuando se llame a g_print() la cadena resultante se pasar como parmetro a func. Parmetros:
2.3.1.1
Ejemplo
En el siguiente ejemplo mostraremos la facilidad de uso y versatilidad de g_print() usando un manejador simple.
Listado de Programa 2.3.1
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: glib-gprint.c * Descripcion: Uso del manejador de g_print() * Comentarios: Demuestra el funcionamiento de g_print() y * allana el camino para GTK+ introduciendo los * punteros a funciones. * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <glib.h> /*Para usar g_printf()*/ #include <glib/gprintf.h> /* mi_gprint Funcion manejadora de g_print */ void mi_manejador (const gchar *string){ g_fprintf(stdout,"mi_manejador:"); g_fprintf(stdout,string); } /* Programa principal */ int main (int argc, char **argv){
21
GPrintFunc viejo; g_print("Usando g_print() sin manejador\n"); g_print("Estableciendo el nuevo manejador de g_print() ..\n\n"); viejo = g_set_print_handler(&mi_manejador); g_print ("Impresion Normal\n"); g_print ("Impresion de numeros: %i, %f, 0x%x\n",1,1.01,0xa1); g_print("Restableciendo el antiguo manejador de g_print() ..\n\n"); viejo = g_set_print_handler(viejo); g_print("Fin\n"); return (0);
El programa listado imprime un par de mensajes usando el manejador por defecto de g_print(), lo cual no presenta demasiada dificultad. La parte ms importante viene a continuacin. Usando la variable viejo guardamos el puntero al manejador por defecto de g_print() e inmediatamente establecemos el nuevo manejador, el cual es nuestra propia funcin:
mi_manejador(). Inmediatamente se pone a prueba nuestro nuevo manejador imprimiendo algunos mensajes de texto y nmeros. Tomemos en cuenta que el manejador solo recibe una cadena y no tiene que estar lidiando con parmetros variables y quien se encarga de esto es Glib. Posteriormente se restablece el manejador original de g_print() y todo vuelve a la normalidad. La comprensin de este sencillo ejemplo es vital para todo el curso, pues no estamos trabajando con instrucciones comunes y corrientes en el lenguaje C, si no con punteros a funciones y estructuras complejas de datos. Este tipo de tpicos por lo general es evitado en los cursos universitarios del lenguaje C debido a que implica un conocimiento claro del lenguaje C. El siguiente ejemplo es un mtodo interactivo para seleccionar el comportamiento de g_print().
22
void muestra_ayuda( void ) { printf("\nError, no ha indicado ningun parametro, o es invalido.\n"); printf("uso:\n\t--normal g_print normal\n\t--manejador g_print con manejador\n"); } /* */ int main (int argc, char **argv) { GPrintFunc viejo; if (argc <= 1){ muestra_ayuda(); return 0; } if (g_str_equal(argv[1],"--normal")){ printf("--== Usando tratamiento normal de mensajes ==--\n"); } else if (g_str_equal(argv[1],"--manejador")) { printf("--== Usando tratamiento con manejador ==--\n"); viejo = g_set_print_handler(&mi_manejador); } Programa principal
23
else { muestra_ayuda(); return 0; } /*Imprime algunos mensajes*/ g_print ("Hola mundo!\n"); if (g_str_equal(argv[1],"--manejador")) g_set_print_handler(viejo); return 0; }
El manejador de g_print() es el mismo que en el listado de programa 2.3.1. Este ejemplo es un programa pensado para la lnea de comandos. Si se ejecuta este programa sin ningn parmetro se ejecutar la funcin muestra_ayuda(). Ocurre lo mismo si no se especifican los parmetros correctos. Solo se aceptan dos parmetros que permiten elegir entre usar o no el manejador de g_print().
2.3.2
Es muy buena prctica el clasificar nuestros mensajes debido a su severidad. Para esta tarea GTK+ nos ofrece tres herramientas:
g_message() es una macro que registra e imprime un mensaje en la salida estndar. Este mensaje se considera informativo e inocuo.
g_debug() es una macro que registra e imprime un mensaje en la salida de error estndar. Este mensaje es til para propsito de depurado de la aplicacin.
g_warning() se utiliza normalmente para avisar acerca de algn evento que ha ocurrido el cual no es lo suficientemente fatal como para que el programa no pueda continuar.
2.3.2.1
Ejemplo
24
g_message("Abriendo dispositivo de adquisicion de datos"); g_debug ("La direccion del dispositivo es 0x378"); g_warning ("No fue posible abrir el dispositivo de adquisicion de datos"); } return 0;
Figura 2.3.1: La salida generada por las diferentes macros de registro de mensajes
25
2.3.3
g_critical() avisa de algn error crtico en la aplicacin. Un error crtico se define dependiendo de cada aplicacin, para algunos un error critico es recuperable y para otros no. Este error se dirige a la salida de error estndar. g_error() avisa de un error grave en un programa. Slo se debe utilizar g_error() para avisar para comunicar errores que de todas formas haran que la aplicacin terminara. El uso de esta macro ocasionar que la aplicacin termine.
Listado de Programa 2.3.4
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: glib-error.c * Descripcion: Uso de macros de registro de mensajes de Glib * Comentarios: Estos mensajes son de indole grave o fatal. * * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <glib.h> int main (int { argc, char **argv)
g_critical("La frecuencia de muestreo es demasiado alta."); g_error("Se ocasiono un sobreflujo de datos. \nImposible continuar "); } return 0;
26
Perspectiva procedimental: Glib ofrecer una vasta coleccin de rutinas de manejo de cadenas similares a las encontradas en la librera string.h de la librera estndar de C. Algunas adiciones buscan facilitar las tareas del programador.
Perspectiva orientada a objetos: Glib pone a disposicin de nosotros GString, un objeto cuyo funcionamiento esta basado en las cadenas del estndar de C, pero tratando de mejorar los problemas que encontremos al manejar cadenas de la manera tradicional. 27
2.4.1
Perspectiva procedimental
Existe una gran variedad de funciones de tratamiento de cadenas en Glib. Resultara ineficaz el tratar todas en este documento. A continuacin haremos resea de un pequeo conjunto de funciones tiles en el tratamiento de cadenas. gchar* g_strdup (const gchar *str);
Valor de retorno: La cadena duplicada en otra regin de memoria. Si NULL se ha especificado como parmetro de entrada, el valor de retorno tambin ser NULL. El programador es responsable de liberar la memoria de la nueva cadena.
gchar*
g_strrstr
Descripcin: Busca una aguja(needle) dentro de un pajar (haystack). Las cadenas de entrada debe estar terminadas con el carcter nulo. Parmetros:
haystack: La cadena donde se busca (pajar). needle: El texto que se busca (aguja).
Valor de retorno: Se regresa un puntero a donde se encontr la primera ocurrencia de la aguja dentro del pajar. Si no se encontraron coincidencias entonces se regresa NULL. gchar* g_strstr_len (const gchar *haystack, gssize haystack_len, 28
const gchar *needle); Descripcin: Esta es una versin de la funcin g_strstr(). Esta versin limitar su bsqueda en el pajar a un nmero de caracteres igual a haystack_len. Parmetros:
haystack: La cadena donde se busca (pajar). haystack_len: Nmero mximo de caracteres que se examinarn del pajar. needle: El texto que se busca (aguja).
Valor de retorno: Se regresa un puntero a donde se encontr la primera ocurrencia de la aguja dentro del pajar. Si no se encontraron coincidencias entonces se regresa NULL. gboolean g_str_has_prefix (const gchar *str, const gchar *prefix); Descripcin: Nos dice si la cadena str tiene el prefijo especificado. Parmetros:
Valor de retorno: Regresa TRUE si la cadena comienza con prefix. FALSE en caso contrario. gboolean g_str_has_suffix (const gchar *str, const gchar *suffix); Descripcin: Nos dice si la cadena str tiene el sufijo especificado. Parmetros:
Valor de retorno: Regresa TRUE si la cadena termina con suffix. FALSE en caso contrario. 29
gboolean
g_str_equal
Descripcin: Esta funcin verifica que las dos cadenas sean iguales. Parmetros:
v1: Una cadena. v2: Otra cadena que se comparar contra v1..
Valor de retorno: Regresa TRUE si ambas cadenas son idnticas. Esta funcin esta preparada para ser usada en estructuras de datos que necesiten comparacin, como listas enlazadas, tablas de claves o arboles binarios5.
2.4.2
Gstring se comporta de igual manera como una cadena de texto de C, pero con la ventaja de que una instancia de Gstring crecer automticamente si el espacio es necesario: GString gestiona automticamente el espacio de memoria. Todas las operaciones son invisibles al usuario.
2.4.2.1
Propiedades
GString define tres miembros pblicos a los que se puede acceder directamente. typedef struct { gchar *str; gsize len; gsize allocated_len; } GString;
30
La propiedad str contendr el texto de la instancia, mientras que la propiedad len contendr la longitud de la cadena, sin contar los caracteres de terminacin de cadena.
2.4.2.2
El constructor de clase para GString es: GString* g_string_new (const gchar *init);
init: Valor inicial de la nueva instancia. Si deseamos instanciar una cadena vaca deberemos pasar la macro NULL como parmetro.
Valor de retorno: la referencia a una nueva instancia de GString. GString* g_string_new_len (const gchar *init, gssize len); Descripcin: Crea una nueva instancia de Gstring. La cadena contendr al inicio una cantidad de bytes determinada por len. Parmetros:
init: Valor inicial de la nueva instancia. Como la cadena se crear con un lmite de almacenamiento entonces no es necesario que la cadena init deba estar delimitada con un carcter nulo y, en cambio, puede contener caracteres nulos embebidos. Si deseamos instanciar una cadena vaca deberemos pasar la macro NULL como parmetro.
2.4.2.3
Mtodos de clase
Una vez instanciado el objeto GString podemos manipular su contenido. El programador no tendr nada que ver con reserva y liberacin de memoria. 31
GString*
string: Una instancia de GString. val: La cadena que que ser asignada a GString. Cualquier contenido previo se sobrescribir.
Valor de retorno: la referencia a la misma instancia de GString que se especific en el primer parmetro. GString* g_string_append (GString *string, const gchar *val);
string: Una instancia de GString. val: La cadena que ser aadida al final de GString.
Valor de retorno: la referencia a la misma instancia de GString que se especific en el primer parmetro. GString* g_string_append_c (GString *string, gchar c);
string: Una instancia de GString. c: El carcter que ser aadido al final de GString.
Valor de retorno: la referencia a la misma instancia de GString que se especific en el primer parmetro. GString* g_string_prepend (GString *string, const gchar *val);
32
Parmetros:
string: Una instancia de GString. val: La cadena que ser aadida al inicio de GString.
Valor de retorno: la referencia a la misma instancia de GString que se especific en el primer parmetro. GString* g_string_prepend_c (GString *string, gchar c);
string: Una instancia de GString. c: El carcter que ser aadido al inicio de GString.
Valor de retorno: la referencia a la misma instancia de GString que se especific en el primer parmetro. GString* g_string_ascii_up (GString *string);
Descripcin: Convierte todas las letras minsculas a maysculas de una instancia de Gstring. Este mtodo solo debe usarse cuando se manejan cadenas en formato ASCII. A diferencia de la funcin de la librera estndar de C: topupper(), este mtodo de GString solo reconoce caracteres ASCII e ignora las conversiones entre idiomas y localidades Parmetros:
string: Una instancia de GString. c: El carcter que ser aadido al inicio de GString.
Valor de retorno: la referencia a la misma instancia de GString que se especific en el primer parmetro. GString* g_string_ascii_down (GString *string);
33
Descripcin: Convierte todas las letras maysculas a minsculas de una instancia de Gstring. Este mtodo solo debe usarse cuando se manejan cadenas en formato ASCII. A diferencia de la funcin de la librera estndar de C: topupper(), este mtodo de GString solo reconoce caracteres ASCII e ignora las conversiones entre idiomas y localidades Parmetros:
string: Una instancia de GString. c: El carcter que ser aadido al inicio de GString.
Valor de retorno: la referencia a la misma instancia de GString que se especific en el primer parmetro.
2.4.2.4
Destructor de clase
Cuando ya no es de utilidad el objeto GString, este debe desaparecer par dar espacio a otras estructuras de datos. gchar* g_string_free (GString *string, gboolean free_segment); Descripcin: Libera toda la memoria ocupada por la estructura de GString. Entregar el contenido del objeto dependiendo de free_segment. Parmetros:
string: Una instancia de GString free_segment: Especifique TRUE si desea destruir por completo al objeto y su contenido.
Valor de retorno: Si se ha especificado FALSE en free_segment, el contenido de la instancia destruida se entrega por este medio. Si es FALSE se entrega NULL. 34
2.4.2.5
Ciclo de vida
Primero debemos asignar espacio para una estructura del tipo GString con un valor inicial. #include <glib.h> /*....*/ Gstring *cadena; cadena=gstring_new(Hola); Si deseamos inicializar un objeto GString vaco deberemos pasar la macro NULL como parmetro a gstring_new().
programador no tendr nada que ver con reserva y liberacin de memoria ya que Glib se encarga de eso. Podemos manipular el objeto... ... definiendo un nuevo valor para la cadena, g_string_assign(cadena, Nuevo valor); ...aadiendo caracteres al inicio y al final de la cadena almacenada en Gstring, ... g_string_append_c(cadena,'Z'); g_string_prepend_c(cadena,'A'); ...aadir cadenas completas, ... g_string_append (cadena, Aadiendo valor al final); g_string_prepend(candena,Aadiendo valor al Principio); ... truncar la cadena a alguna longitud, por ejemplo 0, que significa que la cadena se limpia... g_string_truncate(cadena,0); ... o convertir la cadena en maysculas o minsculas ... g_string_ascii_up(cadena); g_string_ascii_down(cadena); 35
Cuando termina el ciclo de vida de GString slo queda liberar la memoria. g_string_free(cadena, TRUE); Debemos tener cuidado con el segundo parmetro de g_string_free(). ste parmetro define si junto con el valor de la cadena tambin se destruye el objeto. Si ya no planea utilizar ms este objeto utilice TRUE, en cambio deber usar FALSE si el estructura se est usando en algn otro lado.
2.4.3
Ejemplo
36
g_print("( %i Bytes ) %s\n", cadena->len, cadena->str); /*Se inserta algun texto al fin de la cadena*/ g_string_append(cadena,", caret omni gaudio"); g_print("( %i Bytes ) %s\n", cadena->len, cadena->str); /*Se insertan caracteres al incio y al fin de la cadena*/ g_string_append_c(cadena,'!'); g_string_prepend_c(cadena,'.'); g_print("( %i Bytes ) %s\n", cadena->len, cadena->str); /*Se convierte la cadena a Mayusculas */ g_string_ascii_up(cadena); g_print("( %i Bytes ) %s\n", cadena->len, cadena->str); /*Se convierte la cadena a Mayusculas */ g_string_ascii_down(cadena); g_print("( %i Bytes ) %s\n", cadena->len, cadena->str); g_print("\nFin del programa\n"); g_string_free(cadena,TRUE); return 0;
37
2.5.1
Propiedades
La estructura GSList tiene un esquema similar al que se muestra en la Figura 2.5.1, mientras que su estructura en C es la siguiente: typedef struct { gpointer data; GSList *next; } GSList; El puntero *data almacena los datos que se desean coleccionar, mientras que next apunta hacia al siguiente elemento de la lista enlazada.
38
2.5.2
Constructor de clase
Una lista enlazada simple no tiene constructor de clase en si, pues un puntero con el valor NULL se interpreta como una lista vaca. El puntero *GSList siempre se debe inicializar con NULL. El fin de una lista enlazada se encuentra cuando el puntero next contiene el puntero NULL. De ah que una lista vaca slo es un puntero NULL.
2.5.3
La estructura de datos de GSList indica que nuestras listas enlazadas simples pueden contener cualquier dato. Adems de cualquier dato, tambin contienen un puntero a la siguiente estructura. Los datos contenidos en la estructura de datos pueden ser, por ejemplo, un entero usando cualquiera de las macros de conversin de tipo que se revisaron en el Capitulo 2.2.6, o un puntero a otro tipo de datos como un objeto o una cadena. Una lista enlazada simple slo permite recorrer la estructura de datos en una sola direccin (no hay ningn lugar donde diga como regresar a elemento anterior. Es importante no olvidar estos detalles por que todas los funciones asociadas asumen que el puntero que se les entrega es el inicio de la lista. As mismo, las funciones que modifican las listas enlazadas pueden cambiar la lista de tal manera que una referencia antigua ya no apunte al nuevo inicio de la lista. Con las consideraciones anteriores podemos comenzar con nuestra resea. El siguiente conjunto de funciones sirven para aadir y eliminar elementos.
GSList*
g_slist_append
Descripcin: Aade un elemento al final de la lista. Note que esta funcin tiene que recorrer toda la lista hasta el final para aadir el elemento. Una lista lo suficientemente larga puede crear problemas de velocidad de ejecucin y cuellos de botella, principalmente cuando se aaden varios elementos a la vez. Para estos casos se puede insertar todos los elementos al inicio para posteriorment invertir el orden de la lista. Parmetros:
list: Una lista enlazada simple. data: Los datos del elemento a insertar.
GSList*
g_slist_prepend
Descripcin: Aade un elemento al inicio de la lista. Note que el puntero al nuevo inicio de la lista pudo haber cambiado. Asegrese de guardar el nuevo valor. Parmetros:
list: Una lista enlazada simple. data: Los datos del elemento a insertar.
GSList*
g_slist_insert
40
Descripcin: Inserta un elemento al en la posicin especificada. Note que el puntero al nuevo inicio de la lista pudo haber cambiado. Asegrese de guardar el nuevo valor. Parmetros:
list: Una lista enlazada simple. data: Los datos del elemento a insertar. position: La posicin del elemento a insertar. El elemento se inserta al final si la posicin es negativa o es mayor al nmero de elementos de la lista.
GSList*
g_slist_insert_before
Descripcin: Inserta un elemento antes de algn otro elemento. Note que el puntero al nuevo inicio de la lista pudo haber cambiado. Asegrese de guardar el nuevo valor. Parmetros:
list: Una lista enlazada simple. sibling: El elemento del que deseamos que se inserte datos antes de l. data: Los datos del elemento a insertar.
GSList*
g_slist_insert_sorted
Descripcin: Inserta un elemento de manera ordenada. La ordenacin se lleva a cabo mediante la funcin de comparacin especificada. 41
Parmetros:
list: Una lista enlazada simple. data: Los datos del elemento a insertar. func: La funcin que ser usada para ordenar lo datos de la lista. Esta funcin deber tomar dos parmetros y deber regresar un valor mayor a cero si el primer parmetro debe ir despus del segundo parmetro.
Valor de retorno: El nuevo inicio de la lista enlazada simple. GSList* g_slist_remove (GSList *list, gconstpointer data); Descripcin: Remueve un elemento de la lista. Si dos elementos contienen los mismos datos, slo se remover el primero. Si no se encuentra el elemento a eliminar entonces la lista queda sin cambios. Parmetros:
list: Una lista enlazada simple. gconstpointer: Los datos del elemento a eliminar de la lista.
El siguiente conjunto de funciones son para localizar elementos dentro de la lista enlazada simple. GSList* g_slist_last (GSList *list);
42
Descripcin: Una macro que entrega el siguiente elemento de la lista. Equivale a slist>next. Parmetros:
Valor de retorno: El siguiente elemento de la lista enlazada simple. NULL si la lista esta vaca o se ha llegado al ltimo elemento. GSList* g_slist_nth (GSList *list, guint n); Descripcin: Entrega el n-simo elemento de la lista. Parmetros:
Valor de retorno: El n-simo elemento de la lista enlazada simple. NULL si la lista esta vaca o se ha llegado al ltimo elemento. GSList* g_slist_nth (GSList *list, guint n); Descripcin: Entrega el n-simo elemento de la lista. Parmetros:
list: Una lista enlazada simple. n: la posicin del elemento, iniciando desde 0.
Valor de retorno: El n-simo elemento de la lista enlazada simple. NULL si la lista esta vaca o la posicin buscada est fuera de los lmites de la lista. gpointer g_slist_nth_data (GSList *list, guint n); Descripcin: Entrega los datos del n-simo elemento de la lista. Parmetros: 43
list: Una lista enlazada simple. n: la posicin del elemento, iniciando desde 0.
Valor de retorno: Los datos del n-simo elemento de la lista enlazada simple. NULL si la lista esta vaca o la posicin buscada est fuera de los lmites de la lista. GSList* g_slist_find (GSList *list, gconstpointer data); Descripcin: Encuentra el elemento que contiene los datos especificados. Parmetros:
list: Una lista enlazada simple. data: los datos que se buscan
Valor de retorno: El elemento que contiene los datos. NULL si no se encuentra nada. GSList* g_slist_find_custom (GSList *list, gconstpointer data, GCompareFunc func); Descripcin: Encuentra un elemento aplicando el criterio de la funcin especificada. La lista se recorre y en cada paso se llama a la funcin especificada la cual debe regresar 0 cuando se halla encontrado el elemento deseado. Parmetros:
list: Una lista enlazada simple. data: los datos que se buscan. func: la funcin que se llama por cada elemento. Esta funcin debe de tomar dos punteros de tipo gconstpointer, los cuales son los datos del nodo que se esta iterando y los datos que se buscan, respectivamente
Valor de retorno: El elemento que contiene los datos. NULL si no se encuentra nada.
44
Las siguientes funciones servirn para encontrar el ndice de un elemento dentro de la lista gint g_slist_position (GSList *list, GSList *llink); Descripcin: Encuentra la posicin de un nodo dentro de una lista enlazada simple. Parmetros:
list: Una lista enlazada simple. llink: un elemento/nodo dentro de la lista enlazada simple.
gint
g_slist_index
Descripcin: Encuentra la posicin del elemento que contiene los datos especificados. Parmetros:
list: Una lista enlazada simple. data: los datos que se buscan
Valor de retorno: El ndice del elemento que contiene los datos -1 si no se encuentra nada.
Si deseamos recorrer, iterar o caminar a lo largo de la lista debemos usar la siguiente funcin.
void
g_slist_foreach
Descripcin: Recorre toda la lista enlazada simple ejecutando una funcin para cada nodo de la lista. Parmetros:
45
list: Una lista enlazada simple. func: La funcin que se llamar con cada elemento. Esta funcin debe tomar dos punteros de tipo gpointer. El primero corresponde a los datos del elemento iterado, el segundo a los datos extras proporcionados por el programador.
2.5.4
Destructor de clase
Cuando se termine el uso de la lista enlazada simple se debe de limpiar la memoria que este usando. El destructor de GSList libera la memoria de la estructura de la lista, mas no libera la memoria que esta a la que hace referencia cada elemento de la lista. Visto de otra forma. Una lista enlazada simple es una estructura que contiene espacio para dos punteros: uno apunta al siguiente elemento, el otro apunta a cualquier tipo o estructura de datos. Cuando se libera la memoria de la lista enlazada se libera el espacio que ocupan los dos punteros de cada elemento de la lista, pero los datos y estructuras a los que hacan referencia cada elemento de la lista quedan intactos. Ahora que se ha discutido los detalles del destructor, vemos al resea.
void
g_slist_free
(GSList *list);
Descripcin: Libera toda la memoria ocupada por la estructura de una lista enlazada. Parmetros:
46
2.5.5
Comencemos la descripcin del ciclo de vida de una lista enlazada simple. El primer paso es declarar la estructura e inicializarla con valor NULL. #include <glib.h> GSList *lista=NULL; /* ... */
Ahora podemos manipular la lista a nuestro antojo. Podemos, por ejemplo, aadir una sola cadena al final... lista = g_slist_append (lista,Elemento 1); ...al principio ... list = g_slist_prepend(lista,Elemento 0); ... o insertar elementos en posiciones arbitrarias ... list = g_slist_insert (lista, Elemento insertado,1); ... y no solamente funciona con cadenas, si no tambin con otros tipos de objetos... lista = g_slist_append (lista,G_INT_TO_POINTER(113)); lista = g_slist_append (lista,objeto); Cuando llega el momento de recavar la informacin guardada en la lista tendremos que recordar la estructura en C vista arriba. El mismo puntero GSList que representa la lista enlazada, es a su vez el puntero al primer nodo de la lista. El elemento data del nodo es un puntero a los datos guardados y el elemento next apunta al siguiente nodo de la lista o es NULL si ya no hay ms elementos. La manera correcta de acceder a los datos que contiene un nodo es mediante la notacin de punteros: datos= nodo->data; siguiente = nodo->next; Una manera til de recorrer una lista enlazada simple es mediante un ciclo utilizando for... 47
for (nodo=lista; nodo; nodo=nodo->next) g_print("%s\n",(char *)nodo->data); Otra manera de caminar a lo largo de la lista es utilizar g_slist_for_each() el cual se apoya de una funcin definida por el usuario que debe de corresponder con el siguiente prototipo: void GFunc (gpointer data, gpointer extra_data);
Una vez que se ha terminado de operar con la lista enlazada es necesario liberar la memoria usada, para ello se encuentra g_slist_free().
2.5.6
Ejemplo
Mostraremos dos ejemplos. El primero de ellos mostrar de manera breve el ciclo de vida de GSList.
Listado de Programa 2.5.1
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: glib-gslist1.c * Descripcion: Muestra de ciclo de vida de GSlist * Comentarios: Adems muestra como caminar a traves de la * lista. * * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <glib.h> void imprimir_lista(gpointer data, gpointer user_data){ gchar *mensaje; mensaje = (gchar *) data; g_print("%s\n", mensaje); } int main(){ GSList *lista = NULL; GSList *nodo = NULL;
48
gchar *nombre = "Nombre"; /*Insercin de diferentes tipos de elementos */ lista = g_slist_append(lista, nombre); lista = g_slist_prepend(lista, "Elemento adicionado al principio"); lista = g_slist_insert(lista, "Elemento insertado en posicion 1", 1); /* Primer metodo de acceso a elementos */ g_print("==-Primer metodo de acceso a los elementos de una lista-==\n"); for (nodo = lista; nodo; nodo = nodo->next) g_print("%s\n", (char *) nodo->data); /* segundo metodo */ g_print("==-Segundo metodo de acceso a los elementos de una lista-==\n"); g_slist_foreach(lista, (GFunc) imprimir_lista, NULL); /*Destructor*/ g_slist_free(lista); } return 0;
En el ejemplo anterior se ha mostrado que dos mtodos para recorrer toda la lista, elemento por elemento. El primero es un bucle de ejecucin que itera sobre cada elemento hasta que se halle el elemento final de la lista. El segundo mtodo deja que Glib haga la caminata por la lista y llame una funcin designada por nosotros por cada elemento que encuentre. Como se puede ver en la figura siguiente, los efectos de ambos mtodos son iguales.
49
El segundo ejemplo es una aplicacin prctica de las listas enlazadas simples. El objetivo de este ejemplo es realizar una lista de los dispositivos de captura de datos que existe en la computadora e imprimir una relacin de estos.
Listado de Programa 2.5.2
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: glib-gslist2.c * Descripcion: Aplicacin prctica de GSlist * Comentarios: El siguiente ejemplo buscar todos los * dispositivos de sonido del sistema y los guardar * en una lista enlazada para su posterior * procesamiento * * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <glib.h> #include <glib/gprintf.h> //void llenar_lista(GSList lista){ GSList *llenar_lista(GSList *lista){ gchar *comando = "/usr/bin/hal-find-by-property --key alsa.type --string capture"; gchar *mi_stdout; gchar **disps;
50
gint i=0; /*Ejecuta otro programa sin terminar este */ g_spawn_command_line_sync(comando, &mi_stdout, NULL,NULL, NULL); /*La salida del programa se guard en mi_stdout. Ahora procederemos a separar cada uno de los resultados que vienen separados por caracteres de nueva linea*/ disps = g_strsplit(mi_stdout,"\n",-1); /*Despues de separados, cada uno se inserta en la lista*/ for (i=0;i< (g_strv_length(disps) -1); i++) lista = g_slist_insert_sorted(lista,g_strdup(disps[i]),g_str_equal); /*Liberar la memoria usada por los resultados separados*/ g_free(mi_stdout); g_strfreev(disps); } return lista;
/*Esta funcin averiguar el dispositivo linux correspondiente a cada dispositivo de adquisicion de datos*/ void imprimir_lista(gpointer data, gpointer user_data){ GString *comando; gchar *mi_stdout; /*Preparar el comando a ejecutar */ comando = g_string_new(""); g_string_printf( comando, "/usr/bin/hal-get-property --udi %s --key linux.device_file", (gchar *) data); /*Ejecuta el comando programa sin terminar este */ g_spawn_command_line_sync(comando->str, &mi_stdout, NULL,NULL, NULL); /*Presentar los resultados*/ g_print("====\n"); g_print("HAL UDI : %s\n", (gchar *) data); g_print("DISP. LINUX : %s", mi_stdout); /*Limpiar memoria */ g_string_free(comando,TRUE); g_free(mi_stdout);
void limpiar_lista(gpointer data, gpointer user_data){ g_free(data); } int main(){ GSList *lista = NULL;
51
g_print ("Buscando dispositivos de captura...\n"); lista = llenar_lista(lista); g_print ("Se encontraron %i dispostivos\n",g_slist_length(lista)); g_print ("======= LISTA DE DISPOSITIVOS DE ADQUISICION DE DATOS =======\n"); g_slist_foreach(lista,imprimir_lista,NULL); /*Es hora de liberar toda la memoria*/ g_slist_foreach(lista,limpiar_lista,NULL); g_slist_free(lista); g_print ("=============================================================\n"); return 0; }
La tarea anteriormente expuesta parece difcil, pero los ltimos mejoras del sistema operativo Linux hacen que nuestra tarea no sea titnica. FreeDesktop es un grupo de expertos en computacin que se han reunido para establecer estndares de operacin entre las diferentes versiones (distribuciones) de Linux. Una de esas especificaciones es HAL (Hardware Abstraction Layer). Una serie de utileras en lnea de comandos permiten acceder a detalles del hardware de manera sencilla. La lgica detrs de este ejemplo es la siguiente: La funcin llenar_lista() usa HAL para listar a todos los dispositivos de sonido que sean de captura. Lo anterior implica la ejecucin del programa hal-find-by-property, lo cual queda a cargo de la funcin g_spawn_command_line_sync() que ejecuta la linea de comandos, descrita en una cadena, y entrega la salida del comando en otra cadena (mi_stdout). La salida del comando es una lista de los dispositivos de captura de audio disponibles en el sistema y estn separados por caracteres de nueva lnea. Es necesario entonces dividirlos en cadenas independientes. La funcin g_strsplit() parte la cadena mi_stdout en un arreglo de cadenas, las cuales contienen ya, el identificador de cada dispositivo separado de todos los dems. La funcin
52
Despus de haber separado nuestros identificadores en cadenas de texto individuales se procede a llenar la lista enlazada simple con estos valores. Una vez preparada la lista enlazada, se libera la memoria que ya no sirve y se regresa el puntero de la nueva lista, ya llena. Llega la hora de presentar resultados. El numero de dispositivos encontrados es ahora reportado mediante g_slist_lenght(). Ya hemos visto anteriormente como caminar a travs de todos los elementos de la lista; hacemos lo mismo mediante imprimir_lista() que adems de imprimir los identificadores de los dispositivos, utiliza g_spawn_command_line_sync() para investigar el dispositivo Linux correspondiente a cada dispositivo. Antes de poder liberar la memoria de la estructura de la lista enlazada simple, se debe recorrer y liberar la memoria de cada uno de los elementos de la lista en forma individual. Esto se hace fcilmente con la funcin limpiar_lista(). El producto de nuestro programa se muestra a continuacin.
53
54
Introduccin
A continuacin comenzaremos a estudiar la parte grfica, de interaccin con el usuario. Discutiremos el concepto genrico de widget y como utilizarlos para construir gran cantidad de objetos.
La librera GTK+ sigue un modelo de programacin orientada a objetos(POO). La jerarqua de objetos comienza en GObject de la librera Glib del que hereda GtkObject. Todos los widgets heredan de la clase de objetos GtkWidget, que a su vez hereda directamente de GtkObject. Un diagrama UML que muestra tales relaciones se muestra en la figura 3.1.1.
6 Gadget: Adminculo, artilugio, artefacto que tiene una utilidad o funcin especfica
56
La clase GtkWidget contiene las propiedades comunes a todos los widgets; cada widget particular le aade sus propias propiedades. En GTK+ los widgets presentan una relacin padre/hijo entre s, las aplicaciones suelen tener un widget "ventana" de nivel superior que no tiene padre, pero aparte de l, todos los widgets que se usen en una aplicacin debern tener un widget padre. Al widget padre se le denomina contenedor (Consulte la figura 3.1.2).
Siempre que sea posible, en GTK+ siempre se debe evitar el uso de variables globales. Esto es importante an ms para libreras, ya que las variables globales dentro de sendos mdulos se exportan junto a las funciones (mtodos si consideramos a una librera como un objeto). Todos estos smbolos se integran al espacio global de nombres de la aplicacin que llama a la librera(Global Namespace). Un variable global de una librera cuyo nombre haya sido asignado de manera descuidada puede causar conflictos con otra variable con el mismo nombre que se est usando en una parte del cdigo que utiliza la librera.
Los nombres de las funciones debern tener la forma modulo_submodulo_operacion(). Por ejemplo: gtk_window_new(), g_string_truncate() g_tree_destroy(). Esta simple nomenclatura evita el choque de smbolos o nombres entre diferentes mdulos de las libreras.
Los smbolos (funciones y variables), debern tener nombres descriptivos: en lugar de usar cntusr() deberemos usar contar_usuarios_activos(). Esto hace que el cdigo fuente sea fcil de usar y casi auto-documentado.
Los nombres de funciones deben estar en minsculas y debern usar guin bajo para separar palabras, por ejemplo: g_string_destroy(), gtk_window_new(),
abrir_puerto().
Las macros y enumeraciones deben escribirse en letras maysculas, utilizando guiones bajos para separar palabras, por ejemplo: TRUE o G_DIR_SEPARATOR .
Tipos de datos, objetos y nombres de estructuras son una mezcla entre maysculas y minsculas, por ejemplo: Gslist, GtkWidget.
58
El uso de guin bajo para separar palabras hace que el cdigo fuente luzca menos apretado y fcil de editar ya que se pueden utilizar de mejor manera los comandos de navegacin por palabras de los editores de texto de forma que naveguemos ms rpidamente.
A la hora de escribir una librera se hace comn compartir smbolos (nombres de variables o funciones), entre los diversos componentes de la librera pero no se desea que estos smbolos estn disponibles para los usuarios de la librera. En tal caso, se puede anteponer un guin bajo al nombre de la funcin o variable mientras ste sigue la nomenclatura modulo/submdulo descrita arriba. Por ejemplo: _modulo_objeto_algunmetodo().
"clase" debe sustituirse por el nombre del widget que se desea crear. "metodo" describe la accin que ejecutar la instancia de la clase. Por ejemplo, el constructor de clase GtkWindow tiene la siguiente nomenclatura:
gtk_window_new().
59
La funcin de creacin de un widget gtk_clase_new() siempre debe devolver un puntero (en lenguaje C) a una instancia de tipo GtkWidget y no un puntero a una instancia del tipo creado. Por ejemplo, la funcin gtk_window_new() devuelve un puntero a un objeto de GtkWidget y no una instancia de tipo GtkWindow. Es importante remarcar esta caracterstica primordial de GTK+, ya que si recordamos que el lenguaje C no es un lenguaje orientado a objetos, nosotros deberemos hacernos cargo del correcto moldeo de tipos de clase. La preferencia de los constructores de clase de regresar la referencia a un tipo de dato de la clase base (GtkWidget) en lugar de regresar como un puntero a la clase heredada (GtkWindow) se justifica gracias a que muchos mtodos de la clase base an aplican a la clase heredada. El mejor ejemplo lo encontramos a la hora de hacer visible la instancia del objeto de tipo GtkWindow, para ello se utiliza el mtodo gtk_widget_show(). Si en algn momento se necesitase un puntero del tipo de la clase heredada podemos hacer uso de las macros que define cada objeto de GTK+ y que nos ayudan a moldear nuestro puntero a la clase de conveniencia. Con nuestro ejemplo, si necesitsemos un puntero del tipo GtkWindow utilizando como base al puntero de tipo GtkWidget, recurriramos a la macro GTK_WINDOW. Ahora un mismo objeto se puede comportar de dos formas distintas. Esto es conocido en cualquier lenguaje que soporte programacin orientada a objetos como polimorfismo. Un ejemplo no compilable, pero ilustrativo, se redacta a continuacin.
Listado de Programa 3.3.1
/* Crear una ventana con GTK+ */ /*Primero debemos incluir la librera gtk*/ #include <gtk.h> main(){ GtkWidget *ventana; ... /*Crear la instancia de clase GtkWindow*/ ventana = gtk_window_new(...);
60
/*Cambiar el tamao de la ventana, por ejemplo, para ocupar toda la pantalla del monitor Note que se utiliza la macro GTK_WINDOW que hace que el objeto ventana se comporte como GtkWindow. */ gtk_window_set_full_screen(GTK_WINDOW(ventana)); /*A continuacin hacer visible el objeto grfico utilizando herencia y polimorfismo. Note que el objeto ventana ahora se comporta como GtkWidget. */ gtk_widget_show(ventana); /*Otorgar control completo a la librera GTK+*/ gtk_main(); .... }
Aplicacion GTK
gtk_window_new()
Se crea una instancia de GtkWindow y la referencia se regresa a la aplicacin.
Puntero a una nueva instancia de GtkWindow Configuracion de la ventana. Se entrega el control: gtk_main()
Es importante hacer notar que, en este ejemplo en especfico, el objeto ventana es de tipo GtkWindow, pero a la vez es del tipo GtkWidget. Como el tipo base del puntero ventana es GtkWidget, es necesario moldearlo al tipo GtkWindow para que pueda comportarse como ste tipo de objeto. Si no se hace esto, el compilador se quejar y la aplicacin terminar con una violacin de segmento. El interfaz grfico de una aplicacin se construye combinando diferentes widgets (ventanas, cuadros combinados, cuadros de texto, botones, ...) y se establecen diversas retrollamadas (callbacks),
61
eventos asncronos; de esta forma se obtiene la lgica requerida por el programa a medida que se producen ciertas seales que a su vez provocan las retrollamadas. Las seales se producen por diversos sucesos como oprimir el botn de un ratn que se encuentra sobre un widget botn, pasar el cursor por encima de un widget u oprimir una tecla.
En una aplicacin, como veremos ms tarde, continuamente se estn generando seales y eventos, sin embargo no todos son atendidos y slo conectamos retrollamadas para aquellos eventos o seales que son de nuestro inters. Cuando deseamos atender a la escucha de una seal o retrollamada, se asocia un widget y una funcin en C. As, tambin se puede asociar retrollamadas a ms de un widget ahorrando cdigo que deba escribirse.
3.4.1
En GTK+ seales y eventos se administran casi de la misma manera, la distincin entre estos dos grupos se debe a que las seales son provocadas por el sistema de objetos de Glib / GTK+ y los eventos son una corriente de mensajes que llegan desde el subsistema grfico. Desde una perspectiva del programador resulta sencillo pensar en los eventos como cualquier seal causada por la interaccin del usuario con el programa Dos de las seales bsicas en GTK+ son delete_event y destroy. El evento
delete_event generalmente se enva a una ventana cuando el usuario trata de cerrarla. Por su parte, la seal destroy se manda a un objeto cuando su mtodo de destruccin debe ser invocado. Una ventana de nivel superior siempre debe conectar una funcin retrollamada al evento delete_event. Si el usuario quiere cerrar la ventana, entonces la aplicacin deber terminar correctamente. Una retrollamada es una funcin en C como cualquier otra. Sin embargo, dependiendo de la seal o evento a escuchar es como se declarar el tipo dato de regreso y los parmetros. Una vez escrita adecuadamente, se registra esta rutina ante GTK+ usando la macro g_signal_connect(). #define g_signal_connect(instance, detailed_signal, c_handler, data)
63
Descripcin: Conecta una funcin retrollamada que atender una seal de un objeto en particular. Parmetros:
instance : Es la referencia al widget u objeto del que queremos escuchar seales y eventos. Este puntero debe estar moldeado al tipo GObject ya que GtkWidget est es un derivado de ste. Para esto deberemos usar la macro G_OBJECT(). detailed_signal : Es una cadena que especifica la seal o evento a escuchar. c_handler : El puntero de la funcin retrollamada. Este puntero debe estar moldeado mediante la macro G_CALLBACK() al tipo de puntero GCallback. El prototipo de cada funcin retrollamada se determina por el contexto en el que ser usada; visto de otra manera: el prototipo de cada funcin se determina por el tipo de seal a la que ser conectada. data : Este ltimo argumento permite adjuntar algn dato extra a la retrollamada, de tal manera que se evite el uso de variables globales y en su lugar se pasen estructuras o valores directamente a la funcin retrollamada cuando sta sea invocada.
La funcin retrollamada cambia dependiendo de la seal que se desea escuchar, pero hay una funcin retrollamada prototipo que se usa como base para todas las dems: void (*Gcallback) (void);
Lo anterior no significa que todas las funciones retrollamadas no deban tomar parmetros y regresar void. Una funcin retrollamada muy comn en GTK+ y puede tener el siguiente prototipo: void funcion_retrollamada ( Gtkwidget *widget, gpointer datos); El primer argumento es un puntero al widget que recibe el evento o genera la seal. El segundo argumento es un puntero a los datos extras que se mandaron cuando se conect la seal a la retro llamada. De nuevo hay que hacer notar que el perfil de retro llamada descrito arriba es slo una forma general. Hay algunas retrollamadas generadas por widgets especiales que requieren diferentes parmetros.
64
3.4.2
Eventos
En complemento al mecanismo de seales descrito arriba, hay un conjunto de eventos que reflejan el mecanismo de eventos del subsistema grfico del sistema operativo (En UNIX ser X-window). Las funciones retrollamada tambin se pueden conectar a estos. Son:
event button_press_event button_release_event scroll_event motion_notify_event delete_event destroy_event expose_event key_press_event key_release_event enter_notify_event leave_notify_event configure_event focus_in_event
focus_out_event map_event unmap_event property_notify_event selection_clear_event selection_request_event selection_notify_event proximity_in_event proximity_ouLevent visibility_notify_event client_event no_expose_event window_state_event
Para poder conectar una funcin retro llamada a alguno de estos eventos, se usar la funcin g_signal_connect() , tal y como se ha descrito arriba usando alguno de los nombres que se dan como el parmetro seal. La funcin retro llamada para eventos es un poco diferente a la que se usa para las seales:
gint funcion_retrollamada(
En C, GdkEvent es una unin, de la cual su tipo depender de cual de los eventos mostrados arriba se han producido y esta construido mediante diferentes mscaras de eventos.. Para poder decirnos que tipo de evento ha ocurrido, cada una de las posibles alternativas tiene un miembro type 65
que muestra que evento ocurri. Los otros elementos de la estructura dependern de que tipo de evento se gener. Las mscaras de los tipos posibles de eventos son: GDK_NOTHING GDK_DELETE GDK_DESTROY GDK_EXPOSE GDK_MOTION_NOTIFY GDK_BUTTON_PRESS GDK_2BUTTON_PRESS GDK_3BUTTON_PRESS GDK_BUTTON_RELEASE GDK_KEY_PRESS GDK_KEY_RELEASE GDK_ENTER_NOTIFY GDK_LEAVE_NOTIFY GDK_FOCUS_CHANGE GDK_CONFIGURE GDK_MAP GDK_UNMAP GDK_PROPERTY_NOTIFY GDK_SELECION_REQUEST GDK_SELECTION_NOTIFY GDK_PROXIMITY_IN GDK_PROXIMITY_OUT GDK_DRAG_ENTER GDK_DRAG_LEAVE GDK_DRAG_MOTION GDK_DRAG_STATUS GDK_DROP_START GDK_DROP_FINISHED 66
GDK_CLIENTE_EVENT GDK_VISIBILITY_NOTIFY GDK_NO_EXPOSE GDK_SCROLL GDK_WINDOW_STATE GDK_SETTING En resumen: para conectar una retro llamada a uno de esos eventos, usaremos algo como lo que se presenta: g_signal_connect ( G_OBJECT (button), "button_press_event", G_CALLBACK (button_press_callback), NULL); Si asumimos que button es un widget. Cuando el ratn est sobre el botn y el botn sea presionado, se llamar a la funcin button_press_callback(), la cual puede ser declarada como sigue: static gint button_press_callback( GtkWidget *widget, GdkEventButton *event, gpointer data ); Es preciso hacer notar que el segundo argumento lo podemos declarar como tipo GdkEventButton por que ya sabemos cul es el evento que ocurrir para que esta funcin sea invocada. El valor regresado por esta funcin indica si el evento se deber propagar ms all por el mecanismo de manejo de seales de GTK+. Regresar FALSE indica que el evento ya ha sido tratado correctamente y ya no se debe propagar.
3.4.3
El bucle de eventos de GTK+ es el responsable de que el sistema de seales funcione correctamente, ya que el primero no es ms que un bucle interno de GTK+, en el que se van, una y otra vez, comprobando 67
los estados de cada uno de los elementos de la aplicacin, e informando de dichos cambios a los elementos que se hayan registrado para ser informados. Este bucle de eventos GTK+ se traduce bsicamente en dos funciones, que son gtk_main() y gtk_main_quit(). gtk_main() entrega el control de cualquier programa al bucle de eventos de GTK+. Esto significa que, una vez que se haya realizado la llamada a gtk_main(), se cede todo el control de la aplicacin a GTK+. Aunque gtk_main() toma el control de la aplicacin, es posible ejecutar otras porciones de cdigo aprovechando el sistema se seales usando algn manejador (instalado ANTES de llamar a gtk_main()) Dentro de algn manejador o retrollamada se puede llamar a gtk_main_quit() que termina el bucle de eventos de GTK+. El pseudo-cdigo de una tpica aplicacin GTK+ seria: int main (int argc, char *argv[]) { gtk_init (&argc, &argv); /* creacin del interfaz principal */ /* conexin a las distintas seales */ gtk_main (); return 0; } Como puede comprobarse, el programa inicializa GTK+, crea el interfaz bsico, conecta funciones a las distintas seales en las que est interesado (llamadas a g_signal_connect()), para seguidamente entregar el control del programa a GTK+ mediante gtk_main(). Cuando en algn manejador de seal realicemos una llamada a gtk_main_quit(), gtk_main() retornar, tras lo cual la aplicacin termina.
68
3.4.4
Ejemplo
A continuacin se mostrar un sencillo ejemplo mostrando el proceso de creacin del widget ms sencillo (GtkWindow) y el uso de seales. Comencemos recordando el captulo 3.3. El primer widget que aprenderemos a usar es GtkWindow que es ventana comn y corriente.
Listado de Programa 3.4.1
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: bucle1.c * Descripcion: Crea una ventana. * Widgets usados: GtkWindow * Comentarios: * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <gtk/gtk.h> int main( int argc, char *argv[] ) { GtkWidget *window; /* Inicializar la libreria GTK */ gtk_init (&argc, &argv); /*Crea una nueva instancia de GtkWindow*/ window = gtk_window_new(GTK_WINDOW_TOPLEVEL); /*Configura la instancia de GtkWindow*/ gtk_window_set_title (GTK_WINDOW (window), "bucle1.c"); gtk_widget_set_size_request(window,200,100); /*Conectar seales. Cuando la seal "destroy" se emita, se llamar a la funcin gtk_main_quit() que termina el programa */ g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit), NULL); /*Muestra la ventana en la pantalla*/
69
El primer paso es inicializar la librera GTK+ con esta instruccin: gtk_init (&argc, &argv); De no incluirla, nuestros programas fallaran de manera inmediata. El siguiente paso es crear una instancia de una ventana y alojar la referencia al objeto en la variable window: window = gtk_window_new(GTK_WINDOW_TOPLEVEL); El constructor de clase de GtkWindow toma un parmetro, es el tipo de ventana que se desea crear. Las ventanas normales, como la ventana del navegador (Firefox Mozilla) o el administrador de archivos (Nautilus) son ventanas de nivel superior (GTK_WINDOW_TOPLEVEL). El siguiente paso en nuestra aplicacin es establecer el ttulo ... gtk_window_set_title (GTK_WINDOW (window), "bucle1.c"); ... y el tamao: gtk_widget_set_size_request(window,200,100); Observe que el mtodo utilizado para cambiar el tamao de la ventana es un mtodo de GtkWidget y no de GtkWindow. Observe tambin que al establecer el ttulo de la ventana se utiliz una especie de macro con el puntero window como parmetro. Por qu ocurre esto? El constructor de GtkWindow regresa la instancia de GtkWindow como un puntero de GtkWidget y no de GtkWindow. Esto es necesario para que se pueda utilizar el polimorfismo en el
70
lenguaje C. Usando punteros al objeto ms general como GtkWidget nos permite moldearlo a cualquier otro objeto derivado. El mtodo gtk_window_set_title() requiere que el primer parmetro sea un puntero de tipo GtkWindow; la macro GTK_WINDOW() moldea el puntero de tipo GtkWidget a puntero GtkWindow. El mtodo gtk_widget_set_size_request() requiere que el primer parmetro sea un puntero de tipo GtkWindow; en el caso citado anteriormente no es necesario moldear el puntero window pues ya es del tipo deseado. Que ocurrira si decido no usar las macros de moldaje de tipos? El compilador se quejara de punteros de tipos incompatibles. A continuacin viene la instruccin ms importante del programa: g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit), NULL); El prototipo de la macro g_signal_connect() es ya conocida desde el captulo 3.4.1. El objeto window conectar la seal "destroy" a la funcin gtk_main_quit(). La seal "destroy" se emite cuando la ventana es cerrada. Cuando el usuario cierre la ventana tambin ocasionar que el bucle de control de Gtk+ termine y con ello la aplicacin. Qu ocurrira si no conectramos esta seal? Al cerrar la ventana, esta desaparecera pero el programa seguira ejecutndose en memoria.
71
Por ltimo hacemos visible la ventana y entregamos el control de la aplicacin al bucle de GTK+.
72
Como podemos ver en la Figura 3.5.1 GtkWindow tambin puede contener otros widgets, pues desciende de la clase GtkContainer. Pero debido a su descendencia directo con la clase GtkBin slo puede contener un slo widget, eso significa que, a pesar de tener la capacidad de almacenar otros 73
widgets por ser descendiente de GtkContainer, la clase GtkWindow slo puede contener un slo widget debido a su parentesco inmediato con GtkBin. Al igual que GtkWidget, GtkContainer y GtkBin son clases abstractas. Eso quiere decir que no son instanciables y slo sirven de plantillas para otros widgets. La clase GtkBin es muy simple y slo contiene un mtodo que se utiliza de manera errtica. Usaremos, entonces, las siguientes lneas a comentar los mtodos ms importantes de la clase GtkContainer.
3.5.1
void
Descripcin: Inserta un widget dentro de un contenedor. No es posible aadir el mismo widget a mltiples contenedores. Parmetros:
container : Una instancia de un contenedor. Use la macro GTK_CONTAINER() para moldear un puntero de diferente tipo. widget: El widget que se quiere insertar en el contenedor. gtk_container_remove (GtkContainer *container, GtkWidget *widget);
void
container : Una instancia de un contenedor. Use la macro GTK_CONTAINER() para moldear un puntero de diferente tipo. widget: El widget que se quiere remover del contenedor.
Nota: Cada widget creado contiene un contador de referencias. Esto evita que se destruya el widget cuando todava esta en uso. Cuando el contador de referencias llega a cero el sistema de objetos de Glib/GTK+ asume que el
74
widget ya no es de utilidad y se ordena su destruccin. Cuando se remueve un widget de su contenedor se decrementa el contador de referencias, el cual usualmente llega a cero. El efecto es la destruccin del widget. Para evitar esto es necesario referenciar explcitamente el widet usando g_object_ref().
Por el contrario, si ya no desea usar el widget despus de removerlo de un contenedor, la documentacin de GTK+ recomienda usar el destructor de GtkWidget directamente: gtk_widget_destroy(). ste remover el widget del contenedor y adems resolver cualquier otra referencia que se tenga al primero.
void
gtk_container_set_border_width
container : Una instancia de un contenedor. Use la macro GTK_CONTAINER() para moldear un puntero de diferente tipo. border_width: El espacio libre que se desea dejar alrededor del contenedor. Los valores vlidos van de 0 a 65535. gtk_container_get_border_width (GtkContainer *container);
guint
Descripcin: Obtiene el valor actual del ancho de borde del contenedor Parmetros:
container : Una instancia de un contenedor. Use la macro GTK_CONTAINER() para moldear un puntero de diferente tipo.
Hasta ahora hemos visto (al menos en teora), que es posible insertar un widget dentro de otro, para ello usamos el mtodo gtk_container_add(). Pero, Qu pasa si se quiere usar mas de un widget dentro de una ventana?, Cmo se puede controlar la posicin de los widgets?
75
3.6 Cajas
3.6.1 Descripcin
Regresemos un poco a la realidad cotidiana: Si deseamos acomodar algn objeto como un anillo, conseguimos un recipiente adecuado que slo aloje nuestra alhaja. Por otra parte, si nosotros fabricramos telfonos y tuviramos que enviar varios de ellos a un cliente en otro pas, la accin ms comn sera acomodar y empacar todos ellos en una caja y enviarlos a nuestro comprador. En el mundo de GTK+ se hace la misma analoga. Una caja es un widget que organiza un grupo de objetos en un rea rectangular: Si deseamos colocar varios de ellos en una sola ventana usaremos una caja y esta se puede insertar, a su vez, en la ventana. La ventaja principal de usar cajas es el despreocuparnos del lugar donde deben dibujarse cada uno de nuestros objetos grficos, GTK+ toma esa responsabilidad por nosotros. Existen dos tipos de cajas: GtkHBox y GtkVBox. Ambos descienden de la clase abstracta GtkBox (Figura 2.5.2) y son invisibles.
Cuando se empaquetan widgets en una caja horizontal (GtkHBox) se acomodan horizontalmente de izquierda a derecha o viceversa y todos tienen la misma altura.
76
En una caja vertical (GtkVBox) se acomodan de arriba a abajo o viceversa y todos tienen el mismo ancho. Tambin se puede usar una combinacin de cajas dentro o al lado de otras cajas para crear el efecto deseado. GtkBox es una clase abstracta, y las clases derivadas (GtkHBox y GtkVBox) no contienen mtodos de clase. Los constructores de clase son solamente para las cajas verticales u horizontales mientras que los mtodos de clase son de GtkBox.
3.6.2
Constructor de clase.
gtk_hbox_new (gboolean homogeneous, gint spacing);
GtkWidget*
homogeneous : Especifique TRUE si desea que todos los widgets (hijos) que se inserten en la caja les sea asignado un espacio por igual. spacing : El nmero de pixeles que se insertarn entre los widgets hijos.
Valor de retorno: una nueva instancia de GtkHBox. GtkWidget* gtk_vbox_new (gboolean homogeneous, gint spacing); Descripcin: Crea una nueva instancia de una caja vertical. Parmetros:
homogeneous : Especifique TRUE si desea que todos los widgets (hijos) que se inserten en la caja les sea asignado un espacio por igual. spacing : El nmero de pixeles que se insertarn entre los widgets hijos.
77
3.6.3
void
Descripcin: Acomoda un widget en una caja. Los widgets hijos se irn acomodando de arriba a abajo en una caja vertical, mientras que sern acomodados de izquierda a derecha en una caja horizontal. Parmetros:
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado. widget : El widget que ser empacado. gtk_box_pack_end_defaults (GtkBox *box, GtkWidget *widget);
void
Descripcin: Acomoda un widget en una caja. Los widgets hijos se irn acomodando de abajo a arriba en una caja vertical, mientras que sern acomodados de derecha a izquierda en una caja horizontal Parmetros:
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado. widget : El widget que ser empacado.
Nota: Cuando se ha hablado de empacar widgets dentro de una caja siempre hablamos de acomodar en lugar de insertar. Acomodar implica que se van coleccionando los widgets uno tras otro en el orden en el que son empacados.
78
3.6.4
La siguiente coleccin de mtodos exhibe toda la flexibilidad del sistema de empaquetado de GTK+. Las dos principales funciones gtk_box_pack_start() y gtk_box_pack_end() son
complejas, es por eso que se les ha aislado de las dems para una discusin ms detallada. Cinco son los parmetros que gobiernan el comportamiento de cada widget hijo que se acomoda en una caja:
homogeneus y spacing que se determinan en el constructor de clase expand, fill y padding que se determinan cada vez que se empaca un widget en un contenedor.
El parmetro homogeneous controla la cantidad espacio individual asignado a cada uno de los widgets que se empacan en una caja. Si es TRUE entonces el espacio asignado ser igual para todos los widgets hijos. Si es FALSE entonces cada widget hijo podr tener un espacio asignado diferente. El parmetro spacing especifica el nmero de pixeles que se usarn para separar a los widgets hijos. El parmetro expand le permite al widget hijo usar espacio extra. El espacio extra de toda una tabla se divide equitativamente entre todos sus hijos. El parmetro fill permite al widget hijo ocupar todo el espacio que le corresponde, permitiendo llenar por completo el espacio asignado. El widget no tiene permitido ocupar todo el espacio si el parmetro expand es FALSE. Los widgets hijos siempre estn usando todo el espacio vertical cuando estn acomodados en una caja horizontal. Asimismo usarn todo el espacio horizontal si estn situados en una caja vertical. El parmetro padding permite establecer un espacio vaco entre el widget hijo y sus vecinos. Este espacio se aade al establecido por spacing.
79
void
gtk_box_pack_start
(GtkBox *box, GtkWidget *child, gboolean expand, gboolean fill, guint padding);
Descripcin: Acomoda un widget en una caja. Los widgets hijos se irn acomodando de arriba a abajo en una caja vertical, mientras que sern acomodados de izquierda a derecha en una caja horizontal. Parmetros:
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado. child : El widget que ser empacado. expand : Si es TRUE al widget hijo podr asignrsele espacio extra. fill : Si es TRUE el widget podr ocupar el espacio extra que se le asigne. padding : El permetro de espacio vaci del hijo, especificado en pixeles. gtk_box_pack_end (GtkBox *box, GtkWidget *child, gboolean expand, gboolean fill, guint padding);
void
Descripcin: Acomoda un widget en una caja. Los widgets hijos se irn acomodando de abajo a arriba en una caja vertical, mientras que sern acomodados de derecha a izquierda en una caja horizontal Parmetros:
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado. child : El widget que ser empacado. expand : Si es TRUE al widget hijo podr asignrsele espacio extra. fill : Si es TRUE el widget podr ocupar el espacio extra que se le asigne. padding : El permetro de espacio vaci del hijo, especificado en pixeles.
80
void
gtk_box_set_homogeneous
Descripcin: Establece la propiedad "homogeneous" que define cuando los widgets hijos deben de tener el mismo tamao. Parmetros:
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado. homogeneous : Especifique TRUE si desea que todos los widgets (hijos) que se inserten en la caja les sea asignado un espacio por igual. gtk_box_get_homogeneous (GtkBox *box);
gboolean
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado.
Valor de retorno: El valor de la propiedad "homogeneous". void gtk_box_set_spacing (GtkBox *box, gint spacing); Descripcin: Establece la propiedad "homogeneous" que define cuando los widgets hijos deben de tener el mismo tamao. Parmetros:
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado. homogeneous : Especifique TRUE si desea que todos los widgets (hijos) que se inserten en la caja les sea asignado un espacio por igual. gtk_box_get_spacing (GtkBox *box);
gint
81
Parmetros:
box : Una instancia de GtkBox. Use la macro GTK_BOX() para moldear las referencias de cajas verticales y horizontales al tipo adecuado.
Valor de retorno: El nmero de pixeles que hay entre los widgets hijos de la instancia de GtkBox.
3.7 Tablas
3.7.1 Descripcin
Una tabla es una rejilla en donde se colocan widgets. Los widgets pueden ocupar los espacios que se especifiquen (1 o ms celdas).
Como es comn en GTK+, un contenedor no tiene una representacin grfica pero afecta la posicin y tamao de los elementos que contiene Cada widget se inserta en un rectngulo invisible dentro de la cuadrcula de la tabla. Segn podemos ver en la Figura 3.7.2, un widget hijo puede ocupar el espacio de uno o ms celdas de la siguiente lnea o columna, o ambas. Las coordenadas de ese rectngulo definen de qu celda a qu celda ocupar un widget.
82
Figura 3.7.2: Espaciado, alineacin y distribucin de elementos de GtkTable. Los cuadros grises son widgets insertados en la tabla.
El sistema de espaciados contiene diferentes variables que controlar y por tanto puede ocasionar confusin a ms de uno. Para una mejor explicacin debemos hacer distincin entre las propiedades de la tabla y las propiedades de los widgets hijos. Parmetros de comportamiento de GtkTable.
Espaciado entre columnas. Define el espacio (en pixeles) que habr entre dos columnas consecutivas. Este valor se controla mediante la propiedad "column-spacing".
Espaciado entre filas. Define el espacio (en pixeles) que habr entre dos filas consecutivas. Este valor se controla mediante la propiedad "row-spacing".
Numero de columnas. Define el nmero de columnas que contendr la tabla. Un widget puede ocupar ms de dos columnas consecutivas. 83
Numero de filas. Define el nmero de filas que contendr la tabla. Un widget puede ocupar ms de dos columnas consecutivas.
Homogeneidad. Define si las todas las celdas de la tabla tienen el mismo ancho y alto.
Columna. La columna donde se encuentra un widget se numera de izquierda a derecha a partir del numero cero.
Fila. La fila donde se encuentra un widget se numera de arriba a abajo comenzando desde cero. Comportamiento vertical y horizontal. Definen el comportamiento de una celda dentro de una tabla. Estos comportamientos pueden ser:
Expandirse para ocupar todo el espacio extra que la tabla le pueda otorgar. Encogerse para ocupar el espacio mnimo necesario. Expandirse para ocupar el espacio exacto que la tabla le ha otorgado.
Relleno vertical y horizontal. Define el espacio en pixeles que habr entre celdas adyacentes. Coordenadas de la celda. Resulta comn describir el inicio y el fin de una celda utilizando solamente la coordenada superior izquierda de la celda y la coordenada superior izquierda de la celda transpuesta.
Coordenada superior izquierda. Estas coordenadas se forman tomando el numero de la columna que comienza a la izquierda y el numero de la fila que comienza por arriba.
Coordenada inferior derecha. Estas coordenadas se forman tomando el numero de la columna que comienza a la derecha y el numero de la fila que comienza por abajo.
3.7.2
Constructor de clase
GtkWidget*
gtk_table_new
Descripcin: Crea una nueva instancia de una tabla que acomodar widgets a manera de rejilla. Parmetros:
rows : El nmero de filas de la tabla. columns : El nmero de columnas de la tabla. homogeneous : Si este valor es TRUE, entonces las celdas de la tabla se ajustan al tamao del widget ms largo de la tabla. Si es FALSE, las celdas de la tabla se ajustan al tamao del widget ms alto de la fila y el ms ancho de la columna.
3.7.3
void
Mtodos de clase
gtk_table_resize (GtkTable *table, guint rows, guint columns);
Descripcin: Cambia el tamao de la tabla una vez que esta ha sido creada. Parmetros:
table : Una instancia de GtkTable. rows : El nmero de filas que tendr la nueva tabla. columns : El nmero de columnas que tendr la nueva tabla. gtk_table_attach_defaults (GtkTable *table, GtkWidget *widget, guint left_attach, guint right_attach, guint top_attach, guint bottom_attach);
void
Descripcin: Acomoda un widget en la celda de una caja. El widget se insertar en la celda definida por las coordenadas definidas por la esquina superior derecha y la esquina inferior izquierda. Para ocupar una o ms celdas contiguas especifique la coordenada superior izquierda de la primera
85
celda y la coordenada inferior de la ltima celda. Usando este mtodo de clase el relleno de la celda ser 0 pixeles y esta llenar todo el espacio disponible para la celda. Parmetros:
table : Una instancia de GtkTable. widget : El widget que ser acomodado en una celda o celdas adyacentes. left_attach : ordenada de la esquina superior izquierda. right_attach : ordenada de la esquina inferior derecha. top_attach : abscisa de la esquina superior izquierda. bottom_attach : abscisa de la esquina inferior derecha. gtk_table_set_row_spacings (GtkTable *table, guint spacing);
void
table : Una instancia de GtkTable. spacing : El nuevo espaciado en pixeles. gtk_table_set_col_spacings (GtkTable *table, guint spacing);
void
table : Una instancia de GtkTable. spacing : El nuevo espaciado en pixeles. gtk_table_set_row_spacing (GtkTable *table, guint row, guint spacing);
void
Descripcin: Establece el espaciado de una sola fila de la tabla con respecto a las filas adyacentes. Parmetros:
table : Una instancia de GtkTable. row : El numero de la fila, comenzando desde cero. 86
spacing : El nuevo espaciado en pixeles. gtk_table_set_col_spacing (GtkTable *table, guint col, guint spacing);
void
Descripcin: Establece el espaciado de una sola columna de la tabla con respecto a las columnas adyacentes. Parmetros:
table : Una instancia de GtkTable. col : El numero de la columna, comenzando desde cero. spacing : El nuevo espaciado en pixeles. gtk_table_set_homogeneous (GtkTable *table, gboolean homogeneous);
void
table : Una instancia de GtkTable. homogeneous : TRUE si se desea que todas las celdas de la tabla tengan el mismo tamao. Establecer a FALSE si se desea que cada celda se comporte de manera independiente. gtk_table_get_default_row_spacing (GtkTable *table);
guint
Descripcin: Devuelve el espacio que se asigna por defecto a cada fila que se aade. Parmetros:
Valor de retorno: El espaciado de la fila. guint gtk_table_get_default_col_spacing (GtkTable *table); Descripcin: Devuelve el espacio que se asigna por defecto a cada columna que se aade. Parmetros: 87
Valor de retorno: El espaciado de la columna. guint gtk_table_get_row_spacing (GtkTable *table, guint row); Descripcin: Devuelve el espacio que existe entre la fila y la fila subyacente. Parmetros:
table : Una instancia de GtkTable. row : el nmero de la fila comenzando desde cero.
guint
gtk_table_get_col_spacing
Descripcin: Devuelve el espacio que existe entre la columna y la columna adyacente. Parmetros:
table : Una instancia de GtkTable. column : el nmero de la columna comenzando desde cero.
88
3.8 Etiquetas
3.8.1 Descripcin
GtkLabel es til para desplegar cantidades moderadas de informacin en forma de texto el cual se puede alinear a la izquierda, derecha y de forma centrada. La opcin de lenguaje de marcado (similar a HTML) mejora la calidad y cantidad de informacin desplegada usando tipos de letra (itlica, negritas, subrayado) y colores.
Figura 3.8.2: GtkLabel, junto con otros widgets, desciende de GtkMisc y GtkWidget.
3.8.2
Constructor de clase
Solo existe un constructor de clase para GtkLabel. GtkWidget* gtk_label_new (const gchar *str); 89
Descripcin: Crea una nueva instancia de una etiqueta GtkLabel que despliega el texto str. Parmetros:
str : El texto que contendr la etiqueta. Si no se desea ningn texto adentro de la etiqueta se puede pasar NULL como parmetro para una etiqueta vaca.
3.8.3
Los mtodos de clase bsicos son los que se usaran con mas frecuencia y se reducen a escribir el texto de la etiqueta y obtenerlo. Si se desea borrar el texto de una etiqueta solo es necesario escribir en ella un texto vaco. void gtk_label_set_text (GtkLabel *label, const gchar *str); Descripcin: Establece el texto que mostrara la instancia de una etiqueta. Parmetros:
label : Una instancia de GtkLabel str : Un puntero a una cadena que contiene el texto que desplegara la etiqueta. Si especifica NULL entonces se desplegara una etiqueta vaca. (GtkLabel *label);
Descripcin: Obtiene el texto que esta almacenado actualmente en la instancia de la etiqueta. Parmetros:
Valor de retorno: un puntero a la cadena que esta almacenada en la etiqueta. La instancia de GtkLabel es duea de la cadena y por tanto la esta no debe ser modificada.
90
3.8.4
La siguiente coleccin de mtodos indican como realizar un control mas avanzado sobre la etiqueta y as mejorar la presentacin y sencillez de uso de nuestros programas. void gtk_label_set_justify (GtkLabel *label, GtkJustification jtype); Descripcin: Establece el valor de la propiedad "justify" de GtkLabel. Esta propiedad define la alineacin entre las diferentes lineas del texto con respecto unas de otras. Por defecto todas las etiquetas estn alineadas a la izquierda. Parmetros:
label : Una instancia de GtkLabel. jtype : El tipo de alineacin del las lineas de texto en relacin con las dems. Lo anterior implica que no hay efecto visible para las etiquetas que contienen solo una linea. Las diferentes alineaciones son: GTK_JUSTIFY_LEFT, GTK_JUSTIFY_RIGHT, GTK_JUSTIFY_CENTER, GTK_JUSTIFY_FILL Es importante hacer notar que esta funcin establece la alineacin del las lneas texto en relacin de unas con otras. Este mtodo NO establece la alineacin de todo el texto, esa tarea le corresponde a gtk_misc_set_aligment(). (GtkLabel *label);
PangoEllipsizeMode gtk_label_get_ellipsize
Descripcin: Describe la manera en que se esta dibujando una elipsis en la etiqueta label. Parmetros:
Valor de retorno: el modo en que se esta dibujando la elipsis. Este puede ser cualquiera de PANGO_ELLIPSIZE_NONE, PANGO_ELLIPSIZE_START, PANGO_ELLIPSIZE_MIDDLE y PANGO_ELLIPSIZE_END.
91
void
gtk_label_set_ellipsize
Descripcin: Establece el valor de la propiedad "ellipsize" de GtkLabel. Esta propiedad define el comportamiento de GtkLabel cuando no existe suficiente espacio para dibujar el texto de la etiqueta. Parmetros:
label : Una instancia de GtkLabel mode : Se debe establecer a cualquiera de los cuatro modos definidos en la enumeracin PangoEllipsizeMode, a saber: PANGO_ELLIPSIZE_NONE, PANGO_ELLIPSIZE_START, PANGO_ELLIPSIZE_MIDDLE y PANGO_ELLIPSIZE_END. Estos cuatro modos definen si se dibujara una elipsis ("...") cuando no haya suficiente espacio para dibujar todo el texto que contiene la etiqueta. Se omitirn los caracteres suficientes para insertar la elipsis. Si se especifica PANGO_ELLIPSIZE_NONE no se dibujara la elipsis. Si se especifica PANGO_ELLIPSIZE_START entonces se omitirn caracteres del principio de la cadena en favor de la elipsis. Si se especifica PANGO_ELLIPSIZE_MIDDLE los caracteres se omitirn desde la mitad de la cadena hacia los extremos. Si se especifica PANGO_ELLIPSIZE_END los ltimos caracteres se eliminaran en favor de la elipsis. (GtkLabel *label);
PangoEllipsizeMode gtk_label_get_ellipsize
Descripcin: Describe la manera en que se esta dibujando una elipsis en la etiqueta label. Parmetros:
Valor de retorno: el modo en que se esta dibujando la elipsis. Este puede ser cualquiera de PANGO_ELLIPSIZE_NONE, PANGO_ELLIPSIZE_START, PANGO_ELLIPSIZE_MIDDLE y PANGO_ELLIPSIZE_END. void gtk_label_set_markup (GtkLabel *label, const gchar *str);
92
Descripcin: Examina el texto pasado en la cadena str. El texto introducido se formatea de acuerdo al lenguaje de marcado de la librera Pango (similar a HTML). Con este mtodo tenemos la capacidad de desplegar texto con colores o en negritas. Parmetros:
label : Una instancia de GtkLabel str : Un puntero a una cadena que contiene el texto que desplegara la etiqueta y en el lenguaje de marcado de Pango. Si especifica NULL entonces se desplegara una etiqueta vaca. Si el texto no coincide con el lenguaje de marcado de Pango entonces recibir un mensaje de error en tiempo de ejecucin (y no en tiempo de compilacin) y la etiqueta o parte de ella no se mostrar. Vea la Tabla 5 para una breve descripcin de las etiquetas vlidas. Tabla 5: Etiquetas vlidas para el lenguaje de marcado de Pango Etiqueta Descripcin Texto en negritas Texto en un tamao mas grande en relacin con otro texto. Texto en itlica Texto rayado Texto a subndice Texto a superndice Texto en un tamao mas pequeo en relacin con otro texto. Texto monoespaciado Texto subrayado Texto en color azul. Texto con fondo negro.
<b> Texto </b> <big> Texto </big> <i> Texto </i> <s> Texto </s> <sub> Texto </sub> <sup> Texto </sub> <small> Texto </small> <tt> Texto </tt> <u> Texto </u> <span foreground='blue' > Texto </span> <span foreground='#0000FF' > Texto </span> <span background='black' > Texto </span> <span foreground='#000000' > Texto </span>
3.8.5
Ejemplos
El primer ejemplo sirve para demostrar el uso bsico de GtkLabel. Este se muestra en el siguiente listado. 93
94
return 0;
La aplicacin anterior crear una ventana con una etiqueta adentro. Vea la Figura 3.8.3. Inmediatamente despus de inicializar GTK+ (con gtk_init()), se crea una instancia de una etiqueta. Despus de eso se crea una ventana, se conecta el evento delete-event con gtk_main_quit() correctamente. A continuacin se ajustan las opciones cosmticas: (a)Establecer el titulo a label1.c y (b) definir el tamao de la ventana a 200 pixeles de ancho por 150 de alto usando gtk_widget_set_size_request(). Una parte importante que no hay que olvidar es que una aplicacin GTK+ se construye acomodando widgets adentro de otros widgets. De esa forma es como se logra relacionar el comportamiento entre diferentes partes de una interfaz grfica. Una ventana es un contenedor que solo puede alojar un solo widget y en este ejemplo el husped ser la etiqueta que ya hemos creado. La insercin queda a cargo de gtk_container_add(). Slo queda mostrar todos los widgets usando gtk_widget_show_all() y entregarle el control de la aplicacin a GTK+. de manera que cuando se presione el botn de cerrar la aplicacin termine
95
El ejemplo anterior muestra de la manera mas sencilla cmo instanciar una etiqueta e insertarla en un contenedor. El siguiente ejemplo es una muestra de las principales caractersticas avanzadas de GtkLabel.
Listado de Programa 3.8.2
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: label2.c * Descripcion: Alineacin del texto de etiquetas. * Widgets usados: GtkLabel, GtkBox(GtkVBox), GtkWindow y * GtkScrolledWindow * Comentarios: Basado en el ejemplo disponible en el tutorial * original de GTK. (http://www.gtk.org/tutorial/) * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <gtk/gtk.h> int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *vbox; GtkWidget *frame; GtkWidget *label; GtkWidget *scrollw; /* Inicializar la libreria GTK */ gtk_init (&argc, &argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "label2.c"); scrollw = gtk_scrolled_window_new(NULL,NULL); vbox = gtk_vbox_new(FALSE,10); gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(scrollw),vbox); gtk_container_add(GTK_CONTAINER(window),scrollw); gtk_widget_set_size_request(window,450,200); g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit), NULL); frame = gtk_frame_new ("Modo normal"); label = gtk_label_new ("INSTITUTO TECNOLOGICO DE PUEBLA"); gtk_container_add (GTK_CONTAINER (frame), label); gtk_box_pack_start_defaults (GTK_BOX (vbox), frame);
96
frame = gtk_frame_new ("Etiqueta en modo normal con varias lneas"); label = gtk_label_new ("O Freunde, nicht diese Tne!\n"\ "Sondern lat uns angenehmere\n" \ "anstimmen, und freudenvollere!"); gtk_container_add (GTK_CONTAINER (frame), label); gtk_box_pack_start_defaults (GTK_BOX (vbox), frame); frame = gtk_frame_new ("Justificada a la izquierda (GTK_JUSTIFY_LEFT)"); label = gtk_label_new ("Circa mea pectora\nmulta sunt suspiria\n"\ "de tua pulchritudine,\nque me ledunt misere."); gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_LEFT); gtk_container_add (GTK_CONTAINER (frame), label); gtk_box_pack_start_defaults (GTK_BOX (vbox), frame); frame = gtk_frame_new ("Justificada a la derecha (GTK_JUSTIFY_RIGHT)"); label = gtk_label_new ("Como quien viaja a lomos de una llegua sombra,\n"\ "por la ciudad camino, no pregunteis a dnde\n"\ "busco, acaso, un encuentro que me ilumne el dia.\n"\ "Y no encuentro ms que puertas que niegan lo que esconden,\n"); gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_RIGHT); gtk_container_add (GTK_CONTAINER (frame), label); gtk_box_pack_start_defaults (GTK_BOX (vbox), frame); frame = gtk_frame_new ("Texto distribuido en la etiqueta (GTK_JUSTIFY_FILL)"); label = gtk_label_new ("FAUSTO.- Quin soy yo, pues, si no me es dado llegar "\ "a esa corona de la humanidad a la que aspiran todos mis sentidos?\n"\ "MEFISTFELES. - T eres, en ltimo resultado, lo que debes ser: "\ "colca sobre tu cabeza una peluca de miles de bucles, calza tus"\ "pies con conturnos de una vara de alto, que no por ello dejars"\ "de ser lo que eres."); gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_FILL); gtk_label_set_line_wrap (GTK_LABEL (label), TRUE); gtk_container_add (GTK_CONTAINER (frame), label); gtk_box_pack_start_defaults (GTK_BOX (vbox), frame); gtk_widget_show_all (window); gtk_main (); } return 0;
97
Este ejemplo se vuelve un poco ms complicado pues ahora hacemos uso de 5 tipos de widgets: GtkWindow, GtkLabel, GtkVBox, GtkFrame y GtkScrolledWindow. Esto se ha hecho debido a que ahora debemos transmitir una mayor cantidad de informacin en una sola ventana(De paso aprenderemos a trabajar con nuevos objetos de los que conocemos muy poco). Se han creado cinco diferentes etiquetas y cada una contiene un texto diferente. A cada una de estas etiquetas se le ha aplicado un modo de alineacin diferente. Para evitar la confusin y mejorar la apariencia del programa se ha decorado cada una de las diferentes etiquetas con un cuadro que describe el tipo de modo que se quiere mostrar. La clase GtkFrame se comporta como un contenedor ms (esta clase se describir mas a fondo en el apartado dedicado a widgets para decoracin). Debido a que desplegaremos toda la informacin al mismo tiempo es necesario usar una caja vertical (GtkVBox) para acomodar todos los marcos y las etiquetas. Por ltimo se utiliz la clase GtkScrolledWindow para aadir barras de desplazamiento y as evitar que la ventana tenga un tamao grande y desgradable. En resmen: cinco etiquetas (GtkLabel) con diferente alineacin se insertan con sendos marcos(GtkFrame), los cuales se alojan en una caja vertical(GtkVBox). Esta caja se mete dentro
98
de una ventana que contiene barras de desplazamiento(GtkScrolledWindow) que a su vez se inserta en la ventana de nivel principal (GtkWindow). Hay otros dos ejemplos que hay que mostrar. El primero(Listado de Programa 3.8.3) muestra la forma de usar el lenguaje de marcado de Pango para definir diferentes estilos de texto (colores, fuentes, etc.).
Listado de Programa 3.8.3
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: label3.c * Descripcion: Marcado de atributos de texto * Widgets usados: GtkLabel, y GtkWindow * Comentarios: Este ejemplo muestra como utilizar un lenguaje de * marcado de texto similar a HTML para definir el * estilo de texto desplegado en cualquier etiqueta. * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <gtk/gtk.h> int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *label; /* Inicializar la libreria GTK */ gtk_init (&argc, &argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "label3.c"); gtk_widget_set_size_request(window,400,150); g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit), NULL); label = gtk_label_new (NULL); gtk_label_set_markup(GTK_LABEL(label),"<big><b>Lorelei</b></big>\n\ <i>Lorelei</i>,\n\ <s>A poet of tragedies</s>, (<u>scribe I lauds to Death</u>),\n\ Yet who the hell was I to dare?\n\ <sub><i>Lorelei</i></sub>\ <span foreground=\"blue\" background=\"white\"> \
99
Canst thou not see thou to me needful art?</span>\n\ <sup><i>Lorelei</i></sup>\ <span foreground='#00FF00' background='#000000' weight='ultrabold'>\ Canst thou not see the loss of loe painful is?</span>"); gtk_container_add(GTK_CONTAINER(window),label); gtk_widget_show_all (window); gtk_main (); return 0; }
El segundo ejemplo (Listado de Programa 3.8.4) muestra como funciona las elipsis.
100
int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *vbox; GtkWidget *label; gtk_init (&argc, &argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "label4.c"); vbox = gtk_vbox_new(FALSE,5); gtk_container_add(GTK_CONTAINER(window),vbox); g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit), NULL); label = gtk_label_new ("Texto sin elipsis"); gtk_box_pack_start_defaults (GTK_BOX (vbox), label); label = gtk_label_new ("Texto con elipsis: 123456789"); gtk_label_set_ellipsize(GTK_LABEL(label),PANGO_ELLIPSIZE_START); gtk_box_pack_start_defaults (GTK_BOX (vbox), label); label = gtk_label_new ("Texto con elipsis: 123456789"); gtk_label_set_ellipsize(GTK_LABEL(label),PANGO_ELLIPSIZE_MIDDLE); gtk_box_pack_start_defaults (GTK_BOX (vbox), label); label = gtk_label_new ("Texto con elipsis: 123456789"); gtk_label_set_ellipsize(GTK_LABEL(label),PANGO_ELLIPSIZE_END); gtk_box_pack_start_defaults (GTK_BOX (vbox), label); gtk_widget_show_all (window); gtk_main (); return 0;
Con esto hemos cubierto gran parte de la funcionalidad de las etiquetas. Ms informacin se puede hallar en el manual de referencia de GTK+.
101
3.9 Botones
3.9.1 Descripcin
GtkButton es un widget que emite una seal cuando es presionado. Un botn es a su vez un contenedor. Por lo general contiene una etiqueta, una imagen o ambas. GtkButton es punto de partida para la creacin de otros tipos de botones (Vea la Figura 3.9.1).
Ms adelante analizaremos el funcionamiento de GtkToggleButton y GtkOptionMenu. Los dos restantes no sern cubiertos en este manual. GtkColorButton es un botn que al ser presionado muestra una ventana de seleccin de color y GtkFontButton mostrar una ventana de seleccin de fuente al ser presionado.
3.9.2
Constructores de clase
Existen cuatro constructores de clase para GtkButton. Se puede usar el constructor gtk_button_new_with_label() gtk_button_new_with_mnemonic() para crear un botn con etiqueta(normal y con acelerador, respectivamente); gtk_button_new_with_stock()
102
crear un botn cuya imagen y texto estar determinado por un identificador stock_item, y por ltimo gtk_button_new() instancia un botn vaco.
GtkWidget*
gtk_button_new
(void);
Descripcin: Crea una nueva instancia de un botn GtkButton. Esta nueva instancia de botn no contiene nada. Si desea colocar algn widget dentro de la nueva instancia use gtk_container_add(). Valor de retorno: una nueva instancia de GtkButton.
GtkWidget*
gtk_button_new_with_label
Descripcin: Crea una nueva instancia de un botn GtkButton. El nuevo botn contendr una etiqueta con el texto especificado. Parmetros: 103
Valor de retorno: una nueva instancia de GtkButton. GtkWidget* gtk_button_new_with_mnemonic (const gchar *label);
Descripcin: Crea una nueva instancia de un botn GtkButton. El nuevo botn contendr una etiqueta con el texto especificado. Cualquier letra que venga precedida de un guin bajo ('_'), aparecer como texto subrayado. La primera letra que sea precedida con un guin bajo se convierte en el acelerador del botn, es decir, presionando la tecla Alt y la letra activan el botn(Causan que se emita la seal "clicked"). Parmetros:
label : El texto que contendr la etiqueta dentro del botn. Anteponga un guin bajo a un carcter para convertirlo en acelerador.
Valor de retorno: una nueva instancia de GtkButton. GtkWidget* gtk_button_new_from_stock (const gchar *label);
Descripcin: Crea una nueva instancia de un botn GtkButton. El nuevo botn contendr una imagen y una etiqueta predeterminados(stock item) . Es una forma sencilla de hacer botones vistosos con mensajes usuales como si, no, cancelar y abrir. Al usar elementos predeterminados (stock items) nos aseguramos que los botones sigan el tema y el idioma elegidos en el entorno GNOME. Parmetros:
label : El nombre del elemento predeterminado (stock item). Una lista de los elementos predeterminados se muestra en el ANEXO 4.6.1.3 : STOCK ITEMS.
3.9.3
void
Mtodos de clase
gtk_button_set_label (GtkWidget button, const gchar *label); 104
Descripcin: Establece el mensaje que mostrar la etiqueta de un botn. El nuevo botn contendr una etiqueta con el texto especificado. Si hay otro widget dentro del botn, entonces GTK+ lo eliminar y en su lugar insertar una etiqueta. Parmetros:
button : Una instancia de GtkButton. label : El texto que contendr la etiqueta dentro del botn. gtk_button_get_label (GtkButton *button);
const gchar*
Descripcin: Regresa el texto contenido en la etiqueta de un botn si el botn ha sido creado con gtk_button_new_with_label() o se ha establecido el texto de la etiqueta con el mtodo gtk_button_set_label(). Si lo anterior no se cumple el valor regresado es NULL. Parmetros:
Valor de retorno: el texto de la etiqueta del botn. La cadena regresada por este mtodo es propiedad de Gtk+, no la libere ni la manipule u obtendr un fallo de segmentacin.
void
gtk_button_set_use_stock
Descripcin: Si esta propiedad se establece a verdadero entonces el texto de la etiqueta del botn se usar para seleccionar un elemento predeterminado(stock item) para el botn. Use gtk_button_set_text() para establecer un elemento predeterminado. Parmetros:
button : Una instancia de GtkButton. use_stock : TRUE si el botn deber mostrar elementos predeterminados (stock item). gtk_button_get_use_stock (GtkButton *button);
gboolean
105
Descripcin: Determina si la instancia del botn muestra elementos predeterminados (stock item). Parmetros:
void
gtk_button_set_use_underline
Descripcin: Si esta propiedad se establece a verdadero entonces cualquier letra que venga precedida de un guin bajo ('_'), aparecer como texto subrayado. La primera letra que sea precedida con un guin bajo se convierte en el acelerador del botn. Use gtk_button_set_text() para establecer el texto subrayado y/o aceleradores. Parmetros:
button : Una instancia de GtkButton. use_stock : TRUE si el botn deber subrayar elementos y generar mnemnicos. gtk_button_get_use_underline (GtkButton *button);
gboolean
Descripcin: Determina si la instancia del botn subraya caracteres y genera mnemnicos y atajos de teclado. Parmetros:
3.9.4 3.9.4.1
void
Descripcin: Esta seal se emite cuando se ha activado el botn. Lo anterior implica dos eventos: el usuario presiona el botn y lo libera (button-press-event y button-release-event). Lo anterior es importante debido a la confusin que ocasiona la sutil diferencia entre seales y eventos (Consulte el captulo 3.4, Teora de seales y retrollamadas). Como condicin para emitir la seal "clicked", el usuario debe presionar el botn y al liberarlo el cursor del ratn debe permanecer en el botn. Parmetros:
button : La instancia de GtkButton que recibe la seal. user_data : Datos extras que se registran cuando se conecta la seal a esta retrollamada.
3.9.5
Ejemplos
El primer ejemplo tendr como objetivo mostrar el producto de los 4 constructores de clase de GtkButton.
Listado de Programa 3.9.1
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: button1.c * Descripcion: Ejemplo basico de uso de GtkButton * Widgets usados: GtkButton, GtkVBox y GtkWindow * Comentarios: Este ejemplo muestra el producto de los cuatro * diferentes constructores de clase de GtkButton. * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <gtk/gtk.h> int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *button; GtkWidget *box;
107
/* Inicializar la libreria GTK */ gtk_init (&argc, &argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "button1.c"); gtk_widget_set_size_request(window,200,200); g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit), NULL); /*Creamos una caja vertical con espaciado homogeneo y 5 pixels entre cada elemento*/ box = gtk_vbox_new(TRUE,5); gtk_container_add(GTK_CONTAINER(window),box); /*Un boton sin nada adentro*/ button = gtk_button_new(); gtk_box_pack_start_defaults(GTK_BOX(box),button); /*Un boton con una etiqueta*/ button = gtk_button_new_with_label("Electronica"); gtk_box_pack_start_defaults(GTK_BOX(box),button); /*Un boton con un mnemonico*/ button = gtk_button_new_with_mnemonic("_Encender motor"); gtk_box_pack_start_defaults(GTK_BOX(box),button); /*Un boton con elemento predeterminado*/ button = gtk_button_new_from_stock(GTK_STOCK_CONNECT); gtk_box_pack_start_defaults(GTK_BOX(box),button); gtk_widget_show_all (window); gtk_main (); } return 0;
El programa anterior crea una ventana y una caja vertical donde se insertan cuatro botones (cada uno instanciado con un constructor de clase diferente).
108
Figura 3.9.4: Cuatro botones creados con cuatro constructores diferente. Ponga atencin en el ltimo botn de ambas ventanas.
En la Figura 3.6.10 se muestra el resultado de nuestro programa. Recordemos que GTK+ es una librera que soporta varios idiomas. Cuando el entorno GNOME o GTK+ estn configurados para el idioma ingls (por ejemplo), los elementos predeterminados del ltimo botn se traducen automticamente, de ah la importancia de usar elementos predeterminados (stock items), cada vez que se tenga la oportunidad. El segundo ejemplo se vuelve un poco mas complicado pues se comienza a usar las retrollamadas. En este caso hacemos uso de la seal "clicked" para implementar una pequea mquina de estados que nos ayude a mostrar el efecto de los diferentes mtodos de clase de GtkButton.
Listado de Programa 3.9.2
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: button2.c * Descripcion: Metodos de clase GtkButton * Widgets usados: GtkButton, GtkVBox, GtkLabel, GtkWindow * Comentarios: Este ejemplo muestra el producto de los cuatro * diferentes constructores de clase de GtkButton. * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/
109
#include <gtk/gtk.h> void retrollamada (GtkWidget *button, gpointer datos_extra){ GtkLabel *label = GTK_LABEL(datos_extra); static gint contador =0; //Si, es una maquina de estados. switch (contador){ case 0: //Lo convertiremos en un boton con una etiqueta gtk_label_set_markup(label, "<b>Ahora es un botn con una etiqueta.\n" "Presione el botn para activar \nla opcion de subrayado</b>"); gtk_button_set_label(GTK_BUTTON(button),"_Siguiente"); break; case 1: //Ahora sera un boton con un mnemonico gtk_label_set_markup(label, "<b>Ahora es un botn con un mnemnico.\n" "Pruebe presionando las teclas <u>Alt</u> y <u>S</u> o\n" "presionando el botn para convertirlo\n" "en un botn con un elemento\npredeterminado</b>\n"); gtk_button_set_use_underline(GTK_BUTTON(button), TRUE); gtk_button_set_label(GTK_BUTTON(button),"_Siguiente"); break; case 2: gtk_label_set_markup(label, "<span color='blue'>Fin de la demostracion.</span>\n"); gtk_button_set_use_stock(GTK_BUTTON(button), TRUE); gtk_button_set_label (GTK_BUTTON(button),GTK_STOCK_CLOSE); break; default : gtk_main_quit(); } contador++; } int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *button; GtkWidget *box; GtkWidget *label; /* Inicializar la libreria GTK */ gtk_init (&argc, &argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "button2.c"); g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit),
110
NULL); /*Creamos una caja vertical sin espaciado homogeneo y 5 pixels entre cada elemento*/ box = gtk_vbox_new(FALSE,5); gtk_container_add(GTK_CONTAINER(window),box); /*En la caja insertamos una nueva etiqueta */ label = gtk_label_new("<b>Este es un botn vaco.\n" "Presione el botn para convertirlo en un botn con etiqueta</b>"); gtk_label_set_use_markup(GTK_LABEL(label),TRUE); gtk_label_set_line_wrap(GTK_LABEL(label),TRUE); gtk_widget_set_size_request(label,200,200); gtk_box_pack_start_defaults(GTK_BOX(box),label); /*Tambien se aade un boton sin nada adentro*/ button = gtk_button_new(); gtk_widget_set_size_request(button,200,40); gtk_box_pack_start_defaults(GTK_BOX(box),button); /*Ahora conectamos la seal "clicked" a la funcion retrollamada*/ g_signal_connect(G_OBJECT(button),"clicked", G_CALLBACK (retrollamada), label); /*Atencion: ahora hemos enviado la etiqueta label como un parametro extra (en lugar de NULL)*/ gtk_widget_show_all (window); gtk_main (); return 0; }
Comencemos por la estructura de la aplicacin: En una ventana se inserta una caja vertical, una etiqueta y un botn. Para mejorar la presentacin visual de la aplicacin (algo muy importante), los mensajes que se usen en la etiqueta usarn el lenguaje de marcado de Pango. Debido al comportamiento de la caja vertical (que intentar cambiar el tamao de los widgets), se ha usado gtk_widget_set_size_request() en el botn y la etiqueta para fijar el tamao de ambos. Como ha sido usual hasta ahora se conecta el evento "delete-event" de la ventana principal con la funcin gtk_main_quit(), esto ocasiona que cuando se presione el botn de cerrar en la ventana el programa termine.
111
En este ejemplo hemos hecho uso de la seal "clicked". Esta seal se conect a la funcin retrollamada(). Cuando el usuario presione el botn se llamar a esta funcin. Por primera vez hemos usado el ltimo parmetro de la funcin g_signal_connect(). Casi siempre hemos utilizado la macro NULL en este espacio, indicndole a glib que no deseamos enviar ningn parmetro extra a la funcin retrollamada(), como fue en el caso de gtk_main_quit() la cual no toma parmetros. Ahora haremos uso de ese espacio envindole el puntero de la etiqueta que usamos en la ventana a la funcin retrollamada(): Cada vez que esta funcin se ejecute tendremos disponible una referencia al botn y a la etiqueta sin la necesidad de usar variables globales, pues estas estn dentro de main() y no son visibles desde dentro de la funcin. Dentro de la funcin retrollamada() se ha implementado una pequea maquina de estados: cada vez que presionemos el botn este cambiar de aspecto usando los mtodos de clase que hemos discutido aqu. Al iniciar la aplicacin, esta tendr un aspecto parecido al de la Figura 3.9.5.
Figura 3.9.5: La primera forma del botn. Aun no entramos en la funcin retrollamada()
Cuando se presiona el botn se llama a la funcin retrollamada(). La mquina de estados reconoce que es la primera vez que se entra a la funcin (el contador es 0), as que cambia el
112
mensaje que despliega la etiqueta y usa el mtodo gtk_button_set_label() el cual, en este especfico caso, inserta una etiqueta en el botn con un mensaje(Figura 3.9.6)
Figura 3.9.6: Segunda fase de nuestro programa. La maquina de estado ha modificado la aplicacin.
En el siguiente estado de la mquina (cuando el contador es 1) se activar la propiedad "useunderline" mediante el mtodo gtk_button_set_use_underline().
Cuando el contador llega a 2, la maquina de estados insertar un elemento predeterminado, lo cual implica establecer la propiedad "use-stock" a TRUE utilizando
113
Por ltimo, presionando el botn se termina la aplicacin. En este ltimo ejemplo nosotros hemos aprendido a utilizar los diferentes mtodos de clase de GtkButton. Tambin hemos aprendido a usar la seal clicked e implementar acciones con ella. Por ltimo hemos aprendido una leccin importante: las interfases grficas diseadas con GTK+ no son estticas, si no dinmicas y pueden cambiar dependiendo de las necesidades de la aplicacin y del usuario.
GtkEntry es un widget de entrada de texto. Puede almacenar slo una cantidad limitada de informacin debido a que slo despliega una linea de texto. Si el texto que se introduce es ms largo del que se puede mostrar, entonces el contenido de la caja de texto se ir desplazando de tal manera que se pueda visualizar lo que se esta escribiendo.
114
La mayora de los atajos del teclado, comunes en cualquier aplicacin, se encuentran disponibles (ver la Tabla 6). Adems de lo anterior tambin soporta arrastrar y soltar(drag & drop). Con la integracin de Pango como componente base de GTK+, todos los objetos de texto de GTK+ tienen la posibilidad de desplegar textos en otros alfabetos diferentes y soportar mtodos de escritura diferentes al nuestro (por ejemplo, chino, hind o ruso). Tabla 6: Atajos de teclado disponible en cajas de texto Atajo Flechas de direccin () Ctrl + Flechas de direccin () Ctrl + Inicio ( Home ) Fin ( End ) Shift + Flechas de direccin () Shift + Ctrl + Flechas de direccin () Descripcin Navegar en el texto, carcter por carcter. Navegar en el texto, palabra por palabra. Ir al inicio o al fin del texto. Seleccin de texto, carcter por carcter. Seleccin de texto, palabra por palabra.
Shift + Ctrl + Inicio ( Home ) Fin ( End ) Selecciona el texto, desde el cursor hasta el inicio o fin. Ctrl + C Ctrl + Ins ( Insert ) Ctrl + P Shift + Ins ( Insert ) Ctrl + X Shift + Supr ( Delete ) Shift + F10 Copiar el texto seleccionado. Pegar el texto en la posicin del cursor. Cortar el texto seleccionado. Desplegar el men de contexto.
115
3.10.1
Constructor de clase
Slo existe un constructor de clase. En las primeras versiones de GTK+ existieron 2 constructores, sin embargo uno de ellos ha cado en desuso. GtkWidget* gtk_entry_new (void);
Descripcin: Crea una nueva instancia de una caja de texto GtkEntry. Valor de retorno: una nueva instancia de GtkEntry.
116
3.10.2
Mtodos de clase
Algunos de los mtodos de clase, anteriormente disponibles para GtkEntry, ahora han cado en desuso en favor de la interfaz GtkEditable. Esta interfaz provee funcionalidad muy similar para todos los widgets de texto (no solamente GtkEntry). En este captulo solamente discutiremos los mtodos de clase propios de GtkEntry.
void
gtk_entry_set_text
Descripcin: Establece el contenido de la caja de texto. Reemplaza cualquier contenido anterior. Parmetros:
entry : Una instancia de GtkEntry. text : Un puntero a una cadena que contiene el texto que desplegara la caja de texto. Si especifica NULL equivale a limpiar la caja de texto.
Valor de retorno: Un puntero a una cadena con el contenido de la caja de texto. La instancia de GtkEntry es duea de la cadena y por tanto la esta no debe ser modificada. const gchar* gtk_entry_get_text Descripcin: Devuelve el contenido de la caja de texto. Parmetros:
(GtkEntry *entry);
Valor de retorno: Un puntero a una cadena con el contenido de la caja de texto. La instancia de GtkEntry es duea de la cadena y por tanto la esta no debe ser modificada.
void
gtk_entry_set_visibility
117
Descripcin: Define si el contenido de la caja de texto es visible o no. Cuando la propiedad "visibility" est puesta a FALSE el carcter invisible se despliega en lugar del texto. Generalmente el carcter invisible es el asterisco (*) y puede ser cambiado mediante el mtodo gtk_entry_set_invisible_char()7. Parmetros:
entry : Una instancia de GtkEntry. visible : TRUE para desplegar los caracteres tal y como son escritos. gtk_entry_get_visibility (GtkEntry *entry);
gboolean
void
gtk_entry_set_max_length
Descripcin: Establece la longitud mxima del contenido de GtkEntry. Aqu no se puede especificar una longitud en bytes por que otros idiomas como el chino o el hind utilizan cdigos de caracteres de dos bytes. Parmetros:
entry : Una instancia de GtkEntry. max : La longitud mxima del contenido de GtkEntry. Especifique cero si desea usar el lmite mximo permitido que est en el rango de 0 a 65535. gtk_entry_get_max_length (GtkEntry *entry);
gint
118
Parmetros:
Valor de retorno: Regresa el nmero mximo de caracteres . Si es 0 entonces no hay un lmite ms all que el valor mximo de gint.
3.10.3
Seales.
GtkEntry tiene una lista de 10 seales diferentes. Una discusin detallada de las 10 seales diferentes seria una tarea larga. En lugar de hacer una lista detallada aprovecharemos una caracterstica til de GTK+: cada seal define el tipo de retrollamada que quiere usar. Afortunadamente la mayora de las seales no necesitan retrollamadas complejas y utilizan el mismo prototipo. Este es el caso de cinco seales de ms usuales de GtkEntry. En la Tabla 7 hacemos una relacin de esas 5 seales y su descripcin. Todas esta seales usan el mismo prototipo de funcin retrollamada el cual resulta ser muy parecido al de la seal "clicked" de GtkButton. El prototipo genrico es: void funcion_retrollamada ( GtkWidget *widget, gpointer datos); Aunque es comn encontrarlo en esta forma: void funcion_retrollamada ( GtkEntry *entry, gpointer datos); La diferencia entre ambas es que si usamos la primera podemos conectar esa retrollamada a casi cualquier seal de cualquier widget. La segunda tiene la ventaja de habernos hecho el moldeado del widget a GtkEntry. Su desventaja radica en que solo puede ser usada para las seales emitidas por GtkEntry.
119
Tabla 7: Seales de GtkEntry que usan el mismo prototipo de ret8rollamada. Seal "activate" "backspace" "copy-clipboard" Causas de la emisin de la seal. Se ha presionado la tecla enter. Todas las veces que se ha presionado la tecla de retroceso. Cuando el usuario ha copiado texto de la caja, ya sea usando un men contextual o cualquiera de los atajos del teclado. Cuando el usuario ha cortado texto de la caja, ya sea usando un men contextual o cualquiera de los atajos del teclado. Cuando el usuario ha pegado texto en la caja, ya sea usando un men contextual o cualquiera de los atajos del teclado.
"cut-clipboard"
"paste-clipboard"
3.10.4
Ejemplos
Ha llegado el tiempo de mostrar lo que podemos hacer con GtkEntry. El primer ejemplo mostrar como conectar una sola retrollamada a las 6 seales descritas en la Tabla 7 y a un botn.
Listado de Programa 3.10.1
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: entry1.c * Descripcion: Seales de GtkEntry * Widgets usados: GtkEntry, GtkButton, GtkVBox, GtkLabel, GtkWindow * Comentarios: Se mostrar como conectar una sola retrollamada * a las 6 seales descritas en la Tabla 7 y a un * botn. * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <gtk/gtk.h> void callback (GtkWidget *widget, gpointer datos_extra){ gint i = GPOINTER_TO_UINT(datos_extra);
120
if (i == 5) { g_print("El botn ha generado la seal \"clicked\"\n"); return; } g_print("La caja de texto ha generado la seal "); switch (i){ case 0: g_print("\"activate\"\n"); break; case 1: g_print("\"backspace\"\n"); break; case 2: g_print("\"copy-clipboard\"\n"); break; case 3: g_print("\"cut-clipboard\"\n"); break; case 4: g_print("\"paste-clipboard\"\n"); break; } } int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *widget; GtkWidget *box; gchar *signals[] = {"activate", "backspace", "copy-clipboard", "cut-clipboard", "paste-clipboard"}; gint i; /* Inicializar la libreria GTK */ gtk_init (&argc, &argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "entry1.c"); g_signal_connect (G_OBJECT (window), "destroy", G_CALLBACK (gtk_main_quit), NULL); /*Creamos una caja vertical sin espaciado homogeneo y 5 pixels entre cada elemento*/ box = gtk_vbox_new(FALSE,5); gtk_container_add(GTK_CONTAINER(window),box); /*En la caja insertamos una nueva etiqueta */ widget = gtk_label_new("Presione el botn y/o \nescriba en la caja de texto"); gtk_box_pack_start_defaults(GTK_BOX(box),widget); /*Ahora instanciamos una caja de texto y la insertamos en la ventana*/ /* Note que estamos reusando el puntero "widget"*/ widget = gtk_entry_new(); gtk_box_pack_start_defaults(GTK_BOX(box),widget); /*Ahora conectaremos las 5 diferentes seales de GtkEntry*/ for (i=0; i<5; i++) g_signal_connect(G_OBJECT(widget),signals[i], G_CALLBACK (callback),GINT_TO_POINTER(i)); /*Atencion: ahora hemos enviado un nmero entero como un parametro extra (en lugar de NULL)*/ /*Tambien se aade un boton con contenido predeterminado*/
121
widget = gtk_button_new_from_stock(GTK_STOCK_OK); gtk_box_pack_start_defaults(GTK_BOX(box),widget); /*Ahora conectamos la seal "clicked" a la funcion retrollamada*/ g_signal_connect(G_OBJECT(widget),"clicked", G_CALLBACK (callback),GINT_TO_POINTER(i)); gtk_widget_show_all (window); gtk_main (); } return 0;
La estructura de la aplicacin es una ventana con una etiqueta, una caja de texto y un botn (obviando la caja vertical donde se empacaron todos los widgets). Preste atencin al arreglo de cadenas signals. Cuando llega el momento de conectar todas las seales de GtkEntry, se hace mediante un ciclo (for ...) enviando como datos extras el ndice del ciclo. Usamos la macro de conversin de tipos GINT_TO_POINTER() para empacar el ndice en el puntero (Ver la seccin 2.2.6). Tambin se hace lo mismo cuando se conecta la seal de GtkButton.
Cuando ambos widgets generen alguna seal esta ser atendida en la funcin retrollamada callback(). El primer parmetro de esta funcin no nos sirve de mucho en esta ocasin. El segundo parmetro es de ms utilidad pues ah viene empacado un nmero que indica quien gener la seal y qu seal fue. Un par de lineas de condiciones nos permitirn imprimir esa informacin a la consola.
122
El siguiente ejemplo mostrar como usar GtkEntry para crear un dilogo de autenticacin de usuarios.
Listado de Programa 3.10.2
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: entry2.c * Descripcion: Autenticacin de usuarios * Widgets usados: GtkEntry, GtkTable, GtkLabel, GtkWindow * Comentarios: Se mostrar como usar GtkEntry para autenticar * un usuario. * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Autor: Noe Misael Nieto Arroyo [email protected] * ****************************************************************************/ #include <gtk/gtk.h> #include <glib.h> gboolean ignorar (GtkWidget *window, GdkEvent *event, gpointer *data){ /*Si regresamos TRUE, entonces GTK considera que este evento ya no debe propagarse, que es lo que deseamos*/ } return TRUE;
void auth_cb (GtkEntry *entry2, GtkEntry *entry1){ const gchar *usuario="hildebrando117"; const gchar *password="6002perp"; if (g_str_equal(gtk_entry_get_text(entry1), usuario) && g_str_equal(gtk_entry_get_text(entry2), password)) gtk_main_quit(); } int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *table; GtkWidget *widget; GtkWidget *entry1, *entry2; /* Inicializar la libreria GTK */ gtk_init (&argc, &argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "entry1.c");
123
g_signal_connect (G_OBJECT (window), "delete-event", G_CALLBACK (ignorar), NULL); /*Creamos una tabla de tres filas, dos columnas y elementos no homogeneos*/ table = gtk_table_new(3,2,FALSE); gtk_container_add(GTK_CONTAINER(window),table); /*En la tabla insertamos una nueva etiqueta */ widget = gtk_label_new(""); gtk_label_set_use_markup(GTK_LABEL(widget),TRUE); gtk_label_set_markup(GTK_LABEL(widget), "<span color=\"blue\">Bienvenido a la fortaleza secreta</span>\n" "<span color=\"red\"> IDENTIFQUESE !!!</span>\n"); /*Insertamos la etiqueta en la parte superior de la tabla y usar dos celdas*/ gtk_table_attach_defaults(GTK_TABLE(table),widget,0,2,0,1); /* Insertamos otra etiqueta ...*/ widget = gtk_label_new("Usuario : "); gtk_table_attach_defaults(GTK_TABLE(table),widget,0,1,1,2); /* ... y aun lado, una caja de texto*/ entry1 = gtk_entry_new(); gtk_table_attach_defaults(GTK_TABLE(table),entry1,1,2,1,2); /*De igual manera, pero en el siguiente rengln*/ widget = gtk_label_new("Contrasea: "); gtk_table_attach_defaults(GTK_TABLE(table),widget,0,1,2,3); /* Esta es la caja de texto para la contrasea*/ entry2 = gtk_entry_new(); gtk_entry_set_visibility(GTK_ENTRY(entry2),FALSE); /* Presionando <enter> en esta caja de texto activa la autenticacion*/ g_signal_connect(G_OBJECT(entry2),"activate",G_CALLBACK(auth_cb),entry1); gtk_table_attach_defaults(GTK_TABLE(table),entry2,1,2,2,3); gtk_widget_show_all (window); gtk_main (); return 0; }
Comencemos por la estructura de la aplicacin. Una ventana contiene ahora una tabla con tres filas y dos columnas. En la primera fila, ocupando ambas columnas se encuentra una etiqueta con un mensaje de bienvenida. La primera columna de las filas restantes contienen sendas etiquetas especificando el tipo de informacin que se requiere. La segunda columna de las filas 2 y 3 contienen dos cajas de texto diferentes. La primera caja de texto (segunda columna) alojar el nombre de usuario mientras que la ultima caja servir para escribir la contrasea. 124
Si el usuario intenta cerrar la ventana, el manejador instalado ignorar el comando del usuario. Si el usuario intenta introducir un nombre de usuario o contrasea incorrectos el programa no responder en absoluto. La nica forma de cerrar la ventana es introduciendo el nombre de usuario y contrasea correctos. El mtodo para activar la autenticacin es simplemente presionar la tecla Entrar ( Enter ) despus de escribir la contrasea en la caja de texto adecuado. Observe detenidamente la Figura 3.10.4 que describe un diagrama de la lgica secuencial del ejemplo anterior.
125
Introduccin
Glade es una aplicacin que nos permite disear la interfaz grfica de nuestro programa, en una forma visual. Las herramientas de desarrollo visual proveen un enorme ahorro de tiempo y trabajo al evitar tener que codificar la GUI nosotros mismos y el tiempo restante se puede aprovechar para mejorar la lgica del programa. Glade guarda el diseo de la interfaz grfica en un archivo XML. Glade construye la aplicacin a partir del modelo descrito en el archivo XML usando la librera libglade. Nosotros podemos hacer uso de Glade para dibujar la GUI. Una vez guardada la descripcin de la GUI podremos reconstruirla con muy pocas lneas de cdigo. A continuacin proporcionaremos una breve gua de uso de Glade para dibujar nuestras aplicaciones. Al momento de redactar este documento existan dos versiones operativas: glade2 y glade3. Glade3 es la versin de prueba que ha sido rediseada por completo. Se han removido caractersticas obsoletas, mejorado el uso y la presentacin del programa. Tambin esta en planes la integracin de Glade3 con el sistema de documentacin DevHelp, el entorno integrado de desarrollo Anjuta. Sin duda, en los prximos meses se prevee una mejora significativa en los entornos de desarrollo en Linux. Glade3 madura da a da y Glade2 va quedando relegada en el olvido, es por eso que las siguientes secciones se basarn en el Glade3 versin 3.0
127
Al iniciar la aplicacin, aparecern tres ventanas: La ventana de proyecto de Glade, la paleta de widgets y la ventana de propiedades. Vea la Figura 4.1.2.
128
La ventana principal esta compuesta por la barra de mens, la barra de herramientas y una seccin en blanco que contendr una representacin de la estructura de la aplicacin. Los ltimos dos botones permitirn controlar la manera en que se visualiza la representacin de la estructura de nuestra aplicacin. La paleta contiene colecciones de widgets agrupados en las siguientes categorias:
Toplevels - Contiene ventanas y cuadros de dialogo. Containers - Contenedores de todo tipo, como cajas y tablas. Control and Display Widgets de control y despliegue de datos como botones, etiquetas y entradas de texto.
GTK+ Obsolete Este grupo contiene widgets que han sido reemplazados por otros mas flexibles u otros que son de uso muy poco comn, como GtkCurve.
GNOME User Interface Widgets diseados para ser usados con el entorno GNOME.
129
GNOME UI Obsolete Widgets diseados para usarse con el entorno GNOME pero que han sido reemplazados con otros ms flexibles o ya son de uso poco comn. Algunos widgets ya han sido implementados directamente en GTK+.
GNOME Canvas Widgets de dibujo, especialmente diseados para aplicaciones con dibujos vectoriales o elementos grficos interactivos.
La ventana de propiedades contiene cinco pestaas, cada una mostrando informacin de configuracin para cada widget.
General - En esta pestaa se ajustan propiedades generales de un widget como su nombre o el ancho de borde.
Empaquetado - En esta pestaa se encuentran los ajustes para aquellos widgets que se encuentren insertos en en contenedor (Consulte el capitulo 3.5 para una referencia completa del sistema de empaquetado de GTK+).
Comunes - Esta pestaa muestra una coleccin de propiedades ajustables de cualquier widget. Estas operaciones tienen como contra parte algn mtodo de la clase GtkWidget.
Seales - Esta es una lista, agrupada por clase, de todas las seales que puede atrapar un widget. A la izquierda se muestra el nombre de la seal y a la derecha el nombre de la funcin retrollamada.
Tecnologa de asistencia - GTK+ toma en cuenta a las personas con capacidades diferentes. En esta pestaa permite aadir soporte de tecnologa de asistencia a una aplicacin de GTK+.
Una vez que hemos revisado de manera general Glade3 podemos comenzar un proyecto.
130
1. En el grupo Toplevels de la paleta, presione en el primer icono. Una ventana aparecer con un fondo gris: Glade3 acaba de crear una instancia de GtkWindow para usted. Tambin note que en la ventana principal de Glade3 ha aparecido la representacin de la instancia de GtkWindow que acaba de ser creada.
2. El siguiente paso es insertar una etiqueta. El icono se encuentra en el grupo "Control and Display" de la paleta. Haga click en el icono y visualice la ventana que se acaba de crear. Cuando el cursor del ratn circula por el rea gris de la ventana, el puntero del ratn cambia por una cruz. Haga click en cualquier rea gris de la ventana. El rea gris desaparecer y en su lugar aparecer una instancia de GtkLabel (ver Figura 4.2.3).
131
3. Observe que la ventana principal de Glade3 ha actualizado la representacin de la aplicacin que estamos diseando. Se ha vuelto un rbol de donde se desprenden los widgets hijos de cada ventana de nivel superior (ver Figura 4.2.4).
4. Haciendo click en cualquier elemento de la representacin provoca que ese widget obtenga el foco; cuando un widget obtiene el foco es posible configurarlo usando la ventana de propiedades. Haga click en el elemento que representa la instancia de GtkWindow. 5. Centre ahora su atencin en la ventana de propiedades (ver Figura 4.2.5); ah puede cambiar el nombre de la instancia de GtkWindow, que en este ejemplo se llamar ventana. 6. Cambie el ttulo de la ventana a "glade1.c". Esto equivale a llamar al mtodo gtk_window_set_title().
132
7. Seleccione la pestaa "Seales" y busque la seal "delete-event" dentro del grupo de seales pertenecientes a GtkWidget. 8. Haga doble click en la columna de la derecha y escriba "gtk_main_quit". Esto conectar la seal "delete-event" de la instancia de GtkWindow con la retrollamada
gtk_main_quit().
133
9. En la pestaa Comunes, de la ventana de propiedades establezca la peticin de anchura en 260 y la peticin de altura en 60. Esto equivale a usar el mtodo
gtk_widget_set_size_request().
10.Dentro de la misma pestaa asegrese que la propiedad visible este ajustada a Si. Esta propiedad instruye a libglade a que llame el mtodo gtk_widget_show() para el widget que se esta configurando. 11.Seleccione ahora la etiqueta y cambie el texto a "INSTITUTO PUEBLA". Ajuste la propiedad visible a Si. TECNOLGICO DE
El ejemplo est preparado ahora. Guarde el ejemplo con el nombre "glade1.glade" y prosiga con la siguiente seccin para aprender el uso de la librera libglade.
134
4.3.1
El uso de libglade en nuestras aplicaciones permite la separacin de la parte lgica de la aplicacin de la parte grfica. Lo anterior quiere decir que nosotros dibujaremos la aplicacin a nuestro gusto usando Glade3 y posteriormente escribiremos la lgica de comportamiento de la aplicacin en cualquier lenguaje como Python o C. Una vez guardada la descripcin de la GUI en un archivo XML el proceso para reconstruir la GUI se muestra a continuacin.
135
136
int main(int argc, char *argv[]) { /* Este es un puntero a un objeto de tipo GladeXML*/ GladeXML *xml; /*Inicializar las librerias*/ gtk_init(&argc, &argv); /* cargar la GUI desde el archivo XML */ xml = glade_xml_new("app.xml", NULL, NULL); /* Conectar las seales en la GUI */ glade_xml_signal_autoconnect(xml); /* Ceder el control a GTK+*/ gtk_main(); return 0; }
La notable simplicidad de libglade es donde radica su capacidad. Si compilramos el ejemplo anterior tendramos la posibilidad de construir aplicaciones muy sencillas como una ventana con un botn hasta aplicaciones complejas como un editor de texto. Lo nico que tendramos que hacer es intercambiar la descripcin XML de la GUI y liblgade har el trabajo por nosotros. Aunque libglade nos libera de las tareas tediosas de crear GUIs usando el API de GTK+, an as debemos conocer la manera de cmo interactuar con la interfaz grfica generada: tenemos que crear retrollamadas e interactuar con el usuario.
137
El objeto GladeXML representa una instancia de la GUI creada a partir de una descripcin en formato XML. Cuando se crea una instancia de la clase GladeXML, la descripcin se lee desde un archivo y se crea la GUI. Una vez instanciada la clase GaldeXML, esta provee una serie de tiles mtodos para acceder a los widgets de la GUI por medio de una referencia o nombre dentro de la descripcin XML. La clase GladeXML tambin provee mtodos para conectar cualquier retrollamada que haya sido asociada con alguna seal o evento dentro de la descripcin XML. Por ltimo, libglade provee mtodos que buscan nombres de manejadores de seal en la tabla de smbolos de la aplicacin y automticamente conectar tantas retrollamadas como pueda.
GladeXML*
glade_xml_new
Descripcin: Crea una nueva instancia del objeto GladeXML a partir de un archivo de descripcin en formato XML. Opcionalmente se puede comenzar a construir la interfaz a partir de un widget. Es til si se desea construir solamente una barra de men y no toda la aplicacin en la que esta contenida. La descripcin en XML se cachea para acelerar futuras operaciones. Parmetros:
fname : Nombre del archivo que contiene la descripcin XML de la GUI. root : El nodo desde donde se desea comenzar a construir. NULL si desea construir. domain : Dominio de transicin XML.
Valor de retorno: Una nueva instancia de la clase GladeXML que describe una interfaz grfica de usuario. Regresa NULL si la operacin ha fallado.
const char *name); Descripcin: Regresa el puntero del widget con el nombre especificado. Esta funcin permite el acceso a componentes individuales de una GUI despus de que ha sido construida. Parmetros:
Valor de retorno: El puntero del widget cuyo nombre coincida con el especificado. Regresa NULL si el widget no existe.
void
glade_xml_signal_connect
Descripcin: Dentro de la descripcin XML de una GUI, las funciones retrollamada se especifican usando el nombre de la funcin y no un puntero a ella. Esta funcin permite conectar una funcin a todas aquellas seales que hayan especificado esta funcin como funcin retrollamada. Parmetros:
self : Una instancia de GladeXML. handlername : El nombre de la funcin retrollamada. func : Un puntero a la funcin retrollamada. Use la macro G_CALLBACK() para moldear el puntero de la funcin al tipo adecuado. glade_xml_signal_connect_data (GladeXML *self, const char *handlername, GCallback func, gpointer user_data);
void
Descripcin: La diferencia entre este mtodo y glade_xml_signal_connect() es que esta permite pasar el parmetro extra que se acostumbra en g_signal_connect(). Parmetros: 139
self : Una instancia de GladeXML. handlername : El nombre de la funcin retrollamada. handlername : Un puntero a la funcin retrollamada. Use la macro G_CALLBACK() para moldear el puntero de la funcin al tipo adecuado. user_data : Datos extra que se pasarn a la funcin retrollamada.
void
glade_xml_signal_autoconnect
(GladeXML *self);
Descripcin: Este mtodo permite conectar automticamente todas las retrollamadas que hayan sido descritas en la descripcin XML de la GUI. Parmetros:
4.6 Ejemplos.
4.6.1 Ejemplo 1 Ciclo de vida de una aplicacin con libglade
En este primer ejemplo mostraremos el ciclo de vida de una aplicacin con libglade. Se construirn dos interfases diferentes y mostraremos que con slo cambiar el archivo XML podremos cambiar completamente la GUI sin cambiar una sola lnea de cdigo. La primera GUI se retomar del ejemplo que se construy en los captulos 4.3 y 4.3.1: Una ventana con una etiqueta adentro.
140
La segunda GUI ser una ventana con un botn adentro. Con respecto a esta ltima debemos de asegurarnos que:
La instancia de la ventana deber conectar la seal "delete-event" con el mtodo delete_event() de GTK+.
La instancia del botn deber conectar la seal clicked con el mtodo retrollamada() que proveer nuestra aplicacin (ver Figura 4.6.1).
Que tanto la ventana como el botn tengan activada la propiedad visible (en la pestaa Comunes de la ventana de propiedades).
El botn deber tener un ancho de 260 pixeles y una altura de 60 pixeles (en la pestaa Comunes de la ventana de propiedades).
La ventana deber un ancho de borde de 10 pixeles (en la pestaa Generales de la ventana de propiedades).
141
El cdigo fuente de la aplicacin estar basado en el mostrado en los captulos 4.3 y 4.3.1.
Listado de Programa 4.6.1
/*************************************************************************** * Programacion de interfases graficas de usuario con GTK * * Nombre de archivo: glade1.c * Descripcion: Ejemplo del ciclo de vida de una aplicacin * con libglade. * Widgets usados: GtkWidget * Objetos usados: GladeXML * Comentarios: Ejemplo basado del manual de referencia de * libglade * * TESIS PROFESIONAL INSTITUTO TECNOLOGICO DE PUEBLA * INGENIERIA ELECTRONICA * Fuente: Manual de referencia de libglade versin 2.5.0 * ****************************************************************************/ #include <gtk/gtk.h> /*Incluir la cabecera de libglade*/ #include <glade/glade.h> /*Incluir stdlib para usar la funcin exit()*/ #include <stdlib.h> void retrollamada(GtkWidget *widget, gpointer datos_extra) { g_print("Funcion retrollamada\n"); } int main(int argc, char *argv[]) { GladeXML *xml; //GtkWidget *ventana; /* Inicializar la libreria GTK */ gtk_init (&argc, &argv); if (!argv[1]){ g_print("Especifique la GUI que se debera construir\n"); exit(1); } g_print("Construyendo GUI del archivo %s\n",argv[1]); /* cargar la GUI desde el archivo XML */ xml = glade_xml_new(argv[1], NULL, NULL); //ventana = glade_xml_get_widget(xml,"ventana"); /* Conectar las seales en la GUI */ glade_xml_signal_autoconnect(xml); //gtk_widget_show_all(ventana); /* Ceder el control a GTK+*/ gtk_main(); return 0;
142
Este ejemplo, aunque es una GUI, debe de llamarse desde la lnea de comandos y requiere de un parmetro para funcionar: el nombre del archivo XML que contiene la descripcin de la GUI. En este caso puede ser glade1.xml o glade2.xml. En caso de que no se le suministre ningn nombre de archivo el programa imprimir un mensaje informativo y terminar inmediatamente.
La GUI que se muestra en la Figura 4.6.3 ha sido llamada mediante el siguiente comando: ./glade1 glade1.xml
La GUI que se muestra en la Figura 4.6.4 ha sido llamada mediante este otro comando. ./glade1 glade1.xml
Ambas ventanas se cierran automticamente pulsando el botn de cerrar. La segunda aplicacin imprime un mensaje en la consola cuando se presiona el botn.
143
Se ha demostrado que con el mismo cdigo se pueden construir dos interfases diferentes usando libglade. Se ha demostrado, tambin, como conectar seales automticamente usando libglade. Por ltimo se ha demostrado el ciclo de vida bsico de una aplicacin que usa libglade.
144
Bibliografa
[1] "X Window System". .Wikipedia, The Free Encyclopedia. Disponible al 1 Enero 2006 en la URL http://en.wikipedia.org/wiki/X_Window_System [2] "Windows GDI". .Microsoft Corporation. Disponible al 1 de Enero de 2006 en la URL http://msdn.microsoft.com/library/default.asp?url= [3] "Quartz Extreme, Faster graphics". . Apple Computer, Inc.. Disponible al 1 de Enero de 2006 en la URL http://www.apple.com/macosx/features/quartzextreme [4] "The Pango connection, ( 01 Mar 2001)". Tony Graham.IBM Corporation. Disponible al 1 de Enero de 2006 en la URL http://www-128.ibm.com/developerworks/library/l-u[5] Brian Kernighan, Dennis Ritchie, The C Programming Languaje (Second Edition), 1988 [6]Noe Nieto, Christian Alarcon, Sotero I. Fuentes, Micro Laboratorio Virtual, 2004 [7] "Linked List Basics". Nick Parlante.Stanford CS Education Library. Disponible al en la URL [8] "GNOME Programming Guidelines". Federico Mena Quintero, Miguel de Icaza. Morten Welinder.. Disponible al 2 de Febrero de 2006 en la URL http://developer.gnome.org/doc/guides/programming-guidelines/book1.html
145
ANEXOS
Sintaxis.
El compilador GNU C es una herramienta de lnea de comandos y puede aceptar gran cantidad de instrucciones que le indican la manera de comportarse. La lista de todas las opciones, modificadores y conmutadores es muy extensa, sin embargo todas obedecen simples reglas. La sinopsis de uso de gcc es: gcc
Las opciones generalmente van precedidas de un guin (estilo UNIX), o un doble guin (estilo GNU). Las opciones estilo UNIX pueden agruparse y constar de varias letras; Las opciones estilo GNU debern indicarse por separado , en forma de una palabra completa. Algunas opciones requerirn de algn parmetro como un nmero, un directorio,un archivo, una cadena o una frase. La orden gcc se puede utilizar indistintamente, no importando el lenguaje usado: Apegndose a unas sencillas reglas, el compilador ser capaz de determinar la accin a ejecutar dependiendo de la extensin de los archivos
Finalmente, todo lo que no se reconozca como parmetro u opcin, ser tratado como archivo y, dependiendo de su extensin, ste ser procesado como cdigo fuente o cdigo objeto.
Tabla AN1: Extensiones de archivo y su significado para GCC. Extensin de Archivo .c .C .m .cc .cpp .c++ .cp .cxx Descripcin Cdigo fuente en C. Cdigo fuente en C++. Se recomienda usar extensin .cpp Cdigo fuente en Objective C. Un programa hecho en Objective C debe ser ligado a la librera libobjc.a para que pueda funcionar correctamente. C preprocesado C++ preprocesado Cdigo fuente en lenguaje ensamblador Cdigo objeto. Archivo para preprocesador (encabezados), no suele figurar en la lnea de comandos de GCC. Cualquier otro parmetro que no sea archivo de los arriba expuesto o un parmetro vlido, ser tomado como si fuera un archivo objeto.
.i .ii .s .o .h OTRO
Ejemplos
Compila el programa hola.c y genera un archivo ejecutable que se llame a.out: gcc hola.c Compila el programa hola.c y genera un archivo ejecutable que se llame hola: gcc -o hola hola.c Compila el programa hola.cpp y genera un archivo ejecutable que se llame hola: gcc -o hola hola.cpp Los siguientes tres ejemplos compilan el programa hola.c y hola.cpp En este caso ninguno de los tres ejemplos generarn un ejecutable, si no solo el cdigo objeto (hola.o): gcc -c hola.c
gcc -c hola.c -o hola.o gcc -c hola.cpp Generar el ejecutable hola en el directorio bin, dentro del directorio propio del usuario: gcc -o ~/bin/hola hola.c Compilar el programa quetal.c, pero ahora indicarle a GCC dnde buscar las bibliotecas que usa quetal.c. Usaremos la opcin -L; podemos repetirla cuantas veces necesitemos para indicar todos los directorios de libreras que necesitemos. El orden de bsqueda de las libreras es el mismo orden en el que se especificaron en la linea de comandos. gcc -L /lib -L/usr/lib -L~/Librerias -L/opt/ProgramaX/lib quetal.cpp Ahora usaremos -I para indicarle a GCC dnde buscar archivos de cabecera (.h): gcc -I/usr/include/gtk-2.0 -I/opt/ProgramaX/include comoves.c Aunque pueda parecer que GCC es un gran programa que lo hace todo, en realidad es una coleccin de herramientas pequeas que hacen una cosa a la vez. GCC tiene un compilador para cada lenguaje, as si GCC detecta que se desea procesar un programa escrito en C, llamar al compilador de C (gcc), pero si es un programa escrito en C++, llamar al compilador de C++ (g++), y as de manera consecutiva. El proceso de compilacin comprende fases bien definidas. Cuando GCC es invocado, generalmente hace el 1) pre-procesamiento, 2)compilacin, 3)ensamblaje y ligado y por ltimo entrega como resultado final un archivo ejecutable. Una mirada de opciones nos permiten tomar control de cada paso del proceso, por ejemplo, el interruptor -c omite proceso de ligado y entrega solamente el cdigo objeto del programa. Algunas opciones se pasan directamente a alguna de las fases de construccin, algunas otras controlan el pre-procesamiento, otras controlan al ligador o al ensamblador y otras controlan al mismo compilador.
La orden gcc acepta opciones y nombres de archivos como operando. Muchas opciones son operandos de varias letras, as, no es lo mismo especificar la opcin -ab que las opciones -a -b. Salvo los anteriores casos, la mayora de las veces no importa el orden en que se dan los argumentos. Como ya hemos dicho el proceso de compilacin tiene cuatro fases: pre-procesamiento, compilacin, ensamblaje y ligado. Todas ellas siempre en el mismo orden. Las primeras tres de ellas trabajan con un solo archivo de cdigo fuente y terminan produciendo un archivo objeto. El proceso de ligado consiste en combinar a todos los archivos objeto generados, (incluyendo a los que se le han pasado por la linea de comando), resuelve las referencias entre ellos y entrega un archivo ejecutable (cdigo de maquina).
Las opciones ms comunes del compilador GCC son: -c Compila, ensambla, pero no liga. Como resultado obtenemos un archivo objeto por cada archivo de cdigo fuente. Generalmente se les asigna una extensin .o -S Compila pero no ensambla. Se entrega un archivo en ensamblador por cada archivo de cdigo fuente. A los resultados de la salida se les asigna la extensin: .S '-E' Slo realiza la etapa de preproceso. La salida estar en formato del cdigo fuente, procesado con respectivo compilador. -o archivo
Se puede especificar el nombre del archivo de salida que generar el compilador. Esto aplica a cualquier forma de salida que se le est instruyendo al compilador, ya sea, slo ensamblar, compilar, ligar o todos. -Iruta Especifica la ruta hacia el directorio donde se encuentran los archivos marcados para incluir el programa fuente. No debe llevar espacio entre la I y la ruta, as: -I/usr/include. -L Especifica una ruta hacia el directorio donde se encuentran los archivos de biblioteca con el cdigo objeto de las funciones que se usan en el programa. No lleva espacio entre la L y la ruta, as: -L/usr/lib -Wall Muestra todos los mensajes del compilador(advertencias y errores). -g Incluir en el programa generado, la informacin necesaria para poder rastrear posibles errores en un programa usando un depurador, tal como GDB (GNU Debugger). -v Muestra los comandos ejecutados en cada etapa de compilacin , as como la versin del compilador. Es un informe muy detallado.
Etapas de compilacin.
El proceso de compilacin involucra cuatro etapas sucesivas: Pre-procesamiento, compilacin, ensamblaje y enlazado. El proceso de conversin de creacin de un programa a partir del cdigo fuente exige la ejecucin de estas cuatro etapas en forma sucesiva. Los comandos gcc y g++ son capaces de realizar todo el proceso de una sola vez.
Preprocesamiento.
En esta etapa se interpretan las directivas del preprocesador. Entre otras cosas las constantes y macros definidas con #define son sustituidas por su valor en todos los lugares donde aparece su nombre. Usemos como ejemplo este sencillo programa en C. /* Circulo.c: calcula el rea de un crculo. Ejemplo que muestra las etapas de compilacin de GCC */ #include <stdio.h> # define PI main() { float area, radio; radio = 10; area = PI * (radio * radio); printf("Circulo.\n"); printf("%s%f\n\n", "Area de circulo radio 10: ", area); return(0); } El preprocesado puede pedirse llamando directamente al preprocesador (con la orden cpp), o hacindolo mediante GCC (con la orden gcc). Los siguientes dos comandos producen una archivo de salida idntico. $ cpp circulo.c > circulo.i $ gcc -E circulo.c > circulo.i Si examinamos circulo.pp (observe la extensin y compare con la tabla ), podremos observar que la constante PI ha sido substituida por su valor en todos los lugar donde se hacia referencia a ella. 3.1415926535897932384626433832795029L /* pi */
Compilacin.
El proceso de compilacin transforma el cdigo fuente preprocesado en lenguaje ensamblador, propio para el procesador en el que ser usado el programa (tpicamente nuestra propia maquina). Por ejemplo.. $ gcc -S circulo.c
... realiza las primeras dos etapas y crea el archivo circulo.s, si lo examinamos encontraremos cdigo en lenguaje ensamblador.
Ensamblado
El ensamblaje de nuestra aplicacin es el penltimo paso, transforma el archivo circulo.s o cualquier otro cdigo en ensamblador en lenguaje binario ejecutable por la mquina. El ensamblador de GCC es as, he aqu un ejemplo: $ as -o circulo.o circulo.s as crear el archivo en cdigo de mquina o cdigo objeto (circulo.o) a partir de un cdigo en ensamblador (circulo.s). Es muy infrecuente utilizar ensamblado, preprocesado o compilacin por separado, lo usual es realizar todas las etapas anteriores hasta obtener el cdigo objeto: $ gcc -c circulo.c El anterior comando producir el cdigo objeto y lo guardar en el archivo (circulo.o). A diferencia de las etapas anteriores, en programas muy extensos, donde el programa final se debe partir en diferentes mdulos, la prctica comn es usar gcc o g++ con la opcin -c para compilar cada archivo de cdigo fuente por separado y luego unirlos o enlazarlos para formar el programa final.
Enlazado.
Las funciones de C/C++ incluidas en cualquier programa(printf, por ejemplo), se encuentran ya compiladas y ensambladas en las bibliotecas existentes en el sistema. Es necesario incorporar de algn modo el cdigo binario de estas funciones a nuestro programa ejecutable. En esto consiste la etapa de
enlace, donde se renen uno o ms cdigos objeto con el cdigo existente en las bibliotecas del sistema. El enlazador de GCC es la orden ld. A continuacin un ejemplo: $ ld -o circulo circulo.o -lc ld: warning: cannot find entry symbol _start; defaulting to 08048184 El error anterior se debe a la falta de referencias, pues el enlazador no sabe a dnde debe buscar las funciones que el mdulo circulo.c esta usando. Para que esto funcione y obtengamos un ejecutable debera ejecutarse una orden como la que sigue: $ ld -o circulo /usr/lib/gcc-lib/i386-linux/2.95.2/collect2 /lib/ld-linux.so.2 -o /usr/lib/crti.o -m
elf_i386
-dynamic-linker
circulo
/usr/lib/gcc-lib/i386-
-L/usr/lib/gcc-lib/i386-linux/2.95.2
circulo.o -lgcc -lc -lgcc /usr/lib/gcc-lib/i386-linux/2.95.2/crtend.o Esto es incmodo, es por eso que GCC puede ahorrarnos mucho trabajo si le pasamos el nombre del cdigo objeto (o los nombres) que queremos convertir en ejecutable: $ gcc -o circulo circulo.o Crear el programa ejecutable de una manera sencilla y en un slo paso. En un programa con un slo archivo fuente, todo el proceso puede hacerse de una vez por todas: $ gcc -o circulo circulo.c A manera de aprendizaje podramos activar el interruptor -v de GCC que nos mostrar aspectos del proceso de compilacin que normalmente quedan ocultos. Recibiremos un informe detallado de todos los pasos de compilacin.
Dinmico: El cdigo de las funciones permanece en las bibliotecas del sistema, nuestra aplicacin cargar en memoria la librera necesaria y obtendr de ella las funciones que requiere para trabajar.
Confrontemos ambos alcances: Enlazado Dinmico Ventajas y desventajas El enlazado dinmico permite crear un El enlazado esttico crea un programa Enlazado Esttico
archivo ejecutable ms chico, pero requiere que el autnomo pero el precio a pagar es un mayor acceso a las libreras del sistema siempre este tamao. disponible al momento de correr el programa. Compilacin gcc -static -o circulo_s gcc -o circulo_d circulo.c circulo.c Diferencia de tamao8 7.0kB
tzicatl@ubuntu:~/Programar$ ldd circulo_d linux-gate.so.1 => libc.so.6 (0xb7e52000) /lib/ld-linux.so.2 (0xb7f8e000) tzicatl@ubuntu:~/Programar$ => (0xffffe000) /lib/tls/i686/cmov/libc.so.6
Como podemos ver, la versin esttica del programa no muestra dependencia alguna con las libreras del sistema.
Resumen
Si desea producir un ejecutable a partir de un solo archivo de cdigo fuente: $ gcc -o circulo circulo.c Para crear un mdulo objeto, con el mismo nombre del archivo de cdigo fuente y extensin .o: $ gcc -c circulo.c
8 El tamao de ambos ejecutables vara dependiendo del Sistema Operativo, el compilador, las libreras.
Para enlazar los mdulos verde.o, azul.o y rojo.o en un ejecutable llamador colores: $ gcc -o colores verde.o azul.o rojo
La herramienta make
Segn se indica en el manual de make, el propsito de esta utilidad es determinar automticamente qu piezas de un programa necesitan ser recompiladas y, de acuerdo a un conjunto de reglas, lleva a cabo las tareas necesarias para alcanzar el objetivo definido el cual normalmente es un programa ejecutable. make agiliza el proceso de construccin de proyectos con cientos de archivos de cdigo fuente separados en diferentes directorios. De esta forma y con las configuraciones adecuadas, make compila y enlaza todos los programas. Si alguno de los archivos de cdigo fuente sufre alguna modificacin slo ser reconstruido aquel mdulo de cuyos componentes haya cambiado. Por supuesto es necesario indicarle a make que mdulos u objetivos dependen de qu archivos, este listado se concentra en el archivo Makefile.
Comentarios.
Al igual que en cualquier lenguaje de programacin, los comentarios en los archivos Makefile contribuyen a un mejor entendimiento de las reglas definidas en el archivo. Los comentarios se inician con el carcter # y se ignora todo lo que viene despus de este carcter hasta el final de lnea. Ejemplo: # Este es un comentario.
Variables.
Las variables en un Makefile no estn tipeadas (es decir, no es necesario declarar previamente el tipo de valor irn a almacenar), en cambio todas son tratadas como cadenas de texto. Las variables que no estn declaradas simplemente se tratan como si no existieran (por ejemplo son cero, o son una cadena vaca). La asignacin de valores a una variable se hace de una manera sencilla: nombre = dato De esta forma se simplifica el uso de los archivos Makefile. Para obtener el valor de una variable deberemos encerrar el nombre de la variable entre parntesis y anteponer el carcter $. En el caso anterior, todas las instancias de $(nombre) sern reemplazadas por dato. Por ejemplo, la siguiente regla: SRC = main.c Origina la siguiente lnea: gcc $(SRC) Y ser interpretada como: $ gcc main.c
Sin embargo, una variable puede contener ms de un elemento, por ejemplo: objects = modulo_1.o modulo_2.o \ modulo_3.o \ modulo_4.o programa : $(objects)
gcc -o programa $(objects) Debemos hacer notar que la utilidad make hace distincin entre maysculas y minsculas.
Reglas explcitas.
Las reglas explcitas le dictan a make qu archivos dependen de otros y los comandos a usar para lograr un objetivo en especfico. El formato es: objetivo: requisitos comando #para lograr el objetivo Esta regla le instruye a make como crear un objetivo a partir de los requisitos utilizando un comando especfico. Por ejemplo, para generar un ejecutable que se llame main, escribiremos algo por el estilo: main: main.c main.h gcc -o main main.c main.h Esto significa que el requisito para poder lograr el objetivo main(un programa), es que existan los archivos main.c y main.h y para lograr el objetivo deberemos utilizar gcc en la forma descrita.
Reglas implcitas.
La reglas implcitas confan a make el trabajo de adivinar qu tipo de archivo queremos procesar (para ello utiliza las extensiones o sufijos del o los archivos). Las reglas implcitas ahorran el trabajo de tener que indicar qu comandos hay que ejecutar para lograr el objetivo, pues esto se infiere a partir de la extensin del archivo a procesar. Por ejemplo: funciones.o : funcion1.c funcion1.c
origina la siguiente linea: $(CC) $(CFLAGS) -c funcion1.c funcion2.c Existe un conjunto de variables que ya estn predefinidas y se utilizan para las reglas implcitas. De ellas existen dos categoras: (a) aquellas que son nombres de programas (como CC, que invoca al compilador de C), y (b) aquellas que contienen los argumentos para los programas invocados (como CFLAGS, que contiene las opciones que se le pasarn al compilador de C). Todas estas variables ya son provistas y contienen valores predeterminados , sin embargo, pueden ser modificados como se muestra a continuacin: CC = gcc CFLAGS = -g -Wall En el primer caso se indicar que el compilador de C ser GNU GCC y el segundo caso activar todo tipo de avisos del compilador y compilar una versin para depurado.
# La siguiente regla implicita instruye a make en como # procesar los archivos con extensin .c y .o .c.o: $(CC) -c $(CFLAGS) $< # Definicin de variables globales. CC = gcc CFLAGS = -g -Wall -O2 SRC = main.c funciones.c funciones.h OBJ = main.o funciones.o # La regla explicita all indica a make como
# procesar todo el proyecto. all: $(OBJ) $(CC) $(CFLAGS) -o main $(OBJ) # Esta regla indica como limpiar el proyecto de # archivos temporales. clean: $(RM) $(OBJ) main # Reglas implcitas funciones.o: funciones.c \ funciones.h main.o: main.c \ funciones.h En este archivo Makefile se han definido dos reglas explcitas que indican como construir los objetivos all y clean. Para llevar a cabo alguno de los dos objetivos basta ejecutar: $ make ... lo cual ejecutar la primera regla que encuentra, es decir all, la cual compilar los programas definidos en la variable $(OBJECT). Si se desea que se ejecuten las tareas de la regla clean, se deber ejecutar: $ make clean El archivo funciones.h contiene el prototipo de las funciones de las funciones empleadas en el programa main.c y estas, a su vez, se encuentran implementadas en funciones.c. De esta manera, es posible separar en distintos mdulos las funciones, objetos, mtodos, definiciones y variables que necesitemos en un proyecto determinado.
Tambin es posible crear reglas propias para formatos de archivos que no necesariamente han de crear un programa ejecutable. Por ejemplo, se puede mantener un conjunto de documentos, cuyo fuente se encuentran en formato .lyx y que se desea convertir a otros formatos como PDF, TeX, Postcript, etc y cuyos sufijos son desconocidos por make.
A continuacin se describe cmo aadir nuevas reglas con GNU make, el cual puede diferir con versiones antiguas de make. Por compatibilidad, ms adelante se explica cmo definirlo de la antigua forma, que GNU tambin puede interpretar. La forma de definir una regla que permita convertir un archivos PostScript en formato PDF sera de la siguiente manera: %.pdf: %.ps ps2pdf $< Se ha indicado que los archivos cuya extensin son .pdf dependen de los archivos .ps, y que se generan utilizando el programa indicado en la linea siguiente(ps2pdf). El parmetro de entrada para el programa ser el nombre del archivo con extensin .ps. Slo falta indicar la regla que archivos se irn a convertir, por ejemplo: all: documento1.pdf documento2.pdf De esta forma, el objetivo de make ser construir all, para lo cual debe construir documento1.pdf y documento2.pdf. Para lograr este objetivo, make buscar los archivos documento1.ps y documento2.ps, lo cual se traducir en los siguientes comandos: ps2pdf documento1.ps ps2pdf documento2.ps
$ < $*
El nombre del primer requisito. En la definicin de una regla implcita tiene el valor correspondiente al texto que
reemplazar el smbolo %.
$? $@
Es el nombre de todos los prerequisitos. Es el nombre del archivo del objetivo de la regla.
%.pdf : %.ps ps2pdf $ < %.zip: %.pdf echo $*.zip $< PDF = documento1.pdf documento2.pdf ZIP = documento1.zip documento2.zip
pdf: $(PDF) tar -zcvf [email protected] $? zip: $(ZIP) clean: rm -f *.pdf *.tar En el ejemplo, se han definido dos reglas implcitas. La primer indica cmo convertir un archivo PostScript a PDF y la segunda dice cmo comprimir un archivo pdf en formato ZIP. Tambin se han
definido cuatro reglas, dos de ellas son implcitas (pdf y zip), donde slo se han indicado sus requisitos de las otras dos (paquete y clean) son explcitas.
Cuando se ejecute la regla paquete, make analizar las dependencias, es decir, verificar si existen los correspondientes archivos PDF, si no existieren, los construye para luego ejecutar el comando indicado en la regla. La variable $? ser expandida a "documento1.pdf documento2.pdf" y la variable $@ ser expandida a "paquete". De esta forma el comando a ejecutar ser: tar -zcvf paquete.tar.gz documento1.pdf documento2.pdf En el caso de la regla zip, al resolver las dependencias se ejecutar: zip documento1.zip documento1.pdf zip documento2.zip documento2.pdf Es decir, el patrn buscado es documento1 y documento2, los cuales corresponden con la expresin %. Dicha operacin se realizar para cada archivo .pdf.
B C
GTK_STOCK_BOLD GTK_STOCK_CANCEL GTK_STOCK_CDROM GTK_STOCK_CLEAR GTK_STOCK_CLOSE GTK_STOCK_COLOR_PICKER GTK_STOCK_CONVERT GTK_STOCK_CONNECT GTK_STOCK_COPY GTK_STOCK_CUT
GTK_STOCK_DELETE GTK_STOCK_DIALOG_AUTHENTICATION GTK_STOCK_DIALOG_ERROR GTK_STOCK_DIALOG_INFO GTK_STOCK_DIALOG_QUESTION GTK_STOCK_DIALOG_WARNING GTK_STOCK_DIRECTORY GTK_STOCK_DISCONNECT GTK_STOCK_DND GTK_STOCK_DND_MULTIPLE
#define GTK_STOCK_DIALOG_QUESTION "gtk-dialog-question" #define GTK_STOCK_DIALOG_WARNING #define GTK_STOCK_DIRECTORY #define GTK_STOCK_DISCONNECT #define GTK_STOCK_DND #define GTK_STOCK_DND_MULTIPLE #define GTK_STOCK_EDIT "gtk-dialog-warning" "gtk-directory" "gtk-disconnect" "gtk-dnd" "gtk-dnd-multiple" "gtk-edit"
GTK_STOCK_EDIT
GTK_STOCK_FIND_AND_REPLACE
#define GTK_STOCK_FLOPPY #define GTK_STOCK_FULLSCREEN #define GTK_STOCK_GOTO_BOTTOM #define GTK_STOCK_GOTO_FIRST #define GTK_STOCK_GOTO_LAST #define GTK_STOCK_GOTO_TOP #define GTK_STOCK_GO_BACK #define GTK_STOCK_GO_DOWN #define GTK_STOCK_GO_FORWARD #define GTK_STOCK_GO_UP #define GTK_STOCK_HARDDISK #define GTK_STOCK_HELP #define GTK_STOCK_HOME #define GTK_STOCK_INDENT #define GTK_STOCK_INDEX #define GTK_STOCK_INFO #define GTK_STOCK_ITALIC #define GTK_STOCK_JUMP_TO #define GTK_STOCK_JUSTIFY_CENTER #define GTK_STOCK_JUSTIFY_FILL #define GTK_STOCK_JUSTIFY_LEFT #define GTK_STOCK_JUSTIFY_RIGHT #define fullscreen"
"gtk-floppy" "gtk-fullscreen" "gtk-goto-bottom" "gtk-goto-first" "gtk-goto-last" "gtk-goto-top" "gtk-go-back" "gtk-go-down" "gtk-go-forward" "gtk-go-up" "gtk-harddisk" "gtk-help" "gtk-home" "gtk-indent" "gtk-index" "gtk-info" "gtk-italic" "gtk-jump-to" "gtk-justify-center" "gtk-justify-fill" "gtk-justify-left" "gtk-justify-right" "gtk-leave-
GTK_STOCK_LEAVE_FULLSCREEN
GTK_STOCK_LEAVE_FULLSCREEN
#define GTK_STOCK_MEDIA_FORWARD #define GTK_STOCK_MEDIA_NEXT #define GTK_STOCK_MEDIA_PAUSE #define GTK_STOCK_MEDIA_PLAY #define GTK_STOCK_MEDIA_PREVIOUS #define GTK_STOCK_MEDIA_RECORD #define GTK_STOCK_MEDIA_REWIND #define GTK_STOCK_MEDIA_STOP #define GTK_STOCK_MISSING_IMAGE #define GTK_STOCK_NETWORK #define GTK_STOCK_NEW #define GTK_STOCK_NO #define GTK_STOCK_OK #define GTK_STOCK_OPEN #define GTK_STOCK_PASTE #define GTK_STOCK_PREFERENCES #define GTK_STOCK_PRINT #define GTK_STOCK_PRINT_PREVIEW #define GTK_STOCK_PROPERTIES #define GTK_STOCK_QUIT #define GTK_STOCK_REDO #define GTK_STOCK_REFRESH #define GTK_STOCK_REMOVE #define saved" GTK_STOCK_REVERT_TO_SAVED
"gtk-media-forward" "gtk-media-next" "gtk-media-pause" "gtk-media-play" "gtk-media-previous" "gtk-media-record" "gtk-media-rewind" "gtk-media-stop" "gtk-missing-image" "gtk-network" "gtk-new" "gtk-no" "gtk-ok" "gtk-open" "gtk-paste" "gtk-preferences" "gtk-print" "gtk-print-preview" "gtk-properties" "gtk-quit" "gtk-redo" "gtk-refresh" "gtk-remove" "gtk-revert-to-
O P
Q R
#define GTK_STOCK_SORT_DESCENDING descending" #define GTK_STOCK_SPELL_CHECK #define GTK_STOCK_STOP #define GTK_STOCK_STRIKETHROUGH #define GTK_STOCK_UNDELETE #define GTK_STOCK_UNDERLINE #define GTK_STOCK_UNDO #define GTK_STOCK_UNINDENT #define GTK_STOCK_YES #define GTK_STOCK_ZOOM_100 #define GTK_STOCK_ZOOM_FIT #define GTK_STOCK_ZOOM_IN #define GTK_STOCK_ZOOM_OUT
"gtk-spell-check" "gtk-stop" "gtk-strikethrough" "gtk-undelete" "gtk-underline" "gtk-undo" "gtk-unindent" "gtk-yes" "gtk-zoom-100" "gtk-zoom-fit" "gtk-zoom-in" "gtk-zoom-out"
Y Z