From aa47f9014030e48c4e56210ba950c2dbb3a63031 Mon Sep 17 00:00:00 2001 From: Dmitry Kalyanov Date: Tue, 21 Apr 2009 22:55:53 +0400 Subject: [PATCH] add doc --- doc/Makefile | 10 +++++++ doc/doc.xml | 66 ++++++++++++++++++++++++++++++++++++++++++++ doc/hello.lisp | 27 ++++++++++++++++++ doc/hello_world.png | Bin 0 -> 12826 bytes doc/tutorial.xml | 76 +++++++++++++++++++++++++++++++++++++++++++++++++++ 5 files changed, 179 insertions(+) create mode 100644 doc/Makefile create mode 100644 doc/doc.xml create mode 100644 doc/hello.lisp create mode 100644 doc/hello_world.png create mode 100644 doc/tutorial.xml diff --git a/doc/Makefile b/doc/Makefile new file mode 100644 index 0000000..4209300 --- /dev/null +++ b/doc/Makefile @@ -0,0 +1,10 @@ +all: doc.html index.html tutorial.html + +doc.html: doc.xml + xsltproc -o $@ /usr/share/sgml/docbook/xsl-stylesheets/html/docbook.xsl $< + +tutorial.html: tutorial.xml + xsltproc -o $@ /usr/share/sgml/docbook/xsl-stylesheets/html/docbook.xsl $< + +index.html: doc.xml + xsltproc /usr/share/sgml/docbook/xsl-stylesheets/html/chunk.xsl $< \ No newline at end of file diff --git a/doc/doc.xml b/doc/doc.xml new file mode 100644 index 0000000..b21645a --- /dev/null +++ b/doc/doc.xml @@ -0,0 +1,66 @@ + + + + CL-GTK2 - a Gtk+ binding for Common Lisp + + + Dmitry + Kalyanov + + + + + TODO + Installation and requirements + *Introduction + Goals + Development History + Features + *Examples + *GObject binding: using objects, handling memory + *Using Gtk+ in Lisp: Gtk+, Widgets, main loop, threading + *Advanced GObject: defining objects, creating custom gobject classes, overriding and implementing methods and properties + *Internals: how gobject binding is implemented + + + Introduction + CL-GTK2 package is a Common Lisp binding for Gtk+ and related libraries. This package enables Lisp developers to develop Gtk+-based GUI applications. + + + Goals + The goal of CL-GTK2 is to provide portable (modulo threading, metaobject protocol, FFI), extensible and feature-complete binding for Gtk+ and set of related libraries (GLib, GObject, Gdk, Pango, Cairo, GdkPixbuf, GtkSourceView, etc.) + Gtk+ is writtent in C and uses object-oriented style of programming with support of GObject library. GObject provides classes, objects, memory management via reference counters, classes introspection, signals, properties. Goal includes "lispy" interface to GObject-based libraries. This means that GObject classes should map into CLOS classes, GObject instance into CLOS instances, properties should match to slots, it should be possible to attach closures as signal handlers. It should be possible to define GObject classes, implement properties, methods, signals, interfaces from lisp. + All memory management (object instances, closures, signal handlers, etc.) should be automatic. When applicable, optional support for manual management (using refcounts) should be provided for perfomance reasons. + All Gtk+ classes should be mapped, and the rest of the API should be mapped in lispy way. Support for NIH features should not be provided (e.g., do not expose GSList, GList, GHashtable, GUri but rather transparently convert to/from them). + Error handling should be provided; GObject/Gtk warnings and errors should be exposed as conditions. It should not be possible to bring Gtk+ into incostistent state when error in lisp code called from Gtk occurs (e.g., error in signal handler should not cause fatal errors but rather should be handleable). + CL-GTK2 should be threadsafe: it should be possible to make actions under the Gdk lock, and make actions from withing the GUI thread. + Ease of extending and updating to reflect progress of Gtk+ is also important. + CL-GTK2 should run main-loop threadedly; it should be possible to interactively modify the code while the GUI program is running. + + + Development history + This project was started by Kalyanov Dmitry in 2008. It was a hobby project to address the (perceived or real) lack of good portable GUI libraries for Common Lisp and to be able to use it for writing diploma (or is it called "graduate"?) project during final course of the university. + Initial intent was to use SWIGSimplified Wrapper and Interface Generator; a program that automatically parses C and C++ header files and generates the binding code for many languages, including Common Lisp (targetting CFFI). in order to be able to generate all of the code. After some experimenting, it was found out that use of SWIG would not help to achieve the goals. First of all, in order to create "lispy" bindings, a lot of integration should be done: object systems, memory management, passing closure and values, type mapping (G(S)List to list, GCallback/GCLosure to functions, etc.). SWIG is not of much help here. + So, it was decided to use CFFI and manually map all of the API. First, base infrastructure was created (definition of basic types, maps for basic types, integration with GObject object system and memory management, introspection of GObject types). Then large chunks of API were mapped with automatic generation of classes definitions. Then missing properties were added and functions were mapped. + In order to map GtkTreeModel, preliminary support for subclassing and implementing GObjects was introduced. With its help, lispy ArrayListStore (analog for GtkListStore) was implemented. + At the moment, next version of GObject binding using the CLOS MOP is being developed. It pretty much obsoletes class generation macroses and is neede in order to be able to make lispy maps of GObject methods into CLOS generic functions and to subclass GObject classes. + For future development, using GObjectIntrospection seems the definite way. + + + Features + Short summary of features provided by CL-GTK2: + + Mapping from GObject to CLOS with (partial) support for creating subclasses, overriding methods + Automatic memory management (with optional support for manual reference counting) + Lispy interface to related libraries (currently only to GLib, GObject, Gdk, Gtk) + Lispy mapping of Gtk+ API + Custom ListStore (TreeStore to be done) for GtkTreeModel + Partial error management (closures are invoked with restarts making it possible to cancel signal handling without crashing Gtk+) + Some high-level features based on Gtk+ API: with-progress-bar, with-message-error-handler + + + + GObject binding + At the base of Gtk+ is the GObject library. This library provides basic object-oriented system and various services. + + diff --git a/doc/hello.lisp b/doc/hello.lisp new file mode 100644 index 0000000..3b1e16d --- /dev/null +++ b/doc/hello.lisp @@ -0,0 +1,27 @@ +(defpackage :gtk-hello + (:use :cl :gtk :gobject :glib) + (:export :run)) + +(in-package :gtk-hello) + +(defun run () + (let ((output *standard-output*)) + (with-main-loop + (let ((window (make-instance 'gtk-window + :type :toplevel + :window-position :center + :title "Hello world!" + :default-width 300 + :default-height 100)) + (button (make-instance 'button :label "Hello, world!")) + (counter 0)) + (g-signal-connect button "clicked" + (lambda (b) + (declare (ignore b)) + (format output "Hello, world!~%") + (setf (button-label button) + (format nil + "Hello, world! (clicked ~D times)" + (incf counter))))) + (container-add window button) + (widget-show window :all t))))) \ No newline at end of file diff --git a/doc/hello_world.png b/doc/hello_world.png new file mode 100644 index 0000000000000000000000000000000000000000..6d8c8495007db35e3970a0f3b4bc0e3214335f24 GIT binary patch literal 12826 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 literal 0 HcmV?d00001 diff --git a/doc/tutorial.xml b/doc/tutorial.xml new file mode 100644 index 0000000..61b8f39 --- /dev/null +++ b/doc/tutorial.xml @@ -0,0 +1,76 @@ + +
+ + CL-GTK2 Tutorial + + + Dmitry + Kalyanov + + + +
+ TODO + Introduction + More involved example: simple text editor + More involved example walk-through: widgets, properties, packing, child properties + TreeView example +
+
+ Installation + Unpack the CL-GTK2 sources, and add them to asdf:*central-registry*: + + (push "/path/to/cl-gtk2/glib/" asdf:*central-registry*) + (push "/path/to/cl-gtk2/gdk/" asdf:*central-registry*) + (push "/path/to/cl-gtk2/gtk/" asdf:*central-registry*) + +
+
+ "Hello world" example + Let's start from a simple example. + + Load the GTK system: (asdf:oos 'asdf:load-op :gtk) + Having loaded the GTK system, compile the following code: + +(defpackage :gtk-hello + (:use :cl :gtk :gobject :glib) + (:export :run)) + +(in-package :gtk-hello) + +(defun run () + (let ((output *standard-output*)) + (with-main-loop + (let ((window (make-instance 'gtk-window + :type :toplevel + :window-position :center + :title "Hello world!" + :default-width 300 + :default-height 100)) + (button (make-instance 'button :label "Hello, world!")) + (counter 0)) + (g-signal-connect button "clicked" + (lambda (b) + (declare (ignore b)) + (format output "Hello, world!~%") + (setf (button-label button) + (format nil + "Hello, world! (clicked ~D times)" + (incf counter))))) + (container-add window button) + (widget-show window :all t))))) + + And now, run it:(gtk-hello:run) + This code should create a form with a button. When the button is clicked, this program prints a message to standard output and changes the text of the button. + If you have a window, then congratulations! Otherwise, contact the author for support. +
+
+ Example walk-through + Let's analyze the example step by step. + CL-GTK2 runs the Gtk main loop in background thread. This is done so you could have your application running and interacting with the Lisp system through the REPL. + To execute some code and ensure that Gtk+ main loop is started, WITH-MAIN-LOOP macro is used. It runs the body of code within the Gtk+ main loop. Because all calls to Gtk+ functions require locking, it is neccessary to run this code from th main loop. Because we are running the code in another thread, its dynamic bindings (including *standard-output*) will be lost. To be able to print at REPL, we save reference to the standard output stream in the closure. + Gtk+ objects are created with make-instance and are properly garbage-collected. Some properties (slots) of objects are constructor-only properties and can only be set at object construction time. For example, "type" property of GtkWindow can only be set during its creation. + Call to container-add puts button as a child widget into window, and widget-show shows all widgets of window. + In Gtk+, objects have "signals", to which handlers can be attached. When something happens that results in "emitting" the signal (e.g., button being clicked emits "clicked" signal), all handlers of this signal are called. Handler of GtkButton's "clicked" signal has only one argument - the button that was clicked. CL-GTK2 allows attaching any function (including closures) as signal handlers and ensures that closure is freed properly. +
+
-- 1.7.10.4