mirror of
https://github.com/ToshioCP/Gtk4-tutorial.git
synced 2025-01-12 20:03:28 +01:00
385 lines
16 KiB
Markdown
385 lines
16 KiB
Markdown
Up: [Readme.md](../Readme.md), Prev: [Section 11](sec11.md), Next: [Section 13](sec13.md)
|
|
|
|
# Functions in TfeTextView
|
|
|
|
In this section I will explain each function in TfeTextView object.
|
|
|
|
### tfe.h and tfetextview.h
|
|
|
|
`tfe.h` is a top header file and it includes `gtk.h` and all the header files.
|
|
Every C source files, which are `tfeapplication.c`, `tfenotebook.c` and `tfetextview.c`, include `tfe.h` at the beginning of each file.
|
|
|
|
~~~C
|
|
1 #include <gtk/gtk.h>
|
|
2
|
|
3 #include "tfetextview.h"
|
|
4 #include "tfenotebook.h"
|
|
~~~
|
|
|
|
`tfetextview.h` is a header file which describes the public functions in `tfetextview.c`.
|
|
|
|
~~~C
|
|
1 #define TFE_TYPE_TEXT_VIEW tfe_text_view_get_type ()
|
|
2 G_DECLARE_FINAL_TYPE (TfeTextView, tfe_text_view, TFE, TEXT_VIEW, GtkTextView)
|
|
3
|
|
4 /* "open-response" signal response */
|
|
5 enum
|
|
6 {
|
|
7 TFE_OPEN_RESPONSE_SUCCESS,
|
|
8 TFE_OPEN_RESPONSE_CANCEL,
|
|
9 TFE_OPEN_RESPONSE_ERROR
|
|
10 };
|
|
11
|
|
12 GFile *
|
|
13 tfe_text_view_get_file (TfeTextView *tv);
|
|
14
|
|
15 void
|
|
16 tfe_text_view_open (TfeTextView *tv, GtkWidget *win);
|
|
17
|
|
18 void
|
|
19 tfe_text_view_save (TfeTextView *tv);
|
|
20
|
|
21 void
|
|
22 tfe_text_view_saveas (TfeTextView *tv);
|
|
23
|
|
24 GtkWidget *
|
|
25 tfe_text_view_new_with_file (GFile *file);
|
|
26
|
|
27 GtkWidget *
|
|
28 tfe_text_view_new (void);
|
|
29
|
|
~~~
|
|
|
|
- 1-2: These two lines are used to define TfeTextView.
|
|
- 4-10: Definitions of parameter used in the handler of "open-response" signal.
|
|
- 12-28: Public functions on GtkTextView.
|
|
|
|
Each function will be explained later in this section.
|
|
|
|
## Functions to generate TfeTextView object
|
|
|
|
TfeTextView Object is generated by `tfe_text_view_new` or `tfe_text_view_new_with_file`.
|
|
|
|
~~~C
|
|
GtkWidget *tfe_text_view_new (void);
|
|
~~~
|
|
|
|
`tfe_text_view_new` just generates a new TfeTextView object and returns the pointer to the new object.
|
|
|
|
~~~C
|
|
GtkWidget *tfe_text_view_new_with_file (GFile *file);
|
|
~~~
|
|
|
|
`tfe_text_view_new_with_file` is given a Gfile object as the argument and it loads the file into the GtkTextBuffer object, then returns the pointer to the new object.
|
|
|
|
Parameter:
|
|
|
|
- `file`: a pointer to the GFile object.
|
|
|
|
Return value:
|
|
|
|
- A pointer to the generated TfeTextView object but it is casted to a pointer to GtkWidget.
|
|
If an error occurs during the generation process, NULL is returned.
|
|
|
|
Each function is defined as follows.
|
|
|
|
~~~C
|
|
1 GtkWidget *
|
|
2 tfe_text_view_new_with_file (GFile *file) {
|
|
3 g_return_val_if_fail (G_IS_FILE (file), NULL);
|
|
4
|
|
5 GtkWidget *tv;
|
|
6 GtkTextBuffer *tb;
|
|
7 char *contents;
|
|
8 gsize length;
|
|
9
|
|
10 if (! g_file_load_contents (file, NULL, &contents, &length, NULL, NULL)) /* read error */
|
|
11 return NULL;
|
|
12
|
|
13 tv = tfe_text_view_new();
|
|
14 tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
|
|
15 gtk_text_buffer_set_text (tb, contents, length);
|
|
16 g_free (contents);
|
|
17 TFE_TEXT_VIEW (tv)->file = g_file_dup (file);
|
|
18 return tv;
|
|
19 }
|
|
20
|
|
21 GtkWidget *
|
|
22 tfe_text_view_new (void) {
|
|
23 return GTK_WIDGET (g_object_new (TFE_TYPE_TEXT_VIEW, NULL));
|
|
24 }
|
|
~~~
|
|
|
|
- 21-24: `tfe_text_view_new`.
|
|
Just returns the value from the function `g_object_new` but casted to the pointer to GtkWidget.
|
|
Initialization is done in `tfe_text_view_init` which is called in the process of `gtk_widget_new` function.
|
|
- 1-19: `tfe_text_view_new_with_file`
|
|
- 3: `g_return_val_if_fail` is described in [Glib API reference](https://developer.gnome.org/glib/stable/glib-Warnings-and-Assertions.html#g-return-val-if-fail).
|
|
It tests whether the argument `file` is a pointer to GFile.
|
|
If it's true, then the program goes on to the next line.
|
|
If it's false, then it returns NULL (the second argument) immediately.
|
|
And at the same time it logs out the error message (usually the log is outputted to stderr or stdout).
|
|
This function is used to check the programmer's error.
|
|
If an error occurs, the solution is usually to change the (caller) program and fix the bug.
|
|
You need to distinguish programmer's errors and runtime errors.
|
|
You shouldn't use this function to find runtime errors.
|
|
- 10-11: If an error occurs when reading the file, then return NULL.
|
|
- 13-18: Generate TfeTextView and set the pointer to it to `tv`.
|
|
The pointer to GtkTextBuffer is set to `tb`
|
|
Set the contents read from the file to GtkTextBuffer `tb`.
|
|
Free the memories pointed by `contents`.
|
|
Duplicate `file` and set it to `tv->file`.
|
|
Return `tv`.
|
|
|
|
## Save and saveas functions
|
|
|
|
Save and saveas functions write the contents in GtkTextBuffer to a file.
|
|
|
|
~~~C
|
|
void tfe_text_view_save (TfeTextView *tv)
|
|
~~~
|
|
|
|
`save` function writes the contents in GtkTextBuffer to a file specified by `tv->file`.
|
|
If `tv->file` is NULL, then it shows GtkFileChooserDialog and lets the user to give a file to the program. After that, it saves the contents to the specified file and set the file into `tv->file`.
|
|
|
|
~~~C
|
|
void tfe_text_view_saveas (TfeTextView *tv)
|
|
~~~
|
|
|
|
`saveas` function uses GtkFileChooserDialog and lets the user to give a new file to the program. Then, the function changes `tv->file` and save the contents to the specified new file.
|
|
|
|
If an error occurs, it is shown to the user through the message dialog.
|
|
The error is managed only in the object and no information is notified to the caller.
|
|
|
|
~~~C
|
|
1 static void
|
|
2 saveas_dialog_response (GtkWidget *dialog, gint response, TfeTextView *tv) {
|
|
3 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
|
|
4 GFile *file;
|
|
5
|
|
6 if (response == GTK_RESPONSE_ACCEPT) {
|
|
7 file = gtk_file_chooser_get_file (GTK_FILE_CHOOSER (dialog));
|
|
8 if (G_IS_FILE(file)) {
|
|
9 tv->file = file;
|
|
10 gtk_text_buffer_set_modified (tb, TRUE);
|
|
11 g_signal_emit (tv, tfe_text_view_signals[CHANGE_FILE], 0);
|
|
12 tfe_text_view_save (TFE_TEXT_VIEW (tv));
|
|
13 }
|
|
14 }
|
|
15 gtk_window_destroy (GTK_WINDOW (dialog));
|
|
16 }
|
|
17
|
|
18 void
|
|
19 tfe_text_view_save (TfeTextView *tv) {
|
|
20 g_return_if_fail (TFE_IS_TEXT_VIEW (tv));
|
|
21
|
|
22 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
|
|
23 GtkTextIter start_iter;
|
|
24 GtkTextIter end_iter;
|
|
25 gchar *contents;
|
|
26 GtkWidget *message_dialog;
|
|
27 GtkWidget *win = gtk_widget_get_ancestor (GTK_WIDGET (tv), GTK_TYPE_WINDOW);
|
|
28 GError *err = NULL;
|
|
29
|
|
30 if (! gtk_text_buffer_get_modified (tb))
|
|
31 return; /* no necessary to save it */
|
|
32 else if (tv->file == NULL)
|
|
33 tfe_text_view_saveas (tv);
|
|
34 else {
|
|
35 gtk_text_buffer_get_bounds (tb, &start_iter, &end_iter);
|
|
36 contents = gtk_text_buffer_get_text (tb, &start_iter, &end_iter, FALSE);
|
|
37 if (g_file_replace_contents (tv->file, contents, strlen (contents), NULL, TRUE, G_FILE_CREATE_NONE, NULL, NULL, &err))
|
|
38 gtk_text_buffer_set_modified (tb, FALSE);
|
|
39 else {
|
|
40 /* It is possible that tv->file is broken. */
|
|
41 /* It is a good idea to set tv->file to NULL. */
|
|
42 if (G_IS_FILE (tv->file))
|
|
43 g_object_unref (tv->file);
|
|
44 tv->file =NULL;
|
|
45 g_signal_emit (tv, tfe_text_view_signals[CHANGE_FILE], 0);
|
|
46 gtk_text_buffer_set_modified (tb, TRUE);
|
|
47 message_dialog = gtk_message_dialog_new (GTK_WINDOW (win), GTK_DIALOG_MODAL,
|
|
48 GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE,
|
|
49 "%s.\n", err->message);
|
|
50 g_signal_connect (message_dialog, "response", G_CALLBACK (gtk_window_destroy), NULL);
|
|
51 gtk_widget_show (message_dialog);
|
|
52 g_error_free (err);
|
|
53 }
|
|
54 }
|
|
55 }
|
|
56
|
|
57 void
|
|
58 tfe_text_view_saveas (TfeTextView *tv) {
|
|
59 g_return_if_fail (TFE_IS_TEXT_VIEW (tv));
|
|
60
|
|
61 GtkWidget *dialog;
|
|
62 GtkWidget *win = gtk_widget_get_ancestor (GTK_WIDGET (tv), GTK_TYPE_WINDOW);
|
|
63
|
|
64 dialog = gtk_file_chooser_dialog_new ("Save file", GTK_WINDOW (win), GTK_FILE_CHOOSER_ACTION_SAVE,
|
|
65 "_Cancel", GTK_RESPONSE_CANCEL,
|
|
66 "_Save", GTK_RESPONSE_ACCEPT,
|
|
67 NULL);
|
|
68 g_signal_connect (dialog, "response", G_CALLBACK (saveas_dialog_response), tv);
|
|
69 gtk_widget_show (dialog);
|
|
70 }
|
|
~~~
|
|
|
|
- 18-55: `Tfe_text_view_save` function.
|
|
- 20: If `tv` is not a pointer to TfeTextView, then it logs an error message and immediately returns.
|
|
This function is similar to `g_return_val_if_fail` function, but no value is returned because `tfe_text_view_save` doesn't return a value.
|
|
- 30-31: If the buffer hasn't modified, then it doesn't need to save it.
|
|
So the function returns.
|
|
- 32-33: If `tv->file` is NULL, no file has given yet.
|
|
It calls `tfe_text_view_saveas`, which lets the user to choose a file to save.
|
|
- 35-36: Get the contents of the GtkTextBuffer and set its pointer to `contents`.
|
|
- 37-38: Save the content to the file.
|
|
If it succeeds, reset the modified bit in the GtkTextBuffer.
|
|
- 39-53: If file writing fails, it assigns NULL to `tv->file`.
|
|
Emits "change-file" signal.
|
|
Shows the error message dialog (47-51).
|
|
Because the handler is `gtk_window_destroy`, the dialog disappears when user clicks on the button in the dialog.
|
|
- 57-70: `tfe_text_view_saveas` function.
|
|
It shows GtkFileChooserDialog and lets the user choose a file and give it to the signal handler.
|
|
- 64-67: Generate GtkFileChooserDialog.
|
|
The title is "Save file".
|
|
Transient parent of the dialog is `win`, which is the top level window.
|
|
The action is save mode.
|
|
The buttons are Cancel and Save.
|
|
- 68: connect the "response" signal of the dialog and `saveas_dialog_response` handler.
|
|
- 1-16: `saveas_dialog_response` signal handler.
|
|
- 6-14: If the response is `GTK_RESPONSE_ACCEPT`, which is set to the argument when the user has clicked on Save button, then gets a pointer to the GFile object, set it to `tv->file`, turn on the modified bit of the GtkTextBuffer, emits "change-file" signal then call `tfe_text_view_save` to save the buffer to the file.
|
|
|
|
![Saveas process](../image/saveas.png)
|
|
|
|
When you use GtkFileChooserDialog, you need to divide the program into two parts.
|
|
They are a function which generates GtkFileChooserDialog and the signal handler.
|
|
The function just generates and shows the dialog.
|
|
The rest is done by the handler.
|
|
It gets Gfile from GtkFileChooserDialog, save the buffer to the file by calling `tfe_text_view_save`.
|
|
|
|
## Open function
|
|
|
|
Open function shows GtkFileChooserDialog to the user and let them choose a file.
|
|
Then read the file and set it to GtkTextBuffer.
|
|
|
|
~~~C
|
|
void tfe_text_view_open (TfeTextView *tv, GtkWidget *win);
|
|
~~~
|
|
|
|
TfeTextView object `tv` has to be generated in advance.
|
|
This function is usually called just after `tv` has been generated.
|
|
And its buffer is empty, `tv->file` is NULL and `tv` has not set to the widget hierarchy.
|
|
Even if the buffer is not empty, `tfe_text_view_open` doesn't treat it as an error.
|
|
If you want to revert the buffer, calling this function is apropreate.
|
|
Otherwise probably bad things will happen.
|
|
|
|
GtkWidget `win` is expected to be the top level window of the application.
|
|
It will be used as a transient parent window for the argument to the function `gtk_file_chooser_dialog_new`.
|
|
|
|
~~~C
|
|
1 static void
|
|
2 open_dialog_response(GtkWidget *dialog, gint response, TfeTextView *tv) {
|
|
3 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
|
|
4 GFile *file;
|
|
5 char *contents;
|
|
6 gsize length;
|
|
7 GtkWidget *message_dialog;
|
|
8 GError *err = NULL;
|
|
9
|
|
10 if (response != GTK_RESPONSE_ACCEPT)
|
|
11 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_CANCEL);
|
|
12 else if (! G_IS_FILE (file = gtk_file_chooser_get_file (GTK_FILE_CHOOSER (dialog))))
|
|
13 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_ERROR);
|
|
14 else if (! g_file_load_contents (file, NULL, &contents, &length, NULL, &err)) { /* read error */
|
|
15 if (G_IS_FILE (file))
|
|
16 g_object_unref (file);
|
|
17 message_dialog = gtk_message_dialog_new (GTK_WINDOW (dialog), GTK_DIALOG_MODAL,
|
|
18 GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE,
|
|
19 "%s.\n", err->message);
|
|
20 g_signal_connect (message_dialog, "response", G_CALLBACK (gtk_window_destroy), NULL);
|
|
21 gtk_widget_show (message_dialog);
|
|
22 g_error_free (err);
|
|
23 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_ERROR);
|
|
24 } else {
|
|
25 gtk_text_buffer_set_text (tb, contents, length);
|
|
26 g_free (contents);
|
|
27 if (G_IS_FILE (tv->file))
|
|
28 g_object_unref (tv->file);
|
|
29 tv->file = file;
|
|
30 gtk_text_buffer_set_modified (tb, FALSE);
|
|
31 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_SUCCESS);
|
|
32 }
|
|
33 gtk_window_destroy (GTK_WINDOW (dialog));
|
|
34 }
|
|
35
|
|
36 void
|
|
37 tfe_text_view_open (TfeTextView *tv, GtkWidget *win) {
|
|
38 g_return_if_fail (TFE_IS_TEXT_VIEW (tv));
|
|
39 g_return_if_fail (GTK_IS_WINDOW (win));
|
|
40
|
|
41 GtkWidget *dialog;
|
|
42
|
|
43 dialog = gtk_file_chooser_dialog_new ("Open file", GTK_WINDOW (win), GTK_FILE_CHOOSER_ACTION_OPEN,
|
|
44 "Cancel", GTK_RESPONSE_CANCEL,
|
|
45 "Open", GTK_RESPONSE_ACCEPT,
|
|
46 NULL);
|
|
47 g_signal_connect (dialog, "response", G_CALLBACK (open_dialog_response), tv);
|
|
48 gtk_widget_show (dialog);
|
|
49 }
|
|
~~~
|
|
|
|
- 36-49: `tfe_text_view_open` function.
|
|
- 43: Generate GtkFileChooserDialog.
|
|
The title is "Open file".
|
|
Ttransient parent window is the top window of the application, which is given by the caller.
|
|
The action is open mode.
|
|
The buttons are Cancel and Open.
|
|
- 47: connect the "reponse" signal of the dialog and `open_dialog_response` signal handler.
|
|
- 48: Show the dialog.
|
|
- 1-34: `open_dialog_response` signal handler.
|
|
- 10-11: If the response from GtkFileChooserDialog is not `GTK_RESPONSE_ACCEPT`, which means the user has clicked on the "Cancel" button or close button, then it emits "open-response" signal with the parameter `TFE_OPEN_RESPONSE_CANCEL`.
|
|
- 12-13: Get a pointer to Gfile by `gtk_file_chooser_get_file`.
|
|
If it is not GFile, maybe an error occured.
|
|
Then it emits "open-response" signal with the parameter `TFE_OPEN_RESPONSE_ERROR`.
|
|
- 14-23: If an error occurs when it read the file, then it decreases the reference count of Gfile, shows a message dialog to report the error to the user and emits "open-response" signal with the parameter `TFE_OPEN_RESPONSE_ERROR`.
|
|
- 24-32: If the file has successfully read, then the text is set to GtkTextBuffer, free the temporary buffer pointed by `contents`, set file to `tv->file` (no duplication or unref is not necessary) and emits "open-response" signal with the parameter `TFE_OPEN_RESPONSE_SUCCESS`.
|
|
- 33: close GtkFileCooserDialog.
|
|
|
|
Now let's think about the whole process between the other object (caller) and TfeTextView.
|
|
It is shown in the following diagram and you would think that it is really complicated.
|
|
Because signal is the only way for GtkFileChooserDialog to communicate with others.
|
|
In Gtk3, `gtk_dialog_run` function is available.
|
|
It simplifies the process.
|
|
However, in Gtk4, `gtk_dialog_run`is unavailable any more.
|
|
|
|
![Caller and TfeTextView](../image/open.png)
|
|
|
|
1. A caller get a pointer `tv` to TfeTextView by calling `tfe_text_view_new`.
|
|
2. The caller connects the handler (left bottom in the diagram) and the signal "open-response".
|
|
3. It calls `tfe_text_view_open` to let the user select a file from GtkFileChooserDialog.
|
|
4. The dialog emits a signal and it invokes the handler `open_dialog_response`.
|
|
5. The handler read the file and set it into GtkTextBuffer and emits a signal to inform the response status.
|
|
6. The handler outside TfeTextView recieves the signal.
|
|
|
|
## Get file function
|
|
|
|
`gtk_text_view_get_file` is a simple function show as follows.
|
|
|
|
~~~C
|
|
1 GFile *
|
|
2 tfe_text_view_get_file (TfeTextView *tv) {
|
|
3 g_return_val_if_fail (TFE_IS_TEXT_VIEW (tv), NULL);
|
|
4
|
|
5 return g_file_dup (tv->file);
|
|
6 }
|
|
~~~
|
|
|
|
The important thing is duplicate `tv->file`.
|
|
Otherwise, if the caller free the GFile object, `tv->file` is no more guaranteed to point the GFile.
|
|
|
|
## Source file of tfetextview.c
|
|
|
|
All the source files are listed in [Section 15](sec15.md).
|
|
|
|
|
|
Up: [Readme.md](../Readme.md), Prev: [Section 11](sec11.md), Next: [Section 13](sec13.md)
|