PIKApp/app/gui/session.c

490 lines
15 KiB
C
Raw Normal View History

2023-09-26 00:35:21 +02:00
/* 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
*
* Session-managment stuff
* Copyright (C) 1998 Sven Neumann <sven@gimp.org>
*
* 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 <https://www.gnu.org/licenses/>.
*/
#include "config.h"
#include <gegl.h>
#include <gtk/gtk.h>
#include "libpikabase/pikabase.h"
#include "libpikaconfig/pikaconfig.h"
#include "gui-types.h"
#include "config/pikaconfig-file.h"
#include "config/pikaguiconfig.h"
#include "core/pika.h"
#include "core/pikaerror.h"
#include "widgets/pikadialogfactory.h"
#include "widgets/pikasessioninfo.h"
#include "widgets/pikawidgets-utils.h"
#include "dialogs/dialogs.h"
#include "session.h"
#include "pika-log.h"
#include "pika-intl.h"
enum
{
SESSION_INFO = 1,
HIDE_DOCKS,
SINGLE_WINDOW_MODE,
SHOW_TABS,
TABS_POSITION,
LAST_TIP_SHOWN
};
static GFile * session_file (Pika *pika);
/* private variables */
static gboolean sessionrc_deleted = FALSE;
/* public functions */
void
session_init (Pika *pika)
{
GFile *file;
GScanner *scanner;
GTokenType token;
GError *error = NULL;
g_return_if_fail (PIKA_IS_PIKA (pika));
file = session_file (pika);
scanner = pika_scanner_new_file (file, &error);
if (! scanner && error->code == PIKA_CONFIG_ERROR_OPEN_ENOENT)
{
g_clear_error (&error);
g_object_unref (file);
file = pika_sysconf_directory_file ("sessionrc", NULL);
scanner = pika_scanner_new_file (file, NULL);
}
if (! scanner)
{
g_clear_error (&error);
g_object_unref (file);
return;
}
if (pika->be_verbose)
g_print ("Parsing '%s'\n", pika_file_get_utf8_name (file));
g_scanner_scope_add_symbol (scanner, 0, "session-info",
GINT_TO_POINTER (SESSION_INFO));
g_scanner_scope_add_symbol (scanner, 0, "hide-docks",
GINT_TO_POINTER (HIDE_DOCKS));
g_scanner_scope_add_symbol (scanner, 0, "single-window-mode",
GINT_TO_POINTER (SINGLE_WINDOW_MODE));
g_scanner_scope_add_symbol (scanner, 0, "show-tabs",
GINT_TO_POINTER (SHOW_TABS));
g_scanner_scope_add_symbol (scanner, 0, "tabs-position",
GINT_TO_POINTER (TABS_POSITION));
g_scanner_scope_add_symbol (scanner, 0, "last-tip-shown",
GINT_TO_POINTER (LAST_TIP_SHOWN));
token = G_TOKEN_LEFT_PAREN;
while (g_scanner_peek_next_token (scanner) == token)
{
token = g_scanner_get_next_token (scanner);
switch (token)
{
case G_TOKEN_LEFT_PAREN:
token = G_TOKEN_SYMBOL;
break;
case G_TOKEN_SYMBOL:
if (scanner->value.v_symbol == GINT_TO_POINTER (SESSION_INFO))
{
PikaDialogFactory *factory = NULL;
PikaSessionInfo *info = NULL;
gchar *factory_name = NULL;
gchar *entry_name = NULL;
PikaDialogFactoryEntry *entry = NULL;
token = G_TOKEN_STRING;
if (! pika_scanner_parse_string (scanner, &factory_name))
break;
/* In versions <= PIKA 2.6 there was a "toolbox", a
* "dock", a "display" and a "toplevel" factory. These
* are now merged to a single pika_dialog_factory_get_singleton (). We
* need the legacy name though, so keep it around.
*/
factory = pika_dialog_factory_get_singleton ();
info = pika_session_info_new ();
/* PIKA 2.6 has the entry name as part of the
* session-info header, so try to get it
*/
pika_scanner_parse_string (scanner, &entry_name);
if (entry_name)
{
/* Previously, PikaDock was a toplevel. That is why
* versions <= PIKA 2.6 has "dock" as the entry name. We
* want "dock" to be interpreted as 'dock window'
* however so have some special-casing for that. When
* the entry name is "dock" the factory name is either
* "dock" or "toolbox".
*/
if (strcmp (entry_name, "dock") == 0)
{
entry =
pika_dialog_factory_find_entry (factory,
(strcmp (factory_name, "toolbox") == 0 ?
"pika-toolbox-window" :
"pika-dock-window"));
}
else
{
entry = pika_dialog_factory_find_entry (factory,
entry_name);
}
}
/* We're done with these now */
g_free (factory_name);
g_free (entry_name);
/* We can get the factory entry either now (the PIKA <=
* 2.6 way), or when we deserialize (the PIKA 2.8 way)
*/
if (entry)
{
pika_session_info_set_factory_entry (info, entry);
}
/* Always try to deserialize */
if (pika_config_deserialize (PIKA_CONFIG (info), scanner, 1, NULL))
{
/* Make sure we got a factory entry either the 2.6
* or 2.8 way
*/
if (pika_session_info_get_factory_entry (info))
{
PIKA_LOG (DIALOG_FACTORY,
"successfully parsed and added session info %p",
info);
pika_dialog_factory_add_session_info (factory, info);
}
else
{
PIKA_LOG (DIALOG_FACTORY,
"failed to parse session info %p, not adding",
info);
}
g_object_unref (info);
}
else
{
g_object_unref (info);
/* set token to left paren so we won't set another
* error below, pika_config_deserialize() already did
*/
token = G_TOKEN_LEFT_PAREN;
goto error;
}
}
else if (scanner->value.v_symbol == GINT_TO_POINTER (HIDE_DOCKS))
{
gboolean hide_docks;
token = G_TOKEN_IDENTIFIER;
if (! pika_scanner_parse_boolean (scanner, &hide_docks))
break;
g_object_set (pika->config,
"hide-docks", hide_docks,
NULL);
}
else if (scanner->value.v_symbol == GINT_TO_POINTER (SINGLE_WINDOW_MODE))
{
gboolean single_window_mode;
token = G_TOKEN_IDENTIFIER;
if (! pika_scanner_parse_boolean (scanner, &single_window_mode))
break;
g_object_set (pika->config,
"single-window-mode", single_window_mode,
NULL);
}
else if (scanner->value.v_symbol == GINT_TO_POINTER (SHOW_TABS))
{
gboolean show_tabs;
token = G_TOKEN_IDENTIFIER;
if (! pika_scanner_parse_boolean (scanner, &show_tabs))
break;
g_object_set (pika->config,
"show-tabs", show_tabs,
NULL);
}
else if (scanner->value.v_symbol == GINT_TO_POINTER (TABS_POSITION))
{
gint tabs_position;
token = G_TOKEN_INT;
if (! pika_scanner_parse_int (scanner, &tabs_position))
break;
g_object_set (pika->config,
"tabs-position", tabs_position,
NULL);
}
else if (scanner->value.v_symbol == GINT_TO_POINTER (LAST_TIP_SHOWN))
{
gint last_tip_shown;
token = G_TOKEN_INT;
if (! pika_scanner_parse_int (scanner, &last_tip_shown))
break;
g_object_set (pika->config,
"last-tip-shown", last_tip_shown,
NULL);
}
token = G_TOKEN_RIGHT_PAREN;
break;
case G_TOKEN_RIGHT_PAREN:
token = G_TOKEN_LEFT_PAREN;
break;
default: /* do nothing */
break;
}
}
error:
if (token != G_TOKEN_LEFT_PAREN)
{
g_scanner_get_next_token (scanner);
g_scanner_unexp_token (scanner, token, NULL, NULL, NULL,
_("fatal parse error"), TRUE);
}
if (error)
{
pika_message_literal (pika, NULL, PIKA_MESSAGE_ERROR, error->message);
g_clear_error (&error);
pika_config_file_backup_on_error (file, "sessionrc", NULL);
}
pika_scanner_unref (scanner);
g_object_unref (file);
dialogs_load_recent_docks (pika);
}
void
session_exit (Pika *pika)
{
g_return_if_fail (PIKA_IS_PIKA (pika));
}
void
session_restore (Pika *pika,
GdkMonitor *monitor)
{
g_return_if_fail (PIKA_IS_PIKA (pika));
g_return_if_fail (GDK_IS_MONITOR (monitor));
pika_dialog_factory_restore (pika_dialog_factory_get_singleton (),
monitor);
/* make sure PikaImageWindow acts upon hide-docks at the right time,
* see bug #678043.
*/
if (PIKA_GUI_CONFIG (pika->config)->single_window_mode &&
PIKA_GUI_CONFIG (pika->config)->hide_docks)
{
g_object_notify (G_OBJECT (pika->config), "hide-docks");
}
}
void
session_save (Pika *pika,
gboolean always_save)
{
PikaConfigWriter *writer;
GFile *file;
GError *error = NULL;
g_return_if_fail (PIKA_IS_PIKA (pika));
if (sessionrc_deleted && ! always_save)
return;
file = session_file (pika);
if (pika->be_verbose)
g_print ("Writing '%s'\n", pika_file_get_utf8_name (file));
writer =
pika_config_writer_new_from_file (file,
TRUE,
"PIKA sessionrc\n\n"
"This file takes session-specific info "
"(that is info, you want to keep between "
"two PIKA sessions). You are not supposed "
"to edit it manually, but of course you "
"can do. The sessionrc will be entirely "
"rewritten every time you quit PIKA. "
"If this file isn't found, defaults are "
"used.",
NULL);
g_object_unref (file);
if (!writer)
return;
pika_dialog_factory_save (pika_dialog_factory_get_singleton (), writer);
pika_config_writer_linefeed (writer);
pika_config_writer_open (writer, "hide-docks");
pika_config_writer_identifier (writer,
PIKA_GUI_CONFIG (pika->config)->hide_docks ?
"yes" : "no");
pika_config_writer_close (writer);
pika_config_writer_open (writer, "single-window-mode");
pika_config_writer_identifier (writer,
PIKA_GUI_CONFIG (pika->config)->single_window_mode ?
"yes" : "no");
pika_config_writer_close (writer);
pika_config_writer_open (writer, "show-tabs");
pika_config_writer_printf (writer,
PIKA_GUI_CONFIG (pika->config)->show_tabs ?
"yes" : "no");
pika_config_writer_close (writer);
pika_config_writer_open (writer, "tabs-position");
pika_config_writer_printf (writer, "%d",
PIKA_GUI_CONFIG (pika->config)->tabs_position);
pika_config_writer_close (writer);
pika_config_writer_open (writer, "last-tip-shown");
pika_config_writer_printf (writer, "%d",
PIKA_GUI_CONFIG (pika->config)->last_tip_shown);
pika_config_writer_close (writer);
if (! pika_config_writer_finish (writer, "end of sessionrc", &error))
{
pika_message_literal (pika, NULL, PIKA_MESSAGE_ERROR, error->message);
g_clear_error (&error);
}
dialogs_save_recent_docks (pika);
sessionrc_deleted = FALSE;
}
gboolean
session_clear (Pika *pika,
GError **error)
{
GFile *file;
GError *my_error = NULL;
gboolean success = TRUE;
g_return_val_if_fail (PIKA_IS_PIKA (pika), FALSE);
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
file = session_file (pika);
if (! g_file_delete (file, NULL, &my_error) &&
my_error->code != G_IO_ERROR_NOT_FOUND)
{
success = FALSE;
g_set_error (error, PIKA_ERROR, PIKA_FAILED,
_("Deleting \"%s\" failed: %s"),
pika_file_get_utf8_name (file), my_error->message);
}
else
{
sessionrc_deleted = TRUE;
}
g_clear_error (&my_error);
g_object_unref (file);
return success;
}
static GFile *
session_file (Pika *pika)
{
const gchar *basename;
gchar *filename;
GFile *file;
basename = g_getenv ("PIKA_TESTING_SESSIONRC_NAME");
if (! basename)
basename = "sessionrc";
if (pika->session_name)
filename = g_strconcat (basename, ".", pika->session_name, NULL);
else
filename = g_strdup (basename);
file = pika_directory_file (filename, NULL);
g_free (filename);
return file;
}