From: Dmitry Kalyanov Date: Wed, 24 Jun 2009 07:19:34 +0000 (+0400) Subject: doc: added generated html files to git X-Git-Url: http://repo.macrolet.net/gitweb/?a=commitdiff_plain;h=4d5ead740bb437180b93b5b92f10096d1123bf76;p=cl-gtk2.git doc: added generated html files to git --- diff --git a/doc/Makefile b/doc/Makefile index 4209300..92cc1b9 100644 --- a/doc/Makefile +++ b/doc/Makefile @@ -1,4 +1,4 @@ -all: doc.html index.html tutorial.html +all: doc.html tutorial.html doc.html: doc.xml xsltproc -o $@ /usr/share/sgml/docbook/xsl-stylesheets/html/docbook.xsl $< diff --git a/doc/doc.html b/doc/doc.html new file mode 100644 index 0000000..ac45738 --- /dev/null +++ b/doc/doc.html @@ -0,0 +1 @@ +CL-GTK2 - a Gtk+ binding for Common Lisp

CL-GTK2 - a Gtk+ binding for Common Lisp

Dmitry Kalyanov


Chapter 1. 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

Chapter 2. 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.

Chapter 3. 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.

Chapter 4. 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 SWIG[1] 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.



[1] Simplified 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).

Chapter 5. 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

Chapter 6. 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/schemas.xml b/doc/schemas.xml new file mode 100644 index 0000000..942c168 --- /dev/null +++ b/doc/schemas.xml @@ -0,0 +1,4 @@ + + + + diff --git a/doc/tutorial.html b/doc/tutorial.html new file mode 100644 index 0000000..37ae017 --- /dev/null +++ b/doc/tutorial.html @@ -0,0 +1,11 @@ +CL-GTK2 Tutorial

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

CL-GTK2 is in alpha stage and is unstable and not feature-complete. It is being developed on x86-64 gentoo linux with SBCL. It should, in general, work with other lisp compilers and on other platforms. CL-GTK2 requires some features not present in the Common Lisp Standard, namely it requires CFFI support with callbacks and long-long support (most modern lisp implementations are supported, including clisp) and it requires CLOS MOP (MetaObject Protocol) which is also is present (or is being added to) most modern lisp implementations.

CL-GTK2 requires Gtk+ version 2.16 or later. CL-GTK2 was tested on SBCL-1.0.18 and SBCL-1.0.28

If you have any difficulties installing or using CL-GTK2, contact me (the author of this tutorial and of CL-GTK2) via email Kalyanov.Dmitry@gmail.com or via jabber mo3r@jabber.ru.

First, install CL-GTK2 dependcies. CL-GTK2 has the following dependencies (CL-GTK2 was tested with specified versions; it would probably not work with earlier versions but should work with later versions):

Currently, CL-GTK2 is only available in Git repository at http://repo.or.cz/w/cl-gtk2.git. If you do not want or can not to use Git, download the snapshot from http://repo.or.cz/w/cl-gtk2.git?a=snapshot;h=HEAD;sf=tgz.

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*)
+    

Now you should be able to load the CL-GTK2:

(asdf:oos 'asdf:load-op :gtk)

When the system is loaded, run (gtk-demo:demo-text-editor). A text editor window should pop up:

This is a very simple text editor written in CL-GTK2. Apart from editing the text, it can evaluate expressions: select expression and press the "execute" button. Expression will be evaluated and its result will be put into text view.

"Hello world" example

Let's start from a simple example.

Start Slime, type the following code in the REPL:

+(asdf:oos 'asdf:load-op :gtk)
+
+(gtk:within-main-loop
+  (let ((window (make-instance 'gtk:gtk-window :title "Hello, world!")))
+    (gtk:widget-show window)))
+    

The empty window with title "Hello, world!" should appear.

Let's analyze this example line-by-line.

(asdf:oos 'asdf:load-op :gtk) loads the GTK system into Lisp.

CL-GTK2 runs Gtk+ main loop in background thread (because Lisp development is interactive in its nature; if main loop would block the REPL thread, you would have to restart the Lisp image too often). Because all access to Gtk+ should come from Gtk+ thread, we should run the code in that thread. Macro gtk:within-main-loop does exactly that: it schedules the code to be tun in the Gtk+ thread. You should use this macro whenever you want evaluate the code from the REPL or when you start you application.

Next, we create the window with make-instance and set its title property to "Hello, world!".

When the window is created, it is not yet shown on the screen. To show it, we call (gtk:widget-show window).

After this code executes, you should get back to the REPL (rememer, Gtk+ runs in background thread) and the window should appear on the screen.

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.

Object have properties, which are represented as slots in CL-GTK2. 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. To access properties, you may use slot-value function or slot accessor methods. For property Y declared on class X, method X-Y returns the value of the property. Properties are setfable (with exception of read-only and constructor-only properties).

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.