2 * main.c - this file is part of Geany, a fast and lightweight IDE
4 * Copyright 2005-2012 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
5 * Copyright 2006-2012 Nick Treleaven <nick(dot)treleaven(at)btinternet(dot)com>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
24 * Main program-related commands.
25 * Handles program initialization and cleanup.
36 #include "callbacks.h"
39 #include "encodings.h"
40 #include "filetypes.h"
41 #include "geanyobject.h"
42 #include "highlighting.h"
43 #include "keybindings.h"
46 #include "msgwindow.h"
58 #include "templates.h"
66 #include "gtkcompat.h"
70 #include <sys/types.h>
76 #include <glib/gstdio.h>
84 gboolean ignore_callback
; /* hack workaround for GTK+ toggle button callback problem */
86 GeanyStatus main_status
;
87 CommandLineOptions cl_options
; /* fields initialised in parse_command_line_options */
89 static gchar
*original_cwd
= NULL
;
91 static const gchar geany_lib_versions
[] = "GTK %u.%u.%u, GLib %u.%u.%u";
93 static gboolean want_plugins
;
95 /* command-line options */
96 static gboolean verbose_mode
= FALSE
;
97 static gboolean ignore_global_tags
= FALSE
;
98 static gboolean no_msgwin
= FALSE
;
99 static gboolean show_version
= FALSE
;
100 static gchar
*alternate_config
= NULL
;
102 static gboolean no_vte
= FALSE
;
103 static gchar
*lib_vte
= NULL
;
105 static gboolean generate_tags
= FALSE
;
106 static gboolean no_preprocessing
= FALSE
;
107 static gboolean ft_names
= FALSE
;
108 static gboolean print_prefix
= FALSE
;
110 static gboolean no_plugins
= FALSE
;
112 static gboolean dummy
= FALSE
;
114 /* in alphabetical order of short options */
115 static GOptionEntry entries
[] =
117 { "column", 0, 0, G_OPTION_ARG_INT
, &cl_options
.goto_column
, N_("Set initial column number for the first opened file (useful in conjunction with --line)"), NULL
},
118 { "config", 'c', 0, G_OPTION_ARG_FILENAME
, &alternate_config
, N_("Use an alternate configuration directory"), NULL
},
119 { "ft-names", 0, 0, G_OPTION_ARG_NONE
, &ft_names
, N_("Print internal filetype names"), NULL
},
120 { "generate-tags", 'g', 0, G_OPTION_ARG_NONE
, &generate_tags
, N_("Generate global tags file (see documentation)"), NULL
},
121 { "no-preprocessing", 'P', 0, G_OPTION_ARG_NONE
, &no_preprocessing
, N_("Don't preprocess C/C++ files when generating tags"), NULL
},
123 { "new-instance", 'i', 0, G_OPTION_ARG_NONE
, &cl_options
.new_instance
, N_("Don't open files in a running instance, force opening a new instance"), NULL
},
124 { "socket-file", 0, 0, G_OPTION_ARG_FILENAME
, &cl_options
.socket_filename
, N_("Use this socket filename for communication with a running Geany instance"), NULL
},
125 { "list-documents", 0, 0, G_OPTION_ARG_NONE
, &cl_options
.list_documents
, N_("Return a list of open documents in a running Geany instance"), NULL
},
127 { "line", 'l', 0, G_OPTION_ARG_INT
, &cl_options
.goto_line
, N_("Set initial line number for the first opened file"), NULL
},
128 { "no-msgwin", 'm', 0, G_OPTION_ARG_NONE
, &no_msgwin
, N_("Don't show message window at startup"), NULL
},
129 { "no-ctags", 'n', 0, G_OPTION_ARG_NONE
, &ignore_global_tags
, N_("Don't load auto completion data (see documentation)"), NULL
},
131 { "no-plugins", 'p', 0, G_OPTION_ARG_NONE
, &no_plugins
, N_("Don't load plugins"), NULL
},
133 { "print-prefix", 0, 0, G_OPTION_ARG_NONE
, &print_prefix
, N_("Print Geany's installation prefix"), NULL
},
134 { "read-only", 'r', 0, G_OPTION_ARG_NONE
, &cl_options
.readonly
, N_("Open all FILES in read-only mode (see documention)"), NULL
},
135 { "no-session", 's', G_OPTION_FLAG_REVERSE
, G_OPTION_ARG_NONE
, &cl_options
.load_session
, N_("Don't load the previous session's files"), NULL
},
137 { "no-terminal", 't', 0, G_OPTION_ARG_NONE
, &no_vte
, N_("Don't load terminal support"), NULL
},
138 { "vte-lib", 0, 0, G_OPTION_ARG_FILENAME
, &lib_vte
, N_("Filename of libvte.so"), NULL
},
140 { "verbose", 'v', 0, G_OPTION_ARG_NONE
, &verbose_mode
, N_("Be verbose"), NULL
},
141 { "version", 'V', 0, G_OPTION_ARG_NONE
, &show_version
, N_("Show version and exit"), NULL
},
142 { "dummy", 0, G_OPTION_FLAG_HIDDEN
, G_OPTION_ARG_NONE
, &dummy
, NULL
, NULL
}, /* for +NNN line number arguments */
143 { NULL
, 0, 0, 0, NULL
, NULL
, NULL
}
147 static void setup_window_position(void)
149 /* interprets the saved window geometry */
150 if (!prefs
.save_winpos
)
153 if (ui_prefs
.geometry
[0] != -1 && ui_prefs
.geometry
[1] != -1)
154 gtk_window_move(GTK_WINDOW(main_widgets
.window
),
155 ui_prefs
.geometry
[0], ui_prefs
.geometry
[1]);
157 if (ui_prefs
.geometry
[2] != -1 && ui_prefs
.geometry
[3] != -1)
158 gtk_window_set_default_size(GTK_WINDOW(main_widgets
.window
),
159 ui_prefs
.geometry
[2], ui_prefs
.geometry
[3]);
161 if (ui_prefs
.geometry
[4] == 1)
162 gtk_window_maximize(GTK_WINDOW(main_widgets
.window
));
166 /* special things for the initial setup of the checkboxes and related stuff
167 * an action on a setting is only performed if the setting is not equal to the program default
168 * (all the following code is not perfect but it works for the moment) */
169 static void apply_settings(void)
171 ui_update_fold_items();
173 /* toolbar, message window and sidebar are by default visible, so don't change it if it is true */
175 if (! ui_prefs
.msgwindow_visible
)
177 ignore_callback
= TRUE
;
178 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(ui_lookup_widget(main_widgets
.window
, "menu_show_messages_window1")), FALSE
);
179 gtk_widget_hide(ui_lookup_widget(main_widgets
.window
, "scrolledwindow1"));
180 ignore_callback
= FALSE
;
182 if (! ui_prefs
.sidebar_visible
)
184 ignore_callback
= TRUE
;
185 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(ui_lookup_widget(main_widgets
.window
, "menu_show_sidebar1")), FALSE
);
186 ignore_callback
= FALSE
;
189 toolbar_apply_settings();
192 ui_update_view_editor_menu_items();
194 /* hide statusbar if desired */
195 if (! interface_prefs
.statusbar_visible
)
197 gtk_widget_hide(ui_widgets
.statusbar
);
200 /* set the tab placements of the notebooks */
201 gtk_notebook_set_tab_pos(GTK_NOTEBOOK(main_widgets
.notebook
), interface_prefs
.tab_pos_editor
);
202 gtk_notebook_set_tab_pos(GTK_NOTEBOOK(msgwindow
.notebook
), interface_prefs
.tab_pos_msgwin
);
203 gtk_notebook_set_tab_pos(GTK_NOTEBOOK(main_widgets
.sidebar_notebook
), interface_prefs
.tab_pos_sidebar
);
205 /* whether to show notebook tabs or not */
206 gtk_notebook_set_show_tabs(GTK_NOTEBOOK(main_widgets
.notebook
), interface_prefs
.show_notebook_tabs
);
209 if (! vte_info
.have_vte
)
212 gtk_widget_set_sensitive(
213 ui_lookup_widget(main_widgets
.window
, "send_selection_to_vte1"), FALSE
);
216 if (interface_prefs
.sidebar_pos
!= GTK_POS_LEFT
)
217 ui_swap_sidebar_pos();
219 gtk_orientable_set_orientation(GTK_ORIENTABLE(ui_lookup_widget(main_widgets
.window
, "vpaned1")),
220 interface_prefs
.msgwin_orientation
);
224 static void main_init(void)
226 /* add our icon path in case we aren't installed in the system prefix */
229 gchar
*install_dir
= win32_get_installation_dir();
230 path
= g_build_filename(install_dir
, "share", "icons", NULL
);
233 path
= g_build_filename(GEANY_DATADIR
, "icons", NULL
);
235 gtk_icon_theme_append_search_path(gtk_icon_theme_get_default(), path
);
239 ui_init_stock_items();
243 main_widgets
.window
= NULL
;
245 ui_widgets
.open_fontsel
= NULL
;
246 ui_widgets
.open_colorsel
= NULL
;
247 ui_widgets
.prefs_dialog
= NULL
;
248 main_status
.main_window_realized
= FALSE
;
249 file_prefs
.tab_order_ltr
= FALSE
;
250 file_prefs
.tab_order_beside
= FALSE
;
251 main_status
.quitting
= FALSE
;
252 ignore_callback
= FALSE
;
253 app
->tm_workspace
= tm_get_workspace();
254 ui_prefs
.recent_queue
= g_queue_new();
255 ui_prefs
.recent_projects_queue
= g_queue_new();
256 main_status
.opening_session_files
= FALSE
;
258 main_widgets
.window
= create_window1();
260 /* add recent projects to the Project menu */
261 ui_widgets
.recent_projects_menuitem
= ui_lookup_widget(main_widgets
.window
, "recent_projects1");
262 ui_widgets
.recent_projects_menu_menubar
= gtk_menu_new();
263 gtk_menu_item_set_submenu(GTK_MENU_ITEM(ui_widgets
.recent_projects_menuitem
),
264 ui_widgets
.recent_projects_menu_menubar
);
266 /* store important pointers for later reference */
267 main_widgets
.toolbar
= toolbar_init();
268 main_widgets
.sidebar_notebook
= ui_lookup_widget(main_widgets
.window
, "notebook3");
269 main_widgets
.notebook
= ui_lookup_widget(main_widgets
.window
, "notebook1");
270 main_widgets
.editor_menu
= create_edit_menu1();
271 main_widgets
.tools_menu
= ui_lookup_widget(main_widgets
.window
, "tools1_menu");
272 main_widgets
.message_window_notebook
= ui_lookup_widget(main_widgets
.window
, "notebook_info");
273 main_widgets
.project_menu
= ui_lookup_widget(main_widgets
.window
, "menu_project1_menu");
275 ui_widgets
.toolbar_menu
= create_toolbar_popup_menu1();
278 /* set widget names for matching with .gtkrc-2.0 */
279 gtk_widget_set_name(main_widgets
.window
, "GeanyMainWindow");
280 gtk_widget_set_name(ui_widgets
.toolbar_menu
, "GeanyToolbarMenu");
281 gtk_widget_set_name(main_widgets
.editor_menu
, "GeanyEditMenu");
282 gtk_widget_set_name(ui_lookup_widget(main_widgets
.window
, "menubar1"), "GeanyMenubar");
283 gtk_widget_set_name(main_widgets
.toolbar
, "GeanyToolbar");
285 gtk_window_set_default_size(GTK_WINDOW(main_widgets
.window
),
286 GEANY_WINDOW_DEFAULT_WIDTH
, GEANY_WINDOW_DEFAULT_HEIGHT
);
290 const gchar
*main_get_version_string(void)
292 static gchar full
[] = VERSION
" (git >= " REVISION
")";
294 if (utils_str_equal(REVISION
, "-1"))
301 /* get the full file path of a command-line argument
302 * N.B. the result should be freed and may contain '/../' or '/./ ' */
303 gchar
*main_get_argv_filename(const gchar
*filename
)
307 if (g_path_is_absolute(filename
) || utils_is_uri(filename
))
308 result
= g_strdup(filename
);
311 /* use current dir */
312 gchar
*cur_dir
= NULL
;
313 if (original_cwd
== NULL
)
314 cur_dir
= g_get_current_dir();
316 cur_dir
= g_strdup(original_cwd
);
319 G_DIR_SEPARATOR_S
, cur_dir
, filename
, NULL
);
326 /* get a :line:column specifier from the end of a filename (if present),
327 * return the line/column values, and remove the specifier from the string
328 * (Note that *line and *column must both be set to -1 initially) */
329 static void get_line_and_column_from_filename(gchar
*filename
, gint
*line
, gint
*column
)
332 gint colon_count
= 0;
333 gboolean have_number
= FALSE
;
336 g_assert(*line
== -1 && *column
== -1);
338 if (G_UNLIKELY(EMPTY(filename
)))
341 /* allow to open files like "test:0" */
342 if (g_file_test(filename
, G_FILE_TEST_EXISTS
))
345 len
= strlen(filename
);
346 for (i
= len
- 1; i
>= 1; i
--)
348 gboolean is_colon
= filename
[i
] == ':';
349 gboolean is_digit
= g_ascii_isdigit(filename
[i
]);
351 if (! is_colon
&& ! is_digit
)
356 if (++colon_count
> 1)
357 break; /* bail on 2+ colons in a row */
365 if (is_colon
&& have_number
)
367 gint number
= atoi(&filename
[i
+ 1]);
377 break; /* line and column are set, so we're done */
383 static void change_working_directory_on_windows(void)
385 gchar
*install_dir
= win32_get_installation_dir();
387 /* remember original working directory for use with opening files from the command line */
388 original_cwd
= g_get_current_dir();
390 /* On Windows, change the working directory to the Geany installation path to not lock
391 * the directory of a file passed as command line argument (see bug #2626124).
392 * This also helps if plugins or other code uses relative paths to load
393 * any additional resources (e.g. share/geany-plugins/...). */
394 win32_set_working_directory(install_dir
);
401 static void setup_paths(void)
408 /* use the installation directory(the one where geany.exe is located) as the base for the
409 * documentation and data files */
410 gchar
*install_dir
= win32_get_installation_dir();
412 data_dir
= g_build_filename(install_dir
, "data", NULL
); /* e.g. C:\Program Files\geany\data */
413 doc_dir
= g_build_filename(install_dir
, "doc", NULL
);
417 data_dir
= g_build_filename(GEANY_DATADIR
, "geany", NULL
); /* e.g. /usr/share/geany */
418 doc_dir
= g_build_filename(GEANY_DOCDIR
, "html", NULL
);
421 /* convert path names to locale encoding */
422 app
->datadir
= utils_get_locale_from_utf8(data_dir
);
423 app
->docdir
= utils_get_locale_from_utf8(doc_dir
);
431 * Checks whether the main window has been realized.
432 * This is an easy indicator whether Geany is right now starting up (main window is not
433 * yet realized) or whether it has finished the startup process (main window is realized).
434 * This is because the main window is realized (i.e. actually drawn on the screen) at the
435 * end of the startup process.
437 * @note Maybe you want to use the @link pluginsignals.c @c "geany-startup-complete" signal @endlink
438 * to get notified about the completed startup process.
440 * @return @c TRUE if the Geany main window has been realized or @c FALSE otherwise.
444 gboolean
main_is_realized(void)
446 return main_status
.main_window_realized
;
451 * Initialises the gettext translation system.
452 * This is a convenience function to set up gettext for internationalisation support
453 * in external plugins. You should call this function early in @ref plugin_init().
454 * If the macro HAVE_LOCALE_H is defined, @c setlocale(LC_ALL, "") is called.
455 * The codeset for the message translations is set to UTF-8.
457 * Note that this function only setups the gettext textdomain for you. You still have
458 * to adjust the build system of your plugin to get internationalisation support
461 * If you have already used @ref PLUGIN_SET_TRANSLATABLE_INFO() you
462 * don't need to call main_locale_init() again as it has already been done.
464 * @param locale_dir The location where the translation files should be searched. This is
465 * usually the @c LOCALEDIR macro, defined by the build system.
466 * E.g. @c $prefix/share/locale.
467 * Only used on non-Windows systems. On Windows, the directory is determined
468 * by @c g_win32_get_package_installation_directory().
469 * @param package The package name, usually this is the @c GETTEXT_PACKAGE macro,
470 * defined by the build system.
474 void main_locale_init(const gchar
*locale_dir
, const gchar
*package
)
476 gchar
*l_locale_dir
= NULL
;
479 setlocale(LC_ALL
, "");
484 gchar
*install_dir
= win32_get_installation_dir();
485 /* e.g. C:\Program Files\geany\lib\locale */
486 l_locale_dir
= g_build_filename(install_dir
, "share", "locale", NULL
);
490 l_locale_dir
= g_strdup(locale_dir
);
493 (void) bindtextdomain(package
, l_locale_dir
);
494 (void) bind_textdomain_codeset(package
, "UTF-8");
495 g_free(l_locale_dir
);
499 static void print_filetypes(void)
501 const GSList
*list
, *node
;
503 filetypes_init_types();
504 printf("Geany's filetype names:\n");
506 list
= filetypes_get_sorted_by_name();
507 foreach_slist(node
, list
)
509 GeanyFiletype
*ft
= node
->data
;
511 printf("%s\n", ft
->name
);
513 filetypes_free_types();
517 static void wait_for_input_on_windows(void)
522 geany_debug("Press any key to continue");
529 static void parse_command_line_options(gint
*argc
, gchar
***argv
)
531 GError
*error
= NULL
;
532 GOptionContext
*context
;
534 CommandLineOptions def_clo
= {FALSE
, NULL
, TRUE
, -1, -1, FALSE
, FALSE
, FALSE
};
536 /* first initialise cl_options fields with default values */
537 cl_options
= def_clo
;
539 /* the GLib option parser can't handle the +NNN (line number) option,
540 * so we grab that here and replace it with a no-op */
541 for (i
= 1; i
< (*argc
); i
++)
543 if ((*argv
)[i
][0] != '+')
546 cl_options
.goto_line
= atoi((*argv
)[i
] + 1);
547 (*argv
)[i
] = (gchar
*) "--dummy";
550 context
= g_option_context_new(_("[FILES...]"));
552 g_option_context_add_main_entries(context
, entries
, GETTEXT_PACKAGE
);
553 g_option_group_set_translation_domain(g_option_context_get_main_group(context
), GETTEXT_PACKAGE
);
554 g_option_context_add_group(context
, gtk_get_option_group(FALSE
));
555 g_option_context_parse(context
, argc
, argv
, &error
);
556 g_option_context_free(context
);
560 g_printerr("Geany: %s\n", error
->message
);
565 app
->debug_mode
= verbose_mode
;
568 /* Since GLib 2.32 messages logged with levels INFO and DEBUG aren't output by the
569 * default log handler unless the G_MESSAGES_DEBUG environment variable contains the
570 * domain of the message or is set to the special value "all" */
571 g_setenv("G_MESSAGES_DEBUG", "all", FALSE
);
575 win32_init_debug_code();
580 gchar
*build_date
= utils_parse_and_format_build_date(__DATE__
);
582 printf(PACKAGE
" %s (", main_get_version_string());
583 /* note for translators: library versions are printed after this */
584 printf(_("built on %s with "), build_date
);
585 printf(geany_lib_versions
,
586 GTK_MAJOR_VERSION
, GTK_MINOR_VERSION
, GTK_MICRO_VERSION
,
587 GLIB_MAJOR_VERSION
, GLIB_MINOR_VERSION
, GLIB_MICRO_VERSION
);
590 wait_for_input_on_windows();
596 printf("%s\n", GEANY_PREFIX
);
597 printf("%s\n", GEANY_DATADIR
);
598 printf("%s\n", GEANY_LIBDIR
);
599 printf("%s\n", GEANY_LOCALEDIR
);
600 wait_for_input_on_windows();
604 if (alternate_config
)
606 geany_debug("alternate config: %s", alternate_config
);
607 app
->configdir
= alternate_config
;
611 app
->configdir
= utils_get_user_config_dir();
618 filetypes_init_types();
619 ret
= symbols_generate_global_tags(*argc
, *argv
, ! no_preprocessing
);
620 filetypes_free_types();
621 wait_for_input_on_windows();
628 wait_for_input_on_windows();
633 socket_info
.ignore_socket
= cl_options
.new_instance
;
634 if (cl_options
.socket_filename
)
636 socket_info
.file_name
= cl_options
.socket_filename
;
641 vte_info
.lib_vte
= lib_vte
;
643 cl_options
.ignore_global_tags
= ignore_global_tags
;
645 if (! gtk_init_check(NULL
, NULL
))
646 { /* check whether we have a valid X display and exit if not */
647 g_printerr("Geany: cannot open display\n");
653 static gint
create_config_dir(void)
655 gint saved_errno
= 0;
656 gchar
*conf_file
= NULL
;
657 gchar
*filedefs_dir
= NULL
;
658 gchar
*templates_dir
= NULL
;
660 if (! g_file_test(app
->configdir
, G_FILE_TEST_EXISTS
))
663 /* if we are *not* using an alternate config directory, we check whether the old one
664 * in ~/.geany still exists and try to move it */
665 if (alternate_config
== NULL
)
667 gchar
*old_dir
= g_build_filename(g_get_home_dir(), ".geany", NULL
);
668 /* move the old config dir if it exists */
669 if (g_file_test(old_dir
, G_FILE_TEST_EXISTS
))
671 if (! dialogs_show_question_full(main_widgets
.window
,
672 GTK_STOCK_YES
, GTK_STOCK_QUIT
, _("Move it now?"),
674 _("Geany needs to move your old configuration directory before starting.")))
677 if (! g_file_test(app
->configdir
, G_FILE_TEST_IS_DIR
))
678 utils_mkdir(app
->configdir
, TRUE
);
680 if (g_rename(old_dir
, app
->configdir
) == 0)
682 dialogs_show_msgbox(GTK_MESSAGE_INFO
,
683 _("Your configuration directory has been successfully moved from \"%s\" to \"%s\"."),
684 old_dir
, app
->configdir
);
690 dialogs_show_msgbox(GTK_MESSAGE_WARNING
,
691 /* for translators: the third %s in brackets is the error message which
692 * describes why moving the dir didn't work */
693 _("Your old configuration directory \"%s\" could not be moved to \"%s\" (%s). "
694 "Please move manually the directory to the new location."),
695 old_dir
, app
->configdir
, g_strerror(errno
));
701 geany_debug("creating config directory %s", app
->configdir
);
702 saved_errno
= utils_mkdir(app
->configdir
, TRUE
);
705 conf_file
= g_build_filename(app
->configdir
, "geany.conf", NULL
);
706 filedefs_dir
= g_build_filename(app
->configdir
, GEANY_FILEDEFS_SUBDIR
, NULL
);
707 templates_dir
= g_build_filename(app
->configdir
, GEANY_TEMPLATES_SUBDIR
, NULL
);
709 if (saved_errno
== 0 && ! g_file_test(conf_file
, G_FILE_TEST_EXISTS
))
710 { /* check whether geany.conf can be written */
711 saved_errno
= utils_is_file_writable(app
->configdir
);
714 /* make subdir for filetype definitions */
715 if (saved_errno
== 0)
717 gchar
*filedefs_readme
= g_build_filename(app
->configdir
,
718 GEANY_FILEDEFS_SUBDIR
, "filetypes.README", NULL
);
720 if (! g_file_test(filedefs_dir
, G_FILE_TEST_EXISTS
))
722 saved_errno
= utils_mkdir(filedefs_dir
, FALSE
);
724 if (saved_errno
== 0 && ! g_file_test(filedefs_readme
, G_FILE_TEST_EXISTS
))
726 gchar
*text
= g_strconcat(
727 "Copy files from ", app
->datadir
, " to this directory to overwrite "
728 "them. To use the defaults, just delete the file in this directory.\nFor more information read "
729 "the documentation (in ", app
->docdir
, G_DIR_SEPARATOR_S
"index.html or visit " GEANY_HOMEPAGE
").", NULL
);
730 utils_write_file(filedefs_readme
, text
);
733 g_free(filedefs_readme
);
736 /* make subdir for template files */
737 if (saved_errno
== 0)
739 gchar
*templates_readme
= g_build_filename(app
->configdir
, GEANY_TEMPLATES_SUBDIR
,
740 "templates.README", NULL
);
742 if (! g_file_test(templates_dir
, G_FILE_TEST_EXISTS
))
744 saved_errno
= utils_mkdir(templates_dir
, FALSE
);
746 if (saved_errno
== 0 && ! g_file_test(templates_readme
, G_FILE_TEST_EXISTS
))
748 gchar
*text
= g_strconcat(
749 "There are several template files in this directory. For these templates you can use wildcards.\n\
750 For more information read the documentation (in ", app
->docdir
, G_DIR_SEPARATOR_S
"index.html or visit " GEANY_HOMEPAGE
").",
752 utils_write_file(templates_readme
, text
);
755 g_free(templates_readme
);
758 g_free(filedefs_dir
);
759 g_free(templates_dir
);
766 /* Returns 0 if config dir is OK. */
767 static gint
setup_config_dir(void)
769 gint mkdir_result
= 0;
771 /* convert configdir to locale encoding to avoid troubles */
772 SETPTR(app
->configdir
, utils_get_locale_from_utf8(app
->configdir
));
774 mkdir_result
= create_config_dir();
775 if (mkdir_result
!= 0)
777 if (! dialogs_show_question(
778 _("Configuration directory could not be created (%s).\nThere could be some problems "
779 "using Geany without a configuration directory.\nStart Geany anyway?"),
780 g_strerror(mkdir_result
)))
785 /* make configdir a real path */
786 if (g_file_test(app
->configdir
, G_FILE_TEST_EXISTS
))
787 SETPTR(app
->configdir
, tm_get_real_path(app
->configdir
));
792 /* Signal handling removed since main_quit() uses functions that are
793 * illegal in signal handlers
794 static void signal_cb(gint sig)
803 /* Used for command-line arguments at startup or from socket.
804 * this will strip any :line:col filename suffix from locale_filename */
805 gboolean
main_handle_filename(const gchar
*locale_filename
)
808 gint line
= -1, column
= -1;
811 g_return_val_if_fail(locale_filename
, FALSE
);
813 /* check whether the passed filename is an URI */
814 filename
= utils_get_path_from_uri(locale_filename
);
815 if (filename
== NULL
)
818 get_line_and_column_from_filename(filename
, &line
, &column
);
820 cl_options
.goto_line
= line
;
822 cl_options
.goto_column
= column
;
824 if (g_file_test(filename
, G_FILE_TEST_IS_REGULAR
))
826 doc
= document_open_file(filename
, cl_options
.readonly
, NULL
, NULL
);
827 /* add recent file manually if opening_session_files is set */
828 if (doc
!= NULL
&& main_status
.opening_session_files
)
829 ui_add_recent_document(doc
);
833 else if (file_prefs
.cmdline_new_files
)
834 { /* create new file with the given filename */
835 gchar
*utf8_filename
= utils_get_utf8_from_locale(filename
);
837 doc
= document_new_file(utf8_filename
, NULL
, NULL
);
839 ui_add_recent_document(doc
);
840 g_free(utf8_filename
);
849 /* open files from command line */
850 static void open_cl_files(gint argc
, gchar
**argv
)
854 for (i
= 1; i
< argc
; i
++)
856 gchar
*filename
= main_get_argv_filename(argv
[i
]);
858 if (g_file_test(filename
, G_FILE_TEST_IS_DIR
))
865 /* It seems argv elements are encoded in CP1252 on a German Windows */
866 SETPTR(filename
, g_locale_to_utf8(filename
, -1, NULL
, NULL
, NULL
));
868 if (filename
&& ! main_handle_filename(filename
))
870 const gchar
*msg
= _("Could not find file '%s'.");
872 g_printerr(msg
, filename
); /* also print to the terminal */
874 ui_set_statusbar(TRUE
, msg
, filename
);
881 static void load_session_project_file(void)
883 gchar
*locale_filename
;
885 g_return_if_fail(project_prefs
.session_file
!= NULL
);
887 locale_filename
= utils_get_locale_from_utf8(project_prefs
.session_file
);
889 if (G_LIKELY(!EMPTY(locale_filename
)))
890 project_load_file(locale_filename
);
892 g_free(locale_filename
);
893 g_free(project_prefs
.session_file
); /* no longer needed */
897 static void load_settings(void)
899 configuration_load();
900 /* let cmdline options overwrite configuration settings */
902 vte_info
.have_vte
= (no_vte
) ? FALSE
: vte_info
.load_vte
;
905 ui_prefs
.msgwindow_visible
= FALSE
;
908 want_plugins
= prefs
.load_plugins
&& !no_plugins
;
913 void main_load_project_from_command_line(const gchar
*locale_filename
, gboolean use_session
)
917 pfile
= utils_get_path_from_uri(locale_filename
);
921 project_load_file_with_session(pfile
);
923 project_load_file(pfile
);
929 static void load_startup_files(gint argc
, gchar
**argv
)
931 gboolean load_session
= FALSE
;
933 if (argc
> 1 && g_str_has_suffix(argv
[1], ".geany"))
935 /* project file specified: load it, but decide the session later */
936 main_load_project_from_command_line(argv
[1], FALSE
);
938 /* force session load if using project-based session files */
939 load_session
= project_prefs
.project_session
;
942 /* Load the default session if:
943 * 1. "Load files from the last session" is active.
944 * 2. --no-session is not specified.
945 * 3. We are a primary instance.
946 * Has no effect if a CL project is loaded and using project-based session files. */
947 if (prefs
.load_session
&& cl_options
.load_session
&& !cl_options
.new_instance
)
949 if (app
->project
== NULL
)
950 load_session_project_file();
956 /* load session files into tabs, as they are found in the session_files variable */
957 configuration_open_files();
959 if (gtk_notebook_get_n_pages(GTK_NOTEBOOK(main_widgets
.notebook
)) == 0)
961 ui_update_popup_copy_items(NULL
);
962 ui_update_popup_reundo_items(NULL
);
966 open_cl_files(argc
, argv
);
970 static gboolean
send_startup_complete(gpointer data
)
972 g_signal_emit_by_name(geany_object
, "geany-startup-complete");
977 static const gchar
*get_locale(void)
979 const gchar
*locale
= "unknown";
981 locale
= setlocale(LC_CTYPE
, NULL
);
987 #if ! GTK_CHECK_VERSION(3, 0, 0)
988 /* This prepends our own gtkrc file to the list of RC files to be loaded by GTK at startup.
989 * This function *has* to be called before gtk_init().
991 * We have a custom RC file defining various styles we need, and we want the user to be
992 * able to override them (e.g. if they want -- or need -- other colors). Fair enough, one
993 * would simply call gtk_rc_parse() with the appropriate filename. However, the styling
994 * rules applies in the order they are loaded, then if we load our styles after GTK has
995 * loaded the user's ones we'd override them.
997 * There are 2 solutions to fix this:
998 * 1) set our styles' priority to something with lower than "user" (actually "theme"
999 * priority because rules precedence are first calculated depending on the priority
1000 * no matter of how precise the rules is, so we need to override the theme).
1001 * 2) prepend our custom style to GTK's list while keeping priority to user (which is the
1002 * default), so it gets loaded before real user's ones and so gets overridden by them.
1004 * One would normally go for 1 because it's ways simpler and requires less code: you just
1005 * have to add the priorities to your styles, which is a matter of adding a few ":theme" in
1006 * the RC file. However, KDE being a bitch it doesn't set the gtk-theme-name but rather
1007 * directly includes the style to use in a user gtkrc file, which makes the theme have
1008 * "user" priority, hence overriding our styles. So, we cannot set priorities in the RC
1009 * file if we want to support running under KDE, which pretty much leave us with no choice
1010 * but to go with solution 2, which unfortunately requires writing ugly code since GTK
1011 * don't have a gtk_rc_prepend_default_file() function. Thank you very much KDE.
1013 * Though, as a side benefit it also makes the code work with people using gtk-chtheme,
1014 * which also found it funny to include the theme in the user RC file. */
1015 static void setup_gtk2_styles(void)
1017 gchar
**gtk_files
= gtk_rc_get_default_files();
1018 gchar
**new_files
= g_malloc(sizeof *new_files
* (g_strv_length(gtk_files
) + 2));
1021 new_files
[i
++] = g_build_filename(app
->datadir
, "geany.gtkrc", NULL
);
1022 for (; *gtk_files
; gtk_files
++)
1023 new_files
[i
++] = g_strdup(*gtk_files
);
1024 new_files
[i
] = NULL
;
1026 gtk_rc_set_default_files(new_files
);
1028 g_strfreev(new_files
);
1033 gint
main(gint argc
, gchar
**argv
)
1036 gint config_dir_result
;
1037 const gchar
*locale
;
1039 #if ! GLIB_CHECK_VERSION(2, 36, 0)
1043 log_handlers_init();
1045 app
= g_new0(GeanyApp
, 1);
1046 memset(&main_status
, 0, sizeof(GeanyStatus
));
1047 memset(&prefs
, 0, sizeof(GeanyPrefs
));
1048 memset(&interface_prefs
, 0, sizeof(GeanyInterfacePrefs
));
1049 memset(&toolbar_prefs
, 0, sizeof(GeanyToolbarPrefs
));
1050 memset(&file_prefs
, 0, sizeof(GeanyFilePrefs
));
1051 memset(&search_prefs
, 0, sizeof(GeanySearchPrefs
));
1052 memset(&tool_prefs
, 0, sizeof(GeanyToolPrefs
));
1053 memset(&template_prefs
, 0, sizeof(GeanyTemplatePrefs
));
1054 memset(&ui_prefs
, 0, sizeof(UIPrefs
));
1055 memset(&ui_widgets
, 0, sizeof(UIWidgets
));
1058 #if ! GTK_CHECK_VERSION(3, 0, 0)
1059 setup_gtk2_styles();
1062 main_locale_init(GEANY_LOCALEDIR
, GETTEXT_PACKAGE
);
1064 parse_command_line_options(&argc
, &argv
);
1066 #if ! GLIB_CHECK_VERSION(2, 32, 0)
1067 /* Initialize GLib's thread system in case any plugins want to use it or their
1068 * dependencies (e.g. WebKit, Soup, ...). Deprecated since GLIB 2.32. */
1069 if (!g_thread_supported())
1070 g_thread_init(NULL
);
1073 /* removed as signal handling was wrong, see signal_cb()
1074 signal(SIGTERM, signal_cb); */
1077 /* ignore SIGPIPE signal for preventing sudden death of program */
1078 signal(SIGPIPE
, SIG_IGN
);
1081 config_dir_result
= setup_config_dir();
1083 /* check and create (unix domain) socket for remote operation */
1084 if (! socket_info
.ignore_socket
)
1086 socket_info
.lock_socket
= -1;
1087 socket_info
.lock_socket_tag
= 0;
1088 socket_info
.lock_socket
= socket_init(argc
, argv
);
1089 /* Quit if filenames were sent to first instance or the list of open
1090 * documents has been printed */
1091 if ((socket_info
.lock_socket
== -2 /* socket exists */ && argc
> 1) ||
1092 cl_options
.list_documents
)
1095 gdk_notify_startup_complete();
1096 g_free(app
->configdir
);
1097 g_free(app
->datadir
);
1098 g_free(app
->docdir
);
1102 /* Start a new instance if no command line strings were passed,
1103 * even if the socket already exists */
1104 else if (socket_info
.lock_socket
== -2 /* socket already exists */)
1106 socket_info
.ignore_socket
= TRUE
;
1107 cl_options
.new_instance
= TRUE
;
1113 /* after we initialized the socket code and handled command line args,
1114 * let's change the working directory on Windows to not lock it */
1115 change_working_directory_on_windows();
1118 locale
= get_locale();
1119 geany_debug("Geany %s, %s",
1120 main_get_version_string(),
1122 geany_debug(geany_lib_versions
,
1123 gtk_major_version
, gtk_minor_version
, gtk_micro_version
,
1124 glib_major_version
, glib_minor_version
, glib_micro_version
);
1125 geany_debug("System data dir: %s", app
->datadir
);
1126 geany_debug("User config dir: %s", app
->configdir
);
1128 /* create the object so Geany signals can be connected in init() functions */
1129 geany_object
= geany_object_new();
1137 /* init stash groups before loading keyfile */
1138 configuration_init();
1146 load_settings(); /* load keyfile */
1150 ui_create_insert_menu_items();
1151 ui_create_insert_date_menu_items();
1157 document_init_doclist();
1159 editor_snippets_init();
1164 ui_create_recent_menus();
1166 ui_set_statusbar(TRUE
, _("This is Geany %s."), main_get_version_string());
1167 if (config_dir_result
!= 0)
1168 ui_set_statusbar(TRUE
, _("Configuration directory could not be created (%s)."),
1169 g_strerror(config_dir_result
));
1171 /* apply all configuration options */
1175 /* load any enabled plugins before we open any documents */
1177 plugins_load_active();
1180 ui_sidebar_show_hide();
1182 /* set the active sidebar page after plugins have been loaded */
1183 gtk_notebook_set_current_page(GTK_NOTEBOOK(main_widgets
.sidebar_notebook
), ui_prefs
.sidebar_page
);
1185 /* load keybinding settings after plugins have added their groups */
1186 keybindings_load_keyfile();
1188 /* create the custom command menu after the keybindings have been loaded to have the proper
1189 * accelerator shown for the menu items */
1190 tools_create_insert_custom_command_menu_items();
1192 /* load any command line files or session files */
1193 main_status
.opening_session_files
= TRUE
;
1194 load_startup_files(argc
, argv
);
1195 main_status
.opening_session_files
= FALSE
;
1197 /* open a new file if no other file was opened */
1198 document_new_file_if_non_open();
1200 ui_document_buttons_update();
1201 ui_save_buttons_toggle(FALSE
);
1203 doc
= document_get_current();
1204 sidebar_select_openfiles_item(doc
);
1205 build_menu_update(doc
);
1206 sidebar_update_tag_list(doc
, FALSE
);
1209 /* Manually realise the main window to be able to set the position but don't show it.
1210 * We don't set the position after showing the window to avoid flickering. */
1211 gtk_widget_realize(main_widgets
.window
);
1213 setup_window_position();
1215 /* finally show the window */
1216 document_grab_focus(doc
);
1217 gtk_widget_show(main_widgets
.window
);
1218 main_status
.main_window_realized
= TRUE
;
1220 configuration_apply_settings();
1223 /* register the callback of socket input */
1224 if (! socket_info
.ignore_socket
&& socket_info
.lock_socket
> 0)
1226 socket_info
.read_ioc
= g_io_channel_unix_new(socket_info
.lock_socket
);
1227 socket_info
.lock_socket_tag
= g_io_add_watch(socket_info
.read_ioc
,
1228 G_IO_IN
| G_IO_PRI
| G_IO_ERR
, socket_lock_input_cb
, main_widgets
.window
);
1232 /* when we are really done with setting everything up and the main event loop is running,
1233 * tell other components, mainly plugins, that startup is complete */
1234 g_idle_add_full(G_PRIORITY_LOW
, send_startup_complete
, NULL
, NULL
);
1241 static void queue_free(GQueue
*queue
)
1243 while (! g_queue_is_empty(queue
))
1245 g_free(g_queue_pop_tail(queue
));
1247 g_queue_free(queue
);
1251 static void do_main_quit(void)
1253 geany_debug("Quitting...");
1255 configuration_save();
1257 if (app
->project
!= NULL
)
1258 project_close(FALSE
); /* save project session files */
1260 document_close_all();
1262 main_status
.quitting
= TRUE
;
1275 highlighting_free_styles();
1276 templates_free_templates();
1280 document_finalize();
1284 editor_snippets_free();
1285 encodings_finalize();
1288 configuration_finalize();
1289 filetypes_free_types();
1292 tm_workspace_free();
1293 g_free(app
->configdir
);
1294 g_free(app
->datadir
);
1295 g_free(app
->docdir
);
1296 g_free(prefs
.default_open_path
);
1297 g_free(prefs
.custom_plugin_path
);
1298 g_free(ui_prefs
.custom_date_format
);
1299 g_free(interface_prefs
.editor_font
);
1300 g_free(interface_prefs
.tagbar_font
);
1301 g_free(interface_prefs
.msgwin_font
);
1302 g_free(editor_prefs
.long_line_color
);
1303 g_free(editor_prefs
.comment_toggle_mark
);
1304 g_free(editor_prefs
.color_scheme
);
1305 g_free(tool_prefs
.context_action_cmd
);
1306 g_free(template_prefs
.developer
);
1307 g_free(template_prefs
.company
);
1308 g_free(template_prefs
.mail
);
1309 g_free(template_prefs
.initials
);
1310 g_free(template_prefs
.version
);
1311 g_free(tool_prefs
.term_cmd
);
1312 g_free(tool_prefs
.browser_cmd
);
1313 g_free(tool_prefs
.grep_cmd
);
1314 g_free(printing_prefs
.external_print_cmd
);
1315 g_free(printing_prefs
.page_header_datefmt
);
1316 g_strfreev(ui_prefs
.custom_commands
);
1317 g_strfreev(ui_prefs
.custom_commands_labels
);
1319 queue_free(ui_prefs
.recent_queue
);
1320 queue_free(ui_prefs
.recent_projects_queue
);
1322 if (ui_widgets
.prefs_dialog
&& GTK_IS_WIDGET(ui_widgets
.prefs_dialog
)) gtk_widget_destroy(ui_widgets
.prefs_dialog
);
1323 if (ui_widgets
.open_fontsel
&& GTK_IS_WIDGET(ui_widgets
.open_fontsel
)) gtk_widget_destroy(ui_widgets
.open_fontsel
);
1324 if (ui_widgets
.open_colorsel
&& GTK_IS_WIDGET(ui_widgets
.open_colorsel
)) gtk_widget_destroy(ui_widgets
.open_colorsel
);
1326 if (vte_info
.have_vte
) vte_close();
1327 g_free(vte_info
.lib_vte
);
1328 g_free(vte_info
.dir
);
1330 gtk_widget_destroy(main_widgets
.window
);
1332 /* destroy popup menus */
1333 if (main_widgets
.editor_menu
&& GTK_IS_WIDGET(main_widgets
.editor_menu
))
1334 gtk_widget_destroy(main_widgets
.editor_menu
);
1335 if (ui_widgets
.toolbar_menu
&& GTK_IS_WIDGET(ui_widgets
.toolbar_menu
))
1336 gtk_widget_destroy(ui_widgets
.toolbar_menu
);
1337 if (msgwindow
.popup_status_menu
&& GTK_IS_WIDGET(msgwindow
.popup_status_menu
))
1338 gtk_widget_destroy(msgwindow
.popup_status_menu
);
1339 if (msgwindow
.popup_msg_menu
&& GTK_IS_WIDGET(msgwindow
.popup_msg_menu
))
1340 gtk_widget_destroy(msgwindow
.popup_msg_menu
);
1341 if (msgwindow
.popup_compiler_menu
&& GTK_IS_WIDGET(msgwindow
.popup_compiler_menu
))
1342 gtk_widget_destroy(msgwindow
.popup_compiler_menu
);
1344 g_object_unref(geany_object
);
1345 geany_object
= NULL
;
1347 g_free(original_cwd
);
1350 ui_finalize_builder();
1356 static gboolean
check_no_unsaved(void)
1360 for (i
= 0; i
< documents_array
->len
; i
++)
1362 if (documents
[i
]->is_valid
&& documents
[i
]->changed
)
1367 return TRUE
; /* no unsaved edits */
1371 /* Returns false when quitting is aborted due to user cancellation */
1372 gboolean
main_quit(void)
1374 main_status
.quitting
= TRUE
;
1376 if (! check_no_unsaved())
1378 if (document_account_for_unsaved())
1385 if (! prefs
.confirm_exit
||
1386 dialogs_show_question_full(NULL
, GTK_STOCK_QUIT
, GTK_STOCK_CANCEL
, NULL
,
1387 _("Do you really want to quit?")))
1393 main_status
.quitting
= FALSE
;
1398 * Reloads most of Geany's configuration files without restarting. Currently the following
1399 * files are reloaded: all template files, also new file templates and the 'New (with template)'
1400 * menus will be updated, Snippets (snippets.conf), filetype extensions (filetype_extensions.conf),
1401 * and 'settings' and 'build_settings' sections of the filetype definition files.
1403 * Plugins may call this function if they changed any of these files (e.g. a configuration file
1408 void main_reload_configuration(void)
1410 /* reload templates */
1411 templates_free_templates();
1414 /* reload snippets */
1415 editor_snippets_free();
1416 editor_snippets_init();
1418 filetypes_reload_extensions();
1421 /* C tag names to ignore */
1422 symbols_reload_config_files();
1424 ui_set_statusbar(TRUE
, _("Configuration files reloaded."));