Up: [Readme.md](../Readme.md), Prev: [Section 14](sec14.md), Next: [Section 16](sec16.md) # tfe5 source files ## How to compile and execute tfe text editor. First, source files are shown in the later subsections. How to download them is written at the end of the [previous section](../src/sec14.src.md). The following is the instruction of compilation and execution. - You need meson and ninja. - Set necessary environment variables. If you have installed gtk4 under the instruction in [Section 2](sec2.md), type `. env.sh` to set the environment variables. - change your current directory to `src/tfe5` directory. - type `meson _build` for configuration. - type `ninja -C _build` for compilation. Then the application `tfe` is build under the `_build` directory. - type `_build/tfe` to execute it. Then the window appears. There are four buttons, `New`, `Open`, `Save` and `Close`. - Click on `Open` button, then a FileChooserDialog appears. Choose a file in the list and click on `Open` button. Then the file is read and a new Notebook Page appears. - Edit the file and click on `Save` button, then the text is saved to the original file. - Click `Close`, then the Notebook Page disappears. - Click `Close` again, then the `Untitle` Notebook Page disappears and at the same time the appication quits. This is a very simple editor. It is a good practice for you to add more features. ## meson.buld ~~~meson 1 project('tfe', 'c') 2 3 gtkdep = dependency('gtk4') 4 5 gnome=import('gnome') 6 resources = gnome.compile_resources('resources','tfe.gresource.xml') 7 8 sourcefiles=files('tfeapplication.c', 'tfenotebook.c', 'tfetextview.c') 9 10 executable('tfe', sourcefiles, resources, dependencies: gtkdep) ~~~ ## tfe.gresource.xml ~~~xml 1 2 3 4 tfe.ui 5 6 ~~~ ## tfe.ui ~~~xml 1 2 3 file editor 4 600 5 400 6 7 8 GTK_ORIENTATION_VERTICAL 9 10 11 GTK_ORIENTATION_HORIZONTAL 12 13 14 10 15 16 17 18 19 _New 20 TRUE 21 22 23 24 25 _Open 26 TRUE 27 28 29 30 31 TRUE 32 33 34 35 36 _Save 37 TRUE 38 39 40 41 42 _Close 43 TRUE 44 45 46 47 48 10 49 50 51 52 53 54 55 TRUE 56 TRUE 57 TRUE 58 59 60 61 62 63 64 ~~~ ## tfe.h ~~~C 1 #include 2 3 #include "tfetextview.h" 4 #include "tfenotebook.h" ~~~ ## tfeapplication.c ~~~C 1 #include "tfe.h" 2 3 static void 4 open_clicked (GtkWidget *btno, GtkNotebook *nb) { 5 notebook_page_open (nb); 6 } 7 8 static void 9 new_clicked (GtkWidget *btnn, GtkNotebook *nb) { 10 notebook_page_new (nb); 11 } 12 13 static void 14 save_clicked (GtkWidget *btns, GtkNotebook *nb) { 15 notebook_page_save (nb); 16 } 17 18 static void 19 close_clicked (GtkWidget *btnc, GtkNotebook *nb) { 20 GtkWidget *win; 21 GtkWidget *boxv; 22 gint i; 23 24 if (gtk_notebook_get_n_pages (nb) == 1) { 25 boxv = gtk_widget_get_parent (GTK_WIDGET (nb)); 26 win = gtk_widget_get_parent (boxv); 27 gtk_window_destroy (GTK_WINDOW (win)); 28 } else { 29 i = gtk_notebook_get_current_page (nb); 30 gtk_notebook_remove_page (GTK_NOTEBOOK (nb), i); 31 } 32 } 33 34 static void 35 tfe_activate (GApplication *application) { 36 GtkApplication *app = GTK_APPLICATION (application); 37 GtkWidget *win; 38 GtkWidget *boxv; 39 GtkNotebook *nb; 40 41 win = GTK_WIDGET (gtk_application_get_active_window (app)); 42 boxv = gtk_window_get_child (GTK_WINDOW (win)); 43 nb = GTK_NOTEBOOK (gtk_widget_get_last_child (boxv)); 44 45 notebook_page_new (nb); 46 gtk_widget_show (GTK_WIDGET (win)); 47 } 48 49 static void 50 tfe_open (GApplication *application, GFile ** files, gint n_files, const gchar *hint) { 51 GtkApplication *app = GTK_APPLICATION (application); 52 GtkWidget *win; 53 GtkWidget *boxv; 54 GtkNotebook *nb; 55 int i; 56 57 win = GTK_WIDGET (gtk_application_get_active_window (app)); 58 boxv = gtk_window_get_child (GTK_WINDOW (win)); 59 nb = GTK_NOTEBOOK (gtk_widget_get_last_child (boxv)); 60 61 for (i = 0; i < n_files; i++) 62 notebook_page_new_with_file (nb, files[i]); 63 if (gtk_notebook_get_n_pages (nb) == 0) 64 notebook_page_new (nb); 65 gtk_widget_show (win); 66 } 67 68 69 static void 70 tfe_startup (GApplication *application) { 71 GtkApplication *app = GTK_APPLICATION (application); 72 GtkApplicationWindow *win; 73 GtkNotebook *nb; 74 GtkBuilder *build; 75 GtkButton *btno; 76 GtkButton *btnn; 77 GtkButton *btns; 78 GtkButton *btnc; 79 80 build = gtk_builder_new_from_resource ("/com/github/ToshioCP/tfe/tfe.ui"); 81 win = GTK_APPLICATION_WINDOW (gtk_builder_get_object (build, "win")); 82 nb = GTK_NOTEBOOK (gtk_builder_get_object (build, "nb")); 83 gtk_window_set_application (GTK_WINDOW (win), app); 84 btno = GTK_BUTTON (gtk_builder_get_object (build, "btno")); 85 btnn = GTK_BUTTON (gtk_builder_get_object (build, "btnn")); 86 btns = GTK_BUTTON (gtk_builder_get_object (build, "btns")); 87 btnc = GTK_BUTTON (gtk_builder_get_object (build, "btnc")); 88 g_signal_connect (btno, "clicked", G_CALLBACK (open_clicked), nb); 89 g_signal_connect (btnn, "clicked", G_CALLBACK (new_clicked), nb); 90 g_signal_connect (btns, "clicked", G_CALLBACK (save_clicked), nb); 91 g_signal_connect (btnc, "clicked", G_CALLBACK (close_clicked), nb); 92 g_object_unref(build); 93 94 GdkDisplay *display; 95 96 display = gtk_widget_get_display (GTK_WIDGET (win)); 97 GtkCssProvider *provider = gtk_css_provider_new (); 98 gtk_css_provider_load_from_data (provider, "textview {padding: 10px; font-family: monospace; font-size: 12pt;}", -1); 99 gtk_style_context_add_provider_for_display (display, GTK_STYLE_PROVIDER (provider), GTK_STYLE_PROVIDER_PRIORITY_USER); 100 } 101 102 int 103 main (int argc, char **argv) { 104 GtkApplication *app; 105 int stat; 106 107 app = gtk_application_new ("com.github.ToshioCP.tfe", G_APPLICATION_HANDLES_OPEN); 108 109 g_signal_connect (app, "startup", G_CALLBACK (tfe_startup), NULL); 110 g_signal_connect (app, "activate", G_CALLBACK (tfe_activate), NULL); 111 g_signal_connect (app, "open", G_CALLBACK (tfe_open), NULL); 112 113 stat =g_application_run (G_APPLICATION (app), argc, argv); 114 g_object_unref (app); 115 return stat; 116 } 117 ~~~ ## tfenotebook.h ~~~C 1 void 2 notebook_page_save(GtkNotebook *nb); 3 4 void 5 notebook_page_open (GtkNotebook *nb); 6 7 void 8 notebook_page_new_with_file (GtkNotebook *nb, GFile *file); 9 10 void 11 notebook_page_new (GtkNotebook *nb); 12 ~~~ ## tfenotebook.c ~~~C 1 #include "tfe.h" 2 3 /* The returned string should be freed with g_free() when no longer needed. */ 4 static gchar* 5 get_untitled () { 6 static int c = -1; 7 if (++c == 0) 8 return g_strdup_printf("Untitled"); 9 else 10 return g_strdup_printf ("Untitled%u", c); 11 } 12 13 static void 14 file_changed (TfeTextView *tv, GtkNotebook *nb) { 15 GFile *file; 16 char *filename; 17 GtkWidget *scr; 18 GtkWidget *label; 19 20 file = tfe_text_view_get_file (tv); 21 scr = gtk_widget_get_parent (GTK_WIDGET (tv)); 22 if (G_IS_FILE (file)) 23 filename = g_file_get_basename (file); 24 else 25 filename = get_untitled (); 26 label = gtk_label_new (filename); 27 gtk_notebook_set_tab_label (nb, scr, label); 28 g_object_unref (file); 29 g_free (filename); 30 } 31 32 /* Save the contents in the current page */ 33 void 34 notebook_page_save(GtkNotebook *nb) { 35 gint i; 36 GtkWidget *scr; 37 GtkWidget *tv; 38 39 i = gtk_notebook_get_current_page (nb); 40 scr = gtk_notebook_get_nth_page (nb, i); 41 tv = gtk_scrolled_window_get_child (GTK_SCROLLED_WINDOW (scr)); 42 tfe_text_view_save (TFE_TEXT_VIEW (tv)); 43 } 44 45 static void 46 notebook_page_build (GtkNotebook *nb, GtkWidget *tv, char *filename) { 47 GtkWidget *scr; 48 GtkNotebookPage *nbp; 49 GtkWidget *lab; 50 gint i; 51 scr = gtk_scrolled_window_new (); 52 53 gtk_scrolled_window_set_child (GTK_SCROLLED_WINDOW (scr), tv); 54 lab = gtk_label_new (filename); 55 i = gtk_notebook_append_page (nb, scr, lab); 56 nbp = gtk_notebook_get_page (nb, scr); 57 g_object_set (nbp, "tab-expand", TRUE, NULL); 58 gtk_notebook_set_current_page (nb, i); 59 g_signal_connect (GTK_TEXT_VIEW (tv), "change-file", G_CALLBACK (file_changed), nb); 60 } 61 62 static void 63 open_response (TfeTextView *tv, gint response, GtkNotebook *nb) { 64 GFile *file; 65 char *filename; 66 67 if (response != TFE_OPEN_RESPONSE_SUCCESS) { 68 g_object_ref_sink (tv); 69 g_object_unref (tv); 70 }else if (! G_IS_FILE (file = tfe_text_view_get_file (tv))) { 71 g_object_ref_sink (tv); 72 g_object_unref (tv); 73 }else { 74 filename = g_file_get_basename (file); 75 g_object_unref (file); 76 notebook_page_build (nb, GTK_WIDGET (tv), filename); 77 } 78 } 79 80 void 81 notebook_page_open (GtkNotebook *nb) { 82 g_return_if_fail(GTK_IS_NOTEBOOK (nb)); 83 84 GtkWidget *tv; 85 86 tv = tfe_text_view_new (); 87 g_signal_connect (TFE_TEXT_VIEW (tv), "open-response", G_CALLBACK (open_response), nb); 88 tfe_text_view_open (TFE_TEXT_VIEW (tv), gtk_widget_get_ancestor (GTK_WIDGET (nb), GTK_TYPE_WINDOW)); 89 } 90 91 void 92 notebook_page_new_with_file (GtkNotebook *nb, GFile *file) { 93 g_return_if_fail(GTK_IS_NOTEBOOK (nb)); 94 g_return_if_fail(G_IS_FILE (file)); 95 96 GtkWidget *tv; 97 char *filename; 98 99 if ((tv = tfe_text_view_new_with_file (file)) == NULL) 100 return; /* read error */ 101 filename = g_file_get_basename (file); 102 notebook_page_build (nb, tv, filename); 103 } 104 105 void 106 notebook_page_new (GtkNotebook *nb) { 107 g_return_if_fail(GTK_IS_NOTEBOOK (nb)); 108 109 GtkWidget *tv; 110 char *filename; 111 112 tv = tfe_text_view_new (); 113 filename = get_untitled (); 114 notebook_page_build (nb, tv, filename); 115 } 116 ~~~ ## tfetextview.h ~~~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 ~~~ ## tfetextview.c ~~~C 1 #include "tfe.h" 2 3 struct _TfeTextView 4 { 5 GtkTextView parent; 6 GFile *file; 7 }; 8 9 G_DEFINE_TYPE (TfeTextView, tfe_text_view, GTK_TYPE_TEXT_VIEW); 10 11 enum { 12 CHANGE_FILE, 13 OPEN_RESPONSE, 14 NUMBER_OF_SIGNALS 15 }; 16 17 static guint tfe_text_view_signals[NUMBER_OF_SIGNALS]; 18 19 static void 20 tfe_text_view_dispose (GObject *gobject) { 21 TfeTextView *tv = TFE_TEXT_VIEW (gobject); 22 23 if (G_IS_FILE (tv->file)) 24 g_clear_object (&tv->file); 25 26 G_OBJECT_CLASS (tfe_text_view_parent_class)->dispose (gobject); 27 } 28 29 static void 30 tfe_text_view_init (TfeTextView *tv) { 31 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv)); 32 33 tv->file = NULL; 34 gtk_text_buffer_set_modified (tb, FALSE); 35 gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (tv), GTK_WRAP_WORD_CHAR); 36 } 37 38 static void 39 tfe_text_view_class_init (TfeTextViewClass *class) { 40 GObjectClass *object_class = G_OBJECT_CLASS (class); 41 42 object_class->dispose = tfe_text_view_dispose; 43 tfe_text_view_signals[CHANGE_FILE] = g_signal_newv ("change-file", 44 G_TYPE_FROM_CLASS (class), 45 G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 46 NULL /* closure */, 47 NULL /* accumulator */, 48 NULL /* accumulator data */, 49 NULL /* C marshaller */, 50 G_TYPE_NONE /* return_type */, 51 0 /* n_params */, 52 NULL /* param_types */); 53 GType param_types[] = {G_TYPE_INT}; 54 tfe_text_view_signals[OPEN_RESPONSE] = g_signal_newv ("open-response", 55 G_TYPE_FROM_CLASS (class), 56 G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS, 57 NULL /* closure */, 58 NULL /* accumulator */, 59 NULL /* accumulator data */, 60 NULL /* C marshaller */, 61 G_TYPE_NONE /* return_type */, 62 1 /* n_params */, 63 param_types); 64 } 65 66 GFile * 67 tfe_text_view_get_file (TfeTextView *tv) { 68 g_return_val_if_fail (TFE_IS_TEXT_VIEW (tv), NULL); 69 70 return g_file_dup (tv->file); 71 } 72 73 static void 74 open_dialog_response(GtkWidget *dialog, gint response, TfeTextView *tv) { 75 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv)); 76 GFile *file; 77 char *contents; 78 gsize length; 79 GtkWidget *message_dialog; 80 GError *err = NULL; 81 82 if (response != GTK_RESPONSE_ACCEPT) 83 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_CANCEL); 84 else if (! G_IS_FILE (file = gtk_file_chooser_get_file (GTK_FILE_CHOOSER (dialog)))) 85 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_ERROR); 86 else if (! g_file_load_contents (file, NULL, &contents, &length, NULL, &err)) { /* read error */ 87 if (G_IS_FILE (file)) 88 g_object_unref (file); 89 message_dialog = gtk_message_dialog_new (GTK_WINDOW (dialog), GTK_DIALOG_MODAL, 90 GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, 91 "%s.\n", err->message); 92 g_signal_connect (message_dialog, "response", G_CALLBACK (gtk_window_destroy), NULL); 93 gtk_widget_show (message_dialog); 94 g_error_free (err); 95 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_ERROR); 96 } else { 97 gtk_text_buffer_set_text (tb, contents, length); 98 g_free (contents); 99 if (G_IS_FILE (tv->file)) 100 g_object_unref (tv->file); 101 tv->file = file; 102 gtk_text_buffer_set_modified (tb, FALSE); 103 g_signal_emit (tv, tfe_text_view_signals[OPEN_RESPONSE], 0, TFE_OPEN_RESPONSE_SUCCESS); 104 } 105 gtk_window_destroy (GTK_WINDOW (dialog)); 106 } 107 108 void 109 tfe_text_view_open (TfeTextView *tv, GtkWidget *win) { 110 g_return_if_fail (TFE_IS_TEXT_VIEW (tv)); 111 g_return_if_fail (GTK_IS_WINDOW (win)); 112 113 GtkWidget *dialog; 114 115 dialog = gtk_file_chooser_dialog_new ("Open file", GTK_WINDOW (win), GTK_FILE_CHOOSER_ACTION_OPEN, 116 "Cancel", GTK_RESPONSE_CANCEL, 117 "Open", GTK_RESPONSE_ACCEPT, 118 NULL); 119 g_signal_connect (dialog, "response", G_CALLBACK (open_dialog_response), tv); 120 gtk_widget_show (dialog); 121 } 122 123 static void 124 saveas_dialog_response (GtkWidget *dialog, gint response, TfeTextView *tv) { 125 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv)); 126 GFile *file; 127 128 if (response == GTK_RESPONSE_ACCEPT) { 129 file = gtk_file_chooser_get_file (GTK_FILE_CHOOSER (dialog)); 130 if (G_IS_FILE(file)) { 131 tv->file = file; 132 gtk_text_buffer_set_modified (tb, TRUE); 133 g_signal_emit (tv, tfe_text_view_signals[CHANGE_FILE], 0); 134 tfe_text_view_save (TFE_TEXT_VIEW (tv)); 135 } 136 } 137 gtk_window_destroy (GTK_WINDOW (dialog)); 138 } 139 140 void 141 tfe_text_view_save (TfeTextView *tv) { 142 g_return_if_fail (TFE_IS_TEXT_VIEW (tv)); 143 144 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv)); 145 GtkTextIter start_iter; 146 GtkTextIter end_iter; 147 gchar *contents; 148 GtkWidget *message_dialog; 149 GtkWidget *win = gtk_widget_get_ancestor (GTK_WIDGET (tv), GTK_TYPE_WINDOW); 150 GError *err = NULL; 151 152 if (! gtk_text_buffer_get_modified (tb)) 153 return; /* no necessary to save it */ 154 else if (tv->file == NULL) 155 tfe_text_view_saveas (tv); 156 else { 157 gtk_text_buffer_get_bounds (tb, &start_iter, &end_iter); 158 contents = gtk_text_buffer_get_text (tb, &start_iter, &end_iter, FALSE); 159 if (g_file_replace_contents (tv->file, contents, strlen (contents), NULL, TRUE, G_FILE_CREATE_NONE, NULL, NULL, &err)) 160 gtk_text_buffer_set_modified (tb, FALSE); 161 else { 162 /* It is possible that tv->file is broken. */ 163 /* It is a good idea to set tv->file to NULL. */ 164 if (G_IS_FILE (tv->file)) 165 g_object_unref (tv->file); 166 tv->file =NULL; 167 g_signal_emit (tv, tfe_text_view_signals[CHANGE_FILE], 0); 168 gtk_text_buffer_set_modified (tb, TRUE); 169 message_dialog = gtk_message_dialog_new (GTK_WINDOW (win), GTK_DIALOG_MODAL, 170 GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, 171 "%s.\n", err->message); 172 g_signal_connect (message_dialog, "response", G_CALLBACK (gtk_window_destroy), NULL); 173 gtk_widget_show (message_dialog); 174 g_error_free (err); 175 } 176 } 177 } 178 179 void 180 tfe_text_view_saveas (TfeTextView *tv) { 181 g_return_if_fail (TFE_IS_TEXT_VIEW (tv)); 182 183 GtkWidget *dialog; 184 GtkWidget *win = gtk_widget_get_ancestor (GTK_WIDGET (tv), GTK_TYPE_WINDOW); 185 186 dialog = gtk_file_chooser_dialog_new ("Save file", GTK_WINDOW (win), GTK_FILE_CHOOSER_ACTION_SAVE, 187 "_Cancel", GTK_RESPONSE_CANCEL, 188 "_Save", GTK_RESPONSE_ACCEPT, 189 NULL); 190 g_signal_connect (dialog, "response", G_CALLBACK (saveas_dialog_response), tv); 191 gtk_widget_show (dialog); 192 } 193 194 GtkWidget * 195 tfe_text_view_new_with_file (GFile *file) { 196 g_return_val_if_fail (G_IS_FILE (file), NULL); 197 198 GtkWidget *tv; 199 GtkTextBuffer *tb; 200 char *contents; 201 gsize length; 202 203 if (! g_file_load_contents (file, NULL, &contents, &length, NULL, NULL)) /* read error */ 204 return NULL; 205 206 tv = tfe_text_view_new(); 207 tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv)); 208 gtk_text_buffer_set_text (tb, contents, length); 209 g_free (contents); 210 TFE_TEXT_VIEW (tv)->file = g_file_dup (file); 211 return tv; 212 } 213 214 GtkWidget * 215 tfe_text_view_new (void) { 216 return GTK_WIDGET (g_object_new (TFE_TYPE_TEXT_VIEW, NULL)); 217 } 218 ~~~ ## Total number of lines, words and charcters $ LANG=C wc tfe5/meson.build tfe5/tfeapplication.c tfe5/tfe.gresource.xml tfe5/tfe.h tfe5/tfenotebook.c tfe5/tfenotebook.h tfe5/tfetextview.c tfe5/tfetextview.h tfe5/tfe.ui 10 17 279 tfe5/meson.build 117 348 3576 tfe5/tfeapplication.c 6 9 153 tfe5/tfe.gresource.xml 4 6 72 tfe5/tfe.h 116 321 2992 tfe5/tfenotebook.c 12 17 196 tfe5/tfenotebook.h 218 635 7769 tfe5/tfetextview.c 29 49 561 tfe5/tfetextview.h 64 105 2266 tfe5/tfe.ui 576 1507 17864 total Up: [Readme.md](../Readme.md), Prev: [Section 14](sec14.md), Next: [Section 16](sec16.md)