/* PIKA - Photo and Image Kooker Application * a rebranding of The GNU Image Manipulation Program (created with heckimp) * A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio * * Original copyright, applying to most contents (license remains unchanged): * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "config.h" #include #include #include #include #ifdef HAVE_SYS_PARAM_H #include #endif #ifdef HAVE_UNISTD_H #include #endif #include #include #include #include #ifdef G_OS_WIN32 #include #include #endif #undef PIKA_DISABLE_DEPRECATED /* for compat enums */ #include "libpikabase/pikabase.h" #define PIKA_DISABLE_DEPRECATED #include "libpikaconfig/pikaconfig.h" #include "core/core-types.h" #include "config/pikarc.h" #include "gegl/pika-gegl.h" #include "core/pika.h" #include "core/pika-batch.h" #include "core/pika-user-install.h" #include "core/pikaimage.h" #include "file/file-open.h" #ifndef PIKA_CONSOLE_COMPILATION #include #include "dialogs/user-install-dialog.h" #include "gui/pikaapp.h" #include "gui/gui.h" #endif #include "app.h" #include "errors.h" #include "pikaconsoleapp.h" #include "pikacoreapp.h" #include "language.h" #include "sanity.h" #include "pika-debug.h" #include "pika-intl.h" #include "pika-update.h" /* local prototypes */ static void app_init_update_noop (const gchar *text1, const gchar *text2, gdouble percentage); static void app_activate_callback (PikaCoreApp *app, gpointer user_data); static void app_restore_after_callback (Pika *pika, PikaInitStatusFunc status_callback); static gboolean app_exit_after_callback (Pika *pika, gboolean kill_it, GApplication *app); #if 0 /* left here as documentation how to do compat enums */ GType pika_convert_dither_type_compat_get_type (void); /* compat cruft */ #endif /* local variables */ static GObject *initial_monitor = NULL; /* public functions */ void app_libs_init (GOptionContext *context, gboolean no_interface) { #if 0 GQuark quark; #endif /* disable OpenCL before GEGL is even initialized; this way we only * enable if wanted in pikarc, instead of always enabling, and then * disabling again if wanted in pikarc */ g_object_set (gegl_config (), "use-opencl", FALSE, "application-license", "GPL3", NULL); g_option_context_add_group (context, gegl_get_option_group ()); #ifndef PIKA_CONSOLE_COMPILATION if (! no_interface) { gui_libs_init (context); } #endif #if 0 /* left here as documentation how to do compat enums */ /* keep compat enum code in sync with pdb/enumcode.pl */ quark = g_quark_from_static_string ("pika-compat-enum"); g_type_set_qdata (PIKA_TYPE_CONVERT_DITHER_TYPE, quark, (gpointer) pika_convert_dither_type_compat_get_type ()); #endif } void app_abort (gboolean no_interface, const gchar *abort_message) { #ifndef PIKA_CONSOLE_COMPILATION if (no_interface) #endif { g_print ("%s\n\n", abort_message); } #ifndef PIKA_CONSOLE_COMPILATION else { gui_abort (abort_message); } #endif app_exit (EXIT_FAILURE); } void app_exit (gint status) { exit (status); } gint app_run (const gchar *full_prog_name, const gchar **filenames, GFile *alternate_system_pikarc, GFile *alternate_pikarc, const gchar *session_name, const gchar *batch_interpreter, const gchar **batch_commands, gboolean quit, gboolean as_new, gboolean no_interface, gboolean no_data, gboolean no_fonts, gboolean no_splash, gboolean be_verbose, gboolean use_shm, gboolean use_cpu_accel, gboolean console_messages, gboolean use_debug_handler, gboolean show_playground, gboolean show_debug_menu, PikaStackTraceMode stack_trace_mode, PikaPDBCompatMode pdb_compat_mode, const gchar *backtrace_file) { Pika *pika = NULL; GApplication *app = NULL; GFile *default_folder = NULL; GFile *pikadir = NULL; const gchar *abort_message = NULL; gint retval = EXIT_SUCCESS; if (filenames && filenames[0] && ! filenames[1] && g_file_test (filenames[0], G_FILE_TEST_IS_DIR)) { if (g_path_is_absolute (filenames[0])) { default_folder = g_file_new_for_path (filenames[0]); } else { gchar *absolute = g_build_path (G_DIR_SEPARATOR_S, g_get_current_dir (), filenames[0], NULL); default_folder = g_file_new_for_path (absolute); g_free (absolute); } filenames = NULL; } /* Create an instance of the "Pika" object which is the root of the * core object system */ pika = pika_new (full_prog_name, session_name, default_folder, be_verbose, no_data, no_fonts, no_interface, use_shm, use_cpu_accel, console_messages, show_playground, show_debug_menu, stack_trace_mode, pdb_compat_mode); g_clear_object (&default_folder); #ifndef PIKA_CONSOLE_COMPILATION app = pika_app_new (pika, no_splash, quit, as_new, filenames, batch_interpreter, batch_commands); #else app = pika_console_app_new (pika, quit, as_new, filenames, batch_interpreter, batch_commands); #endif pika->app = app; pika_cpu_accel_set_use (use_cpu_accel); /* Check if the user's pika_directory exists */ pikadir = pika_directory_file (NULL); if (g_file_query_file_type (pikadir, G_FILE_QUERY_INFO_NONE, NULL) != G_FILE_TYPE_DIRECTORY) { PikaUserInstall *install = pika_user_install_new (G_OBJECT (pika), be_verbose); #ifdef PIKA_CONSOLE_COMPILATION pika_user_install_run (install, 1); #else if (! (no_interface ? pika_user_install_run (install, 1) : user_install_dialog_run (install))) exit (EXIT_FAILURE); #endif pika_user_install_free (install); } g_object_unref (pikadir); pika_load_config (pika, alternate_system_pikarc, alternate_pikarc); /* Initialize the error handling after creating/migrating the config * directory because it will create some folders for backup and crash * logs in advance. Therefore running this before * pika_user_install_new() would break migration as well as initial * folder creations. * * It also needs to be run after pika_load_config() because error * handling is based on Preferences. It means that early loading code * is not handled by our debug code, but that's not a big deal. */ errors_init (pika, full_prog_name, use_debug_handler, stack_trace_mode, backtrace_file); /* run the late-stage sanity check. it's important that this check is run * after the call to language_init() (see comment in sanity_check_late().) */ abort_message = sanity_check_late (); if (abort_message) app_abort (no_interface, abort_message); /* initialize lowlevel stuff */ pika_gegl_init (pika); /* Connect our restore_after callback before gui_init() connects * theirs, so ours runs first and can grab the initial monitor * before the GUI's restore_after callback resets it. */ g_signal_connect_after (pika, "restore", G_CALLBACK (app_restore_after_callback), NULL); g_signal_connect_after (pika, "exit", G_CALLBACK (app_exit_after_callback), app); g_signal_connect (app, "activate", G_CALLBACK (app_activate_callback), NULL); retval = g_application_run (app, 0, NULL); if (! retval) retval = pika_core_app_get_exit_status (PIKA_CORE_APP (app)); if (pika->be_verbose) g_print ("EXIT: %s\n", G_STRFUNC); g_clear_object (&app); pika_gegl_exit (pika); errors_exit (); while (g_main_context_pending (NULL)) g_main_context_iteration (NULL, TRUE); g_object_unref (pika); pika_debug_instances (); gegl_exit (); return retval; } /* private functions */ static void app_init_update_noop (const gchar *text1, const gchar *text2, gdouble percentage) { /* deliberately do nothing */ } static void app_activate_callback (PikaCoreApp *app, gpointer user_data) { Pika *pika = NULL; PikaInitStatusFunc update_status_func = NULL; const gchar **filenames; const gchar *current_language; gchar *prev_language = NULL; GError *font_error = NULL; gint batch_retval; g_return_if_fail (PIKA_IS_CORE_APP (app)); pika = pika_core_app_get_pika (app); pika_core_app_set_exit_status (app, EXIT_SUCCESS); #ifndef PIKA_CONSOLE_COMPILATION if (! pika->no_interface) update_status_func = gui_init (pika, pika_app_get_no_splash (PIKA_APP (app)), PIKA_APP (app), NULL); #endif if (! update_status_func) update_status_func = app_init_update_noop; /* Create all members of the global Pika instance which need an already * parsed pikarc, e.g. the data factories */ pika_initialize (pika, update_status_func); g_object_get (pika->edit_config, "prev-language", &prev_language, NULL); /* Language was already initialized. I call this again only to get the * actual language information. */ current_language = language_init (NULL); pika->query_all = (prev_language == NULL || g_strcmp0 (prev_language, current_language) != 0); g_free (prev_language); /* Load all data files */ pika_restore (pika, update_status_func, &font_error); /* enable autosave late so we don't autosave when the * monitor resolution is set in gui_init() */ pika_rc_set_autosave (PIKA_RC (pika->edit_config), TRUE); /* check for updates *after* enabling config autosave, so that the timestamp * is saved */ pika_update_auto_check (pika->edit_config, pika); /* Setting properties to be used for the next run. */ g_object_set (pika->edit_config, /* Set this after pika_update_auto_check(). */ "config-version", PIKA_VERSION, /* Set this after pika_restore(). */ "prev-language", current_language, NULL); #ifndef PIKA_CONSOLE_COMPILATION if (! pika->no_interface) { /* Before opening images from command line, check for salvaged images * and query interactively to know if we should recover or discard * them. */ GList *recovered_files; GList *iter; recovered_files = errors_recovered (); if (recovered_files && gui_recover (g_list_length (recovered_files))) { for (iter = recovered_files; iter; iter = iter->next) { GFile *file; PikaImage *image; GError *error = NULL; PikaPDBStatusType status; file = g_file_new_for_path (iter->data); image = file_open_with_display (pika, pika_get_user_context (pika), NULL, file, pika_core_app_get_as_new (app), initial_monitor, &status, &error); if (image) { /* Break ties with the backup directory. */ pika_image_set_file (image, NULL); /* One of the rare exceptions where we should call * pika_image_dirty() directly instead of creating * an undo. We want the image to be dirty from * scratch, without anything to undo. */ pika_image_dirty (image, PIKA_DIRTY_IMAGE); } else { g_error_free (error); } g_object_unref (file); } } /* Delete backup XCF images. */ for (iter = recovered_files; iter; iter = iter->next) { g_unlink (iter->data); } g_list_free_full (recovered_files, g_free); } #endif /* Load the images given on the command-line. */ filenames = pika_core_app_get_filenames (app); if (filenames != NULL) { gint i; for (i = 0; filenames[i] != NULL; i++) { GFile *file = g_file_new_for_commandline_arg (filenames[i]); file_open_from_command_line (pika, file, pika_core_app_get_as_new (app), initial_monitor); g_object_unref (file); } } /* The software is now fully loaded and ready to be used and get * external input. */ pika->initialized = TRUE; if (font_error) { pika_message_literal (pika, NULL, PIKA_MESSAGE_INFO, font_error->message); g_error_free (font_error); } batch_retval = pika_batch_run (pika, pika_core_app_get_batch_interpreter (app), pika_core_app_get_batch_commands (app)); if (pika_core_app_get_quit (app)) { /* Only if we are in batch mode, we want to exit with the * return value of the batch command. */ pika_core_app_set_exit_status (app, batch_retval); /* Emit the "exit" signal, but also properly free all images still * opened. */ pika_exit (pika, TRUE); } } static void app_restore_after_callback (Pika *pika, PikaInitStatusFunc status_callback) { gint dummy; /* Getting the display name for a -1 display returns the initial * monitor during startup. Need to call this from a restore_after * callback, because before restore(), the GUI can't return anything, * after after restore() the initial monitor gets reset. */ g_free (pika_get_display_name (pika, -1, &initial_monitor, &dummy)); } static gboolean app_exit_after_callback (Pika *pika, gboolean kill_it, GApplication *app) { if (pika->be_verbose) g_print ("EXIT: %s\n", G_STRFUNC); /* * In stable releases, we simply call exit() here. This speeds up * the process of quitting PIKA and also works around the problem * that plug-ins might still be running. * * In unstable releases, we shut down PIKA properly in an attempt * to catch possible problems in our finalizers. */ #ifdef PIKA_UNSTABLE g_application_quit (G_APPLICATION (app)); #else pika_gegl_exit (pika); gegl_exit (); exit (pika_core_app_get_exit_status (PIKA_CORE_APP (app))); #endif return FALSE; }