mirror of
https://github.com/ToshioCP/Gtk4-tutorial.git
synced 2025-01-12 20:03:28 +01:00
Add Section 26.
This commit is contained in:
parent
18dcb7da9f
commit
8ed3410f76
13 changed files with 1242 additions and 2 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -26,6 +26,7 @@ src/temp
|
|||
src/le
|
||||
src/list4/_build
|
||||
src/list5/_build
|
||||
src/expression/_build
|
||||
src/column
|
||||
html/*
|
||||
latex/*
|
||||
|
|
|
@ -42,3 +42,4 @@ You can read it without download.
|
|||
1. [Tiny turtle graphics interpreter](gfm/sec23.md)
|
||||
1. [GtkListView](gfm/sec24.md)
|
||||
1. [GtkGridView and activate signal](gfm/sec25.md)
|
||||
1. [GtkExpression](gfm/sec26.md)
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
Up: [Readme.md](../Readme.md), Prev: [Section 24](sec24.md)
|
||||
Up: [Readme.md](../Readme.md), Prev: [Section 24](sec24.md), Next: [Section 26](sec26.md)
|
||||
|
||||
# GtkGridView and activate signal
|
||||
|
||||
|
@ -561,4 +561,4 @@ If you feel some difficulty, it is better for you to separate the ui file.
|
|||
A directory [src/list5](../src/list5) includes the ui file above.
|
||||
|
||||
|
||||
Up: [Readme.md](../Readme.md), Prev: [Section 24](sec24.md)
|
||||
Up: [Readme.md](../Readme.md), Prev: [Section 24](sec24.md), Next: [Section 26](sec26.md)
|
||||
|
|
620
gfm/sec26.md
Normal file
620
gfm/sec26.md
Normal file
|
@ -0,0 +1,620 @@
|
|||
Up: [Readme.md](../Readme.md), Prev: [Section 25](sec25.md)
|
||||
|
||||
# GtkExpression
|
||||
|
||||
GtkExpression is a fundamental type.
|
||||
It is not a descendant of GObject.
|
||||
GtkExpression provides a way to describe references to values.
|
||||
GtkExpression needs to be evaluated to obtain a value.
|
||||
|
||||
It is similar to arithmetic calculation.
|
||||
|
||||
~~~
|
||||
1 + 2 = 3
|
||||
~~~
|
||||
|
||||
`1+2` is an expression.
|
||||
It shows the way how to calculate.
|
||||
`3` is the value comes from the expression.
|
||||
Evaluation is to calculate the expression and get the value.
|
||||
|
||||
GtkExpression is a way to get a value.
|
||||
Evaluation is like a calculation.
|
||||
A value is got by evaluating the expression.
|
||||
|
||||
First, I want to show you the C file of the example for GtkExpression.
|
||||
Its name is `exp.c` and located under [src/expression](../src/expression) directory.
|
||||
You don't need to understand the details now, just look at it.
|
||||
It will be explained in the next subsection.
|
||||
|
||||
~~~C
|
||||
1 #include <gtk/gtk.h>
|
||||
2
|
||||
3 GtkWidget *win1;
|
||||
4 int width, height;
|
||||
5 GtkExpressionWatch *watch_width;
|
||||
6 GtkExpressionWatch *watch_height;
|
||||
7
|
||||
8 /* Notify is called when "default-width" or "default-height" property is changed. */
|
||||
9 static void
|
||||
10 notify (gpointer user_data) {
|
||||
11 GValue value = G_VALUE_INIT;
|
||||
12 char *title;
|
||||
13
|
||||
14 if (watch_width && gtk_expression_watch_evaluate (watch_width, &value))
|
||||
15 width = g_value_get_int (&value);
|
||||
16 g_value_unset (&value);
|
||||
17 if (watch_height && gtk_expression_watch_evaluate (watch_height, &value))
|
||||
18 height = g_value_get_int (&value);
|
||||
19 g_value_unset (&value);
|
||||
20 title = g_strdup_printf ("%d x %d", width, height);
|
||||
21 gtk_window_set_title (GTK_WINDOW (win1), title);
|
||||
22 g_free (title);
|
||||
23 }
|
||||
24
|
||||
25 /* This function is used by closure tag in exp.ui. */
|
||||
26 char *
|
||||
27 set_title (GtkWidget *win, int width, int height) {
|
||||
28 return g_strdup_printf ("%d x %d", width, height);
|
||||
29 }
|
||||
30
|
||||
31 /* ----- activate, open, startup handlers ----- */
|
||||
32 static void
|
||||
33 app_activate (GApplication *application) {
|
||||
34 GtkApplication *app = GTK_APPLICATION (application);
|
||||
35 GtkWidget *box;
|
||||
36 GtkWidget *label1, *label2, *label3;
|
||||
37 GtkWidget *entry;
|
||||
38 GtkEntryBuffer *buffer;
|
||||
39 GtkBuilder *build;
|
||||
40 GtkExpression *expression, *expression1, *expression2;
|
||||
41 GtkExpressionWatch *watch;
|
||||
42 GValue value = G_VALUE_INIT;
|
||||
43 char *s;
|
||||
44
|
||||
45 /* Creates GtkApplicationWindow instance. */
|
||||
46 /* The codes below are complecated. It does the same as "win1 = gtk_application_window_new (app);". */
|
||||
47 /* The codes are written just to show how to use GtkExpression. */
|
||||
48 expression = gtk_cclosure_expression_new (GTK_TYPE_APPLICATION_WINDOW, NULL, 0, NULL,
|
||||
49 G_CALLBACK (gtk_application_window_new), NULL, NULL);
|
||||
50 if (gtk_expression_evaluate (expression, app, &value)) {
|
||||
51 win1 = GTK_WIDGET (g_value_get_object (&value)); /* GtkApplicationWindow */
|
||||
52 g_object_ref (win1);
|
||||
53 g_print ("Got GtkApplicationWindow object.\n");
|
||||
54 }else
|
||||
55 g_print ("The cclosure expression couldn't be evaluated.\n");
|
||||
56 gtk_expression_unref (expression);
|
||||
57 g_value_unset (&value); /* At the same time, the reference count of win1 is decreased by one. */
|
||||
58
|
||||
59 /* Builds a window with components */
|
||||
60 box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 10);
|
||||
61 label1 = gtk_label_new (NULL);
|
||||
62 label2 = gtk_label_new (NULL);
|
||||
63 label3 = gtk_label_new (NULL);
|
||||
64 buffer = gtk_entry_buffer_new (NULL, 0);
|
||||
65 entry = gtk_entry_new_with_buffer (buffer);
|
||||
66 gtk_box_append (GTK_BOX (box), label1);
|
||||
67 gtk_box_append (GTK_BOX (box), label2);
|
||||
68 gtk_box_append (GTK_BOX (box), label3);
|
||||
69 gtk_box_append (GTK_BOX (box), entry);
|
||||
70 gtk_window_set_child (GTK_WINDOW (win1), box);
|
||||
71
|
||||
72 /* Constant expression */
|
||||
73 expression = gtk_constant_expression_new (G_TYPE_INT,100);
|
||||
74 if (gtk_expression_evaluate (expression, NULL, &value)) {
|
||||
75 s = g_strdup_printf ("%d", g_value_get_int (&value));
|
||||
76 gtk_label_set_text (GTK_LABEL (label1), s);
|
||||
77 g_free (s);
|
||||
78 } else
|
||||
79 g_print ("The constant expression couldn't be evaluated.\n");
|
||||
80 gtk_expression_unref (expression);
|
||||
81 g_value_unset (&value);
|
||||
82
|
||||
83 /* Property expression and binding*/
|
||||
84 expression1 = gtk_property_expression_new (GTK_TYPE_ENTRY, NULL, "buffer");
|
||||
85 expression2 = gtk_property_expression_new (GTK_TYPE_ENTRY_BUFFER, expression1, "text");
|
||||
86 watch = gtk_expression_bind (expression2, label2, "label", entry);
|
||||
87
|
||||
88 /* Constant expression instead of "this" instance */
|
||||
89 expression1 = gtk_constant_expression_new (GTK_TYPE_APPLICATION, app);
|
||||
90 expression2 = gtk_property_expression_new (GTK_TYPE_APPLICATION, expression1, "application-id");
|
||||
91 if (gtk_expression_evaluate (expression2, NULL, &value))
|
||||
92 gtk_label_set_text (GTK_LABEL (label3), g_value_get_string (&value));
|
||||
93 else
|
||||
94 g_print ("The property expression couldn't be evaluated.\n");
|
||||
95 gtk_expression_unref (expression1); /* expression 2 is also freed. */
|
||||
96 g_value_unset (&value);
|
||||
97
|
||||
98 width = 800;
|
||||
99 height = 600;
|
||||
100 gtk_window_set_default_size (GTK_WINDOW (win1), width, height);
|
||||
101 notify(NULL);
|
||||
102
|
||||
103 /* GtkExpressionWatch */
|
||||
104 expression1 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-width");
|
||||
105 watch_width = gtk_expression_watch (expression1, win1, notify, NULL, NULL);
|
||||
106 expression2 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-height");
|
||||
107 watch_height = gtk_expression_watch (expression2, win1, notify, NULL, NULL);
|
||||
108
|
||||
109 gtk_widget_show (win1);
|
||||
110
|
||||
111 /* Builds a window with exp.ui resource */
|
||||
112 build = gtk_builder_new_from_resource ("/com/github/ToshioCP/exp/exp.ui");
|
||||
113 GtkWidget *win2 = GTK_WIDGET (gtk_builder_get_object (build, "win2"));
|
||||
114 gtk_window_set_application (GTK_WINDOW (win2), app);
|
||||
115 g_object_unref (build);
|
||||
116
|
||||
117 gtk_widget_show (win2);
|
||||
118 }
|
||||
119
|
||||
120 static void
|
||||
121 app_startup (GApplication *application) {
|
||||
122 }
|
||||
123
|
||||
124 #define APPLICATION_ID "com.github.ToshioCP.exp"
|
||||
125
|
||||
126 int
|
||||
127 main (int argc, char **argv) {
|
||||
128 GtkApplication *app;
|
||||
129 int stat;
|
||||
130
|
||||
131 app = gtk_application_new (APPLICATION_ID, G_APPLICATION_FLAGS_NONE);
|
||||
132
|
||||
133 g_signal_connect (app, "startup", G_CALLBACK (app_startup), NULL);
|
||||
134 g_signal_connect (app, "activate", G_CALLBACK (app_activate), NULL);
|
||||
135 /* g_signal_connect (app, "open", G_CALLBACK (app_open), NULL);*/
|
||||
136
|
||||
137 stat =g_application_run (G_APPLICATION (app), argc, argv);
|
||||
138 g_object_unref (app);
|
||||
139 return stat;
|
||||
140 }
|
||||
141
|
||||
~~~
|
||||
|
||||
`exp.c` consists of five functions.
|
||||
|
||||
- `notify`
|
||||
- `set_title`
|
||||
- `app_activate`. This is a handler of "activate" signal on GtkApplication instance.
|
||||
- `app_startup`. This is a handler of "startup"signal. But nothing is done in `exp.c`.
|
||||
- `main`. This function is called first.
|
||||
|
||||
The role of `main`, `app_startup` and `app_activate` is the same as before.
|
||||
`app_activate` is an actual main body in `exp.c`.
|
||||
|
||||
## Constant expression
|
||||
|
||||
Constant expression provides constant value or instance when it is evaluated.
|
||||
|
||||
- 73-81: An example code of a constant expression.
|
||||
It is extracted and put into here.
|
||||
|
||||
~~~C
|
||||
expression = gtk_constant_expression_new (G_TYPE_INT,100);
|
||||
if (gtk_expression_evaluate (expression, NULL, &value)) {
|
||||
s = g_strdup_printf ("%d", g_value_get_int (&value));
|
||||
gtk_label_set_text (GTK_LABEL (label1), s);
|
||||
g_free (s);
|
||||
} else
|
||||
g_print ("The constant expression couldn't be evaluated.\n");
|
||||
gtk_expression_unref (expression);
|
||||
g_value_unset (&value);
|
||||
~~~
|
||||
|
||||
- Constant expression is created with `gtk_constant_expression_new` function.
|
||||
The parameter of the function is a type (GType) and a value (or instance).
|
||||
- `gtk_expression_evaluate` evaluates the expression.
|
||||
It has three parameters, the expression to evaluate, `this` instance and GValue for being set with the value.
|
||||
`this` instance isn't necessary for constant expressions.
|
||||
Therefore the second argument is NULL.
|
||||
`gtk_expression_evaluate` returns TRUE if it successfully evaluates the expression.
|
||||
Otherwise it returns FALSE.
|
||||
- If it returns TRUE, the GValue `value` is set with the value of the expression.
|
||||
The type is int so it needs to be converted to a string.
|
||||
`g_strdup_printf` creates a new string `s`.
|
||||
- GtkLabel `label1` is set with `s`.
|
||||
- If the evaluation fails a message is displayed.
|
||||
- The expression and GValue are freed.
|
||||
|
||||
Constant expression is usually used to give a constant value or instance to another expression.
|
||||
|
||||
## Property expression
|
||||
|
||||
Property expression looks up a property in a GObject object.
|
||||
For example, a property expression that refers "label" property in GtkLabel object is created like this.
|
||||
|
||||
~~~C
|
||||
expression = gtk_property_expression_new (GTK_TYPE_LABEL, another_expression, "label");
|
||||
~~~
|
||||
|
||||
`another_expression` is expected to give a GtkLabel instance when it is evaluated.
|
||||
For example,
|
||||
|
||||
~~~C
|
||||
label = gtk_label_new ("Hello");
|
||||
another_expression = gtk_constant_expression_new (GTK_TYPE_LABEL, label);
|
||||
expression = gtk_property_expression_new (GTK_TYPE_LABEL, another_expression, "label");
|
||||
~~~
|
||||
|
||||
If `expression` is evaluated, the second parameter `another_expression` is evaluated in advance.
|
||||
Then the value of `another_expression` is `label` (GtkLabel instance).
|
||||
Then, `expression` looks up "label" property of `label` and the evaluation result is "Hello".
|
||||
|
||||
In the example above, the second argument of `gtk_property_expression_new` is another expression.
|
||||
But the second argument can be NULL.
|
||||
If it is NULL, `this` instance is used instead.
|
||||
`this` is given by `gtk_expression_evaluate` function at the evaluation.
|
||||
|
||||
Now look at `exp.c`.
|
||||
The lines from 84 to 86 is extracted here.
|
||||
|
||||
~~~C
|
||||
expression1 = gtk_property_expression_new (GTK_TYPE_ENTRY, NULL, "buffer");
|
||||
expression2 = gtk_property_expression_new (GTK_TYPE_ENTRY_BUFFER, expression1, "text");
|
||||
watch = gtk_expression_bind (expression2, label2, "label", entry);
|
||||
~~~
|
||||
|
||||
- `expression1` looks up "buffer" property of `this` object, which is `GTK_TYPE_ENTRY` type.
|
||||
- `expression2` looks up "text" property of GtkEntryBuffer object given by `epression1`.
|
||||
- `gtk_expression_bind` binds a property to a value given by the expression.
|
||||
In this program, it binds a "label" property in`label2` to the value evaluated with `expresion2` with `entry` as `this` object.
|
||||
The evaluation process is as follows.
|
||||
1. `expression2` is evaluated. But it includes `expression1` so `expression1` is evaluated in advance.
|
||||
2. Because the second argument of `expression1` is NULL, `this`object is used.
|
||||
`this` is given by `gtk_expression_bind`.
|
||||
It is `entry` (GtkEntry instance).
|
||||
`expression1` looks up "buffer" property in `entry`.
|
||||
It is a GtkEntryBuffer instance `buffer`.
|
||||
(See line 64 in `exp.c`.)
|
||||
3. Then, `expression2` looks up "text" property in `buffer`.
|
||||
It is a text held in `entry`.
|
||||
4. The text is assigned to "label" property in `label2`.
|
||||
- `gtk_expression_bind` creates a GtkExpressionWatch, which is assigned to `watch`.
|
||||
It watches `expression2`.
|
||||
And whenever the value from `expression2` changes, it evaluates `expression2` and set "label" property in `label2`.
|
||||
So, the change of the text in `entry` reflects "label" in `label2` immediately.
|
||||
|
||||
## closure expression
|
||||
|
||||
Closure expression calls closure when it is evaluated.
|
||||
A closure is a generic representation of a callback (a pointer to a function).
|
||||
For information about closure, see [GObject API reference](https://developer.gnome.org/gobject/stable/chapter-signal.html#closure).
|
||||
A closure expression is created with `gtk_cclosure_expression_new` function.
|
||||
|
||||
~~~C
|
||||
GtkExpression *
|
||||
gtk_cclosure_expression_new (GType value_type,
|
||||
GClosureMarshal marshal,
|
||||
guint n_params,
|
||||
GtkExpression **params,
|
||||
GCallback callback_func,
|
||||
gpointer user_data,
|
||||
GClosureNotify user_destroy);
|
||||
~~~
|
||||
|
||||
- `value_type` is the type of the value when it is evaluated.
|
||||
- `marshal` is a marshaller.
|
||||
You can assign NULL.
|
||||
If it is NULL, then `g_cclosure_marshal_generic ()` is used as a marshaller.
|
||||
It is a generic marshaller function implemented via libffi.
|
||||
- `n_params` is the number of parameters.
|
||||
- `params` points expressions for each parameter.
|
||||
- `callback_func` is a callback function.
|
||||
- `user_data` is user data.
|
||||
You can add it for the closure.
|
||||
It is like `user_data` in `g_signal_connect`.
|
||||
If it is not necessary, assign NULL.
|
||||
- `user_destroy` is a destroy notify for `user_data`.
|
||||
It is called to destroy `user_data` when it is no longer needed.
|
||||
If NULL is assigned to `user_data`, assign NULL to `user_destroy`, too.
|
||||
|
||||
The following is extracted from `exp.c`.
|
||||
It is from line 48 to line 57.
|
||||
|
||||
~~~C
|
||||
expression = gtk_cclosure_expression_new (GTK_TYPE_APPLICATION_WINDOW, NULL, 0, NULL,
|
||||
G_CALLBACK (gtk_application_window_new), NULL, NULL);
|
||||
if (gtk_expression_evaluate (expression, app, &value)) {
|
||||
win1 = GTK_WIDGET (g_value_get_object (&value)); /* GtkApplicationWindow */
|
||||
g_object_ref (win1);
|
||||
g_print ("Got GtkApplicationWindow object.\n");
|
||||
}else
|
||||
g_print ("The cclosure expression couldn't be evaluated.\n");
|
||||
gtk_expression_unref (expression);
|
||||
g_value_unset (&value); /* At the same time, the reference count of win1 is decreased by one. */
|
||||
~~~
|
||||
|
||||
The callback function is `gtk_application_window_new`.
|
||||
This function has one parameter which is an instance of GtkApplication.
|
||||
And it returns newly created GtkApplicationWindow instance.
|
||||
So, the first argument is `GTK_TYPE_APPLICATION_WINDOW` which is the type of the return value.
|
||||
The second argument is NULL so general marshaller `g_cclosure_marshal_generic ()` will be used.
|
||||
I think assigning NULL works in most cases when you are programming in C language.
|
||||
|
||||
The arguments given to the call back function are `this` object and parameters which are the fourth argument of `gtk_cclosure_expression_new`.
|
||||
So, the number of arguments is `n_params + 1`.
|
||||
Because `gtk_application_window_new` has one parameter, so `n_params` is zero and `**params` is NULL.
|
||||
No user data is necessary, so `user_data` and `user_destroy` are NULL.
|
||||
|
||||
`gtk_expression_evaluate` evaluates the expression.
|
||||
`this` instance will be the first argument for `gtk_application_window_new`, so it is `app`.
|
||||
|
||||
If the evaluation succeeds, the GValue `value` holds a newly created GtkApplicationWindow instance.
|
||||
It is assigned to `win1`.
|
||||
The GValue will be unset when it is no longer used.
|
||||
And when it is unset, the GtkApplicationWindow instance will be released and its reference count will be decreased by one.
|
||||
It is necessary to increase the reference count by one in advance to keep the instance.
|
||||
`gtk_expression_unref` frees `expression` and `value` is unset.
|
||||
|
||||
As a result, we got a GtkApplicationWindow instance `win1`.
|
||||
We can do the same by:
|
||||
|
||||
~~~C
|
||||
win1 = gtk_application_window_new (app);
|
||||
~~~
|
||||
|
||||
The example is more complicated and not practical than this one line code.
|
||||
The aim of the example is just to show how closure expression works.
|
||||
|
||||
Closure expression is flexible than other type of expression because you can specify your own callback function.
|
||||
|
||||
## GtkExpressionWatch
|
||||
|
||||
GtkExpressionWatch watches an expression and if the value of the expression changes it calls its notify handler.
|
||||
|
||||
The example uses GtkExpressionWatch in the line 104 to 107.
|
||||
|
||||
~~~C
|
||||
expression1 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-width");
|
||||
watch_width = gtk_expression_watch (expression1, win1, notify, NULL, NULL);
|
||||
expression2 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-height");
|
||||
watch_height = gtk_expression_watch (expression2, win1, notify, NULL, NULL);
|
||||
~~~
|
||||
|
||||
The expressions above refer to "default-width" and "default-height" properties of GtkWindow.
|
||||
`watch_width` watches `expression1`.
|
||||
The second argument `win1` is `this` instance for `expression1`.
|
||||
So, `watch_width` watches the value of "default-width" property of `win1`.
|
||||
If the value changes, it calls `notify` handler.
|
||||
The fourth and fifth arguments are NULL because no user data is necessary.
|
||||
|
||||
`watch_height` also connects `notify` handler to `expression2`.
|
||||
So, `notiry` is called when "default-width" or "default-height" changes.
|
||||
|
||||
The handler `norify` is as follows.
|
||||
|
||||
~~~C
|
||||
1 static void
|
||||
2 notify (gpointer user_data) {
|
||||
3 GValue value = G_VALUE_INIT;
|
||||
4 char *title;
|
||||
5
|
||||
6 if (watch_width && gtk_expression_watch_evaluate (watch_width, &value))
|
||||
7 width = g_value_get_int (&value);
|
||||
8 g_value_unset (&value);
|
||||
9 if (watch_height && gtk_expression_watch_evaluate (watch_height, &value))
|
||||
10 height = g_value_get_int (&value);
|
||||
11 g_value_unset (&value);
|
||||
12 title = g_strdup_printf ("%d x %d", width, height);
|
||||
13 gtk_window_set_title (GTK_WINDOW (win1), title);
|
||||
14 g_free (title);
|
||||
15 }
|
||||
~~~
|
||||
|
||||
- 6-11: Evaluates `expression1` and `expression2` with `expression_watch_evaluate` function.
|
||||
- 12: Creates a string `title`.
|
||||
It contains the width and height, for example, "800 x 600".
|
||||
- 13: Sets the title of `win1` with the string `title`.
|
||||
|
||||
The title of the window reflects the size of the window.
|
||||
|
||||
## exp.ui
|
||||
|
||||
`exp.c` builds a GtkWindow instance `win2` with `exp.ui`.
|
||||
The ui file `exp.ui` includes tags to create GtkExpressions.
|
||||
The tags are:
|
||||
|
||||
- constant tag to create constant expression
|
||||
- lookup tag to create property expression
|
||||
- closure tag to create closure expression
|
||||
- binding tag to bind a property to an expression
|
||||
|
||||
The window `win2` behaves like `win1`.
|
||||
Because similar expressions are built with the ui file.
|
||||
|
||||
~~~xml
|
||||
1 <?xml version="1.0" encoding="UTF-8"?>
|
||||
2 <interface>
|
||||
3 <object class="GtkWindow" id="win2">
|
||||
4 <binding name="title">
|
||||
5 <closure type="gchararray" function="set_title">
|
||||
6 <lookup name="default-width" type="GtkWindow"></lookup>
|
||||
7 <lookup name="default-height" type="GtkWindow"></lookup>
|
||||
8 </closure>
|
||||
9 </binding>
|
||||
10 <property name="default-width">600</property>
|
||||
11 <property name="default-height">400</property>
|
||||
12 <child>
|
||||
13 <object class="GtkBox">
|
||||
14 <property name="orientation">GTK_ORIENTATION_VERTICAL</property>
|
||||
15 <child>
|
||||
16 <object class="GtkLabel">
|
||||
17 <binding name="label">
|
||||
18 <constant type="gint">100</constant>
|
||||
19 </binding>
|
||||
20 </object>
|
||||
21 </child>
|
||||
22 <child>
|
||||
23 <object class="GtkLabel">
|
||||
24 <binding name="label">
|
||||
25 <lookup name="text">buffer</lookup>
|
||||
26 </binding>
|
||||
27 </object>
|
||||
28 </child>
|
||||
29 <child>
|
||||
30 <object class="GtkLabel">
|
||||
31 <binding name="label">
|
||||
32 <lookup name="application-id">
|
||||
33 <lookup name="application">win2</lookup>
|
||||
34 </lookup>
|
||||
35 </binding>
|
||||
36 </object>
|
||||
37 </child>
|
||||
38 <child>
|
||||
39 <object class="GtkEntry" id="entry">
|
||||
40 <property name="buffer">
|
||||
41 <object class="GtkEntryBuffer" id="buffer"></object>
|
||||
42 </property>
|
||||
43 </object>
|
||||
44 </child>
|
||||
45 </object>
|
||||
46 </child>
|
||||
47 </object>
|
||||
48 </interface>
|
||||
~~~
|
||||
|
||||
### constant tag
|
||||
|
||||
A constant tag corresponds to a constant expression.
|
||||
|
||||
- 18: Constant tag.
|
||||
The constant expression is created with the tag.
|
||||
It returns 100 of which the type is gint when it is evaluated.
|
||||
The type gint is the same as int.
|
||||
- 17-19: Binding tag corresponds to `gtk_expression_bind` function.
|
||||
`name` attribute specifies the "label" property of the GtkLabel object just before the binding tag.
|
||||
Binding tag uses the same GtkLabel object for a `this` object to evaluate the expression.
|
||||
(But the constant expression doesn't use the `this` object at all.)
|
||||
The expression returns a int type GValue.
|
||||
On the other hand "label" property holds a string type GValue.
|
||||
When a GValue is copied to another GValue, the type is automatically converted if possible.
|
||||
In this case, an int `100` is converted to a string `"100"`.
|
||||
|
||||
These binding and constant tag works.
|
||||
But they are not good.
|
||||
A property tag is more straightforward.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkLabel">
|
||||
<property name="label">100</property>
|
||||
</object>
|
||||
~~~
|
||||
|
||||
This example just shows the way how to use constant tag.
|
||||
Constant tag is mainly used to give a constant argument to a closure.
|
||||
|
||||
### lookup tag
|
||||
|
||||
A lookup tag corresponds to a property expression.
|
||||
Line 23 to 27 is copied here.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkLabel">
|
||||
<binding name="label">
|
||||
<lookup name="text">buffer</lookup>
|
||||
</binding>
|
||||
</object>
|
||||
~~~
|
||||
|
||||
- binding tag binds a "label" property in GtkLabel to an expression.
|
||||
The expression is defined with a lookup tag.
|
||||
- The lookup tag defines a property expression looks up a "text" property in `buffer` instance.
|
||||
The `buffer` instance is defined in the line 41.
|
||||
It is a GtkEntryBuffer belongs to a GtkEntry `entry`.
|
||||
A lookup tag takes an object in some ways to look up for a property.
|
||||
- If it has no contents, it takes `this` instance when it is evaluated.
|
||||
- If it has a content of a tag for an expression, which is constant, lookup or closure tag, the value of the expression will be the object to look up when it is evaluated.
|
||||
- If it has a content of an id of an object, then the instance of the object will be taken as the object to lookup.
|
||||
|
||||
As a result, the label of the GtkLabel instance are bound to the text in the field of GtkEntry.
|
||||
If a user input a text in the field, GtkLabel displays the same text.
|
||||
|
||||
Another lookup tag is in the lines from 30 to 36.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkLabel">
|
||||
<binding name="label">
|
||||
<lookup name="application-id">
|
||||
<lookup name="application">win2</lookup>
|
||||
</lookup>
|
||||
</binding>
|
||||
</object>
|
||||
~~~
|
||||
|
||||
- Two expressions are nested.
|
||||
- A lookup tag looks up "application-id" property of the next expression.
|
||||
- The next lookup tag looks up "application" property of `win2` instance.
|
||||
|
||||
As a result, the "label" property in the GtkLabel instance is bound to the "application-id" property.
|
||||
The nested tag makes a chain like:
|
||||
|
||||
~~~
|
||||
"label" <= "application-id" <= "application" <= `win2`
|
||||
~~~
|
||||
|
||||
### closure tag
|
||||
|
||||
The lines from 3 to 9 include a closure tag.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkWindow" id="win2">
|
||||
<binding name="title">
|
||||
<closure type="gchararray" function="set_title">
|
||||
<lookup name="default-width" type="GtkWindow"></lookup>
|
||||
<lookup name="default-height" type="GtkWindow"></lookup>
|
||||
</closure>
|
||||
</binding>
|
||||
~~~
|
||||
|
||||
- A binding tag corresponds to a `gtk_expression_bind` function.
|
||||
`name` attribute specifies the "title" property of `win2`.
|
||||
Binding tag gives `win2` as the `this` object to the expressions, which are the contents of the binding tag.
|
||||
So, closure tag and lookup tags use `win2` as the `this` object when they are evaluated.
|
||||
- A closure tag corresponds to a closure expression.
|
||||
Its callback function is `set_title` and it returns "gchararray" type, which is "an array of characters" i.e. a string.
|
||||
The contents of the closure tag are assigned to parameters of the function.
|
||||
So, `set_title` has three parameters, `win2` (`this` object), default width and default height.
|
||||
- Lookup tags correspond to property expressions.
|
||||
They lookup "default-width" and "default-height" properties of `win2` (`this` object).
|
||||
- Binding tab creates GtkExpressionWatch automatically, so "title" property reflects the changes of "default-width" and "default-height" properties.
|
||||
|
||||
`set_title` function in `exp.c` is as follows.
|
||||
|
||||
~~~C
|
||||
1 char *
|
||||
2 set_title (GtkWidget *win, int width, int height) {
|
||||
3 return g_strdup_printf ("%d x %d", width, height);
|
||||
4 }
|
||||
~~~
|
||||
|
||||
It just creates a string, for example, "800 x 600", and returns it.
|
||||
|
||||
You've probably been notice that ui file is easier and clearer than the corresponding function definitions.
|
||||
One of the most useful case of GtkExpression is building GtkListItem instance with GtkBuilderListItemFatory.
|
||||
Such case has already been described in the prior two sections.
|
||||
|
||||
It will be used in the next section to build GtkListItem in GtkColumnView, which is the most useful view object for GListModel.
|
||||
|
||||
## Compilation and execution
|
||||
|
||||
All the sources are in [src/expression](../src/expression) directory.
|
||||
Change your current directory to the directory above and run meson and ninja.
|
||||
Then, execute the application.
|
||||
|
||||
~~~
|
||||
$ meson _build
|
||||
$ ninja -C _build
|
||||
$ build/exp
|
||||
~~~
|
||||
|
||||
Then, two windows appear.
|
||||
|
||||
![Expression](../image/expression.png)
|
||||
|
||||
If you put some text in the field of the entry, then the same text appears in the second GtkLabel.
|
||||
Because the "label" property of the second GtkLabel instance is bound to the text in the GtkEntryBuffer.
|
||||
|
||||
If you resize the window, then the size appears in the title bar because the "title" property is bound to "default^width" and "default-height" properties.
|
||||
|
||||
|
||||
Up: [Readme.md](../Readme.md), Prev: [Section 25](sec25.md)
|
BIN
image/column.png
Normal file
BIN
image/column.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 60 KiB |
BIN
image/column.xcf
Normal file
BIN
image/column.xcf
Normal file
Binary file not shown.
BIN
image/column_view.png
Normal file
BIN
image/column_view.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 86 KiB |
BIN
image/expression.png
Normal file
BIN
image/expression.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 77 KiB |
141
src/expression/exp.c
Normal file
141
src/expression/exp.c
Normal file
|
@ -0,0 +1,141 @@
|
|||
#include <gtk/gtk.h>
|
||||
|
||||
GtkWidget *win1;
|
||||
int width, height;
|
||||
GtkExpressionWatch *watch_width;
|
||||
GtkExpressionWatch *watch_height;
|
||||
|
||||
/* Notify is called when "default-width" or "default-height" property is changed. */
|
||||
static void
|
||||
notify (gpointer user_data) {
|
||||
GValue value = G_VALUE_INIT;
|
||||
char *title;
|
||||
|
||||
if (watch_width && gtk_expression_watch_evaluate (watch_width, &value))
|
||||
width = g_value_get_int (&value);
|
||||
g_value_unset (&value);
|
||||
if (watch_height && gtk_expression_watch_evaluate (watch_height, &value))
|
||||
height = g_value_get_int (&value);
|
||||
g_value_unset (&value);
|
||||
title = g_strdup_printf ("%d x %d", width, height);
|
||||
gtk_window_set_title (GTK_WINDOW (win1), title);
|
||||
g_free (title);
|
||||
}
|
||||
|
||||
/* This function is used by closure tag in exp.ui. */
|
||||
char *
|
||||
set_title (GtkWidget *win, int width, int height) {
|
||||
return g_strdup_printf ("%d x %d", width, height);
|
||||
}
|
||||
|
||||
/* ----- activate, open, startup handlers ----- */
|
||||
static void
|
||||
app_activate (GApplication *application) {
|
||||
GtkApplication *app = GTK_APPLICATION (application);
|
||||
GtkWidget *box;
|
||||
GtkWidget *label1, *label2, *label3;
|
||||
GtkWidget *entry;
|
||||
GtkEntryBuffer *buffer;
|
||||
GtkBuilder *build;
|
||||
GtkExpression *expression, *expression1, *expression2;
|
||||
GtkExpressionWatch *watch;
|
||||
GValue value = G_VALUE_INIT;
|
||||
char *s;
|
||||
|
||||
/* Creates GtkApplicationWindow instance. */
|
||||
/* The codes below are complecated. It does the same as "win1 = gtk_application_window_new (app);". */
|
||||
/* The codes are written just to show how to use GtkExpression. */
|
||||
expression = gtk_cclosure_expression_new (GTK_TYPE_APPLICATION_WINDOW, NULL, 0, NULL,
|
||||
G_CALLBACK (gtk_application_window_new), NULL, NULL);
|
||||
if (gtk_expression_evaluate (expression, app, &value)) {
|
||||
win1 = GTK_WIDGET (g_value_get_object (&value)); /* GtkApplicationWindow */
|
||||
g_object_ref (win1);
|
||||
g_print ("Got GtkApplicationWindow object.\n");
|
||||
}else
|
||||
g_print ("The cclosure expression couldn't be evaluated.\n");
|
||||
gtk_expression_unref (expression);
|
||||
g_value_unset (&value); /* At the same time, the reference count of win1 is decreased by one. */
|
||||
|
||||
/* Builds a window with components */
|
||||
box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 10);
|
||||
label1 = gtk_label_new (NULL);
|
||||
label2 = gtk_label_new (NULL);
|
||||
label3 = gtk_label_new (NULL);
|
||||
buffer = gtk_entry_buffer_new (NULL, 0);
|
||||
entry = gtk_entry_new_with_buffer (buffer);
|
||||
gtk_box_append (GTK_BOX (box), label1);
|
||||
gtk_box_append (GTK_BOX (box), label2);
|
||||
gtk_box_append (GTK_BOX (box), label3);
|
||||
gtk_box_append (GTK_BOX (box), entry);
|
||||
gtk_window_set_child (GTK_WINDOW (win1), box);
|
||||
|
||||
/* Constant expression */
|
||||
expression = gtk_constant_expression_new (G_TYPE_INT,100);
|
||||
if (gtk_expression_evaluate (expression, NULL, &value)) {
|
||||
s = g_strdup_printf ("%d", g_value_get_int (&value));
|
||||
gtk_label_set_text (GTK_LABEL (label1), s);
|
||||
g_free (s);
|
||||
} else
|
||||
g_print ("The constant expression couldn't be evaluated.\n");
|
||||
gtk_expression_unref (expression);
|
||||
g_value_unset (&value);
|
||||
|
||||
/* Property expression and binding*/
|
||||
expression1 = gtk_property_expression_new (GTK_TYPE_ENTRY, NULL, "buffer");
|
||||
expression2 = gtk_property_expression_new (GTK_TYPE_ENTRY_BUFFER, expression1, "text");
|
||||
watch = gtk_expression_bind (expression2, label2, "label", entry);
|
||||
|
||||
/* Constant expression instead of "this" instance */
|
||||
expression1 = gtk_constant_expression_new (GTK_TYPE_APPLICATION, app);
|
||||
expression2 = gtk_property_expression_new (GTK_TYPE_APPLICATION, expression1, "application-id");
|
||||
if (gtk_expression_evaluate (expression2, NULL, &value))
|
||||
gtk_label_set_text (GTK_LABEL (label3), g_value_get_string (&value));
|
||||
else
|
||||
g_print ("The property expression couldn't be evaluated.\n");
|
||||
gtk_expression_unref (expression1); /* expression 2 is also freed. */
|
||||
g_value_unset (&value);
|
||||
|
||||
width = 800;
|
||||
height = 600;
|
||||
gtk_window_set_default_size (GTK_WINDOW (win1), width, height);
|
||||
notify(NULL);
|
||||
|
||||
/* GtkExpressionWatch */
|
||||
expression1 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-width");
|
||||
watch_width = gtk_expression_watch (expression1, win1, notify, NULL, NULL);
|
||||
expression2 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-height");
|
||||
watch_height = gtk_expression_watch (expression2, win1, notify, NULL, NULL);
|
||||
|
||||
gtk_widget_show (win1);
|
||||
|
||||
/* Builds a window with exp.ui resource */
|
||||
build = gtk_builder_new_from_resource ("/com/github/ToshioCP/exp/exp.ui");
|
||||
GtkWidget *win2 = GTK_WIDGET (gtk_builder_get_object (build, "win2"));
|
||||
gtk_window_set_application (GTK_WINDOW (win2), app);
|
||||
g_object_unref (build);
|
||||
|
||||
gtk_widget_show (win2);
|
||||
}
|
||||
|
||||
static void
|
||||
app_startup (GApplication *application) {
|
||||
}
|
||||
|
||||
#define APPLICATION_ID "com.github.ToshioCP.exp"
|
||||
|
||||
int
|
||||
main (int argc, char **argv) {
|
||||
GtkApplication *app;
|
||||
int stat;
|
||||
|
||||
app = gtk_application_new (APPLICATION_ID, G_APPLICATION_FLAGS_NONE);
|
||||
|
||||
g_signal_connect (app, "startup", G_CALLBACK (app_startup), NULL);
|
||||
g_signal_connect (app, "activate", G_CALLBACK (app_activate), NULL);
|
||||
/* g_signal_connect (app, "open", G_CALLBACK (app_open), NULL);*/
|
||||
|
||||
stat =g_application_run (G_APPLICATION (app), argc, argv);
|
||||
g_object_unref (app);
|
||||
return stat;
|
||||
}
|
||||
|
6
src/expression/exp.gresource.xml
Normal file
6
src/expression/exp.gresource.xml
Normal file
|
@ -0,0 +1,6 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<gresources>
|
||||
<gresource prefix="/com/github/ToshioCP/exp">
|
||||
<file>exp.ui</file>
|
||||
</gresource>
|
||||
</gresources>
|
48
src/expression/exp.ui
Normal file
48
src/expression/exp.ui
Normal file
|
@ -0,0 +1,48 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<interface>
|
||||
<object class="GtkWindow" id="win2">
|
||||
<binding name="title">
|
||||
<closure type="gchararray" function="set_title">
|
||||
<lookup name="default-width" type="GtkWindow"></lookup>
|
||||
<lookup name="default-height" type="GtkWindow"></lookup>
|
||||
</closure>
|
||||
</binding>
|
||||
<property name="default-width">600</property>
|
||||
<property name="default-height">400</property>
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">GTK_ORIENTATION_VERTICAL</property>
|
||||
<child>
|
||||
<object class="GtkLabel">
|
||||
<binding name="label">
|
||||
<constant type="gint">100</constant>
|
||||
</binding>
|
||||
</object>
|
||||
</child>
|
||||
<child>
|
||||
<object class="GtkLabel">
|
||||
<binding name="label">
|
||||
<lookup name="text">buffer</lookup>
|
||||
</binding>
|
||||
</object>
|
||||
</child>
|
||||
<child>
|
||||
<object class="GtkLabel">
|
||||
<binding name="label">
|
||||
<lookup name="application-id">
|
||||
<lookup name="application">win2</lookup>
|
||||
</lookup>
|
||||
</binding>
|
||||
</object>
|
||||
</child>
|
||||
<child>
|
||||
<object class="GtkEntry" id="entry">
|
||||
<property name="buffer">
|
||||
<object class="GtkEntryBuffer" id="buffer"></object>
|
||||
</property>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</interface>
|
11
src/expression/meson.build
Normal file
11
src/expression/meson.build
Normal file
|
@ -0,0 +1,11 @@
|
|||
project('exp', 'c')
|
||||
|
||||
gtkdep = dependency('gtk4')
|
||||
|
||||
gnome=import('gnome')
|
||||
resources = gnome.compile_resources('resources','exp.gresource.xml')
|
||||
|
||||
sourcefiles=files('exp.c')
|
||||
|
||||
executable('exp', sourcefiles, resources, dependencies: gtkdep, export_dynamic: true, install: false)
|
||||
|
412
src/sec26.src.md
Normal file
412
src/sec26.src.md
Normal file
|
@ -0,0 +1,412 @@
|
|||
# GtkExpression
|
||||
|
||||
GtkExpression is a fundamental type.
|
||||
It is not a descendant of GObject.
|
||||
GtkExpression provides a way to describe references to values.
|
||||
GtkExpression needs to be evaluated to obtain a value.
|
||||
|
||||
It is similar to arithmetic calculation.
|
||||
|
||||
~~~
|
||||
1 + 2 = 3
|
||||
~~~
|
||||
|
||||
`1+2` is an expression.
|
||||
It shows the way how to calculate.
|
||||
`3` is the value comes from the expression.
|
||||
Evaluation is to calculate the expression and get the value.
|
||||
|
||||
GtkExpression is a way to get a value.
|
||||
Evaluation is like a calculation.
|
||||
A value is got by evaluating the expression.
|
||||
|
||||
First, I want to show you the C file of the example for GtkExpression.
|
||||
Its name is `exp.c` and located under [src/expression](expression) directory.
|
||||
You don't need to understand the details now, just look at it.
|
||||
It will be explained in the next subsection.
|
||||
|
||||
@@@include
|
||||
expression/exp.c
|
||||
@@@
|
||||
|
||||
`exp.c` consists of five functions.
|
||||
|
||||
- `notify`
|
||||
- `set_title`
|
||||
- `app_activate`. This is a handler of "activate" signal on GtkApplication instance.
|
||||
- `app_startup`. This is a handler of "startup"signal. But nothing is done in `exp.c`.
|
||||
- `main`. This function is called first.
|
||||
|
||||
The role of `main`, `app_startup` and `app_activate` is the same as before.
|
||||
`app_activate` is an actual main body in `exp.c`.
|
||||
|
||||
## Constant expression
|
||||
|
||||
Constant expression provides constant value or instance when it is evaluated.
|
||||
|
||||
- 73-81: An example code of a constant expression.
|
||||
It is extracted and put into here.
|
||||
|
||||
~~~C
|
||||
expression = gtk_constant_expression_new (G_TYPE_INT,100);
|
||||
if (gtk_expression_evaluate (expression, NULL, &value)) {
|
||||
s = g_strdup_printf ("%d", g_value_get_int (&value));
|
||||
gtk_label_set_text (GTK_LABEL (label1), s);
|
||||
g_free (s);
|
||||
} else
|
||||
g_print ("The constant expression couldn't be evaluated.\n");
|
||||
gtk_expression_unref (expression);
|
||||
g_value_unset (&value);
|
||||
~~~
|
||||
|
||||
- Constant expression is created with `gtk_constant_expression_new` function.
|
||||
The parameter of the function is a type (GType) and a value (or instance).
|
||||
- `gtk_expression_evaluate` evaluates the expression.
|
||||
It has three parameters, the expression to evaluate, `this` instance and GValue for being set with the value.
|
||||
`this` instance isn't necessary for constant expressions.
|
||||
Therefore the second argument is NULL.
|
||||
`gtk_expression_evaluate` returns TRUE if it successfully evaluates the expression.
|
||||
Otherwise it returns FALSE.
|
||||
- If it returns TRUE, the GValue `value` is set with the value of the expression.
|
||||
The type is int so it needs to be converted to a string.
|
||||
`g_strdup_printf` creates a new string `s`.
|
||||
- GtkLabel `label1` is set with `s`.
|
||||
- If the evaluation fails a message is displayed.
|
||||
- The expression and GValue are freed.
|
||||
|
||||
Constant expression is usually used to give a constant value or instance to another expression.
|
||||
|
||||
## Property expression
|
||||
|
||||
Property expression looks up a property in a GObject object.
|
||||
For example, a property expression that refers "label" property in GtkLabel object is created like this.
|
||||
|
||||
~~~C
|
||||
expression = gtk_property_expression_new (GTK_TYPE_LABEL, another_expression, "label");
|
||||
~~~
|
||||
|
||||
`another_expression` is expected to give a GtkLabel instance when it is evaluated.
|
||||
For example,
|
||||
|
||||
~~~C
|
||||
label = gtk_label_new ("Hello");
|
||||
another_expression = gtk_constant_expression_new (GTK_TYPE_LABEL, label);
|
||||
expression = gtk_property_expression_new (GTK_TYPE_LABEL, another_expression, "label");
|
||||
~~~
|
||||
|
||||
If `expression` is evaluated, the second parameter `another_expression` is evaluated in advance.
|
||||
Then the value of `another_expression` is `label` (GtkLabel instance).
|
||||
Then, `expression` looks up "label" property of `label` and the evaluation result is "Hello".
|
||||
|
||||
In the example above, the second argument of `gtk_property_expression_new` is another expression.
|
||||
But the second argument can be NULL.
|
||||
If it is NULL, `this` instance is used instead.
|
||||
`this` is given by `gtk_expression_evaluate` function at the evaluation.
|
||||
|
||||
Now look at `exp.c`.
|
||||
The lines from 84 to 86 is extracted here.
|
||||
|
||||
~~~C
|
||||
expression1 = gtk_property_expression_new (GTK_TYPE_ENTRY, NULL, "buffer");
|
||||
expression2 = gtk_property_expression_new (GTK_TYPE_ENTRY_BUFFER, expression1, "text");
|
||||
watch = gtk_expression_bind (expression2, label2, "label", entry);
|
||||
~~~
|
||||
|
||||
- `expression1` looks up "buffer" property of `this` object, which is `GTK_TYPE_ENTRY` type.
|
||||
- `expression2` looks up "text" property of GtkEntryBuffer object given by `epression1`.
|
||||
- `gtk_expression_bind` binds a property to a value given by the expression.
|
||||
In this program, it binds a "label" property in`label2` to the value evaluated with `expresion2` with `entry` as `this` object.
|
||||
The evaluation process is as follows.
|
||||
1. `expression2` is evaluated. But it includes `expression1` so `expression1` is evaluated in advance.
|
||||
2. Because the second argument of `expression1` is NULL, `this`object is used.
|
||||
`this` is given by `gtk_expression_bind`.
|
||||
It is `entry` (GtkEntry instance).
|
||||
`expression1` looks up "buffer" property in `entry`.
|
||||
It is a GtkEntryBuffer instance `buffer`.
|
||||
(See line 64 in `exp.c`.)
|
||||
3. Then, `expression2` looks up "text" property in `buffer`.
|
||||
It is a text held in `entry`.
|
||||
4. The text is assigned to "label" property in `label2`.
|
||||
- `gtk_expression_bind` creates a GtkExpressionWatch, which is assigned to `watch`.
|
||||
It watches `expression2`.
|
||||
And whenever the value from `expression2` changes, it evaluates `expression2` and set "label" property in `label2`.
|
||||
So, the change of the text in `entry` reflects "label" in `label2` immediately.
|
||||
|
||||
## closure expression
|
||||
|
||||
Closure expression calls closure when it is evaluated.
|
||||
A closure is a generic representation of a callback (a pointer to a function).
|
||||
For information about closure, see [GObject API reference](https://developer.gnome.org/gobject/stable/chapter-signal.html#closure).
|
||||
A closure expression is created with `gtk_cclosure_expression_new` function.
|
||||
|
||||
~~~C
|
||||
GtkExpression *
|
||||
gtk_cclosure_expression_new (GType value_type,
|
||||
GClosureMarshal marshal,
|
||||
guint n_params,
|
||||
GtkExpression **params,
|
||||
GCallback callback_func,
|
||||
gpointer user_data,
|
||||
GClosureNotify user_destroy);
|
||||
~~~
|
||||
|
||||
- `value_type` is the type of the value when it is evaluated.
|
||||
- `marshal` is a marshaller.
|
||||
You can assign NULL.
|
||||
If it is NULL, then `g_cclosure_marshal_generic ()` is used as a marshaller.
|
||||
It is a generic marshaller function implemented via libffi.
|
||||
- `n_params` is the number of parameters.
|
||||
- `params` points expressions for each parameter.
|
||||
- `callback_func` is a callback function.
|
||||
- `user_data` is user data.
|
||||
You can add it for the closure.
|
||||
It is like `user_data` in `g_signal_connect`.
|
||||
If it is not necessary, assign NULL.
|
||||
- `user_destroy` is a destroy notify for `user_data`.
|
||||
It is called to destroy `user_data` when it is no longer needed.
|
||||
If NULL is assigned to `user_data`, assign NULL to `user_destroy`, too.
|
||||
|
||||
The following is extracted from `exp.c`.
|
||||
It is from line 48 to line 57.
|
||||
|
||||
~~~C
|
||||
expression = gtk_cclosure_expression_new (GTK_TYPE_APPLICATION_WINDOW, NULL, 0, NULL,
|
||||
G_CALLBACK (gtk_application_window_new), NULL, NULL);
|
||||
if (gtk_expression_evaluate (expression, app, &value)) {
|
||||
win1 = GTK_WIDGET (g_value_get_object (&value)); /* GtkApplicationWindow */
|
||||
g_object_ref (win1);
|
||||
g_print ("Got GtkApplicationWindow object.\n");
|
||||
}else
|
||||
g_print ("The cclosure expression couldn't be evaluated.\n");
|
||||
gtk_expression_unref (expression);
|
||||
g_value_unset (&value); /* At the same time, the reference count of win1 is decreased by one. */
|
||||
~~~
|
||||
|
||||
The callback function is `gtk_application_window_new`.
|
||||
This function has one parameter which is an instance of GtkApplication.
|
||||
And it returns newly created GtkApplicationWindow instance.
|
||||
So, the first argument is `GTK_TYPE_APPLICATION_WINDOW` which is the type of the return value.
|
||||
The second argument is NULL so general marshaller `g_cclosure_marshal_generic ()` will be used.
|
||||
I think assigning NULL works in most cases when you are programming in C language.
|
||||
|
||||
The arguments given to the call back function are `this` object and parameters which are the fourth argument of `gtk_cclosure_expression_new`.
|
||||
So, the number of arguments is `n_params + 1`.
|
||||
Because `gtk_application_window_new` has one parameter, so `n_params` is zero and `**params` is NULL.
|
||||
No user data is necessary, so `user_data` and `user_destroy` are NULL.
|
||||
|
||||
`gtk_expression_evaluate` evaluates the expression.
|
||||
`this` instance will be the first argument for `gtk_application_window_new`, so it is `app`.
|
||||
|
||||
If the evaluation succeeds, the GValue `value` holds a newly created GtkApplicationWindow instance.
|
||||
It is assigned to `win1`.
|
||||
The GValue will be unset when it is no longer used.
|
||||
And when it is unset, the GtkApplicationWindow instance will be released and its reference count will be decreased by one.
|
||||
It is necessary to increase the reference count by one in advance to keep the instance.
|
||||
`gtk_expression_unref` frees `expression` and `value` is unset.
|
||||
|
||||
As a result, we got a GtkApplicationWindow instance `win1`.
|
||||
We can do the same by:
|
||||
|
||||
~~~C
|
||||
win1 = gtk_application_window_new (app);
|
||||
~~~
|
||||
|
||||
The example is more complicated and not practical than this one line code.
|
||||
The aim of the example is just to show how closure expression works.
|
||||
|
||||
Closure expression is flexible than other type of expression because you can specify your own callback function.
|
||||
|
||||
## GtkExpressionWatch
|
||||
|
||||
GtkExpressionWatch watches an expression and if the value of the expression changes it calls its notify handler.
|
||||
|
||||
The example uses GtkExpressionWatch in the line 104 to 107.
|
||||
|
||||
~~~C
|
||||
expression1 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-width");
|
||||
watch_width = gtk_expression_watch (expression1, win1, notify, NULL, NULL);
|
||||
expression2 = gtk_property_expression_new (GTK_TYPE_WINDOW, NULL, "default-height");
|
||||
watch_height = gtk_expression_watch (expression2, win1, notify, NULL, NULL);
|
||||
~~~
|
||||
|
||||
The expressions above refer to "default-width" and "default-height" properties of GtkWindow.
|
||||
`watch_width` watches `expression1`.
|
||||
The second argument `win1` is `this` instance for `expression1`.
|
||||
So, `watch_width` watches the value of "default-width" property of `win1`.
|
||||
If the value changes, it calls `notify` handler.
|
||||
The fourth and fifth arguments are NULL because no user data is necessary.
|
||||
|
||||
`watch_height` also connects `notify` handler to `expression2`.
|
||||
So, `notiry` is called when "default-width" or "default-height" changes.
|
||||
|
||||
The handler `norify` is as follows.
|
||||
|
||||
@@@include
|
||||
expression/exp.c notify
|
||||
@@@
|
||||
|
||||
- 6-11: Evaluates `expression1` and `expression2` with `expression_watch_evaluate` function.
|
||||
- 12: Creates a string `title`.
|
||||
It contains the width and height, for example, "800 x 600".
|
||||
- 13: Sets the title of `win1` with the string `title`.
|
||||
|
||||
The title of the window reflects the size of the window.
|
||||
|
||||
## exp.ui
|
||||
|
||||
`exp.c` builds a GtkWindow instance `win2` with `exp.ui`.
|
||||
The ui file `exp.ui` includes tags to create GtkExpressions.
|
||||
The tags are:
|
||||
|
||||
- constant tag to create constant expression
|
||||
- lookup tag to create property expression
|
||||
- closure tag to create closure expression
|
||||
- binding tag to bind a property to an expression
|
||||
|
||||
The window `win2` behaves like `win1`.
|
||||
Because similar expressions are built with the ui file.
|
||||
|
||||
@@@include
|
||||
expression/exp.ui
|
||||
@@@
|
||||
|
||||
### constant tag
|
||||
|
||||
A constant tag corresponds to a constant expression.
|
||||
|
||||
- 18: Constant tag.
|
||||
The constant expression is created with the tag.
|
||||
It returns 100 of which the type is gint when it is evaluated.
|
||||
The type gint is the same as int.
|
||||
- 17-19: Binding tag corresponds to `gtk_expression_bind` function.
|
||||
`name` attribute specifies the "label" property of the GtkLabel object just before the binding tag.
|
||||
Binding tag uses the same GtkLabel object for a `this` object to evaluate the expression.
|
||||
(But the constant expression doesn't use the `this` object at all.)
|
||||
The expression returns a int type GValue.
|
||||
On the other hand "label" property holds a string type GValue.
|
||||
When a GValue is copied to another GValue, the type is automatically converted if possible.
|
||||
In this case, an int `100` is converted to a string `"100"`.
|
||||
|
||||
These binding and constant tag works.
|
||||
But they are not good.
|
||||
A property tag is more straightforward.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkLabel">
|
||||
<property name="label">100</property>
|
||||
</object>
|
||||
~~~
|
||||
|
||||
This example just shows the way how to use constant tag.
|
||||
Constant tag is mainly used to give a constant argument to a closure.
|
||||
|
||||
### lookup tag
|
||||
|
||||
A lookup tag corresponds to a property expression.
|
||||
Line 23 to 27 is copied here.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkLabel">
|
||||
<binding name="label">
|
||||
<lookup name="text">buffer</lookup>
|
||||
</binding>
|
||||
</object>
|
||||
~~~
|
||||
|
||||
- binding tag binds a "label" property in GtkLabel to an expression.
|
||||
The expression is defined with a lookup tag.
|
||||
- The lookup tag defines a property expression looks up a "text" property in `buffer` instance.
|
||||
The `buffer` instance is defined in the line 41.
|
||||
It is a GtkEntryBuffer belongs to a GtkEntry `entry`.
|
||||
A lookup tag takes an object in some ways to look up for a property.
|
||||
- If it has no contents, it takes `this` instance when it is evaluated.
|
||||
- If it has a content of a tag for an expression, which is constant, lookup or closure tag, the value of the expression will be the object to look up when it is evaluated.
|
||||
- If it has a content of an id of an object, then the instance of the object will be taken as the object to lookup.
|
||||
|
||||
As a result, the label of the GtkLabel instance are bound to the text in the field of GtkEntry.
|
||||
If a user input a text in the field, GtkLabel displays the same text.
|
||||
|
||||
Another lookup tag is in the lines from 30 to 36.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkLabel">
|
||||
<binding name="label">
|
||||
<lookup name="application-id">
|
||||
<lookup name="application">win2</lookup>
|
||||
</lookup>
|
||||
</binding>
|
||||
</object>
|
||||
~~~
|
||||
|
||||
- Two expressions are nested.
|
||||
- A lookup tag looks up "application-id" property of the next expression.
|
||||
- The next lookup tag looks up "application" property of `win2` instance.
|
||||
|
||||
As a result, the "label" property in the GtkLabel instance is bound to the "application-id" property.
|
||||
The nested tag makes a chain like:
|
||||
|
||||
~~~
|
||||
"label" <= "application-id" <= "application" <= `win2`
|
||||
~~~
|
||||
|
||||
### closure tag
|
||||
|
||||
The lines from 3 to 9 include a closure tag.
|
||||
|
||||
~~~xml
|
||||
<object class="GtkWindow" id="win2">
|
||||
<binding name="title">
|
||||
<closure type="gchararray" function="set_title">
|
||||
<lookup name="default-width" type="GtkWindow"></lookup>
|
||||
<lookup name="default-height" type="GtkWindow"></lookup>
|
||||
</closure>
|
||||
</binding>
|
||||
~~~
|
||||
|
||||
- A binding tag corresponds to a `gtk_expression_bind` function.
|
||||
`name` attribute specifies the "title" property of `win2`.
|
||||
Binding tag gives `win2` as the `this` object to the expressions, which are the contents of the binding tag.
|
||||
So, closure tag and lookup tags use `win2` as the `this` object when they are evaluated.
|
||||
- A closure tag corresponds to a closure expression.
|
||||
Its callback function is `set_title` and it returns "gchararray" type, which is "an array of characters" i.e. a string.
|
||||
The contents of the closure tag are assigned to parameters of the function.
|
||||
So, `set_title` has three parameters, `win2` (`this` object), default width and default height.
|
||||
- Lookup tags correspond to property expressions.
|
||||
They lookup "default-width" and "default-height" properties of `win2` (`this` object).
|
||||
- Binding tab creates GtkExpressionWatch automatically, so "title" property reflects the changes of "default-width" and "default-height" properties.
|
||||
|
||||
`set_title` function in `exp.c` is as follows.
|
||||
|
||||
@@@include
|
||||
expression/exp.c set_title
|
||||
@@@
|
||||
|
||||
It just creates a string, for example, "800 x 600", and returns it.
|
||||
|
||||
You've probably been notice that ui file is easier and clearer than the corresponding function definitions.
|
||||
One of the most useful case of GtkExpression is building GtkListItem instance with GtkBuilderListItemFatory.
|
||||
Such case has already been described in the prior two sections.
|
||||
|
||||
It will be used in the next section to build GtkListItem in GtkColumnView, which is the most useful view object for GListModel.
|
||||
|
||||
## Compilation and execution
|
||||
|
||||
All the sources are in [src/expression](expression) directory.
|
||||
Change your current directory to the directory above and run meson and ninja.
|
||||
Then, execute the application.
|
||||
|
||||
~~~
|
||||
$ meson _build
|
||||
$ ninja -C _build
|
||||
$ build/exp
|
||||
~~~
|
||||
|
||||
Then, two windows appear.
|
||||
|
||||
![Expression](../image/expression.png)
|
||||
|
||||
If you put some text in the field of the entry, then the same text appears in the second GtkLabel.
|
||||
Because the "label" property of the second GtkLabel instance is bound to the text in the GtkEntryBuffer.
|
||||
|
||||
If you resize the window, then the size appears in the title bar because the "title" property is bound to "default^width" and "default-height" properties.
|
||||
|
Loading…
Reference in a new issue