zathura/zathura/zathura.c

1516 lines
44 KiB
C
Raw Normal View History

/* See LICENSE file for license and copyright information */
#define _DEFAULT_SOURCE
2011-10-23 20:18:44 +02:00
#define _XOPEN_SOURCE 700
2011-10-21 15:00:22 +02:00
#include <errno.h>
2010-12-28 09:47:09 +01:00
#include <stdlib.h>
2012-02-08 21:34:53 +01:00
#include <math.h>
#include <string.h>
2010-12-28 09:47:09 +01:00
#include <girara/datastructures.h>
#include <girara/utils.h>
#include <girara/session.h>
#include <girara/statusbar.h>
#include <girara/settings.h>
Hide the inputbar and completion menu before saving the adjustments ratios into a new jump structure Since we are now saving the adjustments ratios in the jump structures, we need to take care of the following scenario: - We do an action that results in a new jump structure being added to the jumplist while the inputbar is visible (e.g., search, jumping to a specific page, jumping to a bookmark, or following a link). - Since we are now storing the adjustments ratios in the jump structures, all of the above actions would result in the vertical adjustment ratio being saved while the inputbar and/or the completion menu is visible. - Now we are exactly on the target of the jump (note that the inputbar and completion menu now are hidden), so suppose that we want to go back using ^o (assuming that we didn't change the adjustments after jumping), then the check at sc_jumplist that compares the current adjustments ratios with that of the current jump (the jump that has just been added and which we are currently on it's position) would fail, because after the inputbar (with possibly the completion menu in case of bookmarks) is activated it is hidden, which results in the vertical adjustment upper bound to change, which in turn results in the vertical adjustment ratio returned by zathura_adjustment_get_ratio to become different from what is stored in the current jump structure, even though we haven't changed the adjustments at all after the jump. This would always result in taking us back to the exact position of the jump (which would be slightly different from the current position) when we press ^o. This can be annoying, because it would happen, for example, every time we need to go back quickly after jumping to a link target, a search result, or a bookmark. So, what this patch does is essentially to make the vertical adjustment ratio reflecting the current vertical adjustment after a jump, to always be the same as the one stored in the newly added jump structure, since both are calculated with zathura_adjustment_get_ratio while the inputbar is _not_ visible, so they should be the same. I've elaborated just to make things clear, in case the purpose of the patch isn't obvious. Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-21 14:25:29 +02:00
#include <girara/shortcuts.h>
#include <girara/template.h>
2011-10-21 15:00:22 +02:00
#include <glib/gstdio.h>
2012-03-04 18:35:16 +01:00
#include <glib/gi18n.h>
2011-04-18 17:03:08 +02:00
2014-04-23 17:27:24 +02:00
#ifdef G_OS_UNIX
#include <glib-unix.h>
#include <gio/gunixinputstream.h>
2014-04-23 17:27:24 +02:00
#endif
2011-09-01 15:43:34 +02:00
#include "bookmarks.h"
2010-11-12 13:48:18 +01:00
#include "callbacks.h"
#include "config.h"
#ifdef WITH_SQLITE
#include "database-sqlite.h"
#endif
#include "database-plain.h"
#include "document.h"
2010-11-12 13:48:18 +01:00
#include "shortcuts.h"
2010-11-10 19:18:01 +01:00
#include "zathura.h"
2010-12-28 09:47:09 +01:00
#include "utils.h"
2012-04-21 04:59:58 +02:00
#include "marks.h"
2011-04-18 18:19:41 +02:00
#include "render.h"
2012-03-26 14:44:56 +02:00
#include "page.h"
#include "page-widget.h"
2012-03-27 13:30:04 +02:00
#include "plugin.h"
#include "adjustment.h"
#include "dbus-interface.h"
#include "resources.h"
#include "synctex.h"
#include "content-type.h"
2009-12-26 17:57:46 +01:00
2012-10-09 01:12:18 +02:00
typedef struct zathura_document_info_s {
2011-04-18 21:22:35 +02:00
zathura_t* zathura;
char* path;
char* password;
int page_number;
char* mode;
char* synctex;
2011-04-18 21:22:35 +02:00
} zathura_document_info_t;
static gboolean document_info_open(gpointer data);
2009-12-26 17:57:46 +01:00
2014-04-23 17:27:24 +02:00
#ifdef G_OS_UNIX
static gboolean zathura_signal_sigterm(gpointer data);
#endif
static void
free_document_info(zathura_document_info_t* document_info)
{
if (document_info == NULL) {
return;
}
g_free(document_info->path);
g_free(document_info->password);
g_free(document_info->mode);
g_free(document_info->synctex);
g_free(document_info);
}
2010-11-10 19:18:01 +01:00
/* function implementation */
2011-04-18 17:27:49 +02:00
zathura_t*
2012-08-05 15:34:10 +02:00
zathura_create(void)
2010-06-03 18:05:34 +02:00
{
zathura_t* zathura = g_try_malloc0(sizeof(zathura_t));
if (zathura == NULL) {
return NULL;
}
2012-07-12 10:37:58 +02:00
/* global settings */
zathura->global.search_direction = FORWARD;
2012-07-12 10:37:58 +02:00
/* plugins */
2012-04-01 18:32:16 +02:00
zathura->plugins.manager = zathura_plugin_manager_new();
if (zathura->plugins.manager == NULL) {
2012-08-05 15:34:10 +02:00
goto error_out;
2012-04-01 18:32:16 +02:00
}
2012-08-05 15:34:10 +02:00
/* UI */
zathura->ui.session = girara_session_create();
if (zathura->ui.session == NULL) {
2012-08-05 15:34:10 +02:00
goto error_out;
2011-04-19 14:46:08 +02:00
}
2014-04-23 17:27:24 +02:00
#ifdef G_OS_UNIX
/* signal handler */
zathura->signals.sigterm = g_unix_signal_add(SIGTERM, zathura_signal_sigterm, zathura);
#endif
/* MIME type detection */
zathura->content_type_context = zathura_content_type_new();
2012-08-05 15:34:10 +02:00
zathura->ui.session->global.data = zathura;
return zathura;
error_out:
zathura_free(zathura);
return NULL;
}
2011-04-19 14:46:08 +02:00
static void
create_directories(zathura_t* zathura)
2012-08-05 15:34:10 +02:00
{
static const unsigned int mode = 0700;
if (g_mkdir_with_parents(zathura->config.config_dir, mode) == -1) {
girara_error("Could not create '%s': %s", zathura->config.config_dir,
strerror(errno));
}
if (g_mkdir_with_parents(zathura->config.data_dir, mode) == -1) {
girara_error("Could not create '%s': %s", zathura->config.data_dir,
strerror(errno));
}
}
static bool
init_ui(zathura_t* zathura)
{
2011-12-13 19:59:59 +01:00
if (girara_session_init(zathura->ui.session, "zathura") == false) {
return false;
}
/* girara events */
2012-05-01 19:09:33 +02:00
zathura->ui.session->events.buffer_changed = cb_buffer_changed;
zathura->ui.session->events.unknown_command = cb_unknown_command;
/* zathura signals */
zathura->signals.refresh_view = g_signal_new(
"refresh-view", GTK_TYPE_WIDGET, G_SIGNAL_RUN_LAST, 0, NULL, NULL,
g_cclosure_marshal_generic, G_TYPE_NONE, 1, G_TYPE_POINTER);
g_signal_connect(G_OBJECT(zathura->ui.session->gtk.view), "refresh-view",
G_CALLBACK(cb_refresh_view), zathura);
2010-12-29 11:46:13 +01:00
/* page view */
2012-08-05 02:30:03 +02:00
zathura->ui.page_widget = gtk_grid_new();
2012-08-05 15:56:35 +02:00
gtk_grid_set_row_homogeneous(GTK_GRID(zathura->ui.page_widget), TRUE);
gtk_grid_set_column_homogeneous(GTK_GRID(zathura->ui.page_widget), TRUE);
2012-02-12 16:35:33 +01:00
if (zathura->ui.page_widget == NULL) {
return false;
2010-12-29 11:46:13 +01:00
}
g_signal_connect(G_OBJECT(zathura->ui.session->gtk.window), "size-allocate",
G_CALLBACK(cb_view_resized), zathura);
2012-02-09 01:46:51 +01:00
GtkAdjustment* hadjustment = gtk_scrolled_window_get_hadjustment(
GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view));
/* Connect hadjustment signals */
g_signal_connect(G_OBJECT(hadjustment), "value-changed",
G_CALLBACK(cb_view_hadjustment_value_changed), zathura);
g_signal_connect(G_OBJECT(hadjustment), "changed",
G_CALLBACK(cb_view_hadjustment_changed), zathura);
GtkAdjustment* vadjustment = gtk_scrolled_window_get_vadjustment(
GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view));
/* Connect vadjustment signals */
g_signal_connect(G_OBJECT(vadjustment), "value-changed",
G_CALLBACK(cb_view_vadjustment_value_changed), zathura);
g_signal_connect(G_OBJECT(vadjustment), "changed",
G_CALLBACK(cb_view_vadjustment_changed), zathura);
2011-04-19 21:42:18 +02:00
2012-01-13 17:39:46 +01:00
/* page view alignment */
gtk_widget_set_halign(zathura->ui.page_widget, GTK_ALIGN_CENTER);
gtk_widget_set_valign(zathura->ui.page_widget, GTK_ALIGN_CENTER);
2012-01-13 17:39:46 +01:00
gtk_widget_set_hexpand_set(zathura->ui.page_widget, TRUE);
gtk_widget_set_hexpand(zathura->ui.page_widget, FALSE);
gtk_widget_set_vexpand_set(zathura->ui.page_widget, TRUE);
gtk_widget_set_vexpand(zathura->ui.page_widget, FALSE);
2012-08-05 15:56:35 +02:00
2012-02-07 18:30:46 +01:00
gtk_widget_show(zathura->ui.page_widget);
2010-12-26 01:52:17 +01:00
/* statusbar */
zathura->ui.statusbar.file =
girara_statusbar_item_add(zathura->ui.session, TRUE, TRUE, TRUE, NULL);
2011-04-18 17:27:49 +02:00
if (zathura->ui.statusbar.file == NULL) {
return false;
2010-11-13 12:40:48 +01:00
}
zathura->ui.statusbar.buffer =
girara_statusbar_item_add(zathura->ui.session, FALSE, FALSE, FALSE, NULL);
2011-04-18 17:27:49 +02:00
if (zathura->ui.statusbar.buffer == NULL) {
return false;
2010-11-18 21:22:43 +01:00
}
zathura->ui.statusbar.page_number =
girara_statusbar_item_add(zathura->ui.session, FALSE, FALSE, FALSE, NULL);
2012-02-12 16:35:33 +01:00
if (zathura->ui.statusbar.page_number == NULL) {
return false;
2010-11-18 21:22:43 +01:00
}
girara_statusbar_item_set_text(zathura->ui.session,
zathura->ui.statusbar.file, _("[No name]"));
2010-11-18 21:22:43 +01:00
2010-11-12 13:48:18 +01:00
/* signals */
g_signal_connect(G_OBJECT(zathura->ui.session->gtk.window), "destroy",
G_CALLBACK(cb_destroy), zathura);
2010-12-29 11:46:13 +01:00
return true;
}
static void
init_css(zathura_t* zathura)
{
GiraraTemplate* csstemplate =
girara_session_get_template(zathura->ui.session);
static const char* index_settings[] = {
"index-fg",
"index-bg",
"index-active-fg",
"index-active-bg"
};
for (size_t s = 0; s < LENGTH(index_settings); ++s) {
girara_template_add_variable(csstemplate, index_settings[s]);
char* tmp_value = NULL;
GdkRGBA rgba = {0, 0, 0, 0};
girara_setting_get(zathura->ui.session, index_settings[s], &tmp_value);
if (tmp_value != NULL) {
gdk_rgba_parse(&rgba, tmp_value);
g_free(tmp_value);
}
char* color = gdk_rgba_to_string(&rgba);
girara_template_set_variable_value(csstemplate, index_settings[s], color);
g_free(color);
}
GResource* css_resource = zathura_resources_get_resource();
GBytes* css_data = g_resource_lookup_data(css_resource,
"/org/pwmt/zathura/CSS/zathura.css_t",
G_RESOURCE_LOOKUP_FLAGS_NONE, NULL);
if (css_data != NULL) {
char* css = g_strdup_printf("%s\n%s", girara_template_get_base(csstemplate),
(const char*) g_bytes_get_data(css_data, NULL));
girara_template_set_base(csstemplate, css);
g_free(css);
g_bytes_unref(css_data);
}
}
static void
init_database(zathura_t* zathura)
{
char* database = NULL;
girara_setting_get(zathura->ui.session, "database", &database);
if (g_strcmp0(database, "plain") == 0) {
girara_debug("Using plain database backend.");
zathura->database = zathura_plaindatabase_new(zathura->config.data_dir);
#ifdef WITH_SQLITE
} else if (g_strcmp0(database, "sqlite") == 0) {
girara_debug("Using sqlite database backend.");
char* tmp =
g_build_filename(zathura->config.data_dir, "bookmarks.sqlite", NULL);
zathura->database = zathura_sqldatabase_new(tmp);
g_free(tmp);
#endif
} else if (g_strcmp0(database, "null") != 0) {
girara_error("Database backend '%s' is not supported.", database);
}
if (zathura->database == NULL && g_strcmp0(database, "null") != 0) {
girara_error(
"Unable to initialize database. Bookmarks won't be available.");
}
else {
g_object_set(G_OBJECT(zathura->ui.session->command_history), "io",
zathura->database, NULL);
}
g_free(database);
}
static void
init_jumplist(zathura_t* zathura)
{
int jumplist_size = 20;
girara_setting_get(zathura->ui.session, "jumplist-size", &jumplist_size);
zathura->jumplist.max_size = jumplist_size < 0 ? 0 : jumplist_size;
zathura->jumplist.list = NULL;
zathura->jumplist.size = 0;
zathura->jumplist.cur = NULL;
}
static void
init_shortcut_helpers(zathura_t* zathura)
{
zathura->shortcut.mouse.x = 0;
zathura->shortcut.mouse.y = 0;
zathura->shortcut.toggle_page_mode.pages = 2;
zathura->shortcut.toggle_presentation_mode.pages = 1;
zathura->shortcut.toggle_presentation_mode.first_page_column_list = NULL;
zathura->shortcut.toggle_presentation_mode.zoom = 1.0;
}
bool
zathura_init(zathura_t* zathura)
{
if (zathura == NULL) {
return false;
}
/* create zathura (config/data) directory */
create_directories(zathura);
/* load plugins */
zathura_plugin_manager_load(zathura->plugins.manager);
/* configuration */
config_load_default(zathura);
config_load_files(zathura);
/* UI */
if (!init_ui(zathura)) {
goto error_free;
}
/* database */
init_database(zathura);
/* bookmarks */
zathura->bookmarks.bookmarks = girara_sorted_list_new2(
(girara_compare_function_t)zathura_bookmarks_compare,
(girara_free_function_t)zathura_bookmark_free);
/* jumplist */
init_jumplist(zathura);
/* CSS for index mode */
init_css(zathura);
/* Shortcut helpers */
init_shortcut_helpers(zathura);
/* Start D-Bus service */
bool dbus = true;
girara_setting_get(zathura->ui.session, "dbus-service", &dbus);
if (dbus == true) {
zathura->dbus = zathura_dbus_new(zathura);
}
2012-08-05 15:34:10 +02:00
return true;
2010-12-26 01:52:17 +01:00
error_free:
2012-02-12 16:35:33 +01:00
if (zathura->ui.page_widget != NULL) {
2012-02-07 18:30:46 +01:00
g_object_unref(zathura->ui.page_widget);
2010-12-29 11:46:13 +01:00
}
2012-08-05 15:34:10 +02:00
return false;
2011-04-18 17:27:49 +02:00
}
void
zathura_free(zathura_t* zathura)
{
if (zathura == NULL) {
return;
}
2012-02-20 20:07:24 +01:00
document_close(zathura, false);
/* MIME type detection */
zathura_content_type_free(zathura->content_type_context);
#ifdef G_OS_UNIX
if (zathura->signals.sigterm > 0) {
g_source_remove(zathura->signals.sigterm);
zathura->signals.sigterm = 0;
}
#endif
/* stop D-Bus */
g_clear_object(&zathura->dbus);
2011-04-18 17:27:49 +02:00
if (zathura->ui.session != NULL) {
girara_session_destroy(zathura->ui.session);
}
/* shortcut */
if (zathura->shortcut.toggle_presentation_mode.first_page_column_list != NULL) {
g_free(zathura->shortcut.toggle_presentation_mode.first_page_column_list);
}
2011-10-21 15:00:22 +02:00
/* stdin support */
if (zathura->stdin_support.file != NULL) {
g_unlink(zathura->stdin_support.file);
g_free(zathura->stdin_support.file);
}
2011-09-01 15:43:34 +02:00
/* bookmarks */
girara_list_free(zathura->bookmarks.bookmarks);
/* database */
g_clear_object(&zathura->database);
2011-04-29 00:28:19 +02:00
/* free print settings */
g_clear_object(&zathura->print.settings);
g_clear_object(&zathura->print.page_setup);
2011-04-29 00:28:19 +02:00
2011-04-18 17:27:49 +02:00
/* free registered plugins */
2012-04-01 18:32:16 +02:00
zathura_plugin_manager_free(zathura->plugins.manager);
/* free config variables */
g_free(zathura->config.config_dir);
g_free(zathura->config.data_dir);
g_free(zathura->config.cache_dir);
2011-09-30 12:33:50 +02:00
/* free jumplist */
if (zathura->jumplist.list != NULL) {
girara_list_free(zathura->jumplist.list);
}
if (zathura->jumplist.cur != NULL) {
girara_list_iterator_free(zathura->jumplist.cur);
}
g_free(zathura);
2010-06-03 18:05:34 +02:00
}
2012-08-05 15:34:10 +02:00
void
zathura_set_xid(zathura_t* zathura, Window xid)
{
g_return_if_fail(zathura != NULL);
zathura->ui.session->gtk.embed = xid;
}
void
zathura_set_config_dir(zathura_t* zathura, const char* dir)
{
g_return_if_fail(zathura != NULL);
if (dir != NULL) {
zathura->config.config_dir = g_strdup(dir);
} else {
gchar* path = girara_get_xdg_path(XDG_CONFIG);
zathura->config.config_dir = g_build_filename(path, "zathura", NULL);
g_free(path);
}
}
void
zathura_set_data_dir(zathura_t* zathura, const char* dir)
{
g_return_if_fail(zathura != NULL);
2012-08-05 15:34:10 +02:00
if (dir != NULL) {
zathura->config.data_dir = g_strdup(dir);
} else {
gchar* path = girara_get_xdg_path(XDG_DATA);
zathura->config.data_dir = g_build_filename(path, "zathura", NULL);
g_free(path);
}
}
2012-08-05 15:34:10 +02:00
void
zathura_set_cache_dir(zathura_t* zathura, const char* dir)
{
2012-08-05 15:34:10 +02:00
g_return_if_fail(zathura != NULL);
if (dir != NULL) {
zathura->config.cache_dir = g_strdup(dir);
} else {
gchar* path = girara_get_xdg_path(XDG_CACHE);
zathura->config.cache_dir = g_build_filename(path, "zathura", NULL);
g_free(path);
}
2012-08-05 15:34:10 +02:00
}
void
zathura_set_plugin_dir(zathura_t* zathura, const char* dir)
{
g_return_if_fail(zathura != NULL);
g_return_if_fail(zathura->plugins.manager != NULL);
if (dir != NULL) {
girara_list_t* paths = girara_split_path_array(dir);
GIRARA_LIST_FOREACH(paths, char*, iter, path)
2012-10-09 01:12:18 +02:00
zathura_plugin_manager_add_dir(zathura->plugins.manager, path);
2012-08-05 15:34:10 +02:00
GIRARA_LIST_FOREACH_END(paths, char*, iter, path);
girara_list_free(paths);
} else {
#ifdef ZATHURA_PLUGINDIR
girara_list_t* paths = girara_split_path_array(ZATHURA_PLUGINDIR);
GIRARA_LIST_FOREACH(paths, char*, iter, path)
2012-10-09 01:12:18 +02:00
zathura_plugin_manager_add_dir(zathura->plugins.manager, path);
2012-08-05 15:34:10 +02:00
GIRARA_LIST_FOREACH_END(paths, char*, iter, path);
girara_list_free(paths);
#endif
}
}
void
zathura_set_argv(zathura_t* zathura, char** argv)
{
g_return_if_fail(zathura != NULL);
zathura->global.arguments = argv;
}
#ifdef G_OS_UNIX
2011-10-21 15:00:22 +02:00
static gchar*
prepare_document_open_from_stdin(const char* path)
2011-10-21 15:00:22 +02:00
{
int infileno = -1;
if (g_strcmp0(path, "-") == 0) {
infileno = fileno(stdin);
} else if (g_str_has_prefix(path, "/proc/self/fd/") == true) {
char* begin = g_strrstr(path, "/") + 1;
gint64 temp = g_ascii_strtoll(begin, NULL, 0);
if (temp > INT_MAX || temp < 0) {
return NULL;
}
infileno = (int) temp;
} else {
return NULL;
}
if (infileno == -1) {
girara_error("Can not read from file descriptor.");
return NULL;
}
GInputStream* input_stream = g_unix_input_stream_new(infileno, false);
if (input_stream == NULL) {
girara_error("Can not read from file descriptor.");
return NULL;
}
GFileIOStream* iostream = NULL;
GError* error = NULL;
GFile* tmpfile = g_file_new_tmp("zathura.stdin.XXXXXX", &iostream, &error);
if (tmpfile == NULL) {
if (error != NULL) {
girara_error("Can not create temporary file: %s", error->message);
g_error_free(error);
}
g_object_unref(input_stream);
2011-10-21 15:00:22 +02:00
return NULL;
}
const ssize_t count = g_output_stream_splice(
g_io_stream_get_output_stream(G_IO_STREAM(iostream)), input_stream,
G_OUTPUT_STREAM_SPLICE_NONE, NULL, &error);
g_object_unref(input_stream);
g_object_unref(iostream);
if (count == -1) {
if (error != NULL) {
girara_error("Can not write to temporary file: %s", error->message);
g_error_free(error);
2011-10-21 15:00:22 +02:00
}
g_file_delete(tmpfile, NULL, NULL);
g_object_unref(tmpfile);
2011-10-21 15:00:22 +02:00
return NULL;
}
char* file = g_file_get_path(tmpfile);
g_object_unref(tmpfile);
2011-10-21 15:00:22 +02:00
return file;
}
#endif
2011-10-21 15:00:22 +02:00
2015-12-02 22:56:47 +01:00
static gchar*
prepare_document_open_from_gfile(GFile* source)
2015-12-02 22:56:47 +01:00
{
gchar* file = NULL;
GFileIOStream* iostream = NULL;
GError* error = NULL;
2015-12-02 22:56:47 +01:00
GFile* tmpfile = g_file_new_tmp("zathura.gio.XXXXXX", &iostream, &error);
if (tmpfile == NULL) {
2015-12-02 22:56:47 +01:00
if (error != NULL) {
girara_error("Can not create temporary file: %s", error->message);
g_error_free(error);
}
return NULL;
}
gboolean rc = g_file_copy(source, tmpfile, G_FILE_COPY_OVERWRITE, NULL, NULL,
NULL, &error);
if (rc == FALSE) {
2015-12-02 22:56:47 +01:00
if (error != NULL) {
girara_error("Can not copy to temporary file: %s", error->message);
g_error_free(error);
}
g_object_unref(iostream);
g_object_unref(tmpfile);
return NULL;
}
file = g_file_get_path(tmpfile);
g_object_unref(iostream);
g_object_unref(tmpfile);
return file;
}
static gboolean
2011-04-18 21:22:35 +02:00
document_info_open(gpointer data)
{
zathura_document_info_t* document_info = data;
g_return_val_if_fail(document_info != NULL, FALSE);
2015-12-07 01:13:59 +01:00
char* uri = NULL;
2011-04-18 21:22:35 +02:00
if (document_info->zathura != NULL && document_info->path != NULL) {
2011-10-21 15:00:22 +02:00
char* file = NULL;
if (g_strcmp0(document_info->path, "-") == 0 ||
g_str_has_prefix(document_info->path, "/proc/self/fd/") == true) {
#ifdef G_OS_UNIX
file = prepare_document_open_from_stdin(document_info->path);
#endif
2011-10-21 15:00:22 +02:00
if (file == NULL) {
girara_notify(document_info->zathura->ui.session, GIRARA_ERROR,
2013-11-04 19:57:23 +01:00
_("Could not read file from stdin and write it to a temporary file."));
2011-10-23 17:34:10 +02:00
} else {
document_info->zathura->stdin_support.file = g_strdup(file);
2011-10-21 15:00:22 +02:00
}
} else {
GFile* gf = g_file_new_for_commandline_arg(document_info->path);
if (g_file_is_native(gf) == TRUE) {
/* file was given as a native path */
file = g_file_get_path(gf);
2015-12-02 22:56:47 +01:00
}
else {
/* copy file with GIO */
2015-12-07 01:13:59 +01:00
uri = g_file_get_uri(gf);
file = prepare_document_open_from_gfile(gf);
if (file == NULL) {
girara_notify(document_info->zathura->ui.session, GIRARA_ERROR,
_("Could not read file from GIO and copy it to a temporary file."));
2015-12-02 22:56:47 +01:00
} else {
document_info->zathura->stdin_support.file = g_strdup(file);
2015-12-02 22:56:47 +01:00
}
}
g_object_unref(gf);
2011-10-21 15:00:22 +02:00
}
if (file != NULL) {
if (document_info->synctex != NULL) {
2015-12-07 01:13:59 +01:00
document_open_synctex(document_info->zathura, file, uri,
document_info->password, document_info->synctex);
} else {
2015-12-07 01:13:59 +01:00
document_open(document_info->zathura, file, uri, document_info->password,
document_info->page_number);
}
2011-10-21 15:00:22 +02:00
g_free(file);
2015-12-07 01:13:59 +01:00
g_free(uri);
if (document_info->mode != NULL) {
if (g_strcmp0(document_info->mode, "presentation") == 0) {
sc_toggle_presentation(document_info->zathura->ui.session, NULL, NULL,
0);
} else if (g_strcmp0(document_info->mode, "fullscreen") == 0) {
sc_toggle_fullscreen(document_info->zathura->ui.session, NULL, NULL,
0);
} else {
girara_error("Unknown mode: %s", document_info->mode);
}
}
2011-10-21 15:00:22 +02:00
}
2011-04-18 21:22:35 +02:00
}
free_document_info(document_info);
2011-04-18 21:22:35 +02:00
return FALSE;
2010-06-03 18:05:34 +02:00
}
2015-12-07 01:13:59 +01:00
char*
get_formatted_filename(zathura_t* zathura, bool statusbar)
{
bool basename_only = false;
2015-12-07 01:13:59 +01:00
const char* file_path = zathura_document_get_uri(zathura->document);
if (file_path == NULL) {
2015-12-07 01:13:59 +01:00
file_path = zathura_document_get_path(zathura->document);
}
if (statusbar == true) {
girara_setting_get(zathura->ui.session, "statusbar-basename", &basename_only);
2015-12-07 01:13:59 +01:00
} else {
girara_setting_get(zathura->ui.session, "window-title-basename", &basename_only);
}
if (basename_only == false) {
bool home_tilde = false;
if (statusbar) {
girara_setting_get(zathura->ui.session, "statusbar-home-tilde", &home_tilde);
} else {
girara_setting_get(zathura->ui.session, "window-title-home-tilde", &home_tilde);
}
const size_t file_path_len = file_path ? strlen(file_path) : 0;
if (home_tilde == true) {
char* home = girara_get_home_directory(NULL);
const size_t home_len = home ? strlen(home) : 0;
if (home_len > 1
&& file_path_len >= home_len
&& g_str_has_prefix(file_path, home)
&& (!file_path[home_len] || file_path[home_len] == '/')) {
g_free(home);
2015-10-07 20:10:40 +02:00
return g_strdup_printf("~%s", &file_path[home_len]);
} else {
g_free(home);
2015-10-05 22:59:31 +02:00
return g_strdup(file_path);
}
} else {
2015-10-05 22:59:31 +02:00
return g_strdup(file_path);
}
} else {
const char* basename = zathura_document_get_basename(zathura->document);
2015-10-05 22:59:31 +02:00
return g_strdup(basename);
}
}
static gboolean
document_open_password_dialog(gpointer data)
{
zathura_password_dialog_info_t* password_dialog_info = data;
girara_dialog(password_dialog_info->zathura->ui.session, _("Enter password:"), true, NULL,
cb_password_dialog, password_dialog_info);
return FALSE;
}
2010-12-12 22:04:42 +01:00
bool
2015-12-07 01:13:59 +01:00
document_open(zathura_t* zathura, const char* path, const char* uri, const char* password,
int page_number)
2010-12-12 22:04:42 +01:00
{
if (zathura == NULL || zathura->plugins.manager == NULL || path == NULL) {
2010-12-29 11:46:13 +01:00
goto error_out;
2010-12-12 22:04:42 +01:00
}
zathura_error_t error = ZATHURA_ERROR_OK;
zathura_document_t* document = zathura_document_open(zathura, path, uri, password, &error);
2010-12-12 22:04:42 +01:00
2012-02-12 16:35:33 +01:00
if (document == NULL) {
if (error == ZATHURA_ERROR_INVALID_PASSWORD) {
girara_debug("Invalid or no password.");
zathura_password_dialog_info_t* password_dialog_info = malloc(sizeof(zathura_password_dialog_info_t));
if (password_dialog_info != NULL) {
password_dialog_info->zathura = zathura;
password_dialog_info->path = g_strdup(path);
2015-12-07 01:13:59 +01:00
password_dialog_info->uri = g_strdup(uri);
if (password_dialog_info->path != NULL) {
gdk_threads_add_idle(document_open_password_dialog, password_dialog_info);
goto error_out;
} else {
free(password_dialog_info);
}
}
goto error_out;
}
2013-05-28 18:12:15 +02:00
if (error == ZATHURA_ERROR_OK ) {
girara_notify(zathura->ui.session, GIRARA_ERROR, _("Unsupported file type. Please install the necessary plugin."));
}
2010-12-29 11:46:13 +01:00
goto error_out;
2010-12-12 22:04:42 +01:00
}
const char* file_path = zathura_document_get_path(document);
unsigned int number_of_pages = zathura_document_get_number_of_pages(document);
if (number_of_pages == 0) {
girara_notify(zathura->ui.session, GIRARA_WARNING,
_("Document does not contain any pages"));
goto error_free;
}
zathura->document = document;
/* read history file */
zathura_fileinfo_t file_info = {
.current_page = 0,
.page_offset = 0,
.scale = 1,
.rotation = 0,
.pages_per_row = 0,
.first_page_column_list = NULL,
.position_x = 0,
.position_y = 0
};
bool known_file = false;
if (zathura->database != NULL) {
known_file = zathura_db_get_fileinfo(zathura->database, file_path, &file_info);
}
2012-04-22 10:04:46 +02:00
/* set page offset */
zathura_document_set_page_offset(document, file_info.page_offset);
/* check for valid scale value */
if (file_info.scale <= DBL_EPSILON) {
file_info.scale = 1;
}
2014-10-27 11:55:21 +01:00
zathura_document_set_scale(document,
zathura_correct_scale_value(zathura->ui.session, file_info.scale));
/* check current page number */
/* if it wasn't specified on the command-line, get it from file_info */
if (page_number == ZATHURA_PAGE_NUMBER_UNSPECIFIED)
page_number = file_info.current_page;
if (page_number < 0)
page_number += number_of_pages;
if ((unsigned)page_number > number_of_pages) {
girara_warning("document info: '%s' has an invalid page number", file_path);
zathura_document_set_current_page_number(document, 0);
} else {
zathura_document_set_current_page_number(document, page_number);
}
/* check for valid rotation */
if (file_info.rotation % 90 != 0) {
girara_warning("document info: '%s' has an invalid rotation", file_path);
zathura_document_set_rotation(document, 0);
} else {
zathura_document_set_rotation(document, file_info.rotation % 360);
}
/* jump to first page if setting enabled */
bool always_first_page = false;
girara_setting_get(zathura->ui.session, "open-first-page", &always_first_page);
if (always_first_page == true) {
zathura_document_set_current_page_number(document, 0);
}
/* apply open adjustment */
char* adjust_open = "best-fit";
if (known_file == false && girara_setting_get(zathura->ui.session, "adjust-open", &(adjust_open)) == true) {
if (g_strcmp0(adjust_open, "best-fit") == 0) {
zathura_document_set_adjust_mode(document, ZATHURA_ADJUST_BESTFIT);
} else if (g_strcmp0(adjust_open, "width") == 0) {
zathura_document_set_adjust_mode(document, ZATHURA_ADJUST_WIDTH);
} else {
zathura_document_set_adjust_mode(document, ZATHURA_ADJUST_NONE);
}
g_free(adjust_open);
} else {
zathura_document_set_adjust_mode(document, ZATHURA_ADJUST_NONE);
}
/* initialize bisect state */
zathura->bisect.start = 0;
zathura->bisect.last_jump = zathura_document_get_current_page_number(document);
zathura->bisect.end = number_of_pages - 1;
/* update statusbar */
2015-12-07 01:13:59 +01:00
char* filename = get_formatted_filename(zathura, true);
girara_statusbar_item_set_text(zathura->ui.session, zathura->ui.statusbar.file, filename);
g_free(filename);
2012-02-20 20:07:24 +01:00
/* install file monitor */
if (zathura->file_monitor.monitor == NULL) {
char* filemonitor_backend = NULL;
girara_setting_get(zathura->ui.session, "filemonitor", &filemonitor_backend);
zathura_filemonitor_type_t type = ZATHURA_FILEMONITOR_GLIB;
#ifdef G_OS_UNIX
if (g_strcmp0(filemonitor_backend, "signal") == 0) {
type = ZATHURA_FILEMONITOR_SIGNAL;
}
#endif
g_free(filemonitor_backend);
2012-02-20 20:07:24 +01:00
zathura->file_monitor.monitor = zathura_filemonitor_new(file_path, type);
if (zathura->file_monitor.monitor == NULL) {
goto error_free;
}
g_signal_connect(G_OBJECT(zathura->file_monitor.monitor), "reload-file",
G_CALLBACK(cb_file_monitor), zathura->ui.session);
2012-02-20 20:07:24 +01:00
}
if (password != NULL) {
g_free(zathura->file_monitor.password);
zathura->file_monitor.password = g_strdup(password);
2012-02-20 20:07:24 +01:00
if (zathura->file_monitor.password == NULL) {
goto error_free;
}
}
2012-04-21 04:59:58 +02:00
/* create marks list */
zathura->global.marks = girara_list_new2((girara_free_function_t) mark_free);
if (zathura->global.marks == NULL) {
goto error_free;
}
/* page cache size */
int cache_size = 0;
girara_setting_get(zathura->ui.session, "page-cache-size", &cache_size);
if (cache_size <= 0) {
girara_warning("page-cache-size is not positive, using %d instead",
ZATHURA_PAGE_CACHE_DEFAULT_SIZE);
cache_size = ZATHURA_PAGE_CACHE_DEFAULT_SIZE;
}
/* threads */
zathura->sync.render_thread = zathura_renderer_new(cache_size);
if (zathura->sync.render_thread == NULL) {
goto error_free;
}
/* set up recolor info in ZathuraRenderer */
char* recolor_dark = NULL;
char* recolor_light = NULL;
girara_setting_get(zathura->ui.session, "recolor-darkcolor", &recolor_dark);
girara_setting_get(zathura->ui.session, "recolor-lightcolor", &recolor_light);
zathura_renderer_set_recolor_colors_str(zathura->sync.render_thread,
recolor_light, recolor_dark);
g_free(recolor_dark);
g_free(recolor_light);
bool recolor = false;
girara_setting_get(zathura->ui.session, "recolor", &recolor);
zathura_renderer_enable_recolor(zathura->sync.render_thread, recolor);
girara_setting_get(zathura->ui.session, "recolor-keephue", &recolor);
zathura_renderer_enable_recolor_hue(zathura->sync.render_thread, recolor);
girara_setting_get(zathura->ui.session, "recolor-reverse-video", &recolor);
zathura_renderer_enable_recolor_reverse_video(zathura->sync.render_thread, recolor);
/* get view port size */
GtkAdjustment* hadjustment = gtk_scrolled_window_get_hadjustment(
GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view));
GtkAdjustment* vadjustment = gtk_scrolled_window_get_vadjustment(
GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view));
const unsigned int view_width = (unsigned int)floor(gtk_adjustment_get_page_size(hadjustment));
zathura_document_set_viewport_width(zathura->document, view_width);
const unsigned int view_height = (unsigned int)floor(gtk_adjustment_get_page_size(vadjustment));
zathura_document_set_viewport_height(zathura->document, view_height);
/* create blank pages */
zathura->pages = calloc(number_of_pages, sizeof(GtkWidget*));
if (zathura->pages == NULL) {
goto error_free;
}
for (unsigned int page_id = 0; page_id < number_of_pages; page_id++) {
zathura_page_t* page = zathura_document_get_page(document, page_id);
if (page == NULL) {
goto error_free;
}
GtkWidget* page_widget = zathura_page_widget_new(zathura, page);
if (page_widget == NULL) {
goto error_free;
}
g_object_ref(page_widget);
zathura->pages[page_id] = page_widget;
gtk_widget_set_halign(page_widget, GTK_ALIGN_CENTER);
gtk_widget_set_valign(page_widget, GTK_ALIGN_CENTER);
g_signal_connect(G_OBJECT(page_widget), "text-selected",
G_CALLBACK(cb_page_widget_text_selected), zathura);
g_signal_connect(G_OBJECT(page_widget), "image-selected",
G_CALLBACK(cb_page_widget_image_selected), zathura);
g_signal_connect(G_OBJECT(page_widget), "enter-link",
G_CALLBACK(cb_page_widget_link), (gpointer) true);
g_signal_connect(G_OBJECT(page_widget), "leave-link",
G_CALLBACK(cb_page_widget_link), (gpointer) false);
g_signal_connect(G_OBJECT(page_widget), "scaled-button-release",
G_CALLBACK(cb_page_widget_scaled_button_release), zathura);
}
2011-03-18 18:40:20 +01:00
/* view mode */
unsigned int pages_per_row = 1;
char* first_page_column_list = NULL;
unsigned int page_padding = 1;
girara_setting_get(zathura->ui.session, "page-padding", &page_padding);
if (file_info.pages_per_row > 0) {
pages_per_row = file_info.pages_per_row;
} else {
girara_setting_get(zathura->ui.session, "pages-per-row", &pages_per_row);
}
2012-04-29 06:26:38 +02:00
/* read first_page_column list */
if (file_info.first_page_column_list != NULL && *file_info.first_page_column_list != '\0') {
first_page_column_list = file_info.first_page_column_list;
file_info.first_page_column_list = NULL;
} else {
2015-11-07 20:12:01 +01:00
girara_setting_get(zathura->ui.session, "first-page-column", &first_page_column_list);
}
2015-11-07 20:12:01 +01:00
/* find value for first_page_column */
unsigned int first_page_column = find_first_page_column(first_page_column_list, pages_per_row);
2012-04-29 06:26:38 +02:00
girara_setting_set(zathura->ui.session, "pages-per-row", &pages_per_row);
2015-11-07 20:12:01 +01:00
girara_setting_set(zathura->ui.session, "first-page-column", first_page_column_list);
g_free(file_info.first_page_column_list);
2015-11-07 20:12:01 +01:00
g_free(first_page_column_list);
page_widget_set_mode(zathura, page_padding, pages_per_row, first_page_column);
zathura_document_set_page_layout(zathura->document, page_padding, pages_per_row, first_page_column);
2010-12-28 09:47:09 +01:00
girara_set_view(zathura->ui.session, zathura->ui.page_widget);
2010-12-28 09:47:09 +01:00
2010-12-12 22:04:42 +01:00
2011-09-03 14:21:36 +02:00
/* bookmarks */
if (zathura->database != NULL) {
if (zathura_bookmarks_load(zathura, file_path) == false) {
girara_debug("Failed to load bookmarks.");
}
2011-09-03 14:21:36 +02:00
/* jumplist */
if (zathura_jumplist_load(zathura, file_path) == false) {
zathura->jumplist.list = girara_list_new2(g_free);
}
}
2012-03-16 07:58:55 +01:00
/* update title */
2015-12-07 01:13:59 +01:00
char* formatted_filename = get_formatted_filename(zathura, false);
girara_set_window_title(zathura->ui.session, formatted_filename);
g_free(formatted_filename);
2012-03-16 07:58:55 +01:00
/* adjust_view */
adjust_view(zathura);
for (unsigned int page_id = 0; page_id < number_of_pages; page_id++) {
/* set widget size */
zathura_page_t* page = zathura_document_get_page(document, page_id);
unsigned int page_height = 0;
unsigned int page_width = 0;
/* adjust_view calls render_all in some cases and render_all calls
* gtk_widget_set_size_request. To be sure that it's really called, do it
* here once again. */
const double height = zathura_page_get_height(page);
const double width = zathura_page_get_width(page);
page_calc_height_width(zathura->document, height, width, &page_height, &page_width, true);
gtk_widget_set_size_request(zathura->pages[page_id], page_width, page_height);
/* show widget */
gtk_widget_show(zathura->pages[page_id]);
}
/* Set page */
page_set(zathura, zathura_document_get_current_page_number(document));
/* Set position (only if restoring from history file) */
if (file_info.current_page == zathura_document_get_current_page_number(document) &&
(file_info.position_x != 0 || file_info.position_y != 0)) {
position_set(zathura, file_info.position_x, file_info.position_y);
}
update_visible_pages(zathura);
2010-12-12 22:04:42 +01:00
return true;
2010-12-29 11:46:13 +01:00
error_free:
zathura_document_free(document);
error_out:
return false;
2010-12-12 22:04:42 +01:00
}
bool
2015-12-07 01:13:59 +01:00
document_open_synctex(zathura_t* zathura, const char* path, const char* uri,
const char* password, const char* synctex)
{
2015-12-07 01:13:59 +01:00
bool ret = document_open(zathura, path, password, uri,
ZATHURA_PAGE_NUMBER_UNSPECIFIED);
if (ret == false) {
return false;
}
if (synctex == NULL) {
return true;
}
int line = 0;
int column = 0;
char* input_file = NULL;
if (synctex_parse_input(synctex, &input_file, &line, &column) == false) {
return false;
}
ret = synctex_view(zathura, input_file, line, column);
g_free(input_file);
return ret;
}
2012-08-05 15:34:10 +02:00
void
document_open_idle(zathura_t* zathura, const char* path, const char* password,
int page_number, const char* mode, const char* synctex)
2012-08-05 15:34:10 +02:00
{
g_return_if_fail(zathura != NULL);
g_return_if_fail(path != NULL);
2012-08-05 15:34:10 +02:00
zathura_document_info_t* document_info = g_try_malloc0(sizeof(zathura_document_info_t));
if (document_info == NULL) {
return;
}
2012-08-05 15:34:10 +02:00
document_info->zathura = zathura;
document_info->path = g_strdup(path);
if (password != NULL) {
document_info->password = g_strdup(password);
}
document_info->page_number = page_number;
if (mode != NULL) {
document_info->mode = g_strdup(mode);
}
if (synctex != NULL) {
document_info->synctex = g_strdup(synctex);
}
2012-08-05 15:34:10 +02:00
gdk_threads_add_idle(document_info_open, document_info);
}
2011-09-01 11:51:49 +02:00
bool
document_save(zathura_t* zathura, const char* path, bool overwrite)
{
g_return_val_if_fail(zathura, false);
g_return_val_if_fail(zathura->document, false);
g_return_val_if_fail(path, false);
gchar* file_path = girara_fix_path(path);
/* use current basename if path points to a directory */
if (g_file_test(file_path, G_FILE_TEST_IS_DIR) == TRUE) {
char* basename = g_path_get_basename(zathura_document_get_path(zathura->document));
char* tmp = file_path;
file_path = g_build_filename(file_path, basename, NULL);
g_free(tmp);
g_free(basename);
}
2012-10-09 01:12:18 +02:00
if ((overwrite == false) && g_file_test(file_path, G_FILE_TEST_EXISTS)) {
girara_error("File already exists: %s. Use :write! to overwrite it.", file_path);
2011-10-21 15:00:22 +02:00
g_free(file_path);
2011-09-01 11:51:49 +02:00
return false;
}
2012-03-27 13:30:04 +02:00
zathura_error_t error = zathura_document_save_as(zathura->document, file_path);
2011-09-01 11:51:49 +02:00
g_free(file_path);
2012-03-27 13:30:04 +02:00
return (error == ZATHURA_ERROR_OK) ? true : false;
2011-09-01 11:51:49 +02:00
}
static void
remove_page_from_table(GtkWidget* page, gpointer UNUSED(permanent))
{
gtk_container_remove(GTK_CONTAINER(gtk_widget_get_parent(page)), page);
}
static void
save_fileinfo_to_db(zathura_t* zathura)
{
const char* path = zathura_document_get_path(zathura->document);
zathura_fileinfo_t file_info = {
.current_page = zathura_document_get_current_page_number(zathura->document),
.page_offset = zathura_document_get_page_offset(zathura->document),
.scale = zathura_document_get_scale(zathura->document),
.rotation = zathura_document_get_rotation(zathura->document),
.pages_per_row = 1,
.first_page_column_list = "1:2",
.position_x = zathura_document_get_position_x(zathura->document),
.position_y = zathura_document_get_position_y(zathura->document)
};
girara_setting_get(zathura->ui.session, "pages-per-row",
&(file_info.pages_per_row));
girara_setting_get(zathura->ui.session, "first-page-column",
&(file_info.first_page_column_list));
/* save file info */
zathura_db_set_fileinfo(zathura->database, path, &file_info);
/* save jumplist */
zathura_db_save_jumplist(zathura->database, path, zathura->jumplist.list);
g_free(file_info.first_page_column_list);
}
2010-12-12 22:04:42 +01:00
bool
2012-02-20 20:07:24 +01:00
document_close(zathura_t* zathura, bool keep_monitor)
2010-12-12 22:04:42 +01:00
{
2012-02-20 20:07:24 +01:00
if (zathura == NULL || zathura->document == NULL) {
2010-12-12 22:04:42 +01:00
return false;
}
/* stop rendering */
zathura_renderer_stop(zathura->sync.render_thread);
2012-02-20 20:07:24 +01:00
/* remove monitor */
if (keep_monitor == false) {
g_clear_object(&zathura->file_monitor.monitor);
2012-02-20 20:07:24 +01:00
if (zathura->file_monitor.password != NULL) {
g_free(zathura->file_monitor.password);
zathura->file_monitor.password = NULL;
}
}
2012-04-21 04:59:58 +02:00
/* remove marks */
if (zathura->global.marks != NULL) {
girara_list_free(zathura->global.marks);
zathura->global.marks = NULL;
}
/* store file information */
if (zathura->database != NULL) {
save_fileinfo_to_db(zathura);
}
2011-10-06 17:57:26 +02:00
girara_list_iterator_free(zathura->jumplist.cur);
zathura->jumplist.cur = NULL;
girara_list_free(zathura->jumplist.list);
zathura->jumplist.list = NULL;
zathura->jumplist.size = 0;
/* release render thread */
g_clear_object(&zathura->sync.render_thread);
/* remove widgets */
gtk_container_foreach(GTK_CONTAINER(zathura->ui.page_widget), remove_page_from_table, NULL);
2012-05-08 17:10:57 +02:00
for (unsigned int i = 0; i < zathura_document_get_number_of_pages(zathura->document); i++) {
g_object_unref(zathura->pages[i]);
}
free(zathura->pages);
zathura->pages = NULL;
2011-01-24 12:43:39 +01:00
/* remove document */
2011-04-18 17:27:49 +02:00
zathura_document_free(zathura->document);
zathura->document = NULL;
2012-04-16 09:17:25 +02:00
/* remove index */
if (zathura->ui.index != NULL) {
g_object_ref_sink(zathura->ui.index);
zathura->ui.index = NULL;
}
2012-02-07 18:30:46 +01:00
gtk_widget_hide(zathura->ui.page_widget);
statusbar_page_number_update(zathura);
if (zathura->ui.session != NULL && zathura->ui.statusbar.file != NULL) {
2012-03-16 13:42:15 +01:00
girara_statusbar_item_set_text(zathura->ui.session, zathura->ui.statusbar.file, _("[No name]"));
}
2010-12-12 22:04:42 +01:00
2012-03-16 07:58:55 +01:00
/* update title */
girara_set_window_title(zathura->ui.session, "zathura");
2010-12-12 22:04:42 +01:00
return true;
}
bool
2011-04-18 17:27:49 +02:00
page_set(zathura_t* zathura, unsigned int page_id)
2010-12-12 22:04:42 +01:00
{
if (zathura == NULL || zathura->document == NULL) {
2010-12-26 01:52:17 +01:00
goto error_out;
2010-12-12 22:04:42 +01:00
}
zathura_page_t* page = zathura_document_get_page(zathura->document, page_id);
2012-02-12 16:35:33 +01:00
if (page == NULL) {
goto error_out;
}
zathura_document_set_current_page_number(zathura->document, page_id);
2017-05-21 22:57:58 +02:00
bool continuous_hist_save = false;
girara_setting_get(zathura->ui.session, "continuous-hist-save", &continuous_hist_save);
if (continuous_hist_save) {
save_fileinfo_to_db(zathura);
}
/* negative position means auto */
return position_set(zathura, -1, -1);
2010-12-26 01:52:17 +01:00
error_out:
return false;
2010-12-12 22:04:42 +01:00
}
void
statusbar_page_number_update(zathura_t* zathura)
{
if (zathura == NULL || zathura->ui.statusbar.page_number == NULL) {
return;
}
unsigned int number_of_pages = zathura_document_get_number_of_pages(zathura->document);
unsigned int current_page_number = zathura_document_get_current_page_number(zathura->document);
if (zathura->document != NULL) {
char* page_number_text = g_strdup_printf("[%d/%d]", current_page_number + 1, number_of_pages);
girara_statusbar_item_set_text(zathura->ui.session, zathura->ui.statusbar.page_number, page_number_text);
bool page_number_in_window_title = false;
girara_setting_get(zathura->ui.session, "window-title-page", &page_number_in_window_title);
if (page_number_in_window_title == true) {
2015-12-07 01:13:59 +01:00
char* filename = get_formatted_filename(zathura, false);
char* title = g_strdup_printf("%s %s", filename, page_number_text);
girara_set_window_title(zathura->ui.session, title);
g_free(title);
g_free(filename);
}
g_free(page_number_text);
} else {
girara_statusbar_item_set_text(zathura->ui.session, zathura->ui.statusbar.page_number, "");
}
}
2011-03-18 18:40:20 +01:00
void
page_widget_set_mode(zathura_t* zathura, unsigned int page_padding,
unsigned int pages_per_row, unsigned int first_page_column)
2011-03-18 18:40:20 +01:00
{
2011-04-25 16:54:21 +02:00
/* show at least one page */
if (pages_per_row == 0) {
pages_per_row = 1;
}
2012-10-09 01:12:18 +02:00
/* ensure: 0 < first_page_column <= pages_per_row */
if (first_page_column < 1) {
first_page_column = 1;
}
if (first_page_column > pages_per_row) {
first_page_column = ((first_page_column - 1) % pages_per_row) + 1;
}
if (zathura->document == NULL) {
return;
}
gtk_container_foreach(GTK_CONTAINER(zathura->ui.page_widget), remove_page_from_table, NULL);
unsigned int number_of_pages = zathura_document_get_number_of_pages(zathura->document);
gtk_grid_set_row_spacing(GTK_GRID(zathura->ui.page_widget), page_padding);
gtk_grid_set_column_spacing(GTK_GRID(zathura->ui.page_widget), page_padding);
2012-08-05 02:30:03 +02:00
for (unsigned int i = 0; i < number_of_pages; i++) {
int x = (i + first_page_column - 1) % pages_per_row;
int y = (i + first_page_column - 1) / pages_per_row;
2012-03-26 14:44:56 +02:00
GtkWidget* page_widget = zathura->pages[i];
gtk_grid_attach(GTK_GRID(zathura->ui.page_widget), page_widget, x, y, 1, 1);
2011-03-18 18:40:20 +01:00
}
2012-02-07 18:30:46 +01:00
gtk_widget_show_all(zathura->ui.page_widget);
2011-03-18 18:40:20 +01:00
}
bool
Enhancements/Cleanups for the jumplist mechansim - Don't delete the elements on the right of the current one, when appending a new jump to the jumplist, because this makes no sense at all; the point of the jumplist in the first place is to remember previously jumped-to positions in the document, so there is no need to delete anythings except to trim the oldest entries from the beginning to maintain the maximum size. This also makes us compatible with the Vim way of doing things. - Make the jumplist mechanism functional on the same page; if we followed a link to a target on the same page, remember the adjustments before and after following the link. The same holds for navigating search results on the same page. - Implement position_set and use it instead of position_set_delayed when following links in order to give zathura_jumplist_save a chance to record the exact adjustments of the link target. Otherwise, it will always record the adjustments after going to the target page, but before going to the exact position within it. - Don't consider movements with ^i and ^o as jumps :) - Don't use page_set followed by setting the adjustments in sc_jumplist, because this is redundant and causes clutter when using ^i and ^o, as the adjustments is set twice this way (once in page_set and again in position_set_delayed). It's enough to only update the page number on the statusbar and then set the adjustments. - Hide implementation details (zathura_jumplist_save and zathura_jumplist_append), and make things more consistent by exporting and using only zathura_jumplist_add for adding new entries. The end result: A more slick jumping experience :-) Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-09 05:53:31 +02:00
position_set(zathura_t* zathura, double position_x, double position_y)
{
if (zathura == NULL || zathura->document == NULL) {
return false;
}
Enhancements/Cleanups for the jumplist mechansim - Don't delete the elements on the right of the current one, when appending a new jump to the jumplist, because this makes no sense at all; the point of the jumplist in the first place is to remember previously jumped-to positions in the document, so there is no need to delete anythings except to trim the oldest entries from the beginning to maintain the maximum size. This also makes us compatible with the Vim way of doing things. - Make the jumplist mechanism functional on the same page; if we followed a link to a target on the same page, remember the adjustments before and after following the link. The same holds for navigating search results on the same page. - Implement position_set and use it instead of position_set_delayed when following links in order to give zathura_jumplist_save a chance to record the exact adjustments of the link target. Otherwise, it will always record the adjustments after going to the target page, but before going to the exact position within it. - Don't consider movements with ^i and ^o as jumps :) - Don't use page_set followed by setting the adjustments in sc_jumplist, because this is redundant and causes clutter when using ^i and ^o, as the adjustments is set twice this way (once in page_set and again in position_set_delayed). It's enough to only update the page number on the statusbar and then set the adjustments. - Hide implementation details (zathura_jumplist_save and zathura_jumplist_append), and make things more consistent by exporting and using only zathura_jumplist_add for adding new entries. The end result: A more slick jumping experience :-) Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-09 05:53:31 +02:00
double comppos_x, comppos_y;
const unsigned int page_id = zathura_document_get_current_page_number(zathura->document);
Enhancements/Cleanups for the jumplist mechansim - Don't delete the elements on the right of the current one, when appending a new jump to the jumplist, because this makes no sense at all; the point of the jumplist in the first place is to remember previously jumped-to positions in the document, so there is no need to delete anythings except to trim the oldest entries from the beginning to maintain the maximum size. This also makes us compatible with the Vim way of doing things. - Make the jumplist mechanism functional on the same page; if we followed a link to a target on the same page, remember the adjustments before and after following the link. The same holds for navigating search results on the same page. - Implement position_set and use it instead of position_set_delayed when following links in order to give zathura_jumplist_save a chance to record the exact adjustments of the link target. Otherwise, it will always record the adjustments after going to the target page, but before going to the exact position within it. - Don't consider movements with ^i and ^o as jumps :) - Don't use page_set followed by setting the adjustments in sc_jumplist, because this is redundant and causes clutter when using ^i and ^o, as the adjustments is set twice this way (once in page_set and again in position_set_delayed). It's enough to only update the page number on the statusbar and then set the adjustments. - Hide implementation details (zathura_jumplist_save and zathura_jumplist_append), and make things more consistent by exporting and using only zathura_jumplist_add for adding new entries. The end result: A more slick jumping experience :-) Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-09 05:53:31 +02:00
2017-02-23 01:18:25 +01:00
bool vertical_center = false;
girara_setting_get(zathura->ui.session, "vertical-center", &vertical_center);
/* xalign = 0.5: center horizontally (with the page, not the document) */
if (vertical_center == true) {
2017-02-23 01:18:25 +01:00
/* yalign = 0.0: align page an viewport edges at the top */
page_number_to_position(zathura->document, page_id, 0.5, 0.0, &comppos_x, &comppos_y);
} else {
/* yalign = 0.5: center vertically */
page_number_to_position(zathura->document, page_id, 0.5, 0.5, &comppos_x, &comppos_y);
}
/* automatic horizontal adjustment */
zathura_adjust_mode_t adjust_mode = zathura_document_get_adjust_mode(zathura->document);
/* negative position_x mean: use the computed value */
if (position_x < 0) {
position_x = comppos_x;
bool zoom_center = false;
girara_setting_get(zathura->ui.session, "zoom-center", &zoom_center);
/* center horizontally */
if (adjust_mode == ZATHURA_ADJUST_BESTFIT ||
adjust_mode == ZATHURA_ADJUST_WIDTH ||
zoom_center == true) {
position_x = 0.5;
}
Enhancements/Cleanups for the jumplist mechansim - Don't delete the elements on the right of the current one, when appending a new jump to the jumplist, because this makes no sense at all; the point of the jumplist in the first place is to remember previously jumped-to positions in the document, so there is no need to delete anythings except to trim the oldest entries from the beginning to maintain the maximum size. This also makes us compatible with the Vim way of doing things. - Make the jumplist mechanism functional on the same page; if we followed a link to a target on the same page, remember the adjustments before and after following the link. The same holds for navigating search results on the same page. - Implement position_set and use it instead of position_set_delayed when following links in order to give zathura_jumplist_save a chance to record the exact adjustments of the link target. Otherwise, it will always record the adjustments after going to the target page, but before going to the exact position within it. - Don't consider movements with ^i and ^o as jumps :) - Don't use page_set followed by setting the adjustments in sc_jumplist, because this is redundant and causes clutter when using ^i and ^o, as the adjustments is set twice this way (once in page_set and again in position_set_delayed). It's enough to only update the page number on the statusbar and then set the adjustments. - Hide implementation details (zathura_jumplist_save and zathura_jumplist_append), and make things more consistent by exporting and using only zathura_jumplist_add for adding new entries. The end result: A more slick jumping experience :-) Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-09 05:53:31 +02:00
}
if (position_y < 0) {
position_y = comppos_y;
Enhancements/Cleanups for the jumplist mechansim - Don't delete the elements on the right of the current one, when appending a new jump to the jumplist, because this makes no sense at all; the point of the jumplist in the first place is to remember previously jumped-to positions in the document, so there is no need to delete anythings except to trim the oldest entries from the beginning to maintain the maximum size. This also makes us compatible with the Vim way of doing things. - Make the jumplist mechanism functional on the same page; if we followed a link to a target on the same page, remember the adjustments before and after following the link. The same holds for navigating search results on the same page. - Implement position_set and use it instead of position_set_delayed when following links in order to give zathura_jumplist_save a chance to record the exact adjustments of the link target. Otherwise, it will always record the adjustments after going to the target page, but before going to the exact position within it. - Don't consider movements with ^i and ^o as jumps :) - Don't use page_set followed by setting the adjustments in sc_jumplist, because this is redundant and causes clutter when using ^i and ^o, as the adjustments is set twice this way (once in page_set and again in position_set_delayed). It's enough to only update the page number on the statusbar and then set the adjustments. - Hide implementation details (zathura_jumplist_save and zathura_jumplist_append), and make things more consistent by exporting and using only zathura_jumplist_add for adding new entries. The end result: A more slick jumping experience :-) Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-09 05:53:31 +02:00
}
/* set the position */
zathura_document_set_position_x(zathura->document, position_x);
zathura_document_set_position_y(zathura->document, position_y);
/* trigger a 'change' event for both adjustments */
refresh_view(zathura);
return true;
Enhancements/Cleanups for the jumplist mechansim - Don't delete the elements on the right of the current one, when appending a new jump to the jumplist, because this makes no sense at all; the point of the jumplist in the first place is to remember previously jumped-to positions in the document, so there is no need to delete anythings except to trim the oldest entries from the beginning to maintain the maximum size. This also makes us compatible with the Vim way of doing things. - Make the jumplist mechanism functional on the same page; if we followed a link to a target on the same page, remember the adjustments before and after following the link. The same holds for navigating search results on the same page. - Implement position_set and use it instead of position_set_delayed when following links in order to give zathura_jumplist_save a chance to record the exact adjustments of the link target. Otherwise, it will always record the adjustments after going to the target page, but before going to the exact position within it. - Don't consider movements with ^i and ^o as jumps :) - Don't use page_set followed by setting the adjustments in sc_jumplist, because this is redundant and causes clutter when using ^i and ^o, as the adjustments is set twice this way (once in page_set and again in position_set_delayed). It's enough to only update the page number on the statusbar and then set the adjustments. - Hide implementation details (zathura_jumplist_save and zathura_jumplist_append), and make things more consistent by exporting and using only zathura_jumplist_add for adding new entries. The end result: A more slick jumping experience :-) Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-09 05:53:31 +02:00
}
void
refresh_view(zathura_t* zathura)
{
g_return_if_fail(zathura != NULL);
/* emit a custom refresh-view signal */
g_signal_emit(zathura->ui.session->gtk.view, zathura->signals.refresh_view,
0, zathura);
}
Hide the inputbar and completion menu before saving the adjustments ratios into a new jump structure Since we are now saving the adjustments ratios in the jump structures, we need to take care of the following scenario: - We do an action that results in a new jump structure being added to the jumplist while the inputbar is visible (e.g., search, jumping to a specific page, jumping to a bookmark, or following a link). - Since we are now storing the adjustments ratios in the jump structures, all of the above actions would result in the vertical adjustment ratio being saved while the inputbar and/or the completion menu is visible. - Now we are exactly on the target of the jump (note that the inputbar and completion menu now are hidden), so suppose that we want to go back using ^o (assuming that we didn't change the adjustments after jumping), then the check at sc_jumplist that compares the current adjustments ratios with that of the current jump (the jump that has just been added and which we are currently on it's position) would fail, because after the inputbar (with possibly the completion menu in case of bookmarks) is activated it is hidden, which results in the vertical adjustment upper bound to change, which in turn results in the vertical adjustment ratio returned by zathura_adjustment_get_ratio to become different from what is stored in the current jump structure, even though we haven't changed the adjustments at all after the jump. This would always result in taking us back to the exact position of the jump (which would be slightly different from the current position) when we press ^o. This can be annoying, because it would happen, for example, every time we need to go back quickly after jumping to a link target, a search result, or a bookmark. So, what this patch does is essentially to make the vertical adjustment ratio reflecting the current vertical adjustment after a jump, to always be the same as the one stored in the newly added jump structure, since both are calculated with zathura_adjustment_get_ratio while the inputbar is _not_ visible, so they should be the same. I've elaborated just to make things clear, in case the purpose of the patch isn't obvious. Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
2013-06-21 14:25:29 +02:00
bool
adjust_view(zathura_t* zathura)
{
g_return_val_if_fail(zathura != NULL, false);
if (zathura->ui.page_widget == NULL || zathura->document == NULL) {
goto error_ret;
}
zathura_adjust_mode_t adjust_mode = zathura_document_get_adjust_mode(zathura->document);
if (adjust_mode == ZATHURA_ADJUST_NONE) {
/* there is nothing todo */
goto error_ret;
}
unsigned int cell_height = 0, cell_width = 0;
unsigned int document_height = 0, document_width = 0;
unsigned int view_height = 0, view_width = 0;
zathura_document_get_cell_size(zathura->document, &cell_height, &cell_width);
zathura_document_get_document_size(zathura->document, &document_height, &document_width);
zathura_document_get_viewport_size(zathura->document, &view_height, &view_width);
if (view_height == 0 || view_width == 0 || cell_height == 0 || cell_width == 0) {
goto error_ret;
}
double page_ratio = (double)cell_height / (double)document_width;
double view_ratio = (double)view_height / (double)view_width;
double scale = zathura_document_get_scale(zathura->document);
double newscale = scale;
if (adjust_mode == ZATHURA_ADJUST_WIDTH ||
(adjust_mode == ZATHURA_ADJUST_BESTFIT && page_ratio < view_ratio)) {
newscale *= (double)view_width / (double)document_width;
} else if (adjust_mode == ZATHURA_ADJUST_BESTFIT) {
newscale *= (double)view_height / (double)cell_height;
} else {
goto error_ret;
}
/* save new scale and recompute cell size */
zathura_document_set_scale(zathura->document, newscale);
unsigned int new_cell_height = 0, new_cell_width = 0;
zathura_document_get_cell_size(zathura->document, &new_cell_height, &new_cell_width);
/* if the change in scale changes page cell dimensions by at least one pixel, render */
if (abs((int)new_cell_width - (int)cell_width) > 1 ||
abs((int)new_cell_height - (int)cell_height) > 1) {
render_all(zathura);
refresh_view(zathura);
} else {
/* otherwise set the old scale and leave */
zathura_document_set_scale(zathura->document, scale);
}
error_ret:
return false;
}
2014-04-23 17:27:24 +02:00
#ifdef G_OS_UNIX
static gboolean
zathura_signal_sigterm(gpointer data)
{
if (data == NULL) {
return TRUE;
}
zathura_t* zathura = (zathura_t*) data;
cb_destroy(NULL, zathura);
return TRUE;
}
#endif