Entwickeln von Gnome Applikationen mit Python (Teil 2)

ArticleCategory:

Software Development

AuthorImage:

Hilaire Fernandes

TranslationInfo:[Author + translation history. mailto: or http://homepage]

original in fr Hilaire Fernandes

fr to de G�nther Socher

AboutTheAuthor

Hilaire Fernandes ist Vizepr�sident OFSET, einer Gesellschaft zur F�rderung und Entwicklung von freier Schul- und Lernsoftware f�r das Gnome-Projekt. Er hat ebenfalls Dr. Geo, eine preisgekr�nte Software f�r dynamische Geometrien geschrieben. Zur Zeit besch�ftigt er sich mit Dr. Genius, einer anderen mathematischen Lernsoftware f�r das Gnome Projekt.

Abstract:

Diese Artikelreihe ist vor allem f�r Anf�nger in der Programmierung mit Gnome und GNU/Linux geschrieben. Die gew�hlte Programmiersprache Python ist einfach und vermeidet die gew�hnliche �berforderung des Benutzers wie z. B. mit der Compilersprache C. Um diesen Artikel zu verstehen, sind einige Grundkenntnisse in der Programierung mit Python notwendig.

ArticleIllustration:

Gnome

ArticleBody:

Was man braucht

Eine Liste der notwendigen Software zur Ausf�hrung des beschriebenen Programmes befindet sich im Teil I dieser Artikelreihe.

Ebenfalls ben�tigen Sie :

Die Installation und die Anwendung von Python-Gnome und LibGlade ist ebenfalls im Teil I beschrieben.

Drill, unser Support

Ziel des ersten Teils war es, die verschiedenen Mechanismen und das Zusammenspiel der verschiedenen Teile eines unter Gnome-, Glade-, LibGlade- und Python- geschriebenen Programmes zu zeigen.

Das Anwendungsbeispiel benutzt das Widget GnomeCanvas. Das erm�glicht uns eine farbenreiche Darstellung und zeigt die Vorteile und die Einfachheit der Programmentwicklung mit dieser Konfiguration.

Im folgenden Abschnitt werden die verschiedenen Widgets von Gnome dargestellt. Dieser Artikel besch�ftigt sich haupts�chlich damit, diese Rahmenbedingungen zu erkl�ren. Weitere Artikel, die sich auf diesen beziehen werden, f�gen noch mehr Funktionen hinzu, um die verschiedenen Widgets (=Elemente in einer grafischen Oberfl�che, z.B. ein Knopf) von Gnome zu erkl�ren

Unsere wichtigste Software nennt sich Drill. Es handelt sich um eine Platform zur Schulung, auf die sich unsere Beispiele und �bungen beziehen. Diese Beispiele und �bungen dienen ausschlie�lich zu p�dagogischen Zwecken, um den Umgang mit den Widgets zu zeigen.

Erstellung der Schnittstelle mit Glade

Die Widgets

Das Anwendungsfenster wird mit Hilfe von Glade erstellt. Wie im vorherigen Artikel erstellen Sie zun�chst ein Fenster f�r eine Gnome Applikation. Darin m�ssen Sie die Icons und Men�s l�schen.

Der Haupteil von Drill ist mit dem Widget GtkPaned in zwei Teile unterteilt worden.

[Drill window]
Abb. 1 - Fenster von Drill

Sie sind vertikal unterteilt und diese Unterteilung l�sst sich auch verschieben. Der linke Teil enth�lt ein Baum Widget (GtkTree), in welchem die verschiedenen Abschnitte der �bung angeordnet sind. Der rechte Teil ist leer und dort werden die �bungen von dem Anwender dann ausgef�hrt.

Glade kann das Widget Interface von Drill in Form eines hirarchischen Baumes darstellen. Das erleichtert das Verst�ndnis.

[widgets in tree]
Abb. 2 - Baumdarstellung der Schnittstelle von Drill

In der Abb. 2 kann man das Widget hpanedTree sehen (vom Typ GtkPaned). Es enth�lt nur einen einzigen Widget, frame2 (vom Typ GtkFrame), dieser befindet sich links. frame2 enth�lt den widget exerciceTree. Es ist vorteilhafter zun�chst ein Widget GtkFrame mit der Schattierung des Typs GTK_SHADOW_IN in einem GtkPaned Widget, zu nehmen, was �berschneidungen mit der Trennleiste verhindert.

Zum Schlu� das Dialogfenster Gnome "À propos", das beim Beenden von Drill erscheint.

[Dialog window]
Abb. 3 - Dialogfenster "À propos" von Drill

Die unterschiedlichen Elemente sind von Glade aus im Dialogfenster Widget des Fensters Eigenschaften bearbeitet worden.

Die Namen der Widgets und seine Bedingungsfunktionen

Benutzen Sie die folgenden Namen f�r diese Widgets, um diese schlie�lich unter diesen Namen unter Python zu ver�ndern.

Anwendungsfenster Gnome :
drillApp
Die Trennleiste zwischen dem Baum und der �bung :
hpanedTree
Baum der �bungen :
exerciceTree
Dialogfenster Gnome À-propos :
about

Diese widgets sind sichtbar auf der Abb. 2 bezeichnet

Wir f�hren hier schnell die Bedienungsfunktionen auf. Falls Sie weitergehende Informationen zu diesem Thema ben�tigen, k�nnen Sie diese im Teil I dieser Artikelserie nachschlagen

Name des Widgets Signal Event handler
about clicked gtk_widget_destroy
about close gtk_widget_destroy
about destroy gtk_widget_destroy
button1 (Neues Icon in der
Arbeistleiste)
clicked on_new_activate
new activate on_new_activate
drillApp destroy on_exit_activate
exit activate on_exit_activate
about activate on_about_activate

Letzte Anpassungen

Von Glade aus ist es m�glich, die Geometrie der Widgets zu definieren. In unserem Fall k�nnen Sie die Gr��e der drillApp auf 400x300 im Eigenschaftendialogfenster einstellen. Ebenfalls kann die Position der horizontalen Unterteilung des Fensters auf 100 statt 1 eingestellt werden.

Anschlie�end muss das Widget exerciceTree eingestellt werden, um nur eine Auswahl gleichzeitig zu erlauben. Nur eine �bung gleichzeitig ausgew�hlt werden. Von dem Fenster Eigenschaften, w�hlt man Selection->Single. Die anderen Optionen von diesem Widget sind weniger wichtig

Voilà! Was Drill betrifft, ist die Konfiguration fertig. Jetzt fangen wir an, die �bungen des Artikels auszuf�hren. Nun werden wir sehen, wie man die Schnittstelle von Python aus benutzt und wir werden uns mit den Einstellungen des Widgets GtkTree besch�ftigen.

Der Code von Python

Der vollst�ndige Sourcecode befindet sich am Ende dieses Dokumentes. Er muss im gleichen Verzeichnis wie die Datei drill.glade abgelegt werden.

Die notwendigen Module

from gtk import *
from gnome.ui import *
from GDK import *
from libglade import *

Die graphische Schnittstelle mit LibGlade

Der Aufbau der graphischen Schnittstelle und die Einbindung der Bedienungsfunktionen mit LibGlade wird in analoger Weise wie im vorherigen Beispiel durchgef�hrt. Wir werden auf diesen Aspekt zur�ckkommen.

Im Python Programm definieren wir die globalen Variablen :



Der Baum ist von LibGlade erzeugt worden, der Pointer ist mit folgendem Aufruf erhalten worden:

exerciceTree = wTree.get_widget ("exerciceTree")

Wir ben�tigen ebenfalls den Pointer auf die horizontalen Fenster, genauer gesagt, den Pointer vom Container Widget (GtkPaned), dem horizontalen Fenster mit der Trennleiste. Die linke H�lfte stellt den Baum dar und die rechte H�lfte die �bungen, wo wir jetzt den label gesetzt haben:

paned = wTree.get_widget ("hpanedTree")
label = GtkLabel ("No exercise selected")
label.show ()
paned.pack2 (label)

Es empfiehlt sich, auch die Bedienungsanleitung von GTK+ -- �ber die Objekte GtkLabel und GtkPaned -- und die Quellen von Python /usr/lib/python1.5/site-packages/gtk.py zu studieren.

Das Widget GtkTree

Wir kommen jetzt zum wichtigsten Teil unseres Artikels, um einen Baum des Types GtkTree benutzen zu k�nnen.

Der Baum besteht aus aufeinanderfolgenden Aufrufen der Funktionen addMathExercices(), addFrenchExercices(), addHistoryExercices() et addGeographyExercices(). Diese Funktionen �hneln sich alle. Jede dieser Funktionen f�gt eine neue Unterkategorie (Unterbaum) ebenso wie die Titel der �bung (die Items) hinzu:

def addMathExercices ():
   subtree = addSubtree ("Math�matiques")
   addExercice (subtree, "Exercice 1", "Math. Ex1")
   addExercice (subtree, "Exercice 2", "Math. Ex2")

Der Unterbaum

def addSubtree (name):
   global exerciceTree
   subTree = GtkTree ()
   item = GtkTreeItem (name)
   exerciceTree.append (item)
   item.set_subtree (subTree)
   item.show ()
   item.connect ("select", selectSubtree)
   return subTree

Um einen Unterbaum im existierenden Baum zu erzeugen, muss man zwei Dinge erzeugen: Einen Baum GtkTree und ein Element GtkTreeItem, das nach dem Unterbaum benannt ist. Anschlie�end ist das Element an der Wurzel des Baumes hinzugef�gt worden -- unser Baum beinhaltet alle Kategorien -- dann setzen wir unseren Unterbaum auf das Element mit Hilfe der Methode set_subtree() auf. Schlie�lich ist der Event select an das Element anzuschlie�en. Damit wird die Funktion selectSubtree() aufgerufen sobald ein Kathegorie ausgew�hlte wird.

GtkTreeItem

def addExercice (category, title, idValue):
  item = GtkTreeItem (title)
  item.set_data ("id", idValue)
  category.append (item)
  item.show ()
  item.connect ("select", selectTreeItem)
  item.connect ("deselect", deselectTreeItem)

Die Elemente tragen die gleichen Bezeichnungen wie die Namen der �bungen, hier im allgemeinen nur �bung 1, �bung 2, ... Bei jedem Element wird ein zus�tzliches Attribut hinzugef�gt, id. GTK+ bietet die M�glichkeit jedes Objekt vom Typ GtkObject hinzuzuf�gen. Hierzu existieren zwei Methoden set_data (key, value) und get_data (key), um das Attribut zu initialisieren und auf dessen Wert zur�ckzugreifen. Das Element wird anschlie�end zu seiner Kategorie hinzugef�gt -- einem Unterbaum. Die Methode show() wird aufgerufen um das element anzuzeigen. Anschlie�end werden die Events select und deselect eingebunden. Das Event deselect wird eingesetzt, wenn das Element nicht mehr ausgew�hlt wird. Analog dazu wird die Methode deselectTreeItem() oder selectTreeItem() ausgef�hrt.

Die Bedienungsfunktionen

Wir haben drei Bedienungsfunktionen definiert selectTreeItem(), deselectTreeItem() und selectSubtree(). Diese Funktionen aktualisieren den Text des Labels -- in der rechten Zone -- mit dem Wert des Attributes id.

Schlusswort

Wir haben die Infrastruktur, in der wir die �bungen durchf�hren, aufgebaut -- genauso wie die neu entdeckten Widgets. Wir haben uns haupts�chlich mit dem Widget GtkTree besch�ftigt und wie man Attribute zu den Widgets hinzuf�gen kann. Dieses Verfahren wird sehr oft benutzt, um in den Bedienungsfunktionen noch zus�tzliche Informationen zu erhalten. Bis zum n�chsten Artikel k�nnen Sie versuchen, das Spiel Couleur, womit wir uns im Teil I besch�ftigt haben, als eine �bung in Drill einzubauen.

Anhang: Der vollst�ndige Sourcecode

#!/usr/bin/python
# Drill - Teo Serie
# Copyright Hilaire Fernandes 2001
# Release under the terms of the GPL licence
# You can get a copy of the license at http://www.gnu.org from gtk import *
from gnome.ui import *
from GDK import *
from libglade import * exerciceTree = currentExercice = label = None

def on_about_activate(obj):
    "display the about dialog"
    about = GladeXML ("drill.glade", "about").get_widget ("about")
    about.show ()

def on_new_activate (obj):
    global exerciceTree, currentExercice

def selectTreeItem (item):
    global label
    label.set_text ("L'exercice " +
             item.get_data ("id") + "est sélectionné.")

def deselectTreeItem (item):
    global label
    label.set_text ("L'exercice " +
             item.get_data ("id") + "est désélectionné.")

def selectSubtree (subtree):
    global label
    label.set_text ("No selected exercise")

def addSubtree (name):
    global exerciceTree
    subTree = GtkTree ()
    item = GtkTreeItem (name)
    exerciceTree.append (item)
    item.set_subtree (subTree)
    item.show ()
    item.connect ("select", selectSubtree)
    return subTree

def addExercice (category, title, id):
    item = GtkTreeItem (title)
    item.set_data ("id", id)
    category.append (item)
    item.show ()
    item.connect ("select", selectTreeItem)
    item.connect ("deselect", deselectTreeItem)

def addMathExercices ():
    subtree = addSubtree ("Mathématiques")
    addExercice (subtree, "Exercice 1", "Math. Ex1")
    addExercice (subtree, "Exercice 2", "Math. Ex2")

def addFrenchExercices ():
    subtree = addSubtree ("Français")
    addExercice (subtree, "Exercice 1", "Français Ex1")
    addExercice (subtree, "Exercice 2", "Français Ex2")

def addHistoryExercices ():
    subtree = addSubtree ("Histoire")
    addExercice (subtree, "Exercice 1", "Histoire Ex1")
    addExercice (subtree, "Exercice 2", "Histoire Ex2")

def addGeographyExercices ():
    subtree = addSubtree ("Géographie")
    addExercice (subtree, "Exercice 1", "Géographie Ex1")
    addExercice (subtree, "Exercice 2", "Géographie Ex2")

def initDrill ():
    global exerciceTree, label
    wTree = GladeXML ("drill.glade", "drillApp")
    dic = {"on_about_activate": on_about_activate,
        "on_exit_activate": mainquit,
        "on_new_activate": on_new_activate}
    wTree.signal_autoconnect (dic)
    exerciceTree = wTree.get_widget ("exerciceTree")
    # Temporary until we implement real exercice
    paned = wTree.get_widget ("hpanedTree")
    label = GtkLabel ("No selected exercise")
    label.show ()
    paned.pack2 (label)
    # Free the GladeXML tree
    wTree.destroy ()
    # Add the exercices
    addMathExercices ()
    addFrenchExercices ()
    addHistoryExercices ()
    addGeographyExercices ()

initDrill ()
mainloop ()