Tested with:
- Resolver: LTS Haskell 5.18 (ghc-7.10.3)
- Libraries: gtk3-0.14.4 glib-0.13.2.2
This tutorial shows how to build a graphical user interface (GUI) application in Haskell using bindings to GTK+. While working on a calculator program we'll cover the following topics:
- GTK+ terminology
- Writing handlers that process user's actions (button click, etc.)
- Introducing global state
- Manual creation of forms and creation of forms using Glade
Once you finish with the tutorial you will have a solid understanding of how to move on, read the documentation of the
gtk3
package, and accomplish your tasks.The tutorial does not assume any knowledge of Haskell except for a very basic understanding of how to work with the
IO
monad. The GTK+ binding is very straightforward and imperative in its nature. This may be seen as a downside, but I think it also may make things easier for newcomers to Haskell programming with an imperative background.Available libraries
Before we start with GTK+ bindings, it's reasonable to ask whether there is a better/alternative solution. Indeed, several libraries are available to create GUI in Haskell:
wx
— bindings to wxWidgets. A couple of things about this package I find suspicious: 1) it had delays in development when for a couple of years no new version was released, 2) it's still not present on Stackage. A practical conclusion from the point 2 is that it's not very popular nowadays, or at least not many Haskellers start writing any application with it otherwise it would be added already.
X11
— direct translation of the C binding to X11 graphics library (quote taken from the package description). “Direct translation” means that you pass around pointers. Yes, in Haskell code. For documentation authors suggest to look here (although Haddocks are not blank either). Last release was in May 2014.
hsqml
— the Haskell bindings to Qt Quick. Missing from Stackage. Releases seem to happen once a year.
fltkhs
— the Haskell bindings to the FLTK GUI library. The package seems to be pretty new and I personally don't see any benefit in using it over more mature solutions.
As to GTK+, it seems to be:
- Easy to install.
- Robust and mature. Documentation is quite good and comprehensive.
It's worth mentioning that Haskell has developed much stronger ecosystem with respect to web-development than GUI and desktop development in general. For further reference about the state of standalone GUI applications in Haskell, refer to:
- The section in Gabriel Gonzalez's State of Haskell Ecosystem
- The blog post by Keera Studios
Installation
Please see this article for Gtk2Hs installation. It has instructions for Windows, Linux, Mac, and FreeBSD.
First steps
The calculator application has been chosen because its logic is very straightforward and we can focus on working with the GTK+ framework without much distraction while keeping the tutorial reasonably practical.
Let's start by importing some modules:
import Control.Monad import Control.Monad.IO.Class import Data.IORef import Graphics.UI.Gtk hiding (Action, backspace)
As I said, GTK+ bindings are very imperative. All binding code lives in
IO
monad except for some cases that we will cover shortly.main :: IO () main = do void initGUI -- (1) window <- windowNew -- (2) -- (3) widgetShowAll window -- (4) mainGUI -- (5)
- The first thing we need to do in any program that uses GTK+ is to call the
initGUI
function. This function allocates some resources and prepares GTK+ for work, it also looks up command line arguments that are relevant to GTK+, parses them, and returns all non-parsed arguments. For our purposes, we don't need the command line arguments, so let's wrap it with thevoid
.
- Next, we need a window to build the interface of our calculator inside it. To create a new top-level window we use the
newWindow
action. It returns an opaqueWindow
value that can be used to manipulate the window.
- After creation of a new window we typically want to change some parameters and then render it. For now we just render the window “as is”, but in the next section we will see how to customize widgets using attributes.
widgetShowAll
works with any kind of widget. It performs all the necessary allocations and makes widget passed to it as an argument visible together with all its children widgets.
mainGUI
is the main loop. The loop listens to events such as button click and mouse pointer movement and let appropriate handlers run.
A note about threads. Make sure that all GTK actions happen on the same OS thread (note, this is different from lightweight Haskell threads). This is only important when you compile with multi-threaded runtime, but who knows, maybe the need for concurrent execution will arise later, so my advice is to keep all GTK-related code in one thread and the simplest way to do it is to keep everything is the main thread. For more information about multi-threaded GUIs with GTK+ see here.
If we compile and run the program we will see the following:
First look at the calculator application
Nothing fancy. One nasty detail is that when we close the window the program continues to run. This is because the default handler for click on the “close” button of window just makes it invisible, and the main loop
mainGUI
continues to run. We will see how to handle this situation properly soon.Attributes
Attributes allow to customize widgets, such as our window. There are two methods to specify widget attributes:
- Set them with the
set
action.
- Use Glade to design your UI.
We will touch the second option later, but for now let's become familiar with the
set
action. Typical usage of set
is the following:The GTK modules are structured by widget type, and every module typically has the “attributes” section. This is how to find out which attributes we can tweak. For example a
Window
has the following:windowTitle
windowType
windowResizable
…and many others. Let's change title of our window and make it non-resizeable:
Looks like it works:
Calculator window with adjusted attributes
Containers
Still, even non-resizable window with title is boring. What we would like to do is to put something inside that window. This brings us to the GTK+ notion of container. A container is a widget that can contain another widgets inside. There are two types of containers:
- those that serve purely decorative purpose and can contain only one widget;
- those that help organize forms and can contain several widgets.
Inner widgets are typically called children, while the enclosing widget is called parent.
Most important actions that you will want to perform on containers are:
containerAdd parent child
to addchild
widget toparent
widget
containerRemove parent child
to removechild
widget fromparent
widget
containerGetChildren
to get all children of a container widget
containerForeach
to perform an action on all children of a container
For now we will need a non-editable text area where we will show the number that is being entered and result of computations:
We use the
Entry
widget to display numbers, but it's not editable and right-aligned. We don't hurry to insert it into our window
because we need some sort of “grid” to make the form look like a real calculator.Indeed, there is the
Grid
widget in the Graphics.UI.Gtk.Layout.Grid
module. This is an example of a more complex container that has its own interface for better control of layout. We will be using the following functions from its API:gridNew
and gridSetRowHomogeneous
should be self-explanatory. gridAttach
allows to insert widgets into the grid controlling their position and size. This is very handy for our calculator application, let's use it:gridNew
creates a new grid.
gridSetRowHomogeneous grid True
makes every row have equal height.
- Here we define the
attach
helper function. It attaches given widget to ourgrid
. The argument order of this function helps to use it with(>>=)
.
- We attach the
display
we created previously to thegrid
. It will occupy the entire top row.
- Here we use combination of the
mkBtn
helper andattach
to quickly create buttons and place them on the grid. I'll show whatmkBtn
is in a moment.
- Now the grid itself needs to be inserted into
window
to be visible. This is done with help of the above-mentionedcontainerAdd
function.
mkBtn
is a helper for button creation, right now it's very simple:We create a new button, set its attributes (just label in our case) and return the button.
Our calculator looks like a real calculator
Signals and events
The application looks like a calculator but does not behave like one yet. To fix this, we need to learn about signals and events.
Signal is a name for things that may happen on the form. Almost always signals are connected with user's actions. An example of signal is focus — the moment when a widget becomes active on the form.
To execute some code on signal we use the
on
helper:Here
object
is the widget of interest, callback
is the action that we want to perform. The on
function returns ConnectId
parametrized over the object
type (so we cannot mix up connection identifiers for different types of objects). This is the identifier of signal handler and its sole purpose is to give you a way to disconnect a signal handler if you ever need it. You can use the disconnect
function from System.Glib.Signals
to do that.Every signal dictates the type that
callback
function will have. The following cases are the most frequent:- Just
IO ()
: no information is given to the handler and it is not expected to return anything. Example of such signal isshowSignal
.
- Handlers that are given arguments:
a -> IO Bool
. Example of such signal isfocus
whose handlers have the typeDirectionType -> IO Bool
. Another interesting thing here is returned value of the typeBool
. This is a convention in GTK+ allowing to disable default handling of some signals. If we returnTrue
, default handling will be disabled, whileFalse
will keep it active executing our handler and default handler as well.
There is one more way to get some information from within a signal's handler. Some signals dictate that handler should live in a special monad called
EventM
instead of plain IO
. Signals that like their handlers to be in EventM
are called events.What is the
EventM
monad? Actually it's a type synonym for a simple monad stack with IO
at the bottom:This is just a reader monad transformer on top of
IO
. t
specifies the type of information we can extract and which helper function we can use inside the EventM
monad. These are different for every event. For example, configureEvent
allows to extract information about window size, while keyPressEvent
event provides information about the key that has been pressed, which modifier key was held at that time and so forth. The type system does not allow to try to extract information that particular event does not provide.I would like to quote the docs to accent the importance of returned Boolean value:
Note that an event handler must always return True if the event was handled or False if the event should be dealt with by another event handler. For instance, a handler for a key press should return False if the pressed key is not one of those that the widget reacts to. In this case the event is passed to the parent widgets. This ensures that pressing, say, Alt-F opens the file menu even if the current input focus is in a text entry widget. In order to facilitate writing handlers that may abort handling an event, this module provides the function tryEvent. This function catches pattern match exceptions and returns False. If the signal successfully runs to its end, it returns True.
Knowing all that, we can write a simple handler to run on button activation. Looking at the “signals” section in
Graphics.UI.Gtk.Buttons.Button
, buttonActivated
looks like our friend here:Just for a test, let's re-write
mkBtn
to attach a handler that will update the display with the name of the pressed button (we still don't know a whole lot to make a working calculator):And we need to pass
display
to mkBtn
like this:Another thing that we can deal with now is proper closing of our application. For this we need a way to call the
mainQuit
function:As you may have guessed by now, a convenient place to put the
mainQuit
function is on closing of window
. The event that we're looking for is called deleteEvent
:In our case we just want to close it, so:
Note that
deleteEvent
parametrizes EventM
type by EAny
type-level tag. Its description:Even though it does not carry any event-specific information, a lot of useful information can be extracted, such as current time at the moment when event fired (
eventTime
). See full list of helpers in the “Accessor functions for event information” section of the Graphics.UI.Gtk.Gdk.EventM
module.I encourage you to compile and run the application to see that it responds to button activation and closes properly.
Using IORef
s for application state
Buttons can change display dynamically, but it's still not enough to make our calculator actually useful. For this (as with most other applications), we need some sort of state.
The creators of GTK+ binding didn't give us too many options here because type of handler monad is fixed: it's either plain
IO
or EventM
, which, as we already know, is just ReaderT (Ptr t) IO
. We cannot return anything non-standard from handlers, so the only way to communicate with outside world is via mutable references.There are two most obvious options:
IORef
s — mutable references insideIO
monad.
TVar
s fromControl.Concurrent.STM
.
TVar
s are probably overkill unless you do complex concurrent work. What is good about using TVar
s is that we can update them atomically. This may be not very important for some applications, but I recommend to build with concurrency in mind from the very beginning. But IORef
s can be changed atomically as well with help of atomicModifyIORef
.Now we got to the question how to model calculator logic. Since actual logic is not our primary concern in this tutorial, we will go the easy way.
Value
is our state, it contains textual representation of first argument and optionally representation of action that should be performed on it. The String
s representing arguments are reversed because this way it's faster to add/drop a character at the end of the string. We will reverse the strings back when it's time to turn them into Double
s.We will need a couple of helper functions too. Here they are:
The first two help change and extract the second argument in
Action
, while renderValue
does its best to render current calculator state. Having renderValue
, it's easy to write a function that would update the calculator display:Finally, instead of
mkBtn
let's have mkButton
of the following form:- Just like before we register a handler that will fire on button activation.
atomicModifyIORef
modifies givenIORef
atomically. The callback should return a tuple, first element is the new value to put intoIORef
, second value is the return value of the action. In this case we want the values to be equal.
- We call
updateDisplay
to make results of last action visible to the user.
Now we can define a helper called
mkBtn
in main
:- We need to create
IORef
to keep the program's state there.Value "" Nothing
is its initial value.
- The helper function
mkBtn
uses previously writtenmkButton
and just saves us the boilerplate of passingst
anddisplay
again and again.
- Some examples of
mkBtn
use. By passingid
as state mutating function we make buttons have no effect, but all the machinery for actual work is already in place.
The only thing that remains is state-mutating functions per button. Here I will show some of them:
The calculator is not perfect, but good enough for our purposes. Compile, run it and see how it works. Implementation of the rest of functionality is left as an exercise for the reader.
Using Glade to design forms
You may have probably noticed that manual creation of forms introduces quite a bit of boilerplate in our code. This can become even worse as your forms get more complex. Because of this, I think it's time to try our hand on a modern UI designer for GTK+ called Glade.
Glade is straightforward to install and use. Open the application, you will see panels with various widgets: top-level objects (such as window), containers, and controls (such as buttons).
Here is the plan how to re-create our calculator form with Glade:
- Select window button on the topmost palette and the window will appear on the working area.
- Enter “Calculator” in the title field.
- Don't forget to fill out the “ID” attribute of every widget, this is how you will access widgets on your form in the Haskell code.
- Create grid with id
grid
. When asked about number of rows and columns, choose 7 × 5. Select the “Homogeneous” check box under the “Rows” title.
- Insert entry, use “Drag and resize widgets in the workspace” button on the top tool bar to make it occupy the entire top row.
- Insert buttons to match our existing design.
Building the calculator form in Glade
(Here is a bigger image.)
Hint: the complete Glade form of our calculator is available under the Stack Builders tutorial repository.
To use the form in Haskell code we need
Builder
, which lives in Graphics.UI.Gtk.Builder
module. The Builder
object helps with creating UI from XML files on the fly.Here is how we could use it in our program:
- We need to create a new builder.
- Load our form into it from a file. There are other options such as loading form text (
String
,Text
) and so forth. Consult the docs for more information.
- Now the interesting part is that we can get actual button object knowing its identifier. For example here I'm retrieving button that inputs zeros, it has the
"btn_0"
identifier on my form.castToButton
casts abstract representatin of an object to its typed form. There are manycastToSomething
functions, one per widget (for example we havecastToWindow
for windows).
Having the actual object like that button or the main window itself, it's easy to proceed just like with the manually constructed form to start the main loop.
Conclusion
GTK+ Haskell binding certainly can be used to create professional-looking user interfaces. As I hopefully showed you in this tutorial, using the bindings is very straightforward and doesn't require any special knowledge. For small forms Glade probably doesn't make much sense, but if you write something big, it may save you some tiresome work. Better yet, one doesn't have to be a Haskell programmer to design UI with Glade — this fact makes it easier to divide work between people.