diff --git a/.gitignore b/.gitignore index 487e3b2..ca38e6a 100644 --- a/.gitignore +++ b/.gitignore @@ -1,6 +1,9 @@ -.depend/ *.o *.do -config.h +*~ +*.rej +*.swp +.depend zathura zathura-debug +zathura.pc diff --git a/Makefile b/Makefile index 7c70987..a9d8902 100644 --- a/Makefile +++ b/Makefile @@ -1,42 +1,38 @@ # See LICENSE file for license and copyright information -# zathura - user interface include config.mk include common.mk PROJECT = zathura -SOURCE = zathura.c -OBJECTS = ${SOURCE:.c=.o} -DOBJECTS = ${SOURCE:.c=.do} +SOURCE = $(shell find . -iname "*.c") +OBJECTS = $(patsubst %.c, %.o, $(SOURCE)) +DOBJECTS = $(patsubst %.c, %.do, $(SOURCE)) + +ifneq "$(NEEDS_DL)" "0" +LIBS += -ldl +endif all: options ${PROJECT} options: @echo ${PROJECT} build options: @echo "CFLAGS = ${CFLAGS}" - @echo "LDFLAGS = ${LDFLAGS}" @echo "LIBS = ${LIBS}" @echo "DFLAGS = ${DFLAGS}" @echo "CC = ${CC}" %.o: %.c $(ECHO) CC $< - $(QUIET)${CC} -c ${CFLAGS} -o $@ $< + @mkdir -p .depend + $(QUIET)${CC} -c ${CFLAGS} -o $@ $< -MMD -MF .depend/$@.dep %.do: %.c $(ECHO) CC $< - $(QUIET)${CC} -c ${CFLAGS} ${DFLAGS} -o $@ $< + @mkdir -p .depend + $(QUIET)${CC} -c ${CFLAGS} ${DFLAGS} -o $@ $< -MMD -MF .depend/$@.dep -${OBJECTS}: config.h config.mk -${DOBJECTS}: config.h config.mk - -config.h: config.def.h - @if [ -f $@ ] ; then \ - echo "config.h exists, but config.def.h is newer. Please check your" \ - "config.h or ${PROJECT} might fail to build." ; \ - else \ - cp $< $@ ; \ - fi +${OBJECTS}: config.mk +${DOBJECTS}: config.mk ${PROJECT}: ${OBJECTS} $(ECHO) CC -o $@ @@ -44,17 +40,20 @@ ${PROJECT}: ${OBJECTS} clean: $(QUIET)rm -rf ${PROJECT} ${OBJECTS} ${PROJECT}-${VERSION}.tar.gz \ - ${DOBJECTS} ${PROJECT}-debug - -distclean: clean - $(QUIET)rm -rf config.h + ${DOBJECTS} ${PROJECT}-debug .depend ${PROJECT}.pc ${PROJECT}-debug: ${DOBJECTS} - $(ECHO) CC -o ${PROJECT}-debug - $(QUIET)${CC} ${LDFLAGS} -o ${PROJECT}-debug ${DOBJECTS} ${LIBS} + $(ECHO) CC -o $@ + $(QUIET)${CC} ${LDFLAGS} -o $@ ${DOBJECTS} ${LIBS} debug: ${PROJECT}-debug +${PROJECT}.pc: ${PROJECT}.pc.in config.mk + $(QUIET)echo project=${PROJECT} > ${PROJECT}.pc + $(QUIET)echo version=${VERSION} >> ${PROJECT}.pc + $(QUIET)echo includedir=${PREFIX}/include >> ${PROJECT}.pc + $(QUIET)cat ${PROJECT}.pc.in >> ${PROJECT}.pc + valgrind: debug valgrind --tool=memcheck --leak-check=yes --show-reachable=yes \ ./${PROJECT}-debug @@ -64,34 +63,40 @@ gdb: debug dist: clean $(QUIET)mkdir -p ${PROJECT}-${VERSION} - $(QUIET)cp -R LICENSE Makefile config.mk config.def.h README \ - ${PROJECT}.desktop ${PROJECT}rc.5.rst \ - ${PROJECT}.1 ${SOURCE} ${PROJECT}-${VERSION} + $(QUIET)cp -R LICENSE Makefile config.mk README \ + ${PROJECT}.1 ${SOURCE} ${PROJECT}.pc.in \ + ${PROJECT}-${VERSION} $(QUIET)tar -cf ${PROJECT}-${VERSION}.tar ${PROJECT}-${VERSION} $(QUIET)gzip ${PROJECT}-${VERSION}.tar $(QUIET)rm -rf ${PROJECT}-${VERSION} -install: all +install: all ${PROJECT}.pc $(ECHO) installing executable file $(QUIET)mkdir -p ${DESTDIR}${PREFIX}/bin $(QUIET)install -m 755 ${PROJECT} ${DESTDIR}${PREFIX}/bin + $(ECHO) installing header file + $(QUIET)mkdir -p ${DESTDIR}${PREFIX}/include/${PROJECT} + $(QUIET)cp -f document.h ${DESTDIR}${PREFIX}/include/${PROJECT} + $(QUIET)cp -f zathura.h ${DESTDIR}${PREFIX}/include/${PROJECT} $(ECHO) installing manual page $(QUIET)mkdir -p ${DESTDIR}${MANPREFIX}/man1 $(QUIET)sed "s/VERSION/${VERSION}/g" < ${PROJECT}.1 > ${DESTDIR}${MANPREFIX}/man1/${PROJECT}.1 - $(QUIET)if which rst2man > /dev/null ; then \ - mkdir -p ${DESTDIR}${MANPREFIX}/man5 ; \ - rst2man ${PROJECT}rc.5.rst > ${DESTDIR}${MANPREFIX}/man5/${PROJECT}rc.5 ; \ - fi $(QUIET)chmod 644 ${DESTDIR}${MANPREFIX}/man1/${PROJECT}.1 - $(QUIET)mkdir -p ${DESTDIR}${DESKTOPPREFIX} - $(ECHO) installing desktop file - $(QUIET)install -m 644 ${PROJECT}.desktop ${DESTDIR}${DESKTOPPREFIX} + $(ECHO) installing pkgconfig file + $(QUIET)mkdir -p ${DESTDIR}${PREFIX}/lib/pkgconfig + $(QUIET)cp -f ${PROJECT}.pc ${DESTDIR}${PREFIX}/lib/pkgconfig uninstall: $(ECHO) removing executable file $(QUIET)rm -f ${DESTDIR}${PREFIX}/bin/${PROJECT} + $(ECHO) removing header file + $(QUIET)rm -f ${DESTDIR}${PREFIX}/include/${PROJECT}/document.h + $(QUIET)rm -f ${DESTDIR}${PREFIX}/include/${PROJECT}/zathura.h $(ECHO) removing manual page $(QUIET)rm -f ${DESTDIR}${MANPREFIX}/man1/${PROJECT}.1 - $(QUIET)rm -f ${DESTDIR}${MANPREFIX}/man5/${PROJECT}rc.5 - $(ECHO) removing desktop file - $(QUIET)rm -f ${DESTDIR}${DESKTOPPREFIX}/${PROJECT}.desktop + $(ECHO) removing pkgconfig file + $(QUIET)rm -f ${DESTDIR}${PREFIX}/lib/pkgconfig + +-include $(wildcard .depend/*.dep) + +.PHONY: all options clean debug valgrind gdb dist install uninstall diff --git a/README b/README index c1d4067..ac777c5 100644 --- a/README +++ b/README @@ -1,28 +1,19 @@ -zathura - pdf viewer +zathura - a document viewer -------------------- -zathura is a pdf viewer based on the poppler pdf rendering library +zathura is a highly customizable and functional document viewer based on the +girara user interface library and several document libraries. Requirements ------------ -poppler-glib (>= 0.12.3) -cairo (>= 1.8.8) gtk2 (>= 2.18.6) -glib2 (>= 2.22.4) +girara Please note that you need to have a working pkg-config installation and that the Makefile is only compatible with GNU make. -And also note that rst2man from python-docutils is needed to build -zathurarc.5. If it is not installed, zathurarc.5 won't be built. - -Configuration -------------- -You can modify some parts of zathura by editing the config.h file - Installation ------------ -Customise config.h according to your wishes and run the following -command to build and install zathura: +To build and install zathura: make install @@ -31,9 +22,3 @@ Uninstall: To delete zathura from your system, just type: make uninstall - -Use zathura ------------ -Just run: - - zathura diff --git a/bookmarks.c b/bookmarks.c new file mode 100644 index 0000000..c4c660c --- /dev/null +++ b/bookmarks.c @@ -0,0 +1,97 @@ +/* See LICENSE file for license and copyright information */ + +#include +#include "bookmarks.h" +#include "database.h" +#include "document.h" + +zathura_bookmark_t* +zathura_bookmark_add(zathura_t* zathura, const gchar* id, unsigned int page) +{ + g_return_val_if_fail(zathura && zathura->document && zathura->bookmarks.bookmarks, NULL); + g_return_val_if_fail(id, NULL); + + GIRARA_LIST_FOREACH(zathura->bookmarks.bookmarks, zathura_bookmark_t*, iter, bookmark) + if (strcmp(bookmark->id, id) == 0) { + girara_list_iterator_free(iter); + return NULL; + } + GIRARA_LIST_FOREACH_END(zathura->bookmarks.bookmarks, zathura_bookmark_t*, iter, bookmark) + + zathura_bookmark_t* bookmark = g_malloc0(sizeof(zathura_bookmark_t)); + bookmark->id = g_strdup(id); + bookmark->page = page; + girara_list_append(zathura->bookmarks.bookmarks, bookmark); + if (zathura->database) { + if (!zathura_db_add_bookmark(zathura->database, zathura->document->file_path, bookmark)) { + girara_warning("Failed to add bookmark to database."); + } + } + + return bookmark; +} + +bool +zathura_bookmark_remove(zathura_t* zathura, const gchar* id) +{ + g_return_val_if_fail(zathura && zathura->document && zathura->bookmarks.bookmarks, false); + g_return_val_if_fail(id, false); + + zathura_bookmark_t* bookmark = zathura_bookmark_get(zathura, id); + if (!bookmark) { + return false; + } + + if (zathura->database) { + if (!zathura_db_remove_bookmark(zathura->database, zathura->document->file_path, bookmark->id)) { + girara_warning("Failed to remove bookmark from database."); + } + } + girara_list_remove(zathura->bookmarks.bookmarks, bookmark); + + return true; +} + +zathura_bookmark_t* +zathura_bookmark_get(zathura_t* zathura, const gchar* id) +{ + g_return_val_if_fail(zathura && zathura->bookmarks.bookmarks, NULL); + g_return_val_if_fail(id, NULL); + + GIRARA_LIST_FOREACH(zathura->bookmarks.bookmarks, zathura_bookmark_t*, iter, bookmark) + if (strcmp(bookmark->id, id) == 0) { + girara_list_iterator_free(iter); + return bookmark; + } + GIRARA_LIST_FOREACH_END(zathura->bookmarks.bookmarks, zathura_bookmark_t*, iter, bookmark) + + return NULL; +} + +void +zathura_bookmark_free(zathura_bookmark_t* bookmark) +{ + if (!bookmark) { + return; + } + + g_free(bookmark->id); + g_free(bookmark); +} + +bool +zathura_bookmarks_load(zathura_t* zathura, const gchar* file) { + g_return_val_if_fail(zathura && zathura->database, false); + g_return_val_if_fail(file, false); + + girara_list_t* bookmarks = zathura_db_load_bookmarks(zathura->database, file); + if (!bookmarks) { + return false; + } + + girara_list_free(zathura->bookmarks.bookmarks); + zathura->bookmarks.bookmarks = bookmarks; + return true; +} + + diff --git a/bookmarks.h b/bookmarks.h new file mode 100644 index 0000000..156375a --- /dev/null +++ b/bookmarks.h @@ -0,0 +1,56 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef BOOKMARKS_H +#define BOOKMARKS_H + +#include +#include "zathura.h" + +struct zathura_bookmark_s +{ + gchar* id; + unsigned int page; +}; + +typedef struct zathura_bookmark_s zathura_bookmark_t; + +/** + * Create a bookmark and add it to the list of bookmarks. + * @param zathura The zathura instance. + * @param id The bookmark's id. + * @param page The bookmark's page. + * @return the bookmark instance or NULL on failure. + */ +zathura_bookmark_t* zathura_bookmark_add(zathura_t* zathura, const gchar* id, unsigned int page); + +/** + * Remove a bookmark from the list of bookmarks. + * @param zathura The zathura instance. + * @param id The bookmark's id. + * @return true on success, false otherwise + */ +bool zathura_bookmark_remove(zathura_t* zathura, const gchar* id); + +/** + * Get bookmark from the list of bookmarks. + * @param zathura The zathura instance. + * @param id The bookmark's id. + * @return The bookmark instance if it exists or NULL otherwise. + */ +zathura_bookmark_t* zathura_bookmark_get(zathura_t* zathura, const gchar* id); + +/** + * Free a bookmark instance. + * @param bookmark The bookmark instance. + */ +void zathura_bookmark_free(zathura_bookmark_t* bookmark); + +/** + * Load bookmarks for a specific file. + * @param zathura The zathura instance. + * @param file The file. + * @param true on success, false otherwise + */ +bool zathura_bookmarks_load(zathura_t* zathura, const gchar* file); + +#endif // BOOKMARKS_H diff --git a/callbacks.c b/callbacks.c new file mode 100644 index 0000000..aedd9b5 --- /dev/null +++ b/callbacks.c @@ -0,0 +1,91 @@ +/* See LICENSE file for license and copyright information */ + +#include +#include +#include + +#include "callbacks.h" +#include "zathura.h" +#include "render.h" +#include "document.h" +#include "utils.h" + +gboolean +cb_destroy(GtkWidget* UNUSED(widget), gpointer UNUSED(data)) +{ + return TRUE; +} + +void +buffer_changed(girara_session_t* session) +{ + g_return_if_fail(session != NULL); + g_return_if_fail(session->global.data != NULL); + + zathura_t* zathura = session->global.data; + + char* buffer = girara_buffer_get(session); + + if (buffer) { + girara_statusbar_item_set_text(session, zathura->ui.statusbar.buffer, buffer); + free(buffer); + } else { + girara_statusbar_item_set_text(session, zathura->ui.statusbar.buffer, ""); + } +} + +void +cb_view_vadjustment_value_changed(GtkAdjustment *adjustment, gpointer data) +{ + zathura_t* zathura = data; + if (!zathura || !zathura->document || !zathura->document->pages || !zathura->ui.page_view) { + return; + } + + /* get current adjustment values */ + gdouble lower = gtk_adjustment_get_value(adjustment); + gdouble upper = lower + gtk_adjustment_get_page_size(adjustment); + + /* find page that fits */ + for (unsigned int page_id = 0; page_id < zathura->document->number_of_pages; page_id++) + { + zathura_page_t* page = zathura->document->pages[page_id]; + + page_offset_t* offset = page_calculate_offset(page); + if (offset == NULL) { + continue; + } + + double begin = offset->y; + double end = offset->y + page->height; + + if ( ( (begin >= lower) && (end <= upper) ) /* [> page is in viewport <]*/ + || ( (begin <= lower) && (end >= lower) && (end <= upper) ) /* [> end of the page is in viewport <] */ + || ( (begin >= lower) && (end >= upper) && (begin <= upper) ) /* [> begin of the page is in viewport <] */ + ) { + page->visible = true; + if (page->surface == NULL) { + render_page(zathura->sync.render_thread, page); + } + } else { + page->visible = false; + cairo_surface_destroy(page->surface); + page->surface = NULL; + } + + free(offset); + } +} + +void +cb_pages_per_row_value_changed(girara_session_t* UNUSED(session), girara_setting_t* setting) +{ + int pages_per_row = setting->value.i; + zathura_t* zathura = setting->data; + + if (pages_per_row < 1) { + pages_per_row = 1; + } + + page_view_set_mode(zathura, pages_per_row); +} diff --git a/callbacks.h b/callbacks.h new file mode 100644 index 0000000..1d960c6 --- /dev/null +++ b/callbacks.h @@ -0,0 +1,42 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef CALLBACKS_H +#define CALLBACKS_H + +#include +#include + +/** + * Quits the current zathura session + * + * @param widget The gtk window of zathura + * @param data NULL + * @return TRUE + */ +gboolean cb_destroy(GtkWidget* widget, gpointer data); + +/** + * This function gets called when the buffer of girara changes + * + * @param session The girara session + */ +void buffer_changed(girara_session_t* session); + +/** + * This function gets called when the value of the vertical scrollbars + * changes (e.g.: by scrolling, moving to another page) + * + * @param adjustment The vadjustment of the page view + * @param data NULL + */ +void cb_view_vadjustment_value_changed(GtkAdjustment *adjustment, gpointer data); +/** + * This function gets called when the value of the "pages-per-row" + * variable changes + * + * @param session The current girara session + * @param setting The "pages-per-row" setting + */ +void cb_pages_per_row_value_changed(girara_session_t* session, girara_setting_t* setting); + +#endif // CALLBACKS_H diff --git a/commands.c b/commands.c new file mode 100644 index 0000000..82ba01e --- /dev/null +++ b/commands.c @@ -0,0 +1,169 @@ +/* See LICENSE file for license and copyright information */ + +#include "commands.h" +#include "bookmarks.h" +#include "database.h" +#include "zathura.h" +#include "print.h" +#include "document.h" + +bool +cmd_bookmark_create(girara_session_t* UNUSED(session), girara_list_t* + UNUSED(argument_list)) +{ + return true; +} + +bool +cmd_bookmark_delete(girara_session_t* session, girara_list_t* argument_list) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + if (zathura->document == NULL) { + girara_notify(session, GIRARA_ERROR, "No document opened."); + return false; + } + + const unsigned int argc = girara_list_size(argument_list); + if (argc != 1) { + girara_notify(session, GIRARA_ERROR, "Invalid number of arguments given."); + return false; + } + + const char* bookmark = girara_list_nth(argument_list, 0); + if (zathura_bookmark_remove(zathura, bookmark)) { + girara_notify(session, GIRARA_INFO, "Removed bookmark: %s", bookmark); + } else { + girara_notify(session, GIRARA_ERROR, "Failed to remove bookmark: %s", bookmark); + } + + return true; +} + +bool +cmd_bookmark_open(girara_session_t* UNUSED(session), girara_list_t* + UNUSED(argument_list)) +{ + return true; +} + +bool +cmd_close(girara_session_t* session, girara_list_t* UNUSED(argument_list)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + if (zathura->document == NULL) { + // nothing needs to be done + return true; + } + + document_close(zathura); + + return true; +} + +bool +cmd_info(girara_session_t* UNUSED(session), girara_list_t* + UNUSED(argument_list)) +{ + return true; +} + +bool +cmd_help(girara_session_t* UNUSED(session), girara_list_t* + UNUSED(argument_list)) +{ + return true; +} + +bool +cmd_open(girara_session_t* session, girara_list_t* argument_list) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + + const int argc = girara_list_size(argument_list); + if (argc > 2) { + girara_notify(session, GIRARA_ERROR, "Too many arguments."); + return false; + } + else if (argc >= 1) { + if (zathura->document) { + document_close(zathura); + } + + document_open(zathura, girara_list_nth(argument_list, 0), (argc == 2) ? girara_list_nth(argument_list, 1) : NULL); + } + else { + girara_notify(session, GIRARA_ERROR, "No arguments given."); + return false; + } + + return true; +} + +bool +cmd_print(girara_session_t* session, girara_list_t* UNUSED(argument_list)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + + if (zathura->document == NULL) { + girara_notify(session, GIRARA_ERROR, "No open document."); + return false; + } + + print((zathura_t*) session->global.data); + + return true; +} + +bool +cmd_save(girara_session_t* session, girara_list_t* argument_list) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + + if (zathura->document == NULL) { + girara_notify(session, GIRARA_ERROR, "No open document."); + return false; + } + + if (girara_list_size(argument_list) == 1) { + document_save(zathura, girara_list_nth(argument_list, 0), false); + } + else { + girara_notify(session, GIRARA_ERROR, "Invalid number of arguments."); + return false; + } + + return true; +} + +bool +cmd_savef(girara_session_t* session, girara_list_t* argument_list) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + + if (zathura->document == NULL) { + girara_notify(session, GIRARA_ERROR, "No open document."); + return false; + } + + if (girara_list_size(argument_list) == 1) { + document_save(zathura, girara_list_nth(argument_list, 0), true); + } + else { + girara_notify(session, GIRARA_ERROR, "Invalid number of arguments."); + return false; + } + + return true; +} diff --git a/commands.h b/commands.h new file mode 100644 index 0000000..1f09318 --- /dev/null +++ b/commands.h @@ -0,0 +1,100 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef COMMANDS_H +#define COMMANDS_H + +#include +#include + +/** + * Create a bookmark + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_bookmark_create(girara_session_t* session, girara_list_t* argument_list); + +/** + * Delete a bookmark + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_bookmark_delete(girara_session_t* session, girara_list_t* argument_list); + +/** + * Open a bookmark + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_bookmark_open(girara_session_t* session, girara_list_t* argument_list); + +/** + * Close zathura + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_close(girara_session_t* session, girara_list_t* argument_list); + +/** + * Display document information + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_info(girara_session_t* session, girara_list_t* argument_list); + +/** + * Display help + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_help(girara_session_t* session, girara_list_t* argument_list); + +/** + * Opens a document file + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_open(girara_session_t* session, girara_list_t* argument_list); + +/** + * Print the current file + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_print(girara_session_t* session, girara_list_t* argument_list); + +/** + * Save the current file + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_save(girara_session_t* session, girara_list_t* argument_list); + +/** + * Save the current file and overwrite existing files + * + * @param session The used girara session + * @param argument_list List of passed arguments + * @return true if no error occured + */ +bool cmd_savef(girara_session_t* session, girara_list_t* argument_list); + + +#endif // COMMANDS_H diff --git a/completion.c b/completion.c new file mode 100644 index 0000000..5fc32cf --- /dev/null +++ b/completion.c @@ -0,0 +1,144 @@ +/* See LICENSE file for license and copyright information */ + +#include +#include +#include +#include +#include + +#include "completion.h" +#include "utils.h" + +girara_completion_t* +cc_open(girara_session_t* session, char* input) +{ + g_return_val_if_fail(session != NULL, NULL); + g_return_val_if_fail(session->global.data != NULL, NULL); + zathura_t* zathura = session->global.data; + + girara_completion_t* completion = girara_completion_init(); + girara_completion_group_t* group = girara_completion_group_create(session, NULL); + + gchar* path = NULL; + gchar* current_path = NULL; + + if (completion == NULL || group == NULL) { + goto error_free; + } + + path = girara_fix_path(input); + if (path == NULL) { + goto error_free; + } + + /* If the path does not begin with a slash we update the path with the current + * working directory */ + if (strlen(path) == 0 || path[0] != '/') { + size_t path_max; +#ifdef PATH_MAX + path_max = PATH_MAX; +#else + path_max = pathconf(path,_PC_PATH_MAX); + if (path_max <= 0) + path_max = 4096; +#endif + + char cwd[path_max]; + getcwd(cwd, path_max); + + char* tmp_path = g_strdup_printf("%s/%s", cwd, path); + if (tmp_path == NULL) { + goto error_free; + } + + g_free(path); + path = tmp_path; + } + + /* Append a slash if the given argument is a directory */ + bool is_dir = (path[strlen(path) - 1] == '/') ? true : false; + if ((g_file_test(path, G_FILE_TEST_IS_DIR) == TRUE) && !is_dir) { + char* tmp_path = g_strdup_printf("%s/", path); + if (tmp_path == NULL) { + goto error_free; + } + + g_free(path); + path = tmp_path; + is_dir = true; + } + + /* get current path */ + char* tmp = g_strdup(path); + current_path = is_dir ? g_strdup(tmp) : g_strdup(dirname(tmp)); + g_free(tmp); + + /* get current file */ + gchar* current_file = is_dir ? "" : basename(path); + int current_file_length = strlen(current_file); + + /* read directory */ + if (g_file_test(current_path, G_FILE_TEST_IS_DIR) == TRUE) { + GDir* dir = g_dir_open(current_path, 0, NULL); + if (dir == NULL) { + goto error_free; + } + + /* read files */ + char* name = NULL; + while ((name = (char*) g_dir_read_name(dir)) != NULL) { + char* e_name = g_filename_display_name(name); + int e_length = strlen(e_name); + + if (e_name == NULL) { + goto error_free; + } + + if ((current_file_length > e_length) || strncmp(current_file, e_name, + current_file_length)) { + g_free(e_name); + continue; + } + + char* full_path = g_strdup_printf("%s%s%s", current_path, is_dir ? "" : "/", e_name); + if (full_path == NULL) { + g_free(e_name); + goto error_free; + } + + if (g_file_test(full_path, G_FILE_TEST_IS_DIR) == true) { + char* tmp_path = full_path; + full_path = g_strdup_printf("%s/", full_path); + g_free(tmp_path); + girara_completion_group_add_element(group, full_path, NULL); + } else if (file_valid_extension(zathura, full_path) == true) { + girara_completion_group_add_element(group, full_path, NULL); + } + + g_free(full_path); + g_free(e_name); + } + + g_dir_close(dir); + } + + g_free(path); + + girara_completion_add_group(completion, group); + + return completion; + +error_free: + + if (completion) { + girara_completion_free(completion); + } + if (group) { + girara_completion_group_free(group); + } + + g_free(current_path); + g_free(path); + + return NULL; +} diff --git a/completion.h b/completion.h new file mode 100644 index 0000000..7041d8a --- /dev/null +++ b/completion.h @@ -0,0 +1,18 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef COMPLETION_H +#define COMPLETION_H + +#include + +/** + * Completion for the open command - Creates a list of accesible directories or + * files + * + * @param session The used girara session + * @param input The current input + * @return The completion object or NULL if an error occured + */ +girara_completion_t* cc_open(girara_session_t* session, char* input); + +#endif // COMPLETION_H diff --git a/config.c b/config.c new file mode 100644 index 0000000..b54aea8 --- /dev/null +++ b/config.c @@ -0,0 +1,150 @@ +/* See LICENSE file for license and copyright information */ + +#include "config.h" +#include "commands.h" +#include "completion.h" +#include "callbacks.h" +#include "shortcuts.h" +#include "zathura.h" + +void +config_load_default(zathura_t* zathura) +{ + if (!zathura || !zathura->ui.session) { + return; + } + + int int_value = 0; + char* string_value = NULL; + girara_session_t* gsession = zathura->ui.session; + + /* mode settings */ + zathura->modes.normal = gsession->modes.normal; + zathura->modes.fullscreen = girara_mode_add(gsession, "fullscreen"); + zathura->modes.index = girara_mode_add(gsession, "index"); + zathura->modes.insert = girara_mode_add(gsession, "insert"); + +#define NORMAL zathura->modes.normal +#define INSERT zathura->modes.insert +#define INDEX zathura->modes.index +#define FULLSCREEN zathura->modes.fullscreen + + girara_mode_set(gsession, zathura->modes.normal); + + /* zathura settings */ + int_value = 10; + girara_setting_add(gsession, "zoom-step", &int_value, INT, false, "Zoom step", NULL, NULL); + int_value = 1; + girara_setting_add(gsession, "page-padding", &int_value, INT, true, "Padding between pages", NULL, NULL); + int_value = 2; + girara_setting_add(gsession, "pages-per-row", &int_value, INT, false, "Number of pages per row", cb_pages_per_row_value_changed, zathura); + + string_value = "#FFFFFF"; + girara_setting_add(gsession, "recolor-dark-color", string_value, STRING, false, "Recoloring (dark color)", NULL, NULL); + string_value = "#000000"; + girara_setting_add(gsession, "recolor-light-color", string_value, STRING, false, "Recoloring (light color)", NULL, NULL); + + /* define default shortcuts */ + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_c, NULL, sc_abort, 0, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_Escape, NULL, sc_abort, 0, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_a, NULL, sc_adjust_window, NORMAL, ADJUST_BESTFIT, NULL); + girara_shortcut_add(gsession, 0, GDK_s, NULL, sc_adjust_window, NORMAL, ADJUST_WIDTH, NULL); + girara_shortcut_add(gsession, 0, GDK_i, NULL, sc_change_mode, NORMAL, INSERT, NULL); + girara_shortcut_add(gsession, 0, GDK_m, NULL, sc_change_mode, NORMAL, ADD_MARKER, NULL); + girara_shortcut_add(gsession, 0, GDK_apostrophe, NULL, sc_change_mode, NORMAL, EVAL_MARKER, NULL); + girara_shortcut_add(gsession, 0, GDK_slash, NULL, girara_sc_focus_inputbar, NORMAL, 0, &("/")); + girara_shortcut_add(gsession, GDK_SHIFT_MASK, GDK_slash, NULL, girara_sc_focus_inputbar, NORMAL, 0, &("/")); + girara_shortcut_add(gsession, 0, GDK_question, NULL, girara_sc_focus_inputbar, NORMAL, 0, &("?")); + girara_shortcut_add(gsession, 0, GDK_colon, NULL, girara_sc_focus_inputbar, NORMAL, 0, &(":")); + girara_shortcut_add(gsession, 0, GDK_o, NULL, girara_sc_focus_inputbar, NORMAL, 0, &(":open ")); + girara_shortcut_add(gsession, 0, GDK_O, NULL, girara_sc_focus_inputbar, NORMAL, APPEND_FILEPATH, &(":open ")); + girara_shortcut_add(gsession, 0, GDK_f, NULL, sc_follow, NORMAL, 0, NULL); + girara_shortcut_add(gsession, 0, 0, "gg", sc_goto, NORMAL | FULLSCREEN, TOP, NULL); + girara_shortcut_add(gsession, 0, 0, "G", sc_goto, NORMAL | FULLSCREEN, BOTTOM, NULL); + girara_shortcut_add(gsession, 0, GDK_J, NULL, sc_navigate, NORMAL, NEXT, NULL); + girara_shortcut_add(gsession, 0, GDK_K, NULL, sc_navigate, NORMAL, PREVIOUS, NULL); + girara_shortcut_add(gsession, GDK_MOD1_MASK, GDK_Right, NULL, sc_navigate, NORMAL, NEXT, NULL); + girara_shortcut_add(gsession, GDK_MOD1_MASK, GDK_Left, NULL, sc_navigate, NORMAL, PREVIOUS, NULL); + girara_shortcut_add(gsession, 0, GDK_Left, NULL, sc_navigate, FULLSCREEN, PREVIOUS, NULL); + girara_shortcut_add(gsession, 0, GDK_Up, NULL, sc_navigate, FULLSCREEN, PREVIOUS, NULL); + girara_shortcut_add(gsession, 0, GDK_Down, NULL, sc_navigate, FULLSCREEN, NEXT, NULL); + girara_shortcut_add(gsession, 0, GDK_Right, NULL, sc_navigate, FULLSCREEN, NEXT, NULL); + girara_shortcut_add(gsession, 0, GDK_k, NULL, sc_navigate_index, INDEX, UP, NULL); + girara_shortcut_add(gsession, 0, GDK_j, NULL, sc_navigate_index, INDEX, DOWN, NULL); + girara_shortcut_add(gsession, 0, GDK_h, NULL, sc_navigate_index, INDEX, COLLAPSE, NULL); + girara_shortcut_add(gsession, 0, GDK_l, NULL, sc_navigate_index, INDEX, EXPAND, NULL); + girara_shortcut_add(gsession, 0, GDK_space, NULL, sc_navigate_index, INDEX, SELECT, NULL); + girara_shortcut_add(gsession, 0, GDK_Return, NULL, sc_navigate_index, INDEX, SELECT, NULL); + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_i, NULL, sc_recolor, NORMAL, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_R, NULL, sc_reload, NORMAL, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_r, NULL, sc_rotate, NORMAL, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_h, NULL, sc_scroll, NORMAL, LEFT, NULL); + girara_shortcut_add(gsession, 0, GDK_j, NULL, sc_scroll, NORMAL, DOWN, NULL); + girara_shortcut_add(gsession, 0, GDK_k, NULL, sc_scroll, NORMAL, UP, NULL); + girara_shortcut_add(gsession, 0, GDK_l, NULL, sc_scroll, NORMAL, RIGHT, NULL); + girara_shortcut_add(gsession, 0, GDK_Left, NULL, sc_scroll, NORMAL, LEFT, NULL); + girara_shortcut_add(gsession, 0, GDK_Up, NULL, sc_scroll, NORMAL, UP, NULL); + girara_shortcut_add(gsession, 0, GDK_Down, NULL, sc_scroll, NORMAL, DOWN, NULL); + girara_shortcut_add(gsession, 0, GDK_Right, NULL, sc_scroll, NORMAL, RIGHT, NULL); + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_d, NULL, sc_scroll, NORMAL, HALF_DOWN, NULL); + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_u, NULL, sc_scroll, NORMAL, HALF_UP, NULL); + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_f, NULL, sc_scroll, NORMAL, FULL_DOWN, NULL); + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_b, NULL, sc_scroll, NORMAL, FULL_UP, NULL); + girara_shortcut_add(gsession, 0, GDK_space, NULL, sc_scroll, NORMAL, FULL_DOWN, NULL); + girara_shortcut_add(gsession, GDK_SHIFT_MASK, GDK_space, NULL, sc_scroll, NORMAL, FULL_UP, NULL); + girara_shortcut_add(gsession, 0, GDK_n, NULL, sc_search, NORMAL, FORWARD, NULL); + girara_shortcut_add(gsession, 0, GDK_N, NULL, sc_search, NORMAL, BACKWARD, NULL); + girara_shortcut_add(gsession, 0, GDK_Tab, NULL, sc_toggle_index, NORMAL | INDEX, 0, NULL); + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_m, NULL, girara_sc_toggle_inputbar, NORMAL, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_F5, NULL, sc_toggle_fullscreen, NORMAL | FULLSCREEN, 0, NULL); + girara_shortcut_add(gsession, GDK_CONTROL_MASK, GDK_n, NULL, girara_sc_toggle_statusbar, NORMAL, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_q, NULL, sc_quit, NORMAL, 0, NULL); + girara_shortcut_add(gsession, 0, GDK_plus, NULL, sc_zoom, NORMAL | FULLSCREEN, ZOOM_IN, NULL); + girara_shortcut_add(gsession, 0, GDK_minus, NULL, sc_zoom, NORMAL | FULLSCREEN, ZOOM_OUT, NULL); + girara_shortcut_add(gsession, 0, GDK_equal, NULL, sc_zoom, NORMAL | FULLSCREEN, ZOOM_ORIGINAL, NULL); + girara_shortcut_add(gsession, 0, 0, "zI", sc_zoom, NORMAL | FULLSCREEN, ZOOM_IN, NULL); + girara_shortcut_add(gsession, 0, 0, "zO", sc_zoom, NORMAL | FULLSCREEN, ZOOM_OUT, NULL); + girara_shortcut_add(gsession, 0, 0, "z0", sc_zoom, NORMAL | FULLSCREEN, ZOOM_ORIGINAL, NULL); + + /* define default inputbar commands */ + girara_inputbar_command_add(gsession, "bmark", NULL, cmd_bookmark_create, NULL, "Add a bookmark"); + girara_inputbar_command_add(gsession, "bdelete", NULL, cmd_bookmark_delete, NULL, "Delete a bookmark"); + girara_inputbar_command_add(gsession, "blist", NULL, cmd_bookmark_open, NULL, "List all bookmarks"); + girara_inputbar_command_add(gsession, "close", NULL, cmd_close, NULL, "Close current file"); + girara_inputbar_command_add(gsession, "info", NULL, cmd_info, NULL, "Show file information"); + girara_inputbar_command_add(gsession, "help", NULL, cmd_help, NULL, "Show help"); + girara_inputbar_command_add(gsession, "open", "o", cmd_open, cc_open, "Open document"); + girara_inputbar_command_add(gsession, "print", NULL, cmd_print, NULL, "Print document"); + girara_inputbar_command_add(gsession, "write", NULL, cmd_save, NULL, "Save document"); + girara_inputbar_command_add(gsession, "write!", NULL, cmd_savef, NULL, "Save document (and force overwriting)"); + + /* add shortcut mappings */ + girara_shortcut_mapping_add(gsession, "abort", sc_abort); + girara_shortcut_mapping_add(gsession, "adjust_window", sc_adjust_window); + girara_shortcut_mapping_add(gsession, "change_mode", sc_change_mode); + girara_shortcut_mapping_add(gsession, "follow", sc_follow); + girara_shortcut_mapping_add(gsession, "goto", sc_goto); + girara_shortcut_mapping_add(gsession, "index_navigate", sc_navigate_index); + girara_shortcut_mapping_add(gsession, "navigate", sc_navigate); + girara_shortcut_mapping_add(gsession, "quit", sc_quit); + girara_shortcut_mapping_add(gsession, "recolor", sc_recolor); + girara_shortcut_mapping_add(gsession, "reload", sc_reload); + girara_shortcut_mapping_add(gsession, "rotate", sc_rotate); + girara_shortcut_mapping_add(gsession, "scroll", sc_scroll); + girara_shortcut_mapping_add(gsession, "search", sc_search); + girara_shortcut_mapping_add(gsession, "toggle_fullscreen", sc_toggle_fullscreen); + girara_shortcut_mapping_add(gsession, "toggle_index", sc_toggle_index); + girara_shortcut_mapping_add(gsession, "toggle_inputbar", girara_sc_toggle_inputbar); + girara_shortcut_mapping_add(gsession, "toggle_statusbar", girara_sc_toggle_statusbar); + girara_shortcut_mapping_add(gsession, "zoom", sc_zoom); +} + +void +config_load_file(zathura_t* zathura, char* path) +{ + if (zathura == NULL) { + return; + } + + girara_config_parse(zathura->ui.session, path); +} diff --git a/config.def.h b/config.def.h deleted file mode 100644 index 080796c..0000000 --- a/config.def.h +++ /dev/null @@ -1,368 +0,0 @@ -/* settings */ -int default_width = 800; -int default_height = 600; -float zoom_step = 10; -float zoom_min = 10; -float zoom_max = 400; -float scroll_step = 40; -float transparency = 0.4; -float smooth_scrolling = 0.0; -int n_completion_items = 15; - -/* completion */ -static const char FORMAT_COMMAND[] = "%s"; -static const char FORMAT_DESCRIPTION[] = "%s"; - -/* Use XDG directory specification if no config and data directory are given on - * the command line. Uncomment the next line if you just want to use CONFIG_DIR - * and DATA_DIR instead (these will be the default locations if the XDG_* - * environment variables are not set anyway) */ -/* #define ZATHURA_NO_XDG */ - -/* directories and files */ -static const char BOOKMARK_FILE[] = "bookmarks"; -static const char ZATHURA_RC[] = "zathurarc"; -static const char GLOBAL_RC[] = "/etc/zathurarc"; -static const char CONFIG_DIR[] = "~/.config/zathura"; -static const char DATA_DIR[] = "~/.local/share/zathura"; - -/* bookmarks */ -enum -{ - BM_PAGE_ENTRY = 0, - BM_PAGE_OFFSET, - BM_PAGE_SCALE, - BM_MAX, -}; - -static const char *bm_reserved_names[] = -{ - [BM_PAGE_ENTRY] = "page", - [BM_PAGE_OFFSET] = "offset", - [BM_PAGE_SCALE] = "scale", -}; - -int save_position = 1; -int save_zoom_level = 1; - -/* look */ -char* font = "monospace normal 9"; -char* default_bgcolor = "#000000"; -char* default_fgcolor = "#DDDDDD"; -char* inputbar_bgcolor = "#141414"; -char* inputbar_fgcolor = "#9FBC00"; -char* statusbar_bgcolor = "#000000"; -char* statusbar_fgcolor = "#FFFFFF"; -char* completion_fgcolor = "#DDDDDD"; -char* completion_bgcolor = "#232323"; -char* completion_g_fgcolor = "#DEDEDE"; -char* completion_g_bgcolor = "#FF00FF"; -char* completion_hl_fgcolor = "#232323"; -char* completion_hl_bgcolor = "#9FBC00"; -char* notification_e_bgcolor = "#FF1212"; -char* notification_e_fgcolor = "#FFFFFF"; -char* notification_w_bgcolor = "#FFF712"; -char* notification_w_fgcolor = "#000000"; -char* recolor_darkcolor = "#353535"; -char* recolor_lightcolor = "#DBDBDB"; - -char* search_highlight = "#9FBC00"; -char* select_text = "#000000"; - -/* statusbar */ -char* default_text = "[No Name]"; - -/* printing */ -char* list_printer_command = "lpstat -v | sed -n '/^.*device for \\(.*\\): .*$/s//\\1/p'"; -char* print_command = "lp -d '%s' -P %s %s %s"; /* printer / pages / file */ - -/* open uri */ -char* uri_command = "firefox %s"; /* uri */ - -/* additional settings */ -gboolean show_scrollbars = FALSE; -gboolean scroll_wrap = TRUE; -int adjust_open = ADJUST_BESTFIT; -#define SELECTION_STYLE POPPLER_SELECTION_GLYPH -#define GOTO_MODE GOTO_LABELS /* GOTO_DEFAULT, GOTO_LABELS, GOTO_OFFSET */ - -/* define additional modes */ -#define INSERT (1 << 4) -#define VISUAL (1 << 5) -#define EMACS_CX (1 << 6) - -/* mode names */ -ModeName mode_names[] = { - /* default mode names */ - {"all", ALL, ""}, - {"fullscreen", FULLSCREEN, ""}, - {"index", INDEX, ""}, - {"normal", NORMAL, ""}, - /* additional mode names */ - {"insert", INSERT, "-- INSERT --"}, - {"visual", VISUAL, "-- VISUAL --"} -}; - -/* shortcuts */ -Shortcut shortcuts[] = { - /* mask, key, function, mode, argument */ - {GDK_CONTROL_MASK, GDK_c, sc_abort, ALL, {0} }, - {0, GDK_a, sc_adjust_window, NORMAL, { ADJUST_BESTFIT } }, - {0, GDK_s, sc_adjust_window, NORMAL, { ADJUST_WIDTH } }, - {0, GDK_Escape, sc_abort, ALL, {0} }, - {0, GDK_BackSpace, sc_change_buffer, ALL, { DELETE_LAST } }, - {0, GDK_i, sc_change_mode, NORMAL, { INSERT } }, - {0, GDK_v, sc_change_mode, NORMAL, { VISUAL } }, - {0, GDK_m, sc_change_mode, NORMAL, { ADD_MARKER } }, - {0, GDK_apostrophe, sc_change_mode, NORMAL, { EVAL_MARKER } }, - {0, GDK_slash, sc_focus_inputbar, NORMAL, { .data = "/" } }, - {GDK_SHIFT_MASK, GDK_slash, sc_focus_inputbar, NORMAL, { .data = "/" } }, - {GDK_SHIFT_MASK, GDK_question, sc_focus_inputbar, NORMAL, { .data = "?" } }, - {0, GDK_colon, sc_focus_inputbar, NORMAL, { .data = ":" } }, - {0, GDK_o, sc_focus_inputbar, NORMAL, { .data = ":open " } }, - {0, GDK_O, sc_focus_inputbar, NORMAL, { APPEND_FILEPATH, .data = ":open " } }, - {0, GDK_f, sc_follow, NORMAL, {0} }, - {0, GDK_J, sc_navigate, NORMAL, { NEXT } }, - {0, GDK_K, sc_navigate, NORMAL, { PREVIOUS } }, - {GDK_MOD1_MASK, GDK_Right, sc_navigate, NORMAL, { NEXT } }, - {GDK_MOD1_MASK, GDK_Left, sc_navigate, NORMAL, { PREVIOUS } }, - {0, GDK_Left, sc_navigate, FULLSCREEN, { PREVIOUS } }, - {0, GDK_Up, sc_navigate, FULLSCREEN, { PREVIOUS } }, - {0, GDK_Down, sc_navigate, FULLSCREEN, { NEXT } }, - {0, GDK_Right, sc_navigate, FULLSCREEN, { NEXT } }, - {0, GDK_space, sc_navigate, FULLSCREEN, { NEXT } }, - {0, GDK_k, sc_navigate_index, INDEX, { UP } }, - {0, GDK_j, sc_navigate_index, INDEX, { DOWN } }, - {0, GDK_h, sc_navigate_index, INDEX, { COLLAPSE } }, - {0, GDK_l, sc_navigate_index, INDEX, { EXPAND } }, - {0, GDK_space, sc_navigate_index, INDEX, { SELECT } }, - {0, GDK_Return, sc_navigate_index, INDEX, { SELECT } }, - {GDK_CONTROL_MASK, GDK_i, sc_recolor, NORMAL, {0} }, - {0, GDK_R, sc_reload, NORMAL, {0} }, - {0, GDK_r, sc_rotate, NORMAL, {0} }, - {0, GDK_n, sc_search, NORMAL, { FORWARD } }, - {0, GDK_N, sc_search, NORMAL, { BACKWARD } }, - {0, GDK_h, sc_scroll, NORMAL, { LEFT } }, - {0, GDK_j, sc_scroll, NORMAL, { DOWN } }, - {0, GDK_k, sc_scroll, NORMAL, { UP } }, - {0, GDK_l, sc_scroll, NORMAL, { RIGHT } }, - {0, GDK_Left, sc_scroll, NORMAL, { LEFT } }, - {0, GDK_Up, sc_scroll, NORMAL, { UP } }, - {0, GDK_Down, sc_scroll, NORMAL, { DOWN } }, - {0, GDK_Right, sc_scroll, NORMAL, { RIGHT } }, - {GDK_CONTROL_MASK, GDK_d, sc_scroll, NORMAL, { HALF_DOWN } }, - {GDK_CONTROL_MASK, GDK_u, sc_scroll, NORMAL, { HALF_UP } }, - {GDK_CONTROL_MASK, GDK_f, sc_scroll, NORMAL, { FULL_DOWN } }, - {GDK_CONTROL_MASK, GDK_b, sc_scroll, NORMAL, { FULL_UP } }, - {0, GDK_space, sc_scroll, NORMAL, { FULL_DOWN } }, - {0, GDK_0, sc_switch_goto_mode, NORMAL, {0} }, - {0, GDK_F5, sc_toggle_fullscreen, NORMAL | FULLSCREEN, {0} }, - {0, GDK_Tab, sc_toggle_index, NORMAL | INDEX, {0} }, - {GDK_CONTROL_MASK, GDK_m, sc_toggle_inputbar, NORMAL, {0} }, - {GDK_CONTROL_MASK, GDK_n, sc_toggle_statusbar, NORMAL, {0} }, - {GDK_CONTROL_MASK, GDK_q, sc_quit, ALL, {0} }, - {0, GDK_plus, sc_zoom, NORMAL | FULLSCREEN, { ZOOM_IN } }, - {0, GDK_minus, sc_zoom, NORMAL | FULLSCREEN, { ZOOM_OUT } }, - {0, GDK_equal, sc_zoom, NORMAL | FULLSCREEN, { ZOOM_ORIGINAL } }, -}; - -/* inputbar shortcuts */ -InputbarShortcut inputbar_shortcuts[] = { - /* mask, key, function, argument */ - {0, GDK_Escape, isc_abort, {0} }, - {GDK_CONTROL_MASK, GDK_c, isc_abort, {0} }, - {0, GDK_Up, isc_command_history, {0} }, - {0, GDK_Down, isc_command_history, {0} }, - {0, GDK_Tab, isc_completion, { NEXT } }, - {GDK_CONTROL_MASK, GDK_Tab, isc_completion, { NEXT_GROUP } }, - {0, GDK_ISO_Left_Tab, isc_completion, { PREVIOUS } }, - {GDK_CONTROL_MASK, GDK_ISO_Left_Tab, isc_completion, { PREVIOUS_GROUP } }, - {0, GDK_BackSpace, isc_string_manipulation, { DELETE_LAST_CHAR } }, - {GDK_CONTROL_MASK, GDK_h, isc_string_manipulation, { DELETE_LAST_CHAR } }, - {GDK_CONTROL_MASK, GDK_u, isc_string_manipulation, { DELETE_TO_LINE_START } }, - {GDK_CONTROL_MASK, GDK_w, isc_string_manipulation, { DELETE_LAST_WORD } }, - {GDK_CONTROL_MASK, GDK_f, isc_string_manipulation, { NEXT_CHAR } }, - {GDK_CONTROL_MASK, GDK_b, isc_string_manipulation, { PREVIOUS_CHAR } }, -}; - -/* mouse settings */ -MouseScrollEvent mouse_scroll_events[] = { - /* direction, function, argument */ - {GDK_SCROLL_LEFT, sc_scroll, { LEFT } }, - {GDK_SCROLL_UP, sc_scroll, { UP } }, - {GDK_SCROLL_DOWN, sc_scroll, { DOWN } }, - {GDK_SCROLL_RIGHT, sc_scroll, { RIGHT } }, -}; - -/* commands */ -Command commands[] = { - /* command, abbreviation, function, completion, description */ - {"blist", 0, cmd_open_bookmark, cc_bookmark, "List and open bookmark" }, - {"bmark", "b", cmd_bookmark, 0, "Bookmark current page" }, - {"close", "c", cmd_close, 0, "Close current file" }, - {"coffset", 0, cmd_correct_offset, 0, "Correct page offset" }, - {"delbmark", 0, cmd_delete_bookmark, cc_bookmark, "Delete bookmark" }, - {"export", "e", cmd_export, cc_export, "Export images or attached files" }, - {"info", "i", cmd_info, 0, "Show information about the document" }, - {"map", "m", cmd_map, 0, "Map keybinding to a function" }, - {"open", "o", cmd_open, cc_open, "Open a file" }, - {"print", "p", cmd_print, cc_print, "Print the document" }, - {"quit", "q", cmd_quit, 0, "Quit zathura" }, - {"rotate", "r", cmd_rotate, 0, "Rotate the page" }, - {"set", "s", cmd_set, cc_set, "Set an option" }, - {"write", "w", cmd_save, 0, "Save the document" }, - {"write!", "w!", cmd_savef, 0, "Save the document (and force overwriting)" }, -}; - -/* buffer commands */ -BufferCommand buffer_commands[] = { - /* regex, function, argument */ - {"^gg$", bcmd_goto, { TOP } }, - {"^G$", bcmd_goto, { BOTTOM } }, - {"^[0-9]+G$", bcmd_goto, {0} }, - {"^zI$", bcmd_zoom, { ZOOM_IN } }, - {"^zO$", bcmd_zoom, { ZOOM_OUT } }, - {"^z0$", bcmd_zoom, { ZOOM_ORIGINAL } }, - {"^[0-9]+Z$", bcmd_zoom, { ZOOM_SPECIFIC } }, - {"^[0-9]+%$", bcmd_scroll, {0} }, -}; - -/* special commands */ -SpecialCommand special_commands[] = { - /* identifier, function, a, argument */ - {'/', scmd_search, 1, { DOWN } }, - {'?', scmd_search, 1, { UP } }, -}; - -/* settings */ -Setting settings[] = { - /* name, variable, type, render, re-init, description */ - {"adjust_open", &(adjust_open), 'i', FALSE, FALSE, "Adjust mode"}, - {"browser", &(uri_command), 's', FALSE, FALSE, "Command to open URIs"}, - {"completion_bgcolor", &(completion_bgcolor), 's', FALSE, TRUE, "Completion background color"}, - {"completion_fgcolor", &(completion_fgcolor), 's', FALSE, TRUE, "Completion foreground color"}, - {"completion_g_bgcolor", &(completion_g_bgcolor), 's', FALSE, TRUE, "Completion (group) background color"}, - {"completion_g_fgcolor", &(completion_g_fgcolor), 's', FALSE, TRUE, "Completion (group) foreground color"}, - {"completion_hl_bgcolor", &(completion_hl_bgcolor), 's', FALSE, TRUE, "Completion (highlight) background color"}, - {"completion_hl_fgcolor", &(completion_hl_fgcolor), 's', FALSE, TRUE, "Completion (highlight) foreground color"}, - {"default_bgcolor", &(default_bgcolor), 's', FALSE, TRUE, "Default background color"}, - {"default_fgcolor", &(default_fgcolor), 's', FALSE, TRUE, "Default foreground color"}, - {"default_text", &(default_text), 's', FALSE, FALSE, "Default text"}, - {"font", &(font), 's', FALSE, TRUE, "The used font" }, - {"height", &(default_height), 'i', FALSE, FALSE, "Default window height"}, - {"inputbar_bgcolor", &(inputbar_bgcolor), 's', FALSE, TRUE, "Inputbar background color"}, - {"inputbar_fgcolor", &(inputbar_fgcolor), 's', FALSE, TRUE, "Inputbar foreground color"}, - {"labels", &(Zathura.Global.enable_labelmode), 'b', FALSE, TRUE, "Allow label mode"}, - {"list_printer_command", &(list_printer_command), 's', FALSE, FALSE, "Command to list printers"}, - {"n_completion_items", &(n_completion_items), 'i', FALSE, FALSE, "Number of completion items"}, - {"notification_e_bgcolor", &(notification_e_bgcolor), 's', FALSE, TRUE, "Notification (error) background color"}, - {"notification_e_fgcolor", &(notification_e_fgcolor), 's', FALSE, TRUE, "Notification (error) foreground color"}, - {"notification_w_bgcolor", &(notification_w_bgcolor), 's', FALSE, TRUE, "Notification (warning) background color"}, - {"notification_w_fgcolor", &(notification_w_fgcolor), 's', FALSE, TRUE, "Notification (warning) foreground color"}, - {"offset", &(Zathura.PDF.page_offset), 'i', FALSE, FALSE, "Optional page offset" }, - {"print_command", &(print_command), 's', FALSE, FALSE, "Command to print"}, - {"recolor", &(Zathura.Global.recolor), 'b', TRUE, FALSE, "Invert the image" }, - {"recolor_darkcolor", &(recolor_darkcolor), 's', FALSE, TRUE, "Recoloring (dark color)"}, - {"recolor_lightcolor", &(recolor_lightcolor), 's', FALSE, TRUE, "Recoloring (light color)"}, - {"save_position", &(save_position), 'b', FALSE, FALSE, "Save position in file on quit and restore it on open"}, - {"save_zoom_level", &(save_zoom_level), 'b', FALSE, FALSE, "Save zoom level on quit and restore it on open"}, - {"scroll_step", &(scroll_step), 'f', FALSE, FALSE, "Scroll step"}, - {"scroll_wrap", &(scroll_wrap), 'b', FALSE, FALSE, "Wrap scolling at last page"}, - {"scrollbars", &(show_scrollbars), 'b', FALSE, TRUE, "Show scrollbars"}, - {"show_statusbar", &(Zathura.Global.show_statusbar), 'b', FALSE, TRUE, "Show statusbar"}, - {"show_inputbar", &(Zathura.Global.show_inputbar), 'b', FALSE, TRUE, "Show inputbar"}, - {"search_highlight", &(search_highlight), 's', FALSE, TRUE, "Highlighted results"}, - {"select_text", &(select_text), 's', FALSE, TRUE, "Rectangle of the selected text"}, - {"smooth_scrolling", &(smooth_scrolling), 'f', FALSE, TRUE, "Smooth scrolling"}, - {"statusbar_bgcolor", &(statusbar_bgcolor), 's', FALSE, TRUE, "Statusbar background color"}, - {"statusbar_fgcolor", &(statusbar_fgcolor), 's', FALSE, TRUE, "Statusbar foreground color"}, - {"transparency", &(transparency), 'f', FALSE, FALSE, "Transparency of rectangles"}, - {"uri_command", &(uri_command), 's', FALSE, FALSE, "Command for opening URIs"}, - {"width", &(default_width), 'i', FALSE, FALSE, "Default window width"}, - {"zoom_max", &(zoom_max), 'f', FALSE, FALSE, "Zoom maximum"}, - {"zoom_min", &(zoom_min), 'f', FALSE, FALSE, "Zoom minimum"}, - {"zoom_step", &(zoom_step), 'f', FALSE, FALSE, "Zoom step"}, -}; - -/* shortcut names */ -ShortcutName shortcut_names[] = { - {"abort", sc_abort}, - {"adjust_window", sc_adjust_window}, - {"change_buffer", sc_change_buffer}, - {"change_mode", sc_change_mode}, - {"focus_inputbar", sc_focus_inputbar}, - {"follow", sc_follow}, - {"navigate", sc_navigate}, - {"navigate_index", sc_navigate_index}, - {"quit", sc_quit}, - {"recolor", sc_recolor}, - {"reload", sc_reload}, - {"rotate", sc_rotate}, - {"scroll", sc_scroll}, - {"search", sc_search}, - {"switch_goto_mode", sc_switch_goto_mode}, - {"toggle_fullscreen", sc_toggle_fullscreen}, - {"toggle_index", sc_toggle_index}, - {"toggle_inputbar", sc_toggle_inputbar}, - {"toggle_statusbar", sc_toggle_statusbar}, - {"zoom", sc_zoom}, -}; - -/* argument names */ -ArgumentName argument_names[] = { - {"add_marker", ADD_MARKER}, - {"backward", BACKWARD}, - {"bestfit", ADJUST_BESTFIT}, - {"collapse", COLLAPSE}, - {"delete_last", DELETE_LAST}, - {"down", DOWN}, - {"eval_marker", EVAL_MARKER}, - {"expand", EXPAND}, - {"forward", FORWARD}, - {"full_down", FULL_DOWN}, - {"full_up", FULL_UP}, - {"half_down", HALF_DOWN}, - {"half_up", HALF_UP}, - {"in", ZOOM_IN}, - {"insert", INSERT}, - {"left", LEFT}, - {"next", NEXT}, - {"noadjust", ADJUST_NONE}, - {"original", ZOOM_ORIGINAL}, - {"out", ZOOM_OUT}, - {"previous", PREVIOUS}, - {"right", RIGHT}, - {"select", SELECT}, - {"up", UP}, - {"visual", VISUAL}, - {"width", ADJUST_WIDTH}, -}; - -/* special keys */ -GDKKey gdk_keys[] = { - {"", GDK_BackSpace}, - {"", GDK_Caps_Lock}, - {"", GDK_Down}, - {"", GDK_Escape}, - {"", GDK_F10}, - {"", GDK_F11}, - {"", GDK_F12}, - {"", GDK_F1}, - {"", GDK_F2}, - {"", GDK_F3}, - {"", GDK_F4}, - {"", GDK_F5}, - {"", GDK_F6}, - {"", GDK_F7}, - {"", GDK_F8}, - {"", GDK_F9}, - {"", GDK_Left}, - {"", GDK_Page_Down}, - {"", GDK_Page_Up}, - {"", GDK_Return}, - {"", GDK_Right}, - {"", GDK_space}, - {"", GDK_Super_L}, - {"", GDK_Tab}, - {"", GDK_Up}, -}; diff --git a/config.h b/config.h new file mode 100644 index 0000000..3e30f87 --- /dev/null +++ b/config.h @@ -0,0 +1,25 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef CONFIG_H +#define CONFIG_H + +#define GLOBAL_RC "/etc/zathurarc" +#define ZATHURA_RC "zathurarc" + +#include "zathura.h" + +/** + * This function loads the default values of the configuration + * + * @param zathura the zathura session + */ +void config_load_default(zathura_t* zathura); + +/** + * Loads and evaluates a configuration file + * + * @param path Path to the configuration file + */ +void config_load_file(zathura_t* zathura, char* path); + +#endif // CONFIG_H diff --git a/config.mk b/config.mk index 17d51ff..3363edc 100644 --- a/config.mk +++ b/config.mk @@ -1,28 +1,37 @@ # See LICENSE file for license and copyright information # zathura make config -VERSION = 0.0.8.4 +VERSION = 0.0.8.1 # paths PREFIX ?= /usr MANPREFIX ?= ${PREFIX}/share/man -DESKTOPPREFIX ?= ${PREFIX}/share/applications # libs -GTK_INC = $(shell pkg-config --cflags gtk+-2.0 poppler-glib) -GTK_LIB = $(shell pkg-config --libs gtk+-2.0 gthread-2.0 poppler-glib) -INCS = -I. -I/usr/include ${GTK_INC} -LIBS = -lc ${GTK_LIB} -lpthread -lm +# set this to 0 if you don't need to link against dl +NEEDS_DL ?= 1 -# compiler flags -CFLAGS += -std=c99 -pedantic -Wall $(INCS) +GTK_INC ?= $(shell pkg-config --cflags gtk+-2.0) +GTK_LIB ?= $(shell pkg-config --libs gtk+-2.0 gthread-2.0) -# debug flags +GIRARA_INC ?= $(shell pkg-config --cflags girara-gtk2) +GIRARA_LIB ?= $(shell pkg-config --libs girara-gtk2) + +SQLITE_INC ?= $(shell pkg-config --cflags sqlite3) +SQLITE_LIB ?= $(shell pkg-config --libs sqlite3) + +INCS = ${GIRARA_INC} ${GTK_INC} $(SQLITE_INC) +LIBS = ${GIRARA_LIB} ${GTK_LIB} $(SQLITE_LIB) -lpthread -lm + +# flags +CFLAGS += -std=c99 -pedantic -Wall -Wno-format-zero-length -Wextra $(INCS) + +# debug DFLAGS = -g -# linker flags -LDFLAGS ?= +# ld +LDFLAGS += -rdynamic # compiler CC ?= gcc @@ -30,5 +39,5 @@ CC ?= gcc # strip SFLAGS ?= -s -# set to something != 0 to enable verbose build output +# set to something != 0 if you want verbose build output VERBOSE ?= 0 diff --git a/database-sqlite.c b/database-sqlite.c new file mode 100644 index 0000000..0c71893 --- /dev/null +++ b/database-sqlite.c @@ -0,0 +1,176 @@ +/* See LICENSE file for license and copyright information */ + +#include +#include +#include + +#include "database.h" + +struct zathura_database_s +{ + sqlite3* session; +}; + +zathura_database_t* +zathura_db_init(const char* path) +{ + zathura_database_t* db = g_malloc0(sizeof(zathura_database_t)); + + /* create bookmarks database */ + static const char SQL_BOOKMARK_INIT[] = + "CREATE TABLE IF NOT EXISTS bookmarks (" + "file TEXT PRIMARY KEY," + "id TEXT," + "page INTEGER," + "UNIQUE(file, id));"; + + static const char SQL_FILEINFO_INIT[] = + "CREATE TABLE IF NOT EXISTS fileinfo (" + "file TEXT PRIMARY KEY," + "page INTEGER," + "offset INTEGER," + "scale INTEGER);"; + + if (sqlite3_open(path, &(db->session)) != SQLITE_OK) { + girara_error("Could not open database: %s\n", path); + zathura_db_free(db); + return NULL; + } + + if (sqlite3_exec(db->session, SQL_BOOKMARK_INIT, NULL, 0, NULL) != SQLITE_OK) { + girara_error("Failed to initialize database: %s\n", path); + zathura_db_free(db); + return NULL; + } + + if (sqlite3_exec(db->session, SQL_FILEINFO_INIT, NULL, 0, NULL) != SQLITE_OK) { + girara_error("Failed to initialize database: %s\n", path); + zathura_db_free(db); + return NULL; + } + + return db; +} + +void +zathura_db_free(zathura_database_t* db) +{ + if (db == NULL) { + return; + } + + sqlite3_close(db->session); + g_free(db); +} + +static sqlite3_stmt* +prepare_statement(sqlite3* session, const char* statement) +{ + if (session == NULL || statement == NULL) { + return NULL; + } + + const char* pz_tail = NULL; + sqlite3_stmt* pp_stmt = NULL; + + if (sqlite3_prepare(session, statement, -1, &pp_stmt, &pz_tail) != SQLITE_OK) { + girara_error("Failed to prepare query: %s", statement); + sqlite3_finalize(pp_stmt); + return NULL; + } else if (pz_tail && *pz_tail != '\0') { + girara_error("Unused portion of statement: %s", pz_tail); + sqlite3_finalize(pp_stmt); + return NULL; + } + + return pp_stmt; +} + +bool +zathura_db_add_bookmark(zathura_database_t* db, const char* file, zathura_bookmark_t* bookmark) +{ + g_return_val_if_fail(db && file && bookmark, false); + + static const char SQL_BOOKMARK_ADD[] = + "REPLACE INTO bookmarks (file, id, page) VALUES (?, ?, ?);"; + + sqlite3_stmt* stmt = prepare_statement(db->session, SQL_BOOKMARK_ADD); + if (stmt == NULL) { + return false; + } + + if (sqlite3_bind_text(stmt, 1, file, -1, NULL) != SQLITE_OK || + sqlite3_bind_text(stmt, 2, bookmark->id, -1, NULL) != SQLITE_OK || + sqlite3_bind_int(stmt, 3, bookmark->page) != SQLITE_OK) { + sqlite3_finalize(stmt); + girara_error("Failed to bind arguments."); + return false; + } + + int res = sqlite3_step(stmt); + sqlite3_finalize(stmt); + return res == SQLITE_OK; +} + +bool +zathura_db_remove_bookmark(zathura_database_t* db, const char* file, const char* id) +{ + g_return_val_if_fail(db && file && id, false); + + static const char SQL_BOOKMARK_ADD[] = + "DELETE FROM bookmarks WHERE file = ? && id = ?;"; + + sqlite3_stmt* stmt = prepare_statement(db->session, SQL_BOOKMARK_ADD); + if (stmt == NULL) { + return false; + } + + if (sqlite3_bind_text(stmt, 1, file, -1, NULL) != SQLITE_OK || + sqlite3_bind_text(stmt, 2, id, -1, NULL) != SQLITE_OK) { + sqlite3_finalize(stmt); + girara_error("Failed to bind arguments."); + return false; + } + + int res = sqlite3_step(stmt); + sqlite3_finalize(stmt); + return res == SQLITE_OK; +} + +girara_list_t* +zathura_db_load_bookmarks(zathura_database_t* db, const char* file) +{ + g_return_val_if_fail(db && file, NULL); + + static const char SQL_BOOKMARK_SELECT[] = + "SELECT id, page FROM bookmarks WHERE file = ?;"; + + sqlite3_stmt* stmt = prepare_statement(db->session, SQL_BOOKMARK_SELECT); + if (stmt == NULL) { + return NULL; + } + + if (sqlite3_bind_text(stmt, 1, file, -1, NULL) != SQLITE_OK) { + sqlite3_finalize(stmt); + girara_error("Failed to bind arguments."); + return NULL; + } + + girara_list_t* result = girara_list_new(); + if (result == NULL) { + sqlite3_finalize(stmt); + return NULL; + } + girara_list_set_free_function(result, (girara_free_function_t) zathura_bookmark_free); + + while (sqlite3_step(stmt) == SQLITE_ROW) { + zathura_bookmark_t* bookmark = g_malloc0(sizeof(zathura_bookmark_t)); + bookmark->id = g_strdup((const char*) sqlite3_column_text(stmt, 0)); + bookmark->page = sqlite3_column_int(stmt, 1); + + girara_list_append(result, bookmark); + } + sqlite3_finalize(stmt); + return result; +} + diff --git a/database.h b/database.h new file mode 100644 index 0000000..ae954f3 --- /dev/null +++ b/database.h @@ -0,0 +1,51 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef DATABASE_H +#define DATABASE_H + +#include +#include + +#include "zathura.h" +#include "bookmarks.h" + +/** + * Initialize database system. + * @param path Path to the database file. + * @return A valid zathura_database_t instance or NULL on failure + */ +zathura_database_t* zathura_db_init(const char* path); + +/** + * Free database instance. + * @param The database instance to free. + */ +void zathura_db_free(zathura_database_t* db); + +/** + * Add or update bookmark in the database. + * @param db The database instance + * @param file The file to which the bookmark belongs to. + * @param bookmark The bookmark instance. + * @return true on success, false otherwise + */ +bool zathura_db_add_bookmark(zathura_database_t* db, const char* file, zathura_bookmark_t* bookmark); + +/** + * Add or update bookmark in the database. + * @param db The database instance + * @param file The file to which the bookmark belongs to. + * @param bookmark The bookmark instance. + * @return true on success, false otherwise + */ +bool zathura_db_remove_bookmark(zathura_database_t* db, const char* file, const char* id); + +/** + * Loads all bookmarks from the database belonging to a specific file. + * @param db The database instance. + * @param file The file for which the bookmarks should be loaded. + * @return List of zathura_bookmark_t* or NULL on failure. + */ +girara_list_t* zathura_db_load_bookmarks(zathura_database_t* db, const char* file); + +#endif // DATABASE_H diff --git a/document.c b/document.c new file mode 100644 index 0000000..326e006 --- /dev/null +++ b/document.c @@ -0,0 +1,571 @@ +/* See LICENSE file for license and copyright information */ + +#define _BSD_SOURCE +#define _XOPEN_SOURCE 700 +// TODO: Implement realpath + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "document.h" +#include "utils.h" +#include "zathura.h" +#include "render.h" + +#define LENGTH(x) (sizeof(x)/sizeof((x)[0])) + +void +zathura_document_plugins_load(zathura_t* zathura) +{ + girara_list_iterator_t* iter = girara_list_iterator(zathura->plugins.path); + if (iter == NULL) { + return; + } + + do + { + char* plugindir = girara_list_iterator_data(iter); + + /* read all files in the plugin directory */ + DIR* dir = opendir(plugindir); + if (dir == NULL) { + girara_error("could not open plugin directory: %s", plugindir); + continue; + } + + int fddir = dirfd(dir); + struct dirent* entry; + while ((entry = readdir(dir)) != NULL) { + struct stat statbuf; + if (fstatat(fddir, entry->d_name, &statbuf, 0) != 0) { + girara_error("failed to fstatat %s/%s; errno is %d.", plugindir, entry->d_name, errno); + continue; + } + + /* check if entry is a file */ + if (S_ISREG(statbuf.st_mode) == 0) { + girara_info("%s/%s is not a regular file. Skipping.", plugindir, entry->d_name); + continue; + } + + void* handle = NULL; + zathura_document_plugin_t* plugin = NULL; + char* path = NULL; + + /* get full path */ + path = g_build_filename(plugindir, entry->d_name, NULL); + if (path == NULL) { + g_error("failed to allocate memory!"); + break; + } + + /* load plugin */ + handle = dlopen(path, RTLD_NOW); + if (handle == NULL) { + girara_error("could not load plugin %s (%s)", path, dlerror()); + g_free(path); + continue; + } + + /* resolve symbol */ + zathura_plugin_register_service_t register_plugin; + *(void**)(®ister_plugin) = dlsym(handle, PLUGIN_REGISTER_FUNCTION); + + if (register_plugin == NULL) { + girara_error("could not find '%s' function in plugin %s", PLUGIN_REGISTER_FUNCTION, path); + g_free(path); + dlclose(handle); + continue; + } + + plugin = malloc(sizeof(zathura_document_plugin_t)); + + if (plugin == NULL) { + g_error("failed to allocate memory!"); + break; + } + + plugin->file_extension = NULL; + plugin->open_function = NULL; + + register_plugin(plugin); + + bool r = zathura_document_plugin_register(zathura, plugin, handle); + + if (r == false) { + girara_error("could not register plugin %s", path); + free(plugin); + dlclose(handle); + } + else { + girara_info("successfully loaded plugin %s", path); + } + + g_free(path); + } + + if (closedir(dir) == -1) { + girara_error("could not close plugin directory %s", plugindir); + } + } while (girara_list_iterator_next(iter)); + girara_list_iterator_free(iter); +} + +void +zathura_document_plugins_free(zathura_t* zathura) +{ + if (zathura == NULL) { + return; + } + + girara_list_iterator_t* iter = girara_list_iterator(zathura->plugins.plugins); + if (iter == NULL) { + return; + } + + do { + zathura_document_plugin_t* plugin = (zathura_document_plugin_t*) girara_list_iterator_data(iter); + free(plugin); + } while (girara_list_iterator_next(iter)); + girara_list_iterator_free(iter); +} + +bool +zathura_document_plugin_register(zathura_t* zathura, zathura_document_plugin_t* new_plugin, void* handle) +{ + if( (new_plugin == NULL) || (new_plugin->file_extension == NULL) || (new_plugin->open_function == NULL) + || (handle == NULL) ) { + girara_error("plugin: could not register\n"); + return false; + } + + /* search existing plugins */ + girara_list_iterator_t* iter = girara_list_iterator(zathura->plugins.plugins); + if (iter) { + do { + zathura_document_plugin_t* plugin = (zathura_document_plugin_t*) girara_list_iterator_data(iter); + if (strcmp(plugin->file_extension, new_plugin->file_extension) == 0) { + girara_error("plugin: already registered for filetype %s\n", plugin->file_extension); + girara_list_iterator_free(iter); + return false; + } + } while (girara_list_iterator_next(iter)); + girara_list_iterator_free(iter); + } + + girara_list_append(zathura->plugins.plugins, new_plugin); + return true; +} + +zathura_document_t* +zathura_document_open(zathura_t* zathura, const char* path, const char* password) +{ + if (path == NULL) { + goto error_out; + } + + if (file_exists(path) == false) { + girara_error("File does not exist"); + goto error_out; + } + + const char* file_extension = file_get_extension(path); + if (file_extension == NULL) { + girara_error("Could not determine file type"); + goto error_out; + } + + /* determine real path */ + size_t path_max; +#ifdef PATH_MAX + path_max = PATH_MAX; +#else + path_max = pathconf(path,_PC_PATH_MAX); + if (path_max <= 0) + path_max = 4096; +#endif + + char* real_path = NULL; + zathura_document_t* document = NULL; + + real_path = malloc(sizeof(char) * path_max); + if (real_path == NULL) { + goto error_out; + } + + if (realpath(path, real_path) == NULL) { + goto error_free; + } + + document = malloc(sizeof(zathura_document_t)); + if (document == NULL) { + goto error_free; + } + + document->file_path = real_path; + document->password = password; + document->current_page_number = 0; + document->number_of_pages = 0; + document->scale = 1.0; + document->rotate = 0; + document->data = NULL; + document->pages = NULL; + document->zathura = zathura; + + document->functions.document_free = NULL; + document->functions.document_index_generate = NULL; + document->functions.document_save_as = NULL; + document->functions.document_attachments_get = NULL; + document->functions.page_get = NULL; + document->functions.page_free = NULL; + document->functions.page_search_text = NULL; + document->functions.page_links_get = NULL; + document->functions.page_form_fields_get = NULL; + document->functions.page_render = NULL; + + girara_list_iterator_t* iter = girara_list_iterator(zathura->plugins.plugins); + if (iter == NULL) { + goto error_free; + } + + do { + zathura_document_plugin_t* plugin = (zathura_document_plugin_t*) girara_list_iterator_data(iter); + if (strcmp(file_extension, plugin->file_extension) == 0) { + girara_list_iterator_free(iter); + if (plugin->open_function != NULL) { + if (plugin->open_function(document) == true) { + /* update statusbar */ + girara_statusbar_item_set_text(zathura->ui.session, zathura->ui.statusbar.file, real_path); + + /* read all pages */ + document->pages = calloc(document->number_of_pages, sizeof(zathura_page_t*)); + if (document->pages == NULL) { + goto error_free; + } + + for (unsigned int page_id = 0; page_id < document->number_of_pages; page_id++) { + zathura_page_t* page = zathura_page_get(document, page_id); + if (page == NULL) { + goto error_free; + } + + document->pages[page_id] = page; + } + + return document; + } else { + girara_error("could not open file\n"); + goto error_free; + } + } + } + } while (girara_list_iterator_next(iter)); + girara_list_iterator_free(iter); + + girara_error("unknown file type\n"); + +error_free: + + free(real_path); + + if (document != NULL && document->pages != NULL) { + for (unsigned int page_id = 0; page_id < document->number_of_pages; page_id++) { + zathura_page_free(document->pages[page_id]); + } + + free(document->pages); + } + + free(document); + +error_out: + + return NULL; +} + +bool +zathura_document_free(zathura_document_t* document) +{ + if (document == NULL) { + return false; + } + + /* free pages */ + for (unsigned int page_id = 0; page_id < document->number_of_pages; page_id++) { + zathura_page_free(document->pages[page_id]); + document->pages[page_id] = NULL; + } + + free(document->pages); + + /* free document */ + if (document->functions.document_free == NULL) { + girara_error("%s not implemented", __FUNCTION__); + + if (document->file_path != NULL) { + free(document->file_path); + } + + free(document); + return true; + } + + bool r = document->functions.document_free(document); + + if (document->file_path != NULL) { + free(document->file_path); + } + + free(document); + + return r; +} + +bool +zathura_document_save_as(zathura_document_t* document, const char* path) +{ + if (document == NULL || path == NULL) { + return false; + } + + if (document->functions.document_save_as == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return false; + } + + return document->functions.document_save_as(document, path); +} + +girara_tree_node_t* +zathura_document_index_generate(zathura_document_t* document) +{ + if (document == NULL) { + return NULL; + } + + if (document->functions.document_index_generate == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return NULL; + } + + return document->functions.document_index_generate(document); +} + +zathura_list_t* +zathura_document_attachments_get(zathura_document_t* document) +{ + if (document == NULL) { + return NULL; + } + + if (document->functions.document_attachments_get == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return NULL; + } + + return document->functions.document_attachments_get(document); +} + +bool +zathura_document_attachments_free(zathura_list_t* UNUSED(list)) +{ + return false; +} + +zathura_page_t* +zathura_page_get(zathura_document_t* document, unsigned int page_id) +{ + if (document == NULL) { + return NULL; + } + + if (document->functions.page_get == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return NULL; + } + + zathura_page_t* page = document->functions.page_get(document, page_id); + + if (page != NULL) { + page->number = page_id; + page->visible = false; + page->event_box = gtk_event_box_new(); + page->drawing_area = gtk_drawing_area_new(); + page->surface = NULL; + page->document = document; + g_signal_connect(page->drawing_area, "expose-event", G_CALLBACK(page_expose_event), page); + + gtk_widget_set_size_request(page->drawing_area, page->width * document->scale, page->height * document->scale); + gtk_container_add(GTK_CONTAINER(page->event_box), page->drawing_area); + + g_static_mutex_init(&(page->lock)); + } + + return page; +} + +bool +zathura_page_free(zathura_page_t* page) +{ + if (page == NULL || page->document == NULL) { + return false; + } + + if (page->document->functions.page_free == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return false; + } + + g_static_mutex_free(&(page->lock)); + + return page->document->functions.page_free(page); +} + +zathura_list_t* +zathura_page_search_text(zathura_page_t* page, const char* text) +{ + if (page == NULL || page->document == NULL || text == NULL) { + return NULL; + } + + if (page->document->functions.page_search_text == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return NULL; + } + + return page->document->functions.page_search_text(page, text); +} + +zathura_list_t* +zathura_page_links_get(zathura_page_t* page) +{ + if (page == NULL || page->document == NULL) { + return NULL; + } + + if (page->document->functions.page_links_get == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return NULL; + } + + return page->document->functions.page_links_get(page); +} + +bool +zathura_page_links_free(zathura_list_t* UNUSED(list)) +{ + return false; +} + +zathura_list_t* +zathura_page_form_fields_get(zathura_page_t* page) +{ + if (page == NULL || page->document == NULL) { + return NULL; + } + + if (page->document->functions.page_form_fields_get == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return NULL; + } + + return page->document->functions.page_form_fields_get(page); +} + +bool +zathura_page_form_fields_free(zathura_list_t* UNUSED(list)) +{ + return false; +} + +zathura_image_buffer_t* +zathura_page_render(zathura_page_t* page) +{ + if (page == NULL || page->document == NULL) { + return NULL; + } + + if (page->document->functions.page_render == NULL) { + girara_error("%s not implemented", __FUNCTION__); + return NULL; + } + + return page->document->functions.page_render(page); +} + +zathura_index_element_t* +zathura_index_element_new(const char* title) +{ + if (title == NULL) { + return NULL; + } + + zathura_index_element_t* res = g_malloc0(sizeof(zathura_index_element_t)); + + if (res == NULL) { + return NULL; + } + + res->title = g_strdup(title); + + return res; +} + +void +zathura_index_element_free(zathura_index_element_t* index) +{ + if (index == NULL) { + return; + } + + g_free(index->title); + + if (index->type == ZATHURA_LINK_EXTERNAL) { + g_free(index->target.uri); + } + + g_free(index); +} + +zathura_image_buffer_t* +zathura_image_buffer_create(unsigned int width, unsigned int height) +{ + zathura_image_buffer_t* image_buffer = malloc(sizeof(zathura_image_buffer_t)); + + if (image_buffer == NULL) { + return NULL; + } + + image_buffer->data = calloc(width * height * 3, sizeof(unsigned char)); + + if (image_buffer->data == NULL) { + free(image_buffer); + return NULL; + } + + image_buffer->width = width; + image_buffer->height = height; + image_buffer->rowstride = width * 3; + + return image_buffer; +} + +void +zathura_image_buffer_free(zathura_image_buffer_t* image_buffer) +{ + if (image_buffer == NULL) { + return; + } + + free(image_buffer->data); + free(image_buffer); +} diff --git a/document.h b/document.h new file mode 100644 index 0000000..63517c4 --- /dev/null +++ b/document.h @@ -0,0 +1,379 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef DOCUMENT_H +#define DOCUMENT_H + +#include +#include + +#include +#include "zathura.h" + +#define PLUGIN_REGISTER_FUNCTION "plugin_register" + +typedef struct zathura_list_s zathura_list_t; +// typedef struct zathura_document_s zathura_document_t; + +typedef bool (*zathura_document_open_t)(zathura_document_t* document); + +/** + * Document plugin structure + */ +typedef struct zathura_document_plugin_s +{ + char* file_extension; /**> File extension */ + zathura_document_open_t open_function; /**> Document open function */ + void* handle; /**> DLL handle */ +} zathura_document_plugin_t; + +/** + * Function prototype that is called to register a document plugin + * + * @param The document plugin + */ +typedef void (*zathura_plugin_register_service_t)(zathura_document_plugin_t*); + +/** + * Structure for a list + */ +struct zathura_list_s +{ + void* data; /**> Data value */ + struct zathura_list_s* next; /**> Next element in the list */ +}; + +/** + * Image buffer + */ +typedef struct zathura_image_buffer_s +{ + unsigned char* data; /**> Image buffer data */ + unsigned int height; /**> Height of the image */ + unsigned int width; /**> Width of the image */ + unsigned int rowstride; /**> Rowstride of the image */ +} zathura_image_buffer_t; + +/** + * Creates an image buffer + * + * @param width Width of the image stored in the buffer + * @param height Height of the image stored in the buffer + * @return Image buffer or NULL if an error occured + */ +zathura_image_buffer_t* zathura_image_buffer_create(unsigned int width, unsigned int height); + +/** + * Frees the image buffer + * + * @param zathura_image_buffer_t + */ +void zathura_image_buffer_free(zathura_image_buffer_t*); + +/** + * Rectangle structure + */ +typedef struct zathura_rectangle_s +{ + double x1; /**> X coordinate of point 1 */ + double y1; /**> Y coordinate of point 1 */ + double x2; /**> X coordinate of point 2 */ + double y2; /**> Y coordinate of point 2 */ +} zathura_rectangle_t; + +/** + * Possible link types + */ +typedef enum zathura_link_type_e +{ + ZATHURA_LINK_TO_PAGE, /**> Links to a page */ + ZATHURA_LINK_EXTERNAL, /**> Links to an external source */ +} zathura_link_type_t; + +/** + * Link + */ +typedef struct zathura_link_s +{ + zathura_rectangle_t position; /**> Position of the link */ + zathura_link_type_t type; /**> Link type */ + union + { + unsigned int page_number; /**> Page number */ + char* value; /**> Value */ + } target; +} zathura_link_t; + +/** + * Index element + */ +typedef struct zathura_index_element_s +{ + char* title; /**> Title of the element */ + zathura_link_type_t type; /**> Type */ + union + { + unsigned int page_number; /**> Page number */ + char* uri; /**> Uri */ + } target; +} zathura_index_element_t; + +/** + * Form type + */ +typedef enum zathura_form_type_e +{ + ZATHURA_FORM_CHECKBOX, /**> Checkbox */ + ZATHURA_FORM_TEXTFIELD /**> Textfield */ +} zathura_form_type_t; + +/** + * Form element + */ +typedef struct zathura_form_s +{ + zathura_rectangle_t position; /**> Position */ + zathura_form_type_t type; /**> Type */ +} zathura_form_t; + +/** + * Page + */ +struct zathura_page_s +{ + double height; /**> Page height */ + double width; /**> Page width */ + unsigned int number; /**> Page number */ + zathura_document_t* document; /**> Document */ + void* data; /**> Custom data */ + bool visible; /**> Page is visible */ + GtkWidget* event_box; /**> Widget wrapper for mouse events */ + GtkWidget* drawing_area; /**> Drawing area */ + GStaticMutex lock; /**> Lock */ + cairo_surface_t* surface; /** Cairo surface */ +}; + +/** + * Document + */ +struct zathura_document_s +{ + char* file_path; /**> File path of the document */ + const char* password; /**> Password of the document */ + unsigned int current_page_number; /**> Current page number */ + unsigned int number_of_pages; /**> Number of pages */ + double scale; /**> Scale value */ + int rotate; /**> Rotation */ + void* data; /**> Custom data */ + zathura_t* zathura; /** Zathura object */ + + struct + { + /** + * Frees the document + */ + bool (*document_free)(zathura_document_t* document); + + /** + * Generates the document index + */ + girara_tree_node_t* (*document_index_generate)(zathura_document_t* document); + + /** + * Save the document + */ + bool (*document_save_as)(zathura_document_t* document, const char* path); + + /** + * Get list of attachments + */ + zathura_list_t* (*document_attachments_get)(zathura_document_t* document); + + /** + * Gets the page object + */ + zathura_page_t* (*page_get)(zathura_document_t* document, unsigned int page_id); + + /** + * Search text + */ + zathura_list_t* (*page_search_text)(zathura_page_t* page, const char* text); + + /** + * Get links on a page + */ + zathura_list_t* (*page_links_get)(zathura_page_t* page); + + /** + * Get form fields + */ + zathura_list_t* (*page_form_fields_get)(zathura_page_t* page); + + /** + * Renders the page + */ + zathura_image_buffer_t* (*page_render)(zathura_page_t* page); + + /** + * Free page + */ + bool (*page_free)(zathura_page_t* page); + } functions; + + /** + * Document pages + */ + zathura_page_t** pages; +}; + +/** + * Load all document plugins + * + * @param zathura the zathura session + */ +void zathura_document_plugins_load(zathura_t* zathura); + +/** + * Free all document plugins + * + * @param zathura the zathura session + */ +void zathura_document_plugins_free(zathura_t* zathura); + +/** + * Register document plugin + */ +bool zathura_document_plugin_register(zathura_t* zathura, zathura_document_plugin_t* new_plugin, void* handle); + +/** + * Open the document + * + * @param path Path to the document + * @param password Password of the document or NULL + * @return The document object + */ +zathura_document_t* zathura_document_open(zathura_t* zathura, const char* path, const char* password); + +/** + * Free the document + * + * @param document + * @return true if no error occured, otherwise false + */ +bool zathura_document_free(zathura_document_t* document); + +/** + * Save the document + * + * @param document The document object + * @param path Path for the saved file + * @return true if no error occured, otherwise false + */ +bool zathura_document_save_as(zathura_document_t* document, const char* path); + +/** + * Generate the document index + * + * @param document The document object + * @return Generated index + */ + +girara_tree_node_t* zathura_document_index_generate(zathura_document_t* document); + +/** + * Get list of attachments + * + * @param document The document object + * @return List of attachments + */ +zathura_list_t* zathura_document_attachments_get(zathura_document_t* document); + +/** + * Free document attachments + * + * @param list list of document attachments + * @return + */ +bool zathura_document_attachments_free(zathura_list_t* list); + +/** + * Get the page object + * + * @param document The document + * @param page_id Page number + * @return Page object or NULL if an error occured + */ +zathura_page_t* zathura_page_get(zathura_document_t* document, unsigned int page_id); + +/** + * Frees the page object + * + * @param page The page object + * @return true if no error occured, otherwise false + */ +bool zathura_page_free(zathura_page_t* page); + +/** + * Search page + * + * @param page The page object + * @param text Search item + * @return List of results + */ +zathura_list_t* zathura_page_search_text(zathura_page_t* page, const char* text); + +/** + * Get page links + * + * @param page The page object + * @return List of links + */ +zathura_list_t* zathura_page_links_get(zathura_page_t* page); + +/** + * Free page links + * + * @param list List of links + * @return true if no error occured, otherwise false + */ +bool zathura_page_links_free(zathura_list_t* list); + +/** + * Get list of form fields + * + * @param page The page object + * @return List of form fields + */ +zathura_list_t* zathura_page_form_fields_get(zathura_page_t* page); + +/** + * Free list of form fields + * + * @param list List of form fields + * @return true if no error occured, otherwise false + */ +bool zathura_page_form_fields_free(zathura_list_t* list); + +/** + * Render page + * + * @param page The page object + * @return Image buffer or NULL if an error occured + */ +zathura_image_buffer_t* zathura_page_render(zathura_page_t* page); + +/** + * Create new index element + * + * @param title Title of the index element + * @return Index element + */ +zathura_index_element_t* zathura_index_element_new(const char* title); + +/** + * Free index element + * + * @param index The index element + */ +void zathura_index_element_free(zathura_index_element_t* index); + +#endif // DOCUMENT_H diff --git a/print.c b/print.c new file mode 100644 index 0000000..a9dd536 --- /dev/null +++ b/print.c @@ -0,0 +1,82 @@ +#include "print.h" +#include "document.h" + +void +print(zathura_t* zathura) +{ + g_return_if_fail(zathura != NULL); + g_return_if_fail(zathura->document != NULL); + + GtkPrintOperation* print_operation = gtk_print_operation_new(); + + /* print operation settings */ + if (zathura->print.settings != NULL) { + gtk_print_operation_set_print_settings(print_operation, zathura->print.settings); + } + + if (zathura->print.page_setup != NULL) { + gtk_print_operation_set_default_page_setup(print_operation, zathura->print.page_setup); + } + + gtk_print_operation_set_allow_async(print_operation, TRUE); + gtk_print_operation_set_n_pages(print_operation, zathura->document->number_of_pages); + gtk_print_operation_set_current_page(print_operation, zathura->document->current_page_number); + + /* print operation signals */ + g_signal_connect(print_operation, "draw-page", G_CALLBACK(cb_print_draw_page), zathura); + + /* print */ + GtkPrintOperationResult result = gtk_print_operation_run(print_operation, + GTK_PRINT_OPERATION_ACTION_PRINT_DIALOG, NULL, NULL); + + if (result == GTK_PRINT_OPERATION_RESULT_APPLY) { + /* save previous settings */ + zathura->print.settings = gtk_print_operation_get_print_settings(print_operation); + zathura->print.page_setup = gtk_print_operation_get_default_page_setup(print_operation); + } else if (result == GTK_PRINT_OPERATION_RESULT_ERROR) { + girara_error("Error occured while printing progress"); + } + + g_object_unref(print_operation); +} + +void +cb_print_begin(GtkPrintOperation* UNUSED(print_operation), GtkPrintContext* + UNUSED(context), zathura_t* UNUSED(zathura)) +{ + +} + +void +cb_print_draw_page(GtkPrintOperation* UNUSED(print_operation), GtkPrintContext* + context, gint page_number, zathura_t* zathura) +{ + cairo_t* cairo = gtk_print_context_get_cairo_context(context); + + girara_info("Printing page %d", page_number); + + zathura_page_t* page = zathura->document->pages[page_number]; + + double requested_with = gtk_print_context_get_width(context); + double tmp_scale = zathura->document->scale; + zathura->document->scale = requested_with / page->width; + + g_static_mutex_lock(&(page->lock)); + zathura_image_buffer_t* image_buffer = zathura_page_render(page); + g_static_mutex_unlock(&(page->lock)); + + for (unsigned int y = 0; y < image_buffer->height; y++) { + unsigned char* src = image_buffer->data + y * image_buffer->rowstride; + for (unsigned int x = 0; x < image_buffer->width; x++) { + if (src[0] != 255 && src[1] != 255 && src[2] != 255) { + cairo_set_source_rgb(cairo, src[0], src[1], src[2]); + cairo_rectangle(cairo, x, y, 1, 1); + cairo_fill(cairo); + } + + src += 3; + } + } + + zathura->document->scale = tmp_scale; +} diff --git a/print.h b/print.h new file mode 100644 index 0000000..a8fcb5a --- /dev/null +++ b/print.h @@ -0,0 +1,26 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef PRINT_H +#define PRINT_H + +#include "zathura.h" + +/** + * Opens a print dialog to print the current file + * + * @param zathura + */ +void print(zathura_t* zathura); + +/** + * Callback that is executed for every page that should be printed + * + * @param print_operation Print operation object + * @param context Print context + * @param page_number Current page number + * @param zathura Zathura object + */ +void cb_print_draw_page(GtkPrintOperation* print_operation, GtkPrintContext* + context, gint page_number, zathura_t* zathura); + +#endif // PRINT_H diff --git a/render.c b/render.c new file mode 100644 index 0000000..e6831e9 --- /dev/null +++ b/render.c @@ -0,0 +1,334 @@ +#include "render.h" +#include "zathura.h" +#include "document.h" + +void* render_job(void* data); +bool render(zathura_t* zathura, zathura_page_t* page); + +void* +render_job(void* data) +{ + render_thread_t* render_thread = (render_thread_t*) data; + + while (true) { + g_mutex_lock(render_thread->lock); + + if (girara_list_size(render_thread->list) <= 0) { + g_cond_wait(render_thread->cond, render_thread->lock); + } + + if (girara_list_size(render_thread->list) <= 0) { + /* + * We've been signaled with g_cond_signal(), but the list + * is still empty. This means that the signal came from + * render_free() and current document is being closed. + * We should unlock the mutex and kill the thread. + */ + g_mutex_unlock(render_thread->lock); + g_thread_exit(0); + } + + zathura_page_t* page = (zathura_page_t*) girara_list_nth(render_thread->list, 0); + girara_list_remove(render_thread->list, page); + g_mutex_unlock(render_thread->lock); + + if (render(render_thread->zathura, page) != true) { + girara_error("Rendering failed\n"); + } + } + + return NULL; +} + +render_thread_t* +render_init(zathura_t* zathura) +{ + render_thread_t* render_thread = malloc(sizeof(render_thread_t)); + + if (!render_thread) { + goto error_ret; + } + + /* init */ + render_thread->list = NULL; + render_thread->thread = NULL; + render_thread->cond = NULL; + render_thread->zathura = zathura; + + /* setup */ + render_thread->list = girara_list_new(); + + if (!render_thread->list) { + goto error_free; + } + + render_thread->cond = g_cond_new(); + + if (!render_thread->cond) { + goto error_free; + } + + render_thread->lock = g_mutex_new(); + + if (!render_thread->lock) { + goto error_free; + } + + render_thread->thread = g_thread_create(render_job, render_thread, TRUE, NULL); + + if (!render_thread->thread) { + goto error_free; + } + + return render_thread; + +error_free: + + if (render_thread->list) { + girara_list_free(render_thread->list); + } + + if (render_thread->cond) { + g_cond_free(render_thread->cond); + } + + if (render_thread->lock) { + g_mutex_free(render_thread->lock); + } + + free(render_thread); + +error_ret: + + return NULL; +} + +void +render_free(render_thread_t* render_thread) +{ + if (!render_thread) { + return; + } + + if (render_thread->list) { + girara_list_free(render_thread->list); + } + + if (render_thread->cond) { + g_cond_signal(render_thread->cond); + g_thread_join(render_thread->thread); + g_cond_free(render_thread->cond); + } + + if (render_thread->lock) { + g_mutex_free(render_thread->lock); + } +} + +bool +render_page(render_thread_t* render_thread, zathura_page_t* page) +{ + if (!render_thread || !page || !render_thread->list || page->surface) { + return false; + } + + g_mutex_lock(render_thread->lock); + if (!girara_list_contains(render_thread->list, page)) { + girara_list_append(render_thread->list, page); + } + g_cond_signal(render_thread->cond); + g_mutex_unlock(render_thread->lock); + + return true; +} + +bool +render(zathura_t* zathura, zathura_page_t* page) +{ + if (zathura == NULL || page == NULL) { + return false; + } + + gdk_threads_enter(); + g_static_mutex_lock(&(page->lock)); + zathura_image_buffer_t* image_buffer = zathura_page_render(page); + + if (image_buffer == NULL) { + g_static_mutex_unlock(&(page->lock)); + gdk_threads_leave(); + return false; + } + + /* create cairo surface */ + unsigned int page_width = page->width * zathura->document->scale; + unsigned int page_height = page->height * zathura->document->scale; + + cairo_surface_t* surface = cairo_image_surface_create(CAIRO_FORMAT_RGB24, page_width, page_height); + + int rowstride = cairo_image_surface_get_stride(surface); + unsigned char* image = cairo_image_surface_get_data(surface); + + for (unsigned int y = 0; y < page_height; y++) { + unsigned char* dst = image + y * rowstride; + unsigned char* src = image_buffer->data + y * image_buffer->rowstride; + + for (unsigned int x = 0; x < page_width; x++) { + dst[0] = src[2]; + dst[1] = src[1]; + dst[2] = src[0]; + src += 3; + dst += 4; + } + } + + /* recolor */ + if (zathura->global.recolor) { + /* recolor code based on qimageblitz library flatten() function + (http://sourceforge.net/projects/qimageblitz/) */ + + int r1 = zathura->ui.colors.recolor_dark_color.red / 257; + int g1 = zathura->ui.colors.recolor_dark_color.green / 257; + int b1 = zathura->ui.colors.recolor_dark_color.blue / 257; + int r2 = zathura->ui.colors.recolor_light_color.red / 257; + int g2 = zathura->ui.colors.recolor_light_color.green / 257; + int b2 = zathura->ui.colors.recolor_light_color.blue / 257; + + int min = 0x00; + int max = 0xFF; + int mean = 0x00; + + float sr = ((float) r2 - r1) / (max - min); + float sg = ((float) g2 - g1) / (max - min); + float sb = ((float) b2 - b1) / (max - min); + + for (unsigned int y = 0; y < page_height; y++) { + unsigned char* data = image + y * rowstride; + + for (unsigned int x = 0; x < page_width; x++) { + mean = (data[0] + data[1] + data[2]) / 3; + data[2] = sr * (mean - min) + r1 + 0.5; + data[1] = sg * (mean - min) + g1 + 0.5; + data[0] = sb * (mean - min) + b1 + 0.5; + data += 4; + } + } + } + + /* rotate */ + unsigned int width = page_width; + unsigned int height = page_height; + if (page->document->rotate == 90 || page->document->rotate == 270) { + width = page_height; + height = page_width; + } + + cairo_surface_t* final_surface = cairo_image_surface_create(CAIRO_FORMAT_RGB24, width, height); + cairo_t* cairo = cairo_create(final_surface); + + switch(page->document->rotate) + { + case 90: + cairo_translate(cairo, width, 0); + break; + case 180: + cairo_translate(cairo, width, height); + break; + case 270: + cairo_translate(cairo, 0, height); + break; + } + + if (page->document->rotate != 0) { + cairo_rotate(cairo, page->document->rotate * G_PI / 180.0); + } + + cairo_set_source_surface(cairo, surface, 0, 0); + cairo_set_operator(cairo, CAIRO_OPERATOR_SOURCE); + cairo_paint(cairo); + cairo_destroy(cairo); + cairo_surface_destroy(surface); + + /* draw to gtk widget */ + page->surface = final_surface; + gtk_widget_set_size_request(page->drawing_area, width, height); + gtk_widget_queue_draw(page->drawing_area); + + zathura_image_buffer_free(image_buffer); + g_static_mutex_unlock(&(page->lock)); + + gdk_threads_leave(); + return true; +} + +void +render_all(zathura_t* zathura) +{ + if (zathura->document == NULL) { + return; + } + + /* unmark all pages */ + for (unsigned int page_id = 0; page_id < zathura->document->number_of_pages; page_id++) { + cairo_surface_destroy(zathura->document->pages[page_id]->surface); + zathura->document->pages[page_id]->surface = NULL; + } + + /* redraw current page */ + GtkAdjustment* view_vadjustment = gtk_scrolled_window_get_vadjustment(GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view)); + cb_view_vadjustment_value_changed(view_vadjustment, zathura); +} + +gboolean +page_expose_event(GtkWidget* UNUSED(widget), GdkEventExpose* UNUSED(event), + gpointer data) +{ + zathura_page_t* page = data; + if (page == NULL) { + return FALSE; + } + + g_static_mutex_lock(&(page->lock)); + + cairo_t* cairo = gdk_cairo_create(page->drawing_area->window); + + if (cairo == NULL) { + girara_error("Could not retreive cairo object"); + g_static_mutex_unlock(&(page->lock)); + return FALSE; + } + + if (page->surface != NULL) { + cairo_set_source_surface(cairo, page->surface, 0, 0); + cairo_paint(cairo); + } else { + /* set background color */ + cairo_set_source_rgb(cairo, 255, 255, 255); + cairo_rectangle(cairo, 0, 0, page->width * page->document->scale, page->height * page->document->scale); + cairo_fill(cairo); + + /* write text */ + cairo_set_source_rgb(cairo, 0, 0, 0); + const char* text = "Loading..."; + cairo_select_font_face(cairo, "Sans", CAIRO_FONT_SLANT_NORMAL, CAIRO_FONT_WEIGHT_NORMAL); + cairo_set_font_size(cairo, 16.0); + cairo_text_extents_t extents; + cairo_text_extents(cairo, text, &extents); + double x = (page->width * page->document->scale) / 2 - (extents.width / 2 + extents.x_bearing); + double y = (page->height * page->document->scale) / 2 - (extents.height / 2 + extents.y_bearing); + cairo_move_to(cairo, x, y); + cairo_show_text(cairo, text); + + /* render real page */ + render_page(page->document->zathura->sync.render_thread, page); + + /* update statusbar */ + } + cairo_destroy(cairo); + + page->document->current_page_number = page->number; + statusbar_page_number_update(page->document->zathura); + + g_static_mutex_unlock(&(page->lock)); + + return TRUE; +} diff --git a/render.h b/render.h new file mode 100644 index 0000000..73f2a27 --- /dev/null +++ b/render.h @@ -0,0 +1,66 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef RENDER_H +#define RENDER_H + +#include +#include +#include + +#include "zathura.h" +#include "callbacks.h" + +struct render_thread_s +{ + girara_list_t* list; /**> The list of pages */ + GThread* thread; /**> The thread object */ + GMutex* lock; /**> Lock */ + GCond* cond; /**> Condition */ + zathura_t* zathura; /**> Zathura object */ +}; + +/** + * This function initializes a render thread + * + * @param Zathura object + * @return The render thread object or NULL if an error occured + */ +render_thread_t* render_init(zathura_t* zathura); + +/** + * This function destroys the render thread object + * + * @param render_thread The render thread object + */ +void render_free(render_thread_t* render_thread); + +/** + * This function is used to add a page to the render thread list + * that should be rendered. + * + * @param render_thread The render thread object + * @param page The page that should be rendered + * @return true if no error occured + */ +bool render_page(render_thread_t* render_thread, zathura_page_t* page); + +/** + * This function is used to unmark all pages as not rendered. This should + * be used if all pages should be rendered again (e.g.: the zoom level or the + * colors have changed) + * + * @param zathura Zathura object + */ +void render_all(zathura_t* zathura); + +/** + * Renders page + * + * @param widget Drawing area + * @param event Event + * @param data Optional data + * @return true if no error occured + */ +gboolean page_expose_event(GtkWidget* widget, GdkEventExpose* event, gpointer data); + +#endif // RENDER_H diff --git a/shortcuts.c b/shortcuts.c new file mode 100644 index 0000000..9aed074 --- /dev/null +++ b/shortcuts.c @@ -0,0 +1,388 @@ +/* See LICENSE file for license and copyright information */ + +#include +#include + +#include "callbacks.h" +#include "shortcuts.h" +#include "document.h" +#include "zathura.h" +#include "render.h" +#include "utils.h" + +bool +sc_abort(girara_session_t* session, girara_argument_t* UNUSED(argument), + unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + + girara_mode_set(session, session->modes.normal); + + return false; +} + +bool +sc_adjust_window(girara_session_t* session, girara_argument_t* UNUSED(argument), + unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + + return false; +} + +bool +sc_change_mode(girara_session_t* session, girara_argument_t* argument, unsigned + int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + + girara_mode_set(session, argument->n); + + return false; +} + +bool +sc_follow(girara_session_t* session, girara_argument_t* UNUSED(argument), + unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + + return false; +} + +bool +sc_goto(girara_session_t* session, girara_argument_t* argument, unsigned int t) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(argument != NULL, false); + g_return_val_if_fail(zathura->document != NULL, false); + + if (argument->n == TOP) { + girara_argument_t arg = { TOP, NULL }; + sc_scroll(session, &arg, 0); + + return false; + } else { + if (t == 0) { + girara_argument_t arg = { BOTTOM, NULL }; + sc_scroll(session, &arg, 0); + + return true; + } + + page_set(zathura, t - 1); + } + + return false; +} + +bool +sc_navigate(girara_session_t* session, girara_argument_t* argument, unsigned int + UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(argument != NULL, false); + g_return_val_if_fail(zathura->document != NULL, false); + + unsigned int number_of_pages = zathura->document->number_of_pages; + unsigned int new_page = zathura->document->current_page_number; + + if (argument->n == NEXT) { + new_page = (new_page + 1) % number_of_pages; + } else if (argument->n == PREVIOUS) { + new_page = (new_page + number_of_pages - 1) % number_of_pages; + } + + page_set(zathura, new_page); + + return false; +} + +bool +sc_recolor(girara_session_t* session, girara_argument_t* UNUSED(argument), + unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + + zathura->global.recolor = !zathura->global.recolor; + render_all(zathura); + + return false; +} + +bool +sc_reload(girara_session_t* session, girara_argument_t* UNUSED(argument), + unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + + return false; +} + +bool +sc_rotate(girara_session_t* session, girara_argument_t* UNUSED(argument), + unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(zathura->document != NULL, false); + + /* update rotate value */ + zathura->document->rotate = (zathura->document->rotate + 90) % 360; + + /* render all pages again */ + render_all(zathura); + + return false; +} + +bool +sc_scroll(girara_session_t* session, girara_argument_t* argument, unsigned int + UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(argument != NULL, false); + if (zathura->document == NULL) { + return false; + } + + GtkAdjustment* adjustment = NULL; + if ( (argument->n == LEFT) || (argument->n == RIGHT) ) + adjustment = gtk_scrolled_window_get_hadjustment(GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view)); + else + adjustment = gtk_scrolled_window_get_vadjustment(GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view)); + + gdouble view_size = gtk_adjustment_get_page_size(adjustment); + gdouble value = gtk_adjustment_get_value(adjustment); + gdouble max = gtk_adjustment_get_upper(adjustment) - view_size; + gdouble scroll_step = 40; + + gdouble new_value; + switch(argument->n) + { + case FULL_UP: + new_value = (value - view_size) < 0 ? 0 : (value - view_size); + break; + case FULL_DOWN: + new_value = (value + view_size) > max ? max : (value + view_size); + break; + case HALF_UP: + new_value = (value - (view_size / 2)) < 0 ? 0 : (value - (view_size / 2)); + break; + case HALF_DOWN: + new_value = (value + (view_size / 2)) > max ? max : (value + (view_size / 2)); + break; + case LEFT: + case UP: + new_value = (value - scroll_step) < 0 ? 0 : (value - scroll_step); + break; + case RIGHT: + case DOWN: + new_value = (value + scroll_step) > max ? max : (value + scroll_step); + break; + case TOP: + new_value = 0; + break; + case BOTTOM: + new_value = max; + break; + default: + new_value = 0; + } + + gtk_adjustment_set_value(adjustment, new_value); + + return false; +} + +bool +sc_search(girara_session_t* session, girara_argument_t* argument, unsigned int + UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(argument != NULL, false); + g_return_val_if_fail(zathura->document != NULL, false); + + return false; +} + +bool +sc_navigate_index(girara_session_t* session, girara_argument_t* argument, + unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(argument != NULL, false); + g_return_val_if_fail(zathura->document != NULL, false); + + return false; +} + +bool +sc_toggle_index(girara_session_t* session, girara_argument_t* argument, unsigned + int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(argument != NULL, false); + g_return_val_if_fail(zathura->document != NULL, false); + + girara_tree_node_t* document_index = NULL; + GtkWidget* treeview = NULL; + GtkTreeModel* model = NULL; + GtkCellRenderer* renderer = NULL; + + if (zathura->ui.index == NULL) { + /* create new index widget */ + zathura->ui.index = gtk_scrolled_window_new(NULL, NULL); + + if (zathura->ui.index == NULL) { + goto error_ret; + } + + gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(zathura->ui.index), + GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); + + /* create index */ + document_index = zathura_document_index_generate(zathura->document); + if (document_index == NULL) { + // TODO: Error message + goto error_free; + } + + model = GTK_TREE_MODEL(gtk_tree_store_new(2, G_TYPE_STRING, G_TYPE_POINTER)); + if (model == NULL) { + goto error_free; + } + + treeview = gtk_tree_view_new_with_model(model); + if (treeview == NULL) { + goto error_free; + } + + g_object_unref(model); + + renderer = gtk_cell_renderer_text_new(); + if (renderer == NULL) { + goto error_free; + } + + document_index_build(model, NULL, document_index); + girara_node_free(document_index); + + /* setup widget */ + gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW (treeview), 0, "Title", renderer, "markup", 0, NULL); + gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), FALSE); + g_object_set(G_OBJECT(renderer), "ellipsize", PANGO_ELLIPSIZE_END, NULL); + g_object_set(G_OBJECT(gtk_tree_view_get_column(GTK_TREE_VIEW(treeview), 0)), "expand", TRUE, NULL); + gtk_tree_view_set_cursor(GTK_TREE_VIEW(treeview), gtk_tree_path_new_first(), NULL, FALSE); + /*g_signal_connect(G_OBJECT(treeview), "row-activated", G_CALLBACK(cb_index_row_activated), NULL); TODO*/ + + gtk_container_add(GTK_CONTAINER(zathura->ui.index), treeview); + gtk_widget_show(treeview); + } + + if (gtk_widget_get_visible(GTK_WIDGET(zathura->ui.index))) { + girara_set_view(session, zathura->ui.page_view); + gtk_widget_hide(GTK_WIDGET(zathura->ui.index)); + } else { + girara_set_view(session, zathura->ui.index); + gtk_widget_show(GTK_WIDGET(zathura->ui.index)); + } + + return false; + +error_free: + + if (zathura->ui.index != NULL) { + g_object_ref_sink(zathura->ui.index); + zathura->ui.index = NULL; + } + + if (document_index != NULL) { + girara_node_free(document_index); + } + +error_ret: + + return false; +} + +bool +sc_toggle_fullscreen(girara_session_t* session, girara_argument_t* + UNUSED(argument), unsigned int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + + static bool fullscreen = false; + + if (fullscreen) { + gtk_window_unfullscreen(GTK_WINDOW(session->gtk.window)); + } else { + gtk_window_fullscreen(GTK_WINDOW(session->gtk.window)); + } + + fullscreen = fullscreen ? false : true; + + return false; +} + +bool +sc_quit(girara_session_t* session, girara_argument_t* UNUSED(argument), unsigned + int UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + + girara_argument_t arg = { GIRARA_HIDE, NULL }; + girara_isc_completion(session, &arg, 0); + + cb_destroy(NULL, NULL); + + gtk_main_quit(); + + return false; +} + +bool +sc_zoom(girara_session_t* session, girara_argument_t* argument, unsigned int + UNUSED(t)) +{ + g_return_val_if_fail(session != NULL, false); + g_return_val_if_fail(session->global.data != NULL, false); + zathura_t* zathura = session->global.data; + g_return_val_if_fail(argument != NULL, false); + g_return_val_if_fail(zathura->document != NULL, false); + + /* retreive zoom step value */ + int* value = girara_setting_get(zathura->ui.session, "zoom-step"); + if (value == NULL) { + return false; + } + + float zoom_step = *value / 100.0f; + + if (argument->n == ZOOM_IN) { + zathura->document->scale += zoom_step; + } else if (argument->n == ZOOM_OUT) { + zathura->document->scale -= zoom_step; + } else { + zathura->document->scale = 1.0; + } + + render_all(zathura); + + return false; +} diff --git a/shortcuts.h b/shortcuts.h new file mode 100644 index 0000000..7123b08 --- /dev/null +++ b/shortcuts.h @@ -0,0 +1,168 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef SHORTCUTS_H +#define SHORTCUTS_H + +#include + +/** + * Abort the current action and return to normal mode + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_abort(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Adjust the rendered pages to the window + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_adjust_window(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Change the current mode + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_change_mode(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Follow a link + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_follow(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Go to a specific page or position + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_goto(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Navigate through the document + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_navigate(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Recolor the pages + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_recolor(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Reload the current document + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_reload(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Rotate the pages + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_rotate(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Scroll through the pages + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_scroll(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Search through the document for the latest search item + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_search(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Navigate through the index of the document + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_navigate_index(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Show/Hide the index of the document + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_toggle_index(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Toggle fullscreen mode + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_toggle_fullscreen(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Quit zathura + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_quit(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +/** + * Change the zoom level + * + * @param session The used girara session + * @param argument The used argument + * @param t Number of executions + * @return true if no error occured otherwise false + */ +bool sc_zoom(girara_session_t* session, girara_argument_t* argument, unsigned int t); + +#endif // SHORTCUTS_H diff --git a/utils.c b/utils.c new file mode 100644 index 0000000..45bd735 --- /dev/null +++ b/utils.c @@ -0,0 +1,249 @@ +/* See LICENSE file for license and copyright information */ + +#include +#include +#include +#include +#include +#include +#include + +#include "utils.h" +#include "zathura.h" +#include "document.h" + +#define BLOCK_SIZE 64 + +bool +file_exists(const char* path) +{ + if (!access(path, F_OK)) { + return true; + } else { + return false; + } +} + +const char* +file_get_extension(const char* path) +{ + if (!path) { + return NULL; + } + + unsigned int i = strlen(path); + for (; i > 0; i--) + { + if (*(path + i) != '.') { + continue; + } else { + break; + } + } + + if (!i) { + return NULL; + } + + return path + i + 1; +} + +bool +file_valid_extension(zathura_t* zathura, const char* path) +{ + if (path == NULL) { + return false; + } + + const char* file_extension = file_get_extension(path); + + if (file_extension == NULL) { + return false; + } + + girara_list_iterator_t* iter = girara_list_iterator(zathura->plugins.plugins); + if (iter == NULL) { + return false; + } + + do { + zathura_document_plugin_t* plugin = (zathura_document_plugin_t*) girara_list_iterator_data(iter); + if (!strcmp(file_extension, plugin->file_extension)) { + return true; + } + } while (girara_list_iterator_next(iter)); + girara_list_iterator_free(iter); + + return false; +} + +bool +execute_command(char* const argv[], char** output) +{ + if (!output) { + return false; + } + + int p[2]; + if (pipe(p)) { + return -1; + } + + pid_t pid = fork(); + + if (pid == -1) { // failure + return false; + } else if (pid == 0) { // child + dup2(p[1], 1); + close(p[0]); + + if (execvp(argv[0], argv) == -1) { + return false; + } + } else { // parent + dup2(p[0], 0); + close(p[1]); + + /* read output */ + unsigned int bc = BLOCK_SIZE; + unsigned int i = 0; + char* buffer = malloc(sizeof(char) * bc); + *output = NULL; + + if (!buffer) { + close(p[0]); + return false; + } + + char c; + while (1 == read(p[0], &c, 1)) { + buffer[i++] = c; + + if (i == bc) { + bc += BLOCK_SIZE; + char* tmp = realloc(buffer, sizeof(char) * bc); + + if (!tmp) { + free(buffer); + close(p[0]); + return false; + } + + buffer = tmp; + } + } + + char* tmp = realloc(buffer, sizeof(char) * (bc + 1)); + if (!tmp) { + free(buffer); + close(p[0]); + return false; + } + + buffer = tmp; + buffer[i] = '\0'; + + *output = buffer; + + /* wait for child to terminate */ + waitpid(pid, NULL, 0); + close(p[0]); + } + + return true; +} + +void +document_index_build(GtkTreeModel* UNUSED(model), GtkTreeIter* UNUSED(parent), + girara_tree_node_t* UNUSED(tree)) +{ + /*girara_list_t* list = girara_node_get_children(tree);*/ + /*girara_list_iterator_t* it = girara_list_iterator(list);*/ + + /*do {*/ + /*zathura_index_element_t* index_element = (zathura_index_element_t*) girara_list_iterator_data(it);*/ + /*} while ((it = girara_list_iterator_next(it)));*/ +} + +char* +string_concat(const char* string1, ...) +{ + if(!string1) { + return NULL; + } + + va_list args; + char* s; + int l = strlen(string1) + 1; + + /* calculate length */ + va_start(args, string1); + + s = va_arg(args, char*); + + while(s) { + l += strlen(s); + s = va_arg(args, char*); + } + + va_end(args); + + /* prepare */ + char* c = malloc(sizeof(char) * l); + char* p = c; + + /* copy */ + char* d = p; + char* x = (char*) string1; + + do { + *d++ = *x; + } while (*x++ != '\0'); + + p = d - 1; + + va_start(args, string1); + + s = va_arg(args, char*); + + while(s) { + d = p; + x = s; + + do { + *d++ = *x; + } while (*x++ != '\0'); + + p = d - 1; + s = va_arg(args, char*); + } + + va_end(args); + + return c; +} + + +page_offset_t* +page_calculate_offset(zathura_page_t* page) +{ + if (page == NULL || page->document == NULL || page->document->zathura == NULL) { + return NULL; + } + + page_offset_t* offset = malloc(sizeof(page_offset_t)); + + if (offset == NULL) { + return NULL; + } + + zathura_document_t* document = page->document; + zathura_t* zathura = document->zathura; + + if (gtk_widget_translate_coordinates(page->event_box, zathura->ui.page_view, 0, 0, &(offset->x), &(offset->y)) == false) { + free(offset); + return NULL; + } + + return offset; +} diff --git a/utils.h b/utils.h new file mode 100644 index 0000000..3637b58 --- /dev/null +++ b/utils.h @@ -0,0 +1,82 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef UTILS_H +#define UTILS_H + +#include +#include +#include + +#include "document.h" + +typedef struct page_offset_s +{ + int x; + int y; +} page_offset_t; + +/** + * Checks if the given file exists + * + * @param path Path to the file + * @return true if the file exists, otherwise false + */ +bool file_exists(const char* path); + +/** + * Returns the file extension of a path + * + * @param path Path to the file + * @return The file extension or NULL + */ +const char* file_get_extension(const char* path); + +/** + * This function checks if the file has a valid extension. A extension is + * evaluated as valid if it matches a supported filetype. + * + * @param zathura Zathura object + * @param path The path to the file + * @return true if the extension is valid, otherwise false + */ +bool file_valid_extension(zathura_t* zathura, const char* path); + +/** + * Executes a system command and saves its output into output + * + * @param argv The command + * @param output Pointer where the output will be saved + * @return true if no error occured, otherwise false + */ +bool execute_command(char* const argv[], char** output); + +/** + * Generates the document index based upon the list retreived from the document + * object. + * + * @param model The tree model + * @param tree_it The Tree iterator + * @param list_it The index list iterator + */ +void document_index_build(GtkTreeModel* model, GtkTreeIter* parent, girara_tree_node_t* tree); + +/** + * This function is used to concatenate multiple strings. Argument list has to + * be ended with NULL. Returned string has to be freed with free(). + * + * @param string1 First string + * @param ... Additional strings + * @return Concatenated string or NULL if an error occured + */ +char* string_concat(const char* string1, ...); + +/** + * Calculates the offset of the page to the top of the viewing area as + * well as to the left side of it. The result has to be freed. + * + * @param page The Page + * @return The calculated offset or NULL if an error occured + */ +page_offset_t* page_calculate_offset(zathura_page_t* page); + +#endif // UTILS_H diff --git a/zathura.1 b/zathura.1 index 6d956b4..a38bb6a 100644 --- a/zathura.1 +++ b/zathura.1 @@ -1,248 +1,27 @@ .TH ZATHURA 1 zathura\-VERSION + .SH NAME -zathura \- a PDF viewer +zathura \- a document viewer + .SH SYNOPSIS .B zathura -.RB [-e\ xid] -.RB [-c\ path] -.RB [-d\ path] +.RB [options] .RB [file] .RB [password] -.SH DESCRIPTION -zathura is a highly customizable and functional PDF viewer based on the poppler -rendering library and the GTK+ toolkit. zathura provides a minimalistic and -space saving interface, with a focus on keyboard interaction. -If instead of a filename - is specified on the command line, zathura tries to -render a PDF file piped to zathura via stdin. +.SH DESCRIPTION +zathura is a highly customizable and functional document viewer. + .SH OPTIONS .TP -.B -e xid +.B "-e xid" Reparents to window specified by xid. .TP -.B -c path -Path to the config directory (defaults to .config/zathura) +.B "-c path" +Path to the config directory. .TP -.B -d path -Path to the data directory (defaults to .local/share/zathura) -.SH DEFAULT SETTINGS -.SS Shortcuts +.B "-d path" +Path to the data directory. .TP -.B J -Go to next page -.TP -.B K -Go to previous page -.TP -.B h -Scroll to the left -.TP -.B k -Scroll upwards -.TP -.B j -Scroll downwards -.TP -.B ^f -Scroll page down -.TP -.B ^b -Scroll page up -.TP -.B ^d -Scroll half a page down -.TP -.B ^u -Scroll half a page up -.TP -.B l -Scroll to the right -.TP -.B / -Search forwards -.TP -.B ? -Search backwards -.TP -.B n -Search last keyword forwards -.TP -.B N -Search last keyword backwards -.TP -.B Tab -Toggle index -.TP -.B o -Open a file -.TP -.B a -Zoom to fit -.TP -.B s -Zoom to width -.TP -.B f -Follow a link on the page -.TP -.B m -Mark current position and save it in register specified by a character -.TP -.B ' -Go to saved position specified by the ensuing character -.TP -.B r -Rotate the page -.TP -.B R -Reload the document -.TP -.B O -Change goto mode (L: search labels, D: default, O: manual offset) -.TP -.B ^i -Recolors the page -.TP -.B i -Change to insert mode -.TP -.B v -Change to visual mode -.TP -.B : -Focus inputbar -.TP -.B Backspace -Delete last character in the buffer -.TP -.B F5 -Toggle fullscreen mode -.TP -.B ^n -Toggle statusbar visibility -.TP -.B ^m -Toggle inputbar visibility -.TP -.B + -Zoom in -.TP -.B - -Zoom out -.TP -.B = -Zoom to the original size -.TP -.B Esc | ^c -Abort -.TP -.B ^q -Quit the program -.SS Index commands -.TP -.B k -Navigate to one element above the current position -.TP -.B j -Navigate to one element below the current position -.TP -.B h -Collapse current element -.TP -.B l -Expand current element -.TP -.B Space | Return -Select current element -.SS Buffered commands -.TP -.B gg -Go to the first page -.TP -.B GG -Go to the last page -.TP -.B [0-9]+G -Go to the specified page -.TP -.B zI -Zoom in -.TP -.B zO -Zoom out -.TP -.B z0 -Zoom to the original size -.TP -.B [0-9]+Z -Zoom to the given level -.TP -.B [0-9]+% -Move to the given position -.SS Commands -.TP -.B blist -List and open bookmark -.TP -.B bmark -Bookmark current page -.TP -.B close -Close current file -.TP -.B coffset -Set page offset -.TP -.B delbmark -Delete given bookmark -.TP -.B export -Export images or attached files -.TP -.B info -Show information about the document -.TP -.B map -Map shortcut functions (map ) -.TP -.B open -Open a file -.TP -.B print -Print the document -.TP -.B quit -Quit the program -.TP -.B rotate -Rotate the page -.TP -.B set -Set an option (set ) -.TP -.B write -Save the document -.SS Inputbar shortcuts -.TP -.B Up -Move up in the command history -.TP -.B Down -Move down in the command history -.TP -.B Tab | Shift + Tab -Tab completion -.TP -.B ^w -Delete last word -.SH CONFIGURATION -The complete configuration including the appearance and shortcuts of the program -are defined in a separate file named config.h. In this file you are able to -change and adjust all the settings of zathura according to your wishes. - -In addition you can create a zathurarc file (default path: ~/.config/zathura/zathurarc) -to overwrite settings and keybindings by using the set and map function. For -more information please have a look at \&\fIzathurarc\fR\|(5). - -.SH SEE ALSO -\&\fIzathurarc\fR\|(5) +.B "-p path" +Path to the plugin directory. diff --git a/zathura.c b/zathura.c index 706d18a..6a8a75b 100644 --- a/zathura.c +++ b/zathura.c @@ -1,4929 +1,491 @@ /* See LICENSE file for license and copyright information */ -#define _BSD_SOURCE -#define _XOPEN_SOURCE 500 - -#include -#include #include -#include -#include -#include -#include -#include -#include +#include -#include -#include -#include - -/* macros */ -#define LENGTH(x) (sizeof(x)/sizeof((x)[0])) -#define CLEAN(m) (m & ~(GDK_MOD5_MASK) & ~(GDK_MOD2_MASK) & ~(GDK_BUTTON1_MASK) & ~(GDK_BUTTON2_MASK) & ~(GDK_BUTTON3_MASK) & ~(GDK_BUTTON4_MASK) & ~(GDK_BUTTON5_MASK) & ~(GDK_LEAVE_NOTIFY_MASK)) -#if defined(__GNUC__) || defined(__INTEL_COMPILER) || defined(__ICL) || defined(__ICC) || defined(__ECC) || defined(__clang__) -/* only gcc, clang and Intel's cc seem support this */ -#define NORETURN __attribute__((noreturn)) -#else -#define NORETURN -#endif - -// just remove the #if-#define-#endif block if support for poppler versions -// before 0.15 is dropped -#if !POPPLER_CHECK_VERSION(0,15,0) -#define poppler_page_get_selected_text poppler_page_get_text -#endif - -/* enums */ -enum { NEXT, PREVIOUS, LEFT, RIGHT, UP, DOWN, BOTTOM, TOP, HIDE, HIGHLIGHT, - DELETE_LAST_WORD, DELETE_LAST_CHAR, DEFAULT, ERROR, WARNING, NEXT_GROUP, - PREVIOUS_GROUP, ZOOM_IN, ZOOM_OUT, ZOOM_ORIGINAL, ZOOM_SPECIFIC, FORWARD, - BACKWARD, ADJUST_BESTFIT, ADJUST_WIDTH, ADJUST_NONE, CONTINUOUS, DELETE_LAST, - ADD_MARKER, EVAL_MARKER, EXPAND, COLLAPSE, SELECT, GOTO_DEFAULT, GOTO_LABELS, - GOTO_OFFSET, HALF_UP, HALF_DOWN, FULL_UP, FULL_DOWN, NEXT_CHAR, PREVIOUS_CHAR, - DELETE_TO_LINE_START, APPEND_FILEPATH, NO_SEARCH }; - -/* define modes */ -#define ALL (1 << 0) -#define FULLSCREEN (1 << 1) -#define INDEX (1 << 2) -#define NORMAL (1 << 3) - -/* typedefs */ -struct CElement -{ - char *value; - char *description; - struct CElement *next; -}; - -typedef struct CElement CompletionElement; - -struct CGroup -{ - char *value; - CompletionElement *elements; - struct CGroup *next; -}; - -typedef struct CGroup CompletionGroup; - -typedef struct -{ - CompletionGroup* groups; -} Completion; - -typedef struct -{ - char* command; - char* description; - int command_id; - gboolean is_group; - GtkWidget* row; -} CompletionRow; - -typedef struct -{ - int n; - void *data; -} Argument; - -typedef struct -{ - char* name; - int argument; -} ArgumentName; - -typedef struct -{ - int mask; - int key; - void (*function)(Argument*); - int mode; - Argument argument; -} Shortcut; - -typedef struct -{ - char* name; - int mode; - char* display; -} ModeName; - -typedef struct -{ - char* name; - void (*function)(Argument*); -} ShortcutName; - -typedef struct -{ - int mask; - int key; - void (*function)(Argument*); - Argument argument; -} InputbarShortcut; - -typedef struct -{ - int direction; - void (*function)(Argument*); - Argument argument; -} MouseScrollEvent; - -typedef struct -{ - char* command; - char* abbr; - gboolean (*function)(int, char**); - Completion* (*completion)(char*); - char* description; -} Command; - -typedef struct -{ - char* regex; - void (*function)(char*, Argument*); - Argument argument; -} BufferCommand; - -typedef struct -{ - char identifier; - gboolean (*function)(char*, Argument*); - int always; - Argument argument; -} SpecialCommand; - -struct SCList -{ - Shortcut element; - struct SCList *next; -}; - -typedef struct SCList ShortcutList; - -typedef struct -{ - char* identifier; - int key; -} GDKKey; - -typedef struct -{ - PopplerPage *page; - int id; - char *label; -} Page; - -typedef struct -{ - char* name; - void* variable; - char type; - gboolean render; - gboolean reinit; - char* description; -} Setting; - -typedef struct -{ - int id; - int page; -} Marker; - -typedef struct -{ - char* id; - int page; -} Bookmark; - -/* zathura */ -struct -{ - struct - { - GtkWidget *window; - GtkBox *box; - GtkBox *continuous; - GtkScrolledWindow *view; - GtkViewport *viewport; - GtkWidget *statusbar; - GtkBox *statusbar_entries; - GtkEntry *inputbar; - GtkWidget *index; - GtkWidget *information; - GtkWidget *drawing_area; - GtkWidget *document; - GdkNativeWindow embed; - } UI; - - struct - { - GdkColor default_fg; - GdkColor default_bg; - GdkColor inputbar_fg; - GdkColor inputbar_bg; - GdkColor statusbar_fg; - GdkColor statusbar_bg; - GdkColor completion_fg; - GdkColor completion_bg; - GdkColor completion_g_bg; - GdkColor completion_g_fg; - GdkColor completion_hl_fg; - GdkColor completion_hl_bg; - GdkColor notification_e_fg; - GdkColor notification_e_bg; - GdkColor notification_w_fg; - GdkColor notification_w_bg; - GdkColor recolor_darkcolor; - GdkColor recolor_lightcolor; - GdkColor search_highlight; - GdkColor select_text; - PangoFontDescription *font; - } Style; - - struct - { - GtkLabel *status_text; - GtkLabel *status_buffer; - GtkLabel *status_state; - GString *buffer; - GList *history; - int mode; - int viewing_mode; - gboolean recolor; - gboolean enable_labelmode; - int goto_mode; - int adjust_mode; - gboolean show_index; - gboolean show_statusbar; - gboolean show_inputbar; - } Global; - - struct - { - ShortcutList *sclist; - } Bindings; - - struct - { - gdouble x; - gdouble y; - } SelectPoint; - - struct - { - char* filename; - char* pages; - int scroll_percentage; - } State; - - struct - { - Marker* markers; - int number_of_markers; - int last; - } Marker; - - struct - { - GFileMonitor* monitor; - GFile* file; - } FileMonitor; - - struct - { - GKeyFile *data; - char *file; - Bookmark *bookmarks; - int number_of_bookmarks; - } Bookmarks; - - struct - { - PopplerDocument *document; - char *file; - char *password; - Page **pages; - int page_number; - int page_offset; - int number_of_pages; - int scale; - int rotate; - cairo_surface_t *surface; - } PDF; - - struct - { - GStaticMutex pdflib_lock; - GStaticMutex pdf_obj_lock; - GStaticMutex search_lock; - GStaticMutex select_lock; - } Lock; - - struct - { - GList* results; - int page; - gboolean draw; - gchar* query; - } Search; - - struct - { - GThread* search_thread; - gboolean search_thread_running; - GThread* inotify_thread; - } Thread; - - struct - { - guint inputbar_activate; - guint inputbar_key_press_event; - } Handler; - - struct - { - gchar* config_dir; - gchar* data_dir; - } Config; - - struct - { - gchar* file; - } StdinSupport; -} Zathura; - - -/* function declarations */ -void init_look(void); -void init_directories(void); -void init_bookmarks(void); -void init_keylist(void); -void init_settings(void); -void init_zathura(void); -void add_marker(int); -void build_index(GtkTreeModel*, GtkTreeIter*, PopplerIndexIter*); -void change_mode(int); -void calculate_offset(GtkWidget*, double*, double*); -void close_file(gboolean); -void enter_password(void); -void highlight_result(int, PopplerRectangle*); -void draw(int); -void eval_marker(int); -void notify(int, const char*); -gboolean open_file(char*, char*); -gboolean open_stdin(gchar*); -void open_uri(char*); -void out_of_memory(void) NORETURN; -void* safe_realloc(void** ptr, size_t nmemb, size_t size); -void update_status(void); -void read_bookmarks_file(void); -void write_bookmarks_file(void); -void free_bookmarks(void); -void read_configuration_file(const char*); -void read_configuration(void); -void recalc_rectangle(int, PopplerRectangle*); -void set_completion_row_color(GtkBox*, int, int); -void set_page(int); -void switch_view(GtkWidget*); -GtkEventBox* create_completion_row(GtkBox*, char*, char*, gboolean); -gchar* fix_path(const gchar*); -gchar* path_from_env(const gchar*); -gchar* get_home_dir(void); -gboolean is_reserved_bm_name(const char *); - -Completion* completion_init(void); -CompletionGroup* completion_group_create(char*); -void completion_add_group(Completion*, CompletionGroup*); -void completion_free(Completion*); -void completion_group_add_element(CompletionGroup*, char*, char*); - -/* thread declaration */ -void* search(void*); - -/* shortcut declarations */ -void sc_abort(Argument*); -void sc_adjust_window(Argument*); -void sc_change_buffer(Argument*); -void sc_change_mode(Argument*); -void sc_focus_inputbar(Argument*); -void sc_follow(Argument*); -void sc_navigate(Argument*); -void sc_recolor(Argument*); -void sc_reload(Argument*); -void sc_rotate(Argument*); -void sc_scroll(Argument*); -void sc_search(Argument*); -void sc_switch_goto_mode(Argument*); -void sc_navigate_index(Argument*); -void sc_toggle_index(Argument*); -void sc_toggle_inputbar(Argument*); -void sc_toggle_fullscreen(Argument*); -void sc_toggle_statusbar(Argument*); -void sc_quit(Argument*); -void sc_zoom(Argument*); - -/* inputbar shortcut declarations */ -void isc_abort(Argument*); -void isc_command_history(Argument*); -void isc_completion(Argument*); -void isc_string_manipulation(Argument*); - -/* command declarations */ -gboolean cmd_bookmark(int, char**); -gboolean cmd_open_bookmark(int, char**); -gboolean cmd_close(int, char**); -gboolean cmd_correct_offset(int, char**); -gboolean cmd_delete_bookmark(int, char**); -gboolean cmd_export(int, char**); -gboolean cmd_info(int, char**); -gboolean cmd_map(int, char**); -gboolean cmd_open(int, char**); -gboolean cmd_print(int, char**); -gboolean cmd_rotate(int, char**); -gboolean cmd_set(int, char**); -gboolean cmd_quit(int, char**); -gboolean cmd_save(int, char**); -gboolean cmd_savef(int, char**); - -/* completion commands */ -Completion* cc_bookmark(char*); -Completion* cc_export(char*); -Completion* cc_open(char*); -Completion* cc_print(char*); -Completion* cc_set(char*); - -/* buffer command declarations */ -void bcmd_evalmarker(char*, Argument*); -void bcmd_goto(char*, Argument*); -void bcmd_scroll(char*, Argument*); -void bcmd_setmarker(char*, Argument*); -void bcmd_zoom(char*, Argument*); - -/* special command delcarations */ -gboolean scmd_search(gchar*, Argument*); - -/* callback declarations */ -gboolean cb_destroy(GtkWidget*, gpointer); -gboolean cb_draw(GtkWidget*, GdkEventExpose*, gpointer); -gboolean cb_index_row_activated(GtkTreeView*, GtkTreePath*, GtkTreeViewColumn*, gpointer); -gboolean cb_inputbar_kb_pressed(GtkWidget*, GdkEventKey*, gpointer); -gboolean cb_inputbar_activate(GtkEntry*, gpointer); -gboolean cb_inputbar_form_activate(GtkEntry*, gpointer); -gboolean cb_inputbar_password_activate(GtkEntry*, gpointer); -gboolean cb_view_kb_pressed(GtkWidget*, GdkEventKey*, gpointer); -gboolean cb_view_resized(GtkWidget*, GtkAllocation*, gpointer); -gboolean cb_view_button_pressed(GtkWidget*, GdkEventButton*, gpointer); -gboolean cb_view_button_release(GtkWidget*, GdkEventButton*, gpointer); -gboolean cb_view_motion_notify(GtkWidget*, GdkEventMotion*, gpointer); -gboolean cb_view_scrolled(GtkWidget*, GdkEventScroll*, gpointer); -gboolean cb_watch_file(GFileMonitor*, GFile*, GFile*, GFileMonitorEvent, gpointer); - -/* configuration */ +#include "bookmarks.h" +#include "callbacks.h" #include "config.h" +#include "database.h" +#include "document.h" +#include "shortcuts.h" +#include "zathura.h" +#include "utils.h" +#include "render.h" + +typedef struct zathura_document_info_s +{ + zathura_t* zathura; + const char* path; + const char* password; +} zathura_document_info_t; + +gboolean document_info_open(gpointer data); /* function implementation */ -void -init_look(void) -{ - /* parse */ - gdk_color_parse(default_fgcolor, &(Zathura.Style.default_fg)); - gdk_color_parse(default_bgcolor, &(Zathura.Style.default_bg)); - gdk_color_parse(inputbar_fgcolor, &(Zathura.Style.inputbar_fg)); - gdk_color_parse(inputbar_bgcolor, &(Zathura.Style.inputbar_bg)); - gdk_color_parse(statusbar_fgcolor, &(Zathura.Style.statusbar_fg)); - gdk_color_parse(statusbar_bgcolor, &(Zathura.Style.statusbar_bg)); - gdk_color_parse(completion_fgcolor, &(Zathura.Style.completion_fg)); - gdk_color_parse(completion_bgcolor, &(Zathura.Style.completion_bg)); - gdk_color_parse(completion_g_fgcolor, &(Zathura.Style.completion_g_fg)); - gdk_color_parse(completion_g_fgcolor, &(Zathura.Style.completion_g_fg)); - gdk_color_parse(completion_hl_fgcolor, &(Zathura.Style.completion_hl_fg)); - gdk_color_parse(completion_hl_bgcolor, &(Zathura.Style.completion_hl_bg)); - gdk_color_parse(notification_e_fgcolor, &(Zathura.Style.notification_e_fg)); - gdk_color_parse(notification_e_bgcolor, &(Zathura.Style.notification_e_bg)); - gdk_color_parse(notification_w_fgcolor, &(Zathura.Style.notification_w_fg)); - gdk_color_parse(notification_w_bgcolor, &(Zathura.Style.notification_w_bg)); - gdk_color_parse(recolor_darkcolor, &(Zathura.Style.recolor_darkcolor)); - gdk_color_parse(recolor_lightcolor, &(Zathura.Style.recolor_lightcolor)); - gdk_color_parse(search_highlight, &(Zathura.Style.search_highlight)); - gdk_color_parse(select_text, &(Zathura.Style.select_text)); - - pango_font_description_free(Zathura.Style.font); - Zathura.Style.font = pango_font_description_from_string(font); - - /* window and viewport */ - gtk_widget_modify_bg(GTK_WIDGET(Zathura.UI.window), GTK_STATE_NORMAL, &(Zathura.Style.default_bg)); - gtk_widget_modify_bg(GTK_WIDGET(Zathura.UI.viewport), GTK_STATE_NORMAL, &(Zathura.Style.default_bg)); - - /* drawing area */ - gtk_widget_modify_bg(GTK_WIDGET(Zathura.UI.drawing_area), GTK_STATE_NORMAL, &(Zathura.Style.default_bg)); - - /* statusbar */ - gtk_widget_modify_bg(GTK_WIDGET(Zathura.UI.statusbar), GTK_STATE_NORMAL, &(Zathura.Style.statusbar_bg)); - - gtk_widget_modify_fg(GTK_WIDGET(Zathura.Global.status_text), GTK_STATE_NORMAL, &(Zathura.Style.statusbar_fg)); - gtk_widget_modify_fg(GTK_WIDGET(Zathura.Global.status_state), GTK_STATE_NORMAL, &(Zathura.Style.statusbar_fg)); - gtk_widget_modify_fg(GTK_WIDGET(Zathura.Global.status_buffer), GTK_STATE_NORMAL, &(Zathura.Style.statusbar_fg)); - - gtk_widget_modify_font(GTK_WIDGET(Zathura.Global.status_text), Zathura.Style.font); - gtk_widget_modify_font(GTK_WIDGET(Zathura.Global.status_state), Zathura.Style.font); - gtk_widget_modify_font(GTK_WIDGET(Zathura.Global.status_buffer), Zathura.Style.font); - - /* inputbar */ - gtk_widget_modify_base(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.inputbar_bg)); - gtk_widget_modify_text(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.inputbar_fg)); - gtk_widget_modify_font(GTK_WIDGET(Zathura.UI.inputbar), Zathura.Style.font); - - g_object_set(G_OBJECT(Zathura.UI.inputbar), "has-frame", FALSE, NULL); - - /* scrollbars */ - if(show_scrollbars) - gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(Zathura.UI.view), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); - else - gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(Zathura.UI.view), GTK_POLICY_NEVER, GTK_POLICY_NEVER); - - /* inputbar */ - if(Zathura.Global.show_inputbar) - gtk_widget_show(GTK_WIDGET(Zathura.UI.inputbar)); - else - gtk_widget_hide(GTK_WIDGET(Zathura.UI.inputbar)); - - /* statusbar */ - if(Zathura.Global.show_statusbar) - gtk_widget_show(GTK_WIDGET(Zathura.UI.statusbar)); - else - gtk_widget_hide(GTK_WIDGET(Zathura.UI.statusbar)); -} - -gchar* -fix_path(const gchar* path) -{ - if (!path) - return NULL; - - if (path[0] == '~') - { - gchar* home_path = get_home_dir(); - gchar* res = g_build_filename(home_path, path + 1, NULL); - g_free(home_path); - return res; - } - else - return g_strdup(path); -} - -gchar* path_from_env(const gchar* var) -{ - gchar* env = fix_path(g_getenv(var)); - if (!env) - return NULL; - - gchar* res = g_build_filename(env, "zathura", NULL); - g_free(env); - return res; -} - -gchar* get_home_dir(void) -{ - const gchar* homedir = g_getenv("HOME"); - return g_strdup(homedir ? homedir : g_get_home_dir()); -} - -void -init_directories(void) -{ - /* setup directories */ - if (!Zathura.Config.config_dir) - { -#ifndef ZATHURA_NO_XDG - gchar* env = path_from_env("XDG_CONFIG_HOME"); - if (env) - Zathura.Config.config_dir = env; - else -#endif - Zathura.Config.config_dir = fix_path(CONFIG_DIR); - } - if (!Zathura.Config.data_dir) - { -#ifndef ZATHURA_NO_XDG - gchar* env = path_from_env("XDG_DATA_HOME"); - if (env) - Zathura.Config.data_dir = env; - else -#endif - Zathura.Config.data_dir = fix_path(DATA_DIR); - } - - /* create zathura (config/data) directory */ - g_mkdir_with_parents(Zathura.Config.config_dir, 0771); - g_mkdir_with_parents(Zathura.Config.data_dir, 0771); -} - -void -init_bookmarks(void) -{ - /* init variables */ - Zathura.Bookmarks.number_of_bookmarks = 0; - Zathura.Bookmarks.bookmarks = NULL; - - Zathura.Bookmarks.file = g_build_filename(Zathura.Config.data_dir, BOOKMARK_FILE, NULL); - read_bookmarks_file(); -} - -gboolean -is_reserved_bm_name(const char *bm_name) -{ - int i; - - for(i = 0; i < BM_MAX; i++) - if(strcmp(bm_reserved_names[i], bm_name) == 0) - return TRUE; - - return FALSE; -} - -void -init_keylist(void) -{ - ShortcutList* e = NULL; - ShortcutList* p = NULL; - - int i; - for(i = 0; i < LENGTH(shortcuts); i++) - { - e = malloc(sizeof(ShortcutList)); - if(!e) - out_of_memory(); - - e->element = shortcuts[i]; - e->next = NULL; - - if(!Zathura.Bindings.sclist) - Zathura.Bindings.sclist = e; - if(p) - p->next = e; - - p = e; - } -} - -void -init_settings(void) -{ - Zathura.State.filename = g_strdup((char*) default_text); - Zathura.Global.adjust_mode = adjust_open; - - gtk_window_set_default_size(GTK_WINDOW(Zathura.UI.window), default_width, default_height); -} - -void -init_zathura(void) -{ - /* init mutexes */ - g_static_mutex_init(&(Zathura.Lock.pdflib_lock)); - g_static_mutex_init(&(Zathura.Lock.search_lock)); - g_static_mutex_init(&(Zathura.Lock.pdf_obj_lock)); - g_static_mutex_init(&(Zathura.Lock.select_lock)); - - /* other */ - Zathura.Global.mode = NORMAL; - Zathura.Global.viewing_mode = NORMAL; - Zathura.Global.recolor = 0; - Zathura.Global.goto_mode = GOTO_MODE; - Zathura.Global.show_index = FALSE; - Zathura.Global.show_inputbar = TRUE; - Zathura.Global.show_statusbar = TRUE; - - Zathura.State.pages = g_strdup(""); - Zathura.State.scroll_percentage = 0; - - Zathura.Marker.markers = NULL; - Zathura.Marker.number_of_markers = 0; - Zathura.Marker.last = -1; - - Zathura.Search.results = NULL; - Zathura.Search.page = 0; - Zathura.Search.draw = FALSE; - Zathura.Search.query = NULL; - - Zathura.FileMonitor.monitor = NULL; - Zathura.FileMonitor.file = NULL; - - Zathura.StdinSupport.file = NULL; - - /* window */ - if(Zathura.UI.embed) - Zathura.UI.window = gtk_plug_new(Zathura.UI.embed); - else - Zathura.UI.window = gtk_window_new(GTK_WINDOW_TOPLEVEL); - - /* UI */ - Zathura.UI.box = GTK_BOX(gtk_vbox_new(FALSE, 0)); - Zathura.UI.continuous = GTK_BOX(gtk_vbox_new(FALSE, 0)); - Zathura.UI.view = GTK_SCROLLED_WINDOW(gtk_scrolled_window_new(NULL, NULL)); - Zathura.UI.viewport = GTK_VIEWPORT(gtk_viewport_new(NULL, NULL)); - Zathura.UI.drawing_area = gtk_drawing_area_new(); - Zathura.UI.statusbar = gtk_event_box_new(); - Zathura.UI.statusbar_entries = GTK_BOX(gtk_hbox_new(FALSE, 0)); - Zathura.UI.inputbar = GTK_ENTRY(gtk_entry_new()); - Zathura.UI.document = gtk_event_box_new(); - - /* window */ - gtk_window_set_title(GTK_WINDOW(Zathura.UI.window), "zathura"); - GdkGeometry hints = { 1, 1 }; - gtk_window_set_geometry_hints(GTK_WINDOW(Zathura.UI.window), NULL, &hints, GDK_HINT_MIN_SIZE); - g_signal_connect(G_OBJECT(Zathura.UI.window), "destroy", G_CALLBACK(cb_destroy), NULL); - - /* box */ - gtk_box_set_spacing(Zathura.UI.box, 0); - gtk_container_add(GTK_CONTAINER(Zathura.UI.window), GTK_WIDGET(Zathura.UI.box)); - - /* continuous */ - gtk_box_set_spacing(Zathura.UI.continuous, 5); - - /* events */ - gtk_container_add(GTK_CONTAINER(Zathura.UI.document), GTK_WIDGET(Zathura.UI.drawing_area)); - gtk_widget_add_events(GTK_WIDGET(Zathura.UI.document), GDK_POINTER_MOTION_MASK | GDK_POINTER_MOTION_HINT_MASK | - GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK); - - g_signal_connect(G_OBJECT(Zathura.UI.document), "button-press-event", G_CALLBACK(cb_view_button_pressed), NULL); - g_signal_connect(G_OBJECT(Zathura.UI.document), "button-release-event", G_CALLBACK(cb_view_button_release), NULL); - g_signal_connect(G_OBJECT(Zathura.UI.document), "motion-notify-event", G_CALLBACK(cb_view_motion_notify), NULL); - gtk_widget_show(Zathura.UI.document); - - /* view */ - g_signal_connect(G_OBJECT(Zathura.UI.view), "key-press-event", G_CALLBACK(cb_view_kb_pressed), NULL); - g_signal_connect(G_OBJECT(Zathura.UI.view), "size-allocate", G_CALLBACK(cb_view_resized), NULL); - g_signal_connect(G_OBJECT(Zathura.UI.view), "scroll-event", G_CALLBACK(cb_view_scrolled), NULL); - gtk_container_add(GTK_CONTAINER(Zathura.UI.view), GTK_WIDGET(Zathura.UI.viewport)); - gtk_viewport_set_shadow_type(Zathura.UI.viewport, GTK_SHADOW_NONE); - - /* drawing area */ - gtk_widget_show(Zathura.UI.drawing_area); - g_signal_connect(G_OBJECT(Zathura.UI.drawing_area), "expose-event", G_CALLBACK(cb_draw), NULL); - - /* statusbar */ - Zathura.Global.status_text = GTK_LABEL(gtk_label_new(NULL)); - Zathura.Global.status_state = GTK_LABEL(gtk_label_new(NULL)); - Zathura.Global.status_buffer = GTK_LABEL(gtk_label_new(NULL)); - - gtk_misc_set_alignment(GTK_MISC(Zathura.Global.status_text), 0.0, 0.0); - gtk_misc_set_alignment(GTK_MISC(Zathura.Global.status_state), 1.0, 0.0); - gtk_misc_set_alignment(GTK_MISC(Zathura.Global.status_buffer), 1.0, 0.0); - - gtk_misc_set_padding(GTK_MISC(Zathura.Global.status_text), 2.0, 4.0); - gtk_misc_set_padding(GTK_MISC(Zathura.Global.status_state), 2.0, 4.0); - gtk_misc_set_padding(GTK_MISC(Zathura.Global.status_buffer), 2.0, 4.0); - - gtk_label_set_use_markup(Zathura.Global.status_text, TRUE); - gtk_label_set_use_markup(Zathura.Global.status_state, TRUE); - gtk_label_set_use_markup(Zathura.Global.status_buffer, TRUE); - - gtk_box_pack_start(Zathura.UI.statusbar_entries, GTK_WIDGET(Zathura.Global.status_text), TRUE, TRUE, 2); - gtk_box_pack_start(Zathura.UI.statusbar_entries, GTK_WIDGET(Zathura.Global.status_buffer), FALSE, FALSE, 2); - gtk_box_pack_start(Zathura.UI.statusbar_entries, GTK_WIDGET(Zathura.Global.status_state), FALSE, FALSE, 2); - - gtk_container_add(GTK_CONTAINER(Zathura.UI.statusbar), GTK_WIDGET(Zathura.UI.statusbar_entries)); - - /* inputbar */ - gtk_entry_set_inner_border(Zathura.UI.inputbar, NULL); - gtk_entry_set_has_frame( Zathura.UI.inputbar, FALSE); - gtk_editable_set_editable( GTK_EDITABLE(Zathura.UI.inputbar), TRUE); - - Zathura.Handler.inputbar_key_press_event = - g_signal_connect(G_OBJECT(Zathura.UI.inputbar), "key-press-event", G_CALLBACK(cb_inputbar_kb_pressed), NULL); - Zathura.Handler.inputbar_activate = - g_signal_connect(G_OBJECT(Zathura.UI.inputbar), "activate", G_CALLBACK(cb_inputbar_activate), NULL); - - /* packing */ - gtk_box_pack_start(Zathura.UI.box, GTK_WIDGET(Zathura.UI.view), TRUE, TRUE, 0); - gtk_box_pack_start(Zathura.UI.box, GTK_WIDGET(Zathura.UI.statusbar), FALSE, FALSE, 0); - gtk_box_pack_end( Zathura.UI.box, GTK_WIDGET(Zathura.UI.inputbar), FALSE, FALSE, 0); -} - -void -add_marker(int id) -{ - if( (id < 0x30) || (id > 0x7A)) - return; - - /* current information */ - int page_number = Zathura.PDF.page_number; - - /* search if entry already exists */ - int i; - for(i = 0; i < Zathura.Marker.number_of_markers; i++) - { - if(Zathura.Marker.markers[i].id == id) - { - Zathura.Marker.markers[i].page = page_number; - Zathura.Marker.last = page_number; - return; - } - } - - /* add new marker */ - int marker_index = Zathura.Marker.number_of_markers++; - Zathura.Marker.markers = safe_realloc((void**)&Zathura.Marker.markers, - Zathura.Marker.number_of_markers, sizeof(Marker)); - if(!Zathura.Marker.markers) - out_of_memory(); - - Zathura.Marker.markers[marker_index].id = id; - Zathura.Marker.markers[marker_index].page = page_number; - Zathura.Marker.last = page_number; -} - -void -build_index(GtkTreeModel* model, GtkTreeIter* parent, PopplerIndexIter* index_iter) -{ - do - { - GtkTreeIter tree_iter; - PopplerIndexIter *child; - PopplerAction *action; - gchar *markup; - - action = poppler_index_iter_get_action(index_iter); - if(!action) - continue; - - markup = g_markup_escape_text (action->any.title, -1); - - gtk_tree_store_append(GTK_TREE_STORE(model), &tree_iter, parent); - gtk_tree_store_set(GTK_TREE_STORE(model), &tree_iter, 0, markup, 1, action, -1); - g_object_weak_ref(G_OBJECT(model), (GWeakNotify) poppler_action_free, action); - g_free(markup); - - child = poppler_index_iter_get_child(index_iter); - if(child) - build_index(model, &tree_iter, child); - poppler_index_iter_free(child); - } while(poppler_index_iter_next(index_iter)); -} - -void -draw(int page_id) -{ - if(!Zathura.PDF.document || page_id < 0 || page_id >= Zathura.PDF.number_of_pages) - return; - - double page_width, page_height; - double width, height; - - double scale = ((double) Zathura.PDF.scale / 100.0); - - int rotate = Zathura.PDF.rotate; - - Page *current_page = Zathura.PDF.pages[page_id]; - - if(Zathura.PDF.surface) - cairo_surface_destroy(Zathura.PDF.surface); - Zathura.PDF.surface = NULL; - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_page_get_size(current_page->page, &page_width, &page_height); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if(rotate == 0 || rotate == 180) - { - width = page_width * scale; - height = page_height * scale; - } - else - { - width = page_height * scale; - height = page_width * scale; - } - - cairo_t *cairo; - Zathura.PDF.surface = cairo_image_surface_create(CAIRO_FORMAT_RGB24, width, height); - cairo = cairo_create(Zathura.PDF.surface); - - cairo_save(cairo); - cairo_set_source_rgb(cairo, 1, 1, 1); - cairo_rectangle(cairo, 0, 0, width, height); - cairo_fill(cairo); - cairo_restore(cairo); - cairo_save(cairo); - - switch(rotate) - { - case 90: - cairo_translate(cairo, width, 0); - break; - case 180: - cairo_translate(cairo, width, height); - break; - case 270: - cairo_translate(cairo, 0, height); - break; - default: - cairo_translate(cairo, 0, 0); - } - - if(scale != 1.0) - cairo_scale(cairo, scale, scale); - - if(rotate != 0) - cairo_rotate(cairo, rotate * G_PI / 180.0); - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_page_render(current_page->page, cairo); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - cairo_restore(cairo); - cairo_destroy(cairo); - - if(Zathura.Global.recolor) - { - unsigned char* image = cairo_image_surface_get_data(Zathura.PDF.surface); - int x, y; - - int width = cairo_image_surface_get_width(Zathura.PDF.surface); - int height = cairo_image_surface_get_height(Zathura.PDF.surface); - int rowstride = cairo_image_surface_get_stride(Zathura.PDF.surface); - - /* recolor code based on qimageblitz library flatten() function - (http://sourceforge.net/projects/qimageblitz/) */ - - int r1 = Zathura.Style.recolor_darkcolor.red / 257; - int g1 = Zathura.Style.recolor_darkcolor.green / 257; - int b1 = Zathura.Style.recolor_darkcolor.blue / 257; - int r2 = Zathura.Style.recolor_lightcolor.red / 257; - int g2 = Zathura.Style.recolor_lightcolor.green / 257; - int b2 = Zathura.Style.recolor_lightcolor.blue / 257; - - int min = 0x00; - int max = 0xFF; - int mean; - - float sr = ((float) r2 - r1) / (max - min); - float sg = ((float) g2 - g1) / (max - min); - float sb = ((float) b2 - b1) / (max - min); - - for (y = 0; y < height; y++) - { - unsigned char* data = image + y * rowstride; - - for (x = 0; x < width; x++) - { - mean = (data[0] + data[1] + data[2]) / 3; - data[2] = sr * (mean - min) + r1 + 0.5; - data[1] = sg * (mean - min) + g1 + 0.5; - data[0] = sb * (mean - min) + b1 + 0.5; - data += 4; - } - } - } - - gtk_widget_set_size_request(Zathura.UI.drawing_area, width, height); - gtk_widget_queue_draw(Zathura.UI.drawing_area); -} - -void -change_mode(int mode) -{ - char* mode_text = 0; - for(unsigned int i = 0; i != LENGTH(mode_names); ++i) - if(mode_names[i].mode == mode) - { - mode_text = mode_names[i].display; - break; - } - - if(!mode_text) - { - switch(mode) - { - case ADD_MARKER: - mode_text = ""; - break; - case EVAL_MARKER: - mode_text = ""; - break; - default: - mode_text = ""; - mode = NORMAL; - break; - } - } - - Zathura.Global.mode = mode; - notify(DEFAULT, mode_text); -} - -void -calculate_offset(GtkWidget* widget, double* offset_x, double* offset_y) -{ - double page_width, page_height, width, height; - double scale = ((double) Zathura.PDF.scale / 100.0); - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_page_get_size(Zathura.PDF.pages[Zathura.PDF.page_number]->page, &page_width, &page_height); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if(Zathura.PDF.rotate == 0 || Zathura.PDF.rotate == 180) - { - width = page_width * scale; - height = page_height * scale; - } - else - { - width = page_height * scale; - height = page_width * scale; - } - - int window_x, window_y; - gdk_drawable_get_size(widget->window, &window_x, &window_y); - - if (window_x > width) - *offset_x = (window_x - width) / 2; - else - *offset_x = 0; - - if (window_y > height) - *offset_y = (window_y - height) / 2; - else - *offset_y = 0; -} - -void -close_file(gboolean keep_monitor) -{ - if(!Zathura.PDF.document) - return; - - /* clean up pages */ - int i; - for(i = 0; i < Zathura.PDF.number_of_pages; i++) - { - Page* current_page = Zathura.PDF.pages[i]; - g_object_unref(current_page->page); - if(current_page->label) - g_free(current_page->label); - free(current_page); - } - - /* save bookmarks */ - if(Zathura.Bookmarks.data) - { - read_bookmarks_file(); - - if(save_position) - { - /* set current page */ - g_key_file_set_integer(Zathura.Bookmarks.data, Zathura.PDF.file, - bm_reserved_names[BM_PAGE_ENTRY], Zathura.PDF.page_number); - - /* set page offset */ - g_key_file_set_integer(Zathura.Bookmarks.data, Zathura.PDF.file, - bm_reserved_names[BM_PAGE_OFFSET], Zathura.PDF.page_offset); - } - - if (save_zoom_level) - { - /* set zoom level */ - g_key_file_set_integer(Zathura.Bookmarks.data, Zathura.PDF.file, - bm_reserved_names[BM_PAGE_SCALE], Zathura.PDF.scale); - } - - write_bookmarks_file(); - free_bookmarks(); - } - - /* inotify */ - if(!keep_monitor) - { - g_object_unref(Zathura.FileMonitor.monitor); - Zathura.FileMonitor.monitor = NULL; - - if(Zathura.FileMonitor.file) - { - g_object_unref(Zathura.FileMonitor.file); - Zathura.FileMonitor.file = NULL; - } - } - - /* reset values */ - g_free(Zathura.PDF.pages); - g_object_unref(Zathura.PDF.document); - g_free(Zathura.State.pages); - gtk_window_set_title(GTK_WINDOW(Zathura.UI.window), "zathura"); - - Zathura.State.pages = g_strdup(""); - g_free(Zathura.State.filename); - Zathura.State.filename = g_strdup((char*) default_text); - - g_static_mutex_lock(&(Zathura.Lock.pdf_obj_lock)); - Zathura.PDF.document = NULL; - - if(!keep_monitor) - { - g_free(Zathura.PDF.file); - g_free(Zathura.PDF.password); - Zathura.PDF.file = NULL; - Zathura.PDF.password = NULL; - Zathura.PDF.page_number = 0; - Zathura.PDF.scale = 0; - Zathura.PDF.rotate = 0; - } - Zathura.PDF.number_of_pages = 0; - Zathura.PDF.page_offset = 0; - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - - /* destroy index */ - if(Zathura.UI.index) - { - gtk_widget_destroy(Zathura.UI.index); - Zathura.UI.index = NULL; - } - - /* destroy information */ - if(Zathura.UI.information) - { - gtk_widget_destroy(Zathura.UI.information); - Zathura.UI.information = NULL; - } - - /* free markers */ - if(Zathura.Marker.markers) - free(Zathura.Marker.markers); - Zathura.Marker.number_of_markers = 0; - Zathura.Marker.last = -1; - - update_status(); -} - -void -enter_password(void) -{ - /* replace default inputbar handler */ - g_signal_handler_disconnect((gpointer) Zathura.UI.inputbar, Zathura.Handler.inputbar_activate); - Zathura.Handler.inputbar_activate = g_signal_connect(G_OBJECT(Zathura.UI.inputbar), "activate", G_CALLBACK(cb_inputbar_password_activate), NULL); - - Argument argument; - argument.data = "Enter password: "; - sc_focus_inputbar(&argument); -} - -void -eval_marker(int id) -{ - /* go to last marker */ - if(id == 0x27) - { - int current_page = Zathura.PDF.page_number; - set_page(Zathura.Marker.last); - Zathura.Marker.last = current_page; - return; - } - - /* search markers */ - int i; - for(i = 0; i < Zathura.Marker.number_of_markers; i++) - { - if(Zathura.Marker.markers[i].id == id) - { - set_page(Zathura.Marker.markers[i].page); - return; - } - } -} - -void -highlight_result(int page_id, PopplerRectangle* rectangle) -{ - PopplerRectangle* trect = poppler_rectangle_copy(rectangle); - cairo_t *cairo = cairo_create(Zathura.PDF.surface); - cairo_set_source_rgba(cairo, Zathura.Style.search_highlight.red, Zathura.Style.search_highlight.green, - Zathura.Style.search_highlight.blue, transparency); - - recalc_rectangle(page_id, trect); - cairo_rectangle(cairo, trect->x1, trect->y1, (trect->x2 - trect->x1), (trect->y2 - trect->y1)); - poppler_rectangle_free(trect); - cairo_fill(cairo); - cairo_destroy(cairo); -} - -void -notify(int level, const char* message) -{ - switch(level) - { - case ERROR: - gtk_widget_modify_base(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.notification_e_bg)); - gtk_widget_modify_text(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.notification_e_fg)); - break; - case WARNING: - gtk_widget_modify_base(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.notification_w_bg)); - gtk_widget_modify_text(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.notification_w_fg)); - break; - default: - gtk_widget_modify_base(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.inputbar_bg)); - gtk_widget_modify_text(GTK_WIDGET(Zathura.UI.inputbar), GTK_STATE_NORMAL, &(Zathura.Style.inputbar_fg)); - break; - } - - if(message) - gtk_entry_set_text(Zathura.UI.inputbar, message); -} - -gboolean -open_file(char* path, char* password) -{ - g_static_mutex_lock(&(Zathura.Lock.pdf_obj_lock)); - - /* specify path max */ - size_t pm; -#ifdef PATH_MAX - pm = PATH_MAX; -#else - pm = pathconf(path,_PC_PATH_MAX); - if(pm <= 0) - pm = 4096; -#endif - - char* rpath = NULL; - if(path[0] == '~') - { - gchar* home_path = get_home_dir(); - rpath = g_build_filename(home_path, path + 1, NULL); - g_free(home_path); - } - else - rpath = g_strdup(path); - - /* get filename */ - char* file = (char*) g_malloc0(sizeof(char) * pm); - if (!file) - out_of_memory(); - - if(!realpath(rpath, file)) - { - notify(ERROR, "File does not exist"); - g_free(file); - g_free(rpath); - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - return FALSE; - } - g_free(rpath); - - /* check if file exists */ - if(!g_file_test(file, G_FILE_TEST_IS_REGULAR)) - { - notify(ERROR, "File does not exist"); - g_free(file); - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - return FALSE; - } - - /* close old file */ - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - close_file(FALSE); - g_static_mutex_lock(&(Zathura.Lock.pdf_obj_lock)); - - /* format path */ - GError* error = NULL; - char* file_uri = g_filename_to_uri(file, NULL, &error); - if (!file_uri) - { - if(file) - g_free(file); - char* message = g_strdup_printf("Can not open file: %s", error->message); - notify(ERROR, message); - g_free(message); - g_error_free(error); - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - return FALSE; - } - - /* open file */ - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - Zathura.PDF.document = poppler_document_new_from_file(file_uri, password, &error); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if(!Zathura.PDF.document) - { - if(error->code == 1) - { - g_free(file_uri); - g_error_free(error); - Zathura.PDF.file = file; - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - enter_password(); - return FALSE; - } - else - { - char* message = g_strdup_printf("Can not open file: %s", error->message); - notify(ERROR, message); - g_free(file_uri); - g_free(message); - g_error_free(error); - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - return FALSE; - } - } - - /* save password */ - g_free(Zathura.PDF.password); - Zathura.PDF.password = password ? g_strdup(password) : NULL; - - /* inotify */ - if(!Zathura.FileMonitor.monitor) - { - GFile* file = g_file_new_for_uri(file_uri); - - if(file) - { - Zathura.FileMonitor.monitor = g_file_monitor_file(file, G_FILE_MONITOR_NONE, NULL, NULL); - if(Zathura.FileMonitor.monitor) - g_signal_connect(G_OBJECT(Zathura.FileMonitor.monitor), "changed", G_CALLBACK(cb_watch_file), NULL); - Zathura.FileMonitor.file = file; - } - } - - g_free(file_uri); - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - Zathura.PDF.number_of_pages = poppler_document_get_n_pages(Zathura.PDF.document); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - g_free(Zathura.PDF.file); - Zathura.PDF.file = file; - Zathura.PDF.scale = 100; - Zathura.PDF.rotate = 0; - if(Zathura.State.filename) - g_free(Zathura.State.filename); - Zathura.State.filename = g_markup_escape_text(file, -1); - Zathura.PDF.pages = g_malloc(Zathura.PDF.number_of_pages * sizeof(Page*)); - - if(!Zathura.PDF.pages) - out_of_memory(); - - /* get pages and check label mode */ - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - Zathura.Global.enable_labelmode = FALSE; - - int i; - for(i = 0; i < Zathura.PDF.number_of_pages; i++) - { - Zathura.PDF.pages[i] = malloc(sizeof(Page)); - if(!Zathura.PDF.pages[i]) - out_of_memory(); - - Zathura.PDF.pages[i]->id = i + 1; - Zathura.PDF.pages[i]->page = poppler_document_get_page(Zathura.PDF.document, i); - g_object_get(G_OBJECT(Zathura.PDF.pages[i]->page), "label", &(Zathura.PDF.pages[i]->label), NULL); - - /* check if it is necessary to use the label mode */ - int label_int = atoi(Zathura.PDF.pages[i]->label); - if(label_int == 0 || label_int != (i+1)) - Zathura.Global.enable_labelmode = TRUE; - } - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - /* set correct goto mode */ - if(!Zathura.Global.enable_labelmode && GOTO_MODE == GOTO_LABELS) - Zathura.Global.goto_mode = GOTO_DEFAULT; - - /* start page */ - int start_page = 0; - Zathura.PDF.page_offset = 0; - - /* bookmarks */ - if(Zathura.Bookmarks.data && g_key_file_has_group(Zathura.Bookmarks.data, file)) - { - /* get last opened page */ - if(save_position && g_key_file_has_key(Zathura.Bookmarks.data, file, - bm_reserved_names[BM_PAGE_ENTRY], NULL)) - start_page = g_key_file_get_integer(Zathura.Bookmarks.data, file, - bm_reserved_names[BM_PAGE_ENTRY], NULL); - - /* get page offset */ - if(save_position && g_key_file_has_key(Zathura.Bookmarks.data, file, - bm_reserved_names[BM_PAGE_OFFSET], NULL)) - Zathura.PDF.page_offset = g_key_file_get_integer(Zathura.Bookmarks.data, file, - bm_reserved_names[BM_PAGE_OFFSET], NULL); - if((Zathura.PDF.page_offset != 0) && (Zathura.PDF.page_offset != GOTO_OFFSET)) - Zathura.PDF.page_offset = GOTO_OFFSET; - - /* get zoom level */ - if (save_zoom_level && g_key_file_has_key(Zathura.Bookmarks.data, file, - bm_reserved_names[BM_PAGE_SCALE], NULL)) - { - Zathura.PDF.scale = g_key_file_get_integer(Zathura.Bookmarks.data, file, - bm_reserved_names[BM_PAGE_SCALE], NULL); - Zathura.Global.adjust_mode = ADJUST_NONE; - } - if (Zathura.PDF.scale > zoom_max) - Zathura.PDF.scale = zoom_max; - if (Zathura.PDF.scale < zoom_min) - Zathura.PDF.scale = zoom_min; - - /* open and read bookmark file */ - gsize i = 0; - gsize number_of_keys = 0; - char** keys = g_key_file_get_keys(Zathura.Bookmarks.data, file, &number_of_keys, NULL); - - for(i = 0; i < number_of_keys; i++) - { - if(!is_reserved_bm_name(keys[i])) - { - Zathura.Bookmarks.bookmarks = safe_realloc((void**)&Zathura.Bookmarks.bookmarks, - Zathura.Bookmarks.number_of_bookmarks + 1, sizeof(Bookmark)); - if(!Zathura.Bookmarks.bookmarks) - out_of_memory(); - - Zathura.Bookmarks.bookmarks[Zathura.Bookmarks.number_of_bookmarks].id = g_strdup(keys[i]); - Zathura.Bookmarks.bookmarks[Zathura.Bookmarks.number_of_bookmarks].page = - g_key_file_get_integer(Zathura.Bookmarks.data, file, keys[i], NULL); - - Zathura.Bookmarks.number_of_bookmarks++; - } - } - - g_strfreev(keys); - } - - /* set window title */ - gtk_window_set_title(GTK_WINDOW(Zathura.UI.window), basename(file)); - - /* show document */ - set_page(start_page); - update_status(); - - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - isc_abort(NULL); - return TRUE; -} - -gboolean -open_stdin(gchar* password) -{ - GError* error = NULL; - gchar* file = NULL; - gint handle = g_file_open_tmp("zathura.stdin.XXXXXX.pdf", &file, &error); - if (handle == -1) - { - gchar* message = g_strdup_printf("Can not create temporary file: %s", error->message); - notify(ERROR, message); - g_free(message); - g_error_free(error); - return FALSE; - } - - // read from stdin and dump to temporary file - int stdinfno = fileno(stdin); - if (stdinfno == -1) - { - gchar* message = g_strdup_printf("Can not read from stdin."); - notify(ERROR, message); - g_free(message); - close(handle); - g_unlink(file); - g_free(file); - return FALSE; - - } - - char buffer[BUFSIZ]; - ssize_t count = 0; - while ((count = read(stdinfno, buffer, BUFSIZ)) > 0) - { - if (write(handle, buffer, count) != count) - { - gchar* message = g_strdup_printf("Can not write to temporary file: %s", file); - notify(ERROR, message); - g_free(message); - close(handle); - g_unlink(file); - g_free(file); - return FALSE; - } - } - close(handle); - - if (count != 0) - { - gchar* message = g_strdup_printf("Can not read from stdin."); - notify(ERROR, message); - g_free(message); - g_unlink(file); - g_free(file); - return FALSE; - } - - /* update data */ - if (Zathura.StdinSupport.file) - g_unlink(Zathura.StdinSupport.file); - g_free(Zathura.StdinSupport.file); - Zathura.StdinSupport.file = file; - - return open_file(Zathura.StdinSupport.file, password); -} - -void open_uri(char* uri) -{ - char* escaped_uri = g_shell_quote(uri); - char* uri_cmd = g_strdup_printf(uri_command, escaped_uri); - system(uri_cmd); - g_free(uri_cmd); - g_free(escaped_uri); -} - -void out_of_memory(void) -{ - printf("error: out of memory\n"); - exit(-1); -} - -void* -safe_realloc(void** ptr, size_t nmemb, size_t size) -{ - static const size_t limit = ~((size_t)0u); - void* tmp = NULL; - - /* Check for overflow. */ - if(nmemb > limit / size) - goto failure; - - tmp = realloc(*ptr, nmemb * size); - /* Check for out of memory. */ - if(!tmp) - goto failure; - - *ptr = tmp; - return *ptr; - -/* Error handling. */ -failure: - free(*ptr); - *ptr = NULL; - return NULL; -} - -void -update_status(void) -{ - /* update text */ - gtk_label_set_markup((GtkLabel*) Zathura.Global.status_text, Zathura.State.filename); - - /* update pages */ - if( Zathura.PDF.document && Zathura.PDF.pages ) - { - int page = Zathura.PDF.page_number; - g_free(Zathura.State.pages); - - Zathura.State.pages = g_strdup_printf("[%i/%i]", page + 1, Zathura.PDF.number_of_pages); - } - - /* update state */ - char* zoom_level = (Zathura.PDF.scale != 0) ? g_strdup_printf("%d%%", Zathura.PDF.scale) : g_strdup(""); - char* goto_mode = (Zathura.Global.goto_mode == GOTO_LABELS) ? "L" : - (Zathura.Global.goto_mode == GOTO_OFFSET) ? "O" : "D"; - char* status_text = g_strdup_printf("%s [%s] %s (%d%%)", zoom_level, goto_mode, Zathura.State.pages, Zathura.State.scroll_percentage); - gtk_label_set_markup((GtkLabel*) Zathura.Global.status_state, status_text); - g_free(status_text); - g_free(zoom_level); -} - -void -read_bookmarks_file(void) -{ - /* free it at first */ - if (Zathura.Bookmarks.data) - g_key_file_free(Zathura.Bookmarks.data); - - /* create or open existing bookmark file */ - Zathura.Bookmarks.data = g_key_file_new(); - if(!g_file_test(Zathura.Bookmarks.file, G_FILE_TEST_IS_REGULAR)) - { - /* file does not exist */ - g_file_set_contents(Zathura.Bookmarks.file, "# Zathura bookmarks\n", -1, NULL); - } - - GError* error = NULL; - if(!g_key_file_load_from_file(Zathura.Bookmarks.data, Zathura.Bookmarks.file, - G_KEY_FILE_KEEP_COMMENTS | G_KEY_FILE_KEEP_TRANSLATIONS, &error)) - { - gchar* message = g_strdup_printf("Could not load bookmark file: %s", error->message); - notify(ERROR, message); - g_free(message); - } -} - -void -write_bookmarks_file(void) -{ - if (!Zathura.Bookmarks.data) - /* nothing to do */ - return; - - /* save bookmarks */ - for(int i = 0; i < Zathura.Bookmarks.number_of_bookmarks; i++) - { - g_key_file_set_integer(Zathura.Bookmarks.data, Zathura.PDF.file, - Zathura.Bookmarks.bookmarks[i].id, Zathura.Bookmarks.bookmarks[i].page); - } - - /* convert file and save it */ - gchar* bookmarks = g_key_file_to_data(Zathura.Bookmarks.data, NULL, NULL); - g_file_set_contents(Zathura.Bookmarks.file, bookmarks, -1, NULL); - g_free(bookmarks); -} - -void -free_bookmarks(void) -{ - for(int i = 0; i < Zathura.Bookmarks.number_of_bookmarks; i++) - { - g_free(Zathura.Bookmarks.bookmarks[i].id); - } - free(Zathura.Bookmarks.bookmarks); - Zathura.Bookmarks.bookmarks = NULL; - Zathura.Bookmarks.number_of_bookmarks = 0; -} - -void -read_configuration_file(const char* rcfile) -{ - if(!rcfile) - return; - - if(!g_file_test(rcfile, G_FILE_TEST_IS_REGULAR)) - return; - - char* content = NULL; - if(g_file_get_contents(rcfile, &content, NULL, NULL)) - { - gchar **lines = g_strsplit(content, "\n", -1); - int n = g_strv_length(lines) - 1; - - int i; - for(i = 0; i <= n; i++) - { - if(!strlen(lines[i])) - continue; - - gchar **pre_tokens = g_strsplit_set(lines[i], "\t ", -1); - int pre_length = g_strv_length(pre_tokens); - - gchar** tokens = g_malloc0(sizeof(gchar*) * (pre_length + 1)); - gchar** tokp = tokens; - int length = 0; - for (int i = 0; i != pre_length; ++i) { - if (strlen(pre_tokens[i])) { - *tokp++ = pre_tokens[i]; - ++length; - } - } - - if(!strcmp(tokens[0], "set")) - cmd_set(length - 1, tokens + 1); - else if(!strcmp(tokens[0], "map")) - cmd_map(length - 1, tokens + 1); - - g_free(tokens); - } - - g_strfreev(lines); - g_free(content); - } -} - -void -read_configuration(void) -{ - char* zathurarc = g_build_filename(Zathura.Config.config_dir, ZATHURA_RC, NULL); - read_configuration_file(GLOBAL_RC); - read_configuration_file(zathurarc); - g_free(zathurarc); -} - -void -recalc_rectangle(int page_id, PopplerRectangle* rectangle) -{ - double page_width, page_height; - double x1 = rectangle->x1; - double x2 = rectangle->x2; - double y1 = rectangle->y1; - double y2 = rectangle->y2; - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_page_get_size(Zathura.PDF.pages[page_id]->page, &page_width, &page_height); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - double scale = ((double) Zathura.PDF.scale / 100.0); - - int rotate = Zathura.PDF.rotate; - - switch(rotate) - { - case 90: - rectangle->x1 = y2 * scale; - rectangle->y1 = x1 * scale; - rectangle->x2 = y1 * scale; - rectangle->y2 = x2 * scale; - break; - case 180: - rectangle->x1 = (page_width - x2) * scale; - rectangle->y1 = y2 * scale; - rectangle->x2 = (page_width - x1) * scale; - rectangle->y2 = y1 * scale; - break; - case 270: - rectangle->x1 = (page_height - y1) * scale; - rectangle->y1 = (page_width - x2) * scale; - rectangle->x2 = (page_height - y2) * scale; - rectangle->y2 = (page_width - x1) * scale; - break; - default: - rectangle->x1 = x1 * scale; - rectangle->y1 = (page_height - y1) * scale; - rectangle->x2 = x2 * scale; - rectangle->y2 = (page_height - y2) * scale; - } -} - -GtkEventBox* -create_completion_row(GtkBox* results, char* command, char* description, gboolean group) -{ - GtkBox *col = GTK_BOX(gtk_hbox_new(FALSE, 0)); - GtkEventBox *row = GTK_EVENT_BOX(gtk_event_box_new()); - - GtkLabel *show_command = GTK_LABEL(gtk_label_new(NULL)); - GtkLabel *show_description = GTK_LABEL(gtk_label_new(NULL)); - - gtk_misc_set_alignment(GTK_MISC(show_command), 0.0, 0.0); - gtk_misc_set_alignment(GTK_MISC(show_description), 0.0, 0.0); - - if(group) - { - gtk_misc_set_padding(GTK_MISC(show_command), 2.0, 4.0); - gtk_misc_set_padding(GTK_MISC(show_description), 2.0, 4.0); - } - else - { - gtk_misc_set_padding(GTK_MISC(show_command), 1.0, 1.0); - gtk_misc_set_padding(GTK_MISC(show_description), 1.0, 1.0); - } - - gtk_label_set_use_markup(show_command, TRUE); - gtk_label_set_use_markup(show_description, TRUE); - - gchar* c = g_markup_printf_escaped(FORMAT_COMMAND, command ? command : ""); - gchar* d = g_markup_printf_escaped(FORMAT_DESCRIPTION, description ? description : ""); - gtk_label_set_markup(show_command, c); - gtk_label_set_markup(show_description, d); - g_free(c); - g_free(d); - - if(group) - { - gtk_widget_modify_fg(GTK_WIDGET(show_command), GTK_STATE_NORMAL, &(Zathura.Style.completion_g_fg)); - gtk_widget_modify_fg(GTK_WIDGET(show_description), GTK_STATE_NORMAL, &(Zathura.Style.completion_g_fg)); - gtk_widget_modify_bg(GTK_WIDGET(row), GTK_STATE_NORMAL, &(Zathura.Style.completion_g_bg)); - } - else - { - gtk_widget_modify_fg(GTK_WIDGET(show_command), GTK_STATE_NORMAL, &(Zathura.Style.completion_fg)); - gtk_widget_modify_fg(GTK_WIDGET(show_description), GTK_STATE_NORMAL, &(Zathura.Style.completion_fg)); - gtk_widget_modify_bg(GTK_WIDGET(row), GTK_STATE_NORMAL, &(Zathura.Style.completion_bg)); - } - - gtk_widget_modify_font(GTK_WIDGET(show_command), Zathura.Style.font); - gtk_widget_modify_font(GTK_WIDGET(show_description), Zathura.Style.font); - - gtk_box_pack_start(GTK_BOX(col), GTK_WIDGET(show_command), TRUE, TRUE, 2); - gtk_box_pack_start(GTK_BOX(col), GTK_WIDGET(show_description), FALSE, FALSE, 2); - - gtk_container_add(GTK_CONTAINER(row), GTK_WIDGET(col)); - - gtk_box_pack_start(results, GTK_WIDGET(row), FALSE, FALSE, 0); - - return row; -} - -void -set_completion_row_color(GtkBox* results, int mode, int id) -{ - GtkEventBox *row = (GtkEventBox*) g_list_nth_data(gtk_container_get_children(GTK_CONTAINER(results)), id); - - if(row) - { - GtkBox *col = (GtkBox*) g_list_nth_data(gtk_container_get_children(GTK_CONTAINER(row)), 0); - GtkLabel *cmd = (GtkLabel*) g_list_nth_data(gtk_container_get_children(GTK_CONTAINER(col)), 0); - GtkLabel *cdesc = (GtkLabel*) g_list_nth_data(gtk_container_get_children(GTK_CONTAINER(col)), 1); - - if(mode == NORMAL) - { - gtk_widget_modify_fg(GTK_WIDGET(cmd), GTK_STATE_NORMAL, &(Zathura.Style.completion_fg)); - gtk_widget_modify_fg(GTK_WIDGET(cdesc), GTK_STATE_NORMAL, &(Zathura.Style.completion_fg)); - gtk_widget_modify_bg(GTK_WIDGET(row), GTK_STATE_NORMAL, &(Zathura.Style.completion_bg)); - } - else - { - gtk_widget_modify_fg(GTK_WIDGET(cmd), GTK_STATE_NORMAL, &(Zathura.Style.completion_hl_fg)); - gtk_widget_modify_fg(GTK_WIDGET(cdesc), GTK_STATE_NORMAL, &(Zathura.Style.completion_hl_fg)); - gtk_widget_modify_bg(GTK_WIDGET(row), GTK_STATE_NORMAL, &(Zathura.Style.completion_hl_bg)); - } - } -} - -void -set_page(int page) -{ - if(page >= Zathura.PDF.number_of_pages || page < 0) - { - notify(WARNING, "Could not open page"); - return; - } - - Zathura.PDF.page_number = page; - Zathura.Search.draw = FALSE; - - Argument argument; - argument.n = TOP; - - switch_view(Zathura.UI.document); - draw(page); - sc_scroll(&argument); -} - -void -switch_view(GtkWidget* widget) -{ - GtkWidget* child = gtk_bin_get_child(GTK_BIN(Zathura.UI.viewport)); - if(child == widget) - return; - if(child) - { - g_object_ref(child); - gtk_container_remove(GTK_CONTAINER(Zathura.UI.viewport), child); - } - - gtk_container_add(GTK_CONTAINER(Zathura.UI.viewport), GTK_WIDGET(widget)); -} - -Completion* -completion_init(void) -{ - Completion *completion = malloc(sizeof(Completion)); - if(!completion) - out_of_memory(); - - completion->groups = NULL; - - return completion; -} - -CompletionGroup* -completion_group_create(char* name) -{ - CompletionGroup* group = malloc(sizeof(CompletionGroup)); - if(!group) - out_of_memory(); - - group->value = name ? g_strdup(name) : NULL; - group->elements = NULL; - group->next = NULL; - - return group; -} - -void -completion_add_group(Completion* completion, CompletionGroup* group) -{ - CompletionGroup* cg = completion->groups; - - while(cg && cg->next) - cg = cg->next; - - if(cg) - cg->next = group; - else - completion->groups = group; -} - -void completion_free(Completion* completion) -{ - CompletionGroup* group = completion->groups; - CompletionElement *element; - - while(group) - { - element = group->elements; - while(element) - { - CompletionElement* ne = element->next; - g_free(element->value); - g_free(element->description); - free(element); - element = ne; - } - - CompletionGroup *ng = group->next; - g_free(group->value); - free(group); - group = ng; - } - free(completion); -} - -void completion_group_add_element(CompletionGroup* group, char* name, char* description) -{ - CompletionElement* el = group->elements; - - while(el && el->next) - el = el->next; - - CompletionElement* new_element = malloc(sizeof(CompletionElement)); - if(!new_element) - out_of_memory(); - - new_element->value = name ? g_strdup(name) : NULL; - new_element->description = description ? g_strdup(description) : NULL; - new_element->next = NULL; - - if(el) - el->next = new_element; - else - group->elements = new_element; -} - - -/* thread implementation */ -void* -search(void* parameter) -{ - Argument* argument = (Argument*) parameter; - - static char* search_item; - static int direction; - static int next_page = 0; - gchar* old_query = NULL; - GList* results = NULL; - - if(argument->n != NO_SEARCH) - { - /* search document */ - if(argument->n) - direction = (argument->n == BACKWARD) ? -1 : 1; - - if(argument->data) - { - if(search_item) - g_free(search_item); - - search_item = g_strdup((char*) argument->data); - } - g_free(argument->data); - g_free(argument); - - g_static_mutex_lock(&(Zathura.Lock.pdf_obj_lock)); - if(!Zathura.PDF.document || !search_item || !strlen(search_item)) - { - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - g_static_mutex_lock(&(Zathura.Lock.search_lock)); - Zathura.Thread.search_thread_running = FALSE; - g_static_mutex_unlock(&(Zathura.Lock.search_lock)); - g_thread_exit(NULL); - } - - old_query = Zathura.Search.query; - - /* delete old results */ - if(Zathura.Search.results) - { - g_list_free(Zathura.Search.results); - Zathura.Search.results = NULL; - } - - Zathura.Search.query = g_strdup(search_item); - - int number_of_pages = Zathura.PDF.number_of_pages; - int page_number = Zathura.PDF.page_number; - - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - - int page_counter = (g_strcmp0(old_query,search_item) == 0) ? 1 : 0; - for( ; page_counter <= number_of_pages; page_counter++) - { - g_static_mutex_lock(&(Zathura.Lock.search_lock)); - if(Zathura.Thread.search_thread_running == FALSE) - { - g_static_mutex_unlock(&(Zathura.Lock.search_lock)); - g_free(old_query); - g_thread_exit(NULL); - } - g_static_mutex_unlock(&(Zathura.Lock.search_lock)); - - next_page = (number_of_pages + page_number + - page_counter * direction) % number_of_pages; - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - PopplerPage* page = poppler_document_get_page(Zathura.PDF.document, next_page); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if(!page) - { - g_free(old_query); - g_thread_exit(NULL); - } - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - results = poppler_page_find_text(page, search_item); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - g_object_unref(page); - - if(results) - break; - } - } - else - { - Zathura.Search.draw = TRUE; - g_free(argument->data); - g_free(argument); - } - - /* draw results */ - if(results) - { - gdk_threads_enter(); - - set_page(next_page); - - if(Zathura.Search.results) - g_list_free(Zathura.Search.results); - - Zathura.Search.results = results; - Zathura.Search.page = next_page; - Zathura.Search.draw = TRUE; - Zathura.Search.query = g_strdup(search_item); - - gdk_threads_leave(); - } - - g_static_mutex_lock(&(Zathura.Lock.search_lock)); - Zathura.Thread.search_thread_running = FALSE; - g_static_mutex_unlock(&(Zathura.Lock.search_lock)); - - g_free(old_query); - g_thread_exit(NULL); - return NULL; -} - -/* shortcut implementation */ -void -sc_abort(Argument* argument) -{ - /* Clear buffer */ - if(Zathura.Global.buffer) - { - g_string_free(Zathura.Global.buffer, TRUE); - Zathura.Global.buffer = NULL; - gtk_label_set_markup((GtkLabel*) Zathura.Global.status_buffer, ""); - } - - if(!Zathura.Global.show_inputbar) - gtk_widget_hide(GTK_WIDGET(Zathura.UI.inputbar)); - - /* Set back to normal mode */ - change_mode(NORMAL); - switch_view(Zathura.UI.document); -} - -void -sc_adjust_window(Argument* argument) -{ - if(!Zathura.PDF.document) - return; - - Zathura.Global.adjust_mode = argument->n; - - GtkAdjustment* adjustment; - double view_size; - double page_width; - double page_height; - - if(argument->n == ADJUST_BESTFIT) - adjustment = gtk_scrolled_window_get_vadjustment(Zathura.UI.view); - else if(argument->n == ADJUST_WIDTH) - adjustment = gtk_scrolled_window_get_hadjustment(Zathura.UI.view); - else - return; - - view_size = gtk_adjustment_get_page_size(adjustment); - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_page_get_size(Zathura.PDF.pages[Zathura.PDF.page_number]->page, &page_width, &page_height); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if ((Zathura.PDF.rotate == 90) || (Zathura.PDF.rotate == 270)) - { - double swap = page_width; - page_width = page_height; - page_height = swap; - } - - if(argument->n == ADJUST_BESTFIT) - Zathura.PDF.scale = (view_size / page_height) * 100; - else - Zathura.PDF.scale = (view_size / page_width) * 100; - - draw(Zathura.PDF.page_number); - update_status(); -} - -void -sc_change_buffer(Argument* argument) -{ - if(!Zathura.Global.buffer) - return; - - int buffer_length = Zathura.Global.buffer->len; - - if(argument->n == DELETE_LAST) - { - if((buffer_length - 1) == 0) - { - g_string_free(Zathura.Global.buffer, TRUE); - Zathura.Global.buffer = NULL; - gtk_label_set_markup((GtkLabel*) Zathura.Global.status_buffer, ""); - } - else - { - GString* temp = g_string_new_len(Zathura.Global.buffer->str, buffer_length - 1); - g_string_free(Zathura.Global.buffer, TRUE); - Zathura.Global.buffer = temp; - gtk_label_set_markup((GtkLabel*) Zathura.Global.status_buffer, Zathura.Global.buffer->str); - } - } -} - -void -sc_change_mode(Argument* argument) -{ - if(argument) - change_mode(argument->n); -} - -void -sc_focus_inputbar(Argument* argument) -{ - if(!(GTK_WIDGET_VISIBLE(GTK_WIDGET(Zathura.UI.inputbar)))) - gtk_widget_show(GTK_WIDGET(Zathura.UI.inputbar)); - - if(argument->data) - { - char* data = argument->data; - - if(argument->n == APPEND_FILEPATH) - data = g_strdup_printf("%s%s", data, Zathura.PDF.file); - else - data = g_strdup(data); - - notify(DEFAULT, data); - g_free(data); - gtk_widget_grab_focus(GTK_WIDGET(Zathura.UI.inputbar)); - gtk_editable_set_position(GTK_EDITABLE(Zathura.UI.inputbar), -1); - } -} - -void -sc_follow(Argument* argument) -{ - if(!Zathura.PDF.document) - return; - - Page* current_page = Zathura.PDF.pages[Zathura.PDF.page_number]; - int link_id = 1; - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - GList *link_list = poppler_page_get_link_mapping(current_page->page); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - link_list = g_list_reverse(link_list); - - if(g_list_length(link_list) <= 0) - return; - - GList *links; - for(links = link_list; links; links = g_list_next(links)) - { - PopplerLinkMapping *link_mapping = (PopplerLinkMapping*) links->data; - PopplerRectangle* link_rectangle = &link_mapping->area; - PopplerAction *action = link_mapping->action; - - /* only handle URI and internal links */ - if(action->type == POPPLER_ACTION_URI || action->type == POPPLER_ACTION_GOTO_DEST) - { - highlight_result(Zathura.PDF.page_number, link_rectangle); - - /* draw text */ - recalc_rectangle(Zathura.PDF.page_number, link_rectangle); - cairo_t *cairo = cairo_create(Zathura.PDF.surface); - cairo_select_font_face(cairo, font, CAIRO_FONT_SLANT_NORMAL, CAIRO_FONT_WEIGHT_BOLD); - cairo_set_font_size(cairo, 10); - cairo_move_to(cairo, link_rectangle->x1 + 1, link_rectangle->y1 - 1); - char* link_number = g_strdup_printf("%i", link_id++); - cairo_show_text(cairo, link_number); - cairo_destroy(cairo); - g_free(link_number); - } - } - - gtk_widget_queue_draw(Zathura.UI.drawing_area); - poppler_page_free_link_mapping(link_list); - - /* replace default inputbar handler */ - g_signal_handler_disconnect((gpointer) Zathura.UI.inputbar, Zathura.Handler.inputbar_activate); - Zathura.Handler.inputbar_activate = g_signal_connect(G_OBJECT(Zathura.UI.inputbar), "activate", G_CALLBACK(cb_inputbar_form_activate), NULL); - - argument->data = "Follow hint: "; - sc_focus_inputbar(argument); -} - -void -sc_navigate(Argument* argument) -{ - if(!Zathura.PDF.document) - return; - - int number_of_pages = Zathura.PDF.number_of_pages; - int new_page = Zathura.PDF.page_number; - - if(argument->n == NEXT) - new_page = scroll_wrap ? ((new_page + 1) % number_of_pages) : (new_page + 1); - else if(argument->n == PREVIOUS) - new_page = scroll_wrap ? ((new_page + number_of_pages - 1) % number_of_pages) : (new_page - 1); - - if (!scroll_wrap && (new_page < 0 || new_page >= number_of_pages)) - return; - - set_page(new_page); - update_status(); -} - -void -sc_recolor(Argument* argument) -{ - Zathura.Global.recolor = !Zathura.Global.recolor; - draw(Zathura.PDF.page_number); -} - -void -sc_reload(Argument* argument) -{ - draw(Zathura.PDF.page_number); - - GtkAdjustment* vadjustment = gtk_scrolled_window_get_vadjustment(Zathura.UI.view); - GtkAdjustment* hadjustment = gtk_scrolled_window_get_hadjustment(Zathura.UI.view); - - /* save old information */ - g_static_mutex_lock(&(Zathura.Lock.pdf_obj_lock)); - char* path = Zathura.PDF.file ? strdup(Zathura.PDF.file) : NULL; - char* password = Zathura.PDF.password ? strdup(Zathura.PDF.password) : NULL; - int scale = Zathura.PDF.scale; - int page = Zathura.PDF.page_number; - int rotate = Zathura.PDF.rotate; - gdouble va = gtk_adjustment_get_value(vadjustment); - gdouble ha = gtk_adjustment_get_value(hadjustment); - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - - /* reopen and restore settings */ - close_file(TRUE); - open_file(path, password); - - g_static_mutex_lock(&(Zathura.Lock.pdf_obj_lock)); - Zathura.PDF.scale = scale; - Zathura.PDF.rotate = rotate; - gtk_adjustment_set_value(vadjustment, va); - gtk_adjustment_set_value(hadjustment, ha); - g_static_mutex_unlock(&(Zathura.Lock.pdf_obj_lock)); - - if (Zathura.PDF.number_of_pages != 0) { - if (page >= Zathura.PDF.number_of_pages - 1) - page = Zathura.PDF.number_of_pages - 1; - Zathura.PDF.page_number = page; - draw(Zathura.PDF.page_number); - } - - if(path) - free(path); - if(password) - free(password); -} - - -void -sc_rotate(Argument* argument) -{ - Zathura.PDF.rotate = (Zathura.PDF.rotate + 90) % 360; - Zathura.Search.draw = TRUE; - - draw(Zathura.PDF.page_number); -} - -void -sc_scroll(Argument* argument) -{ - GtkAdjustment* adjustment; - - if( (argument->n == LEFT) || (argument->n == RIGHT) ) - adjustment = gtk_scrolled_window_get_hadjustment(Zathura.UI.view); - else - adjustment = gtk_scrolled_window_get_vadjustment(Zathura.UI.view); - - gdouble view_size = gtk_adjustment_get_page_size(adjustment); - gdouble value = gtk_adjustment_get_value(adjustment); - gdouble max = gtk_adjustment_get_upper(adjustment) - view_size; - static gboolean ss = FALSE; - - if((argument->n == UP || argument->n == HALF_UP || argument->n == FULL_UP) && value == 0) - { - int old_page = Zathura.PDF.page_number; - Argument arg; - arg.n = PREVIOUS; - sc_navigate(&arg); - if (scroll_wrap || (Zathura.PDF.page_number < old_page)) - { - arg.n = BOTTOM; - ss = TRUE; - sc_scroll(&arg); - } - return; - } - else if((argument->n == DOWN || argument->n == HALF_DOWN || argument->n == FULL_DOWN) && value == max) - { - Argument arg; - arg.n = NEXT; - ss = TRUE; - sc_navigate(&arg); - return; - } - - gdouble new_value; - if(argument->n == FULL_UP) - new_value = (value - view_size) < 0 ? 0 : (value - view_size); - else if(argument->n == FULL_DOWN) - new_value = (value + view_size) > max ? max : (value + view_size); - else if(argument->n == HALF_UP) - new_value = (value - (view_size / 2)) < 0 ? 0 : (value - (view_size / 2)); - else if(argument->n == HALF_DOWN) - new_value = (value + (view_size / 2)) > max ? max : (value + (view_size / 2)); - else if((argument->n == LEFT) || (argument->n == UP)) - new_value = (value - scroll_step) < 0 ? 0 : (value - scroll_step); - else if(argument->n == TOP) - new_value = 0; - else if(argument->n == BOTTOM) - new_value = max; - else - new_value = (value + scroll_step) > max ? max : (value + scroll_step); - - if( !((argument->n == LEFT) || (argument->n == RIGHT)) ) - Zathura.State.scroll_percentage = max == 0 ? 0 : (new_value*100/max); - - if(smooth_scrolling && !ss) - { - gdouble i; - if(new_value > value) - for(i = value; (i + smooth_scrolling) < new_value; i += smooth_scrolling) - gtk_adjustment_set_value(adjustment, i); - else - for(i = value; (i + smooth_scrolling) > new_value; i -= smooth_scrolling) - gtk_adjustment_set_value(adjustment, i); - } - - gtk_adjustment_set_value(adjustment, new_value); - ss = FALSE; - - update_status(); -} - -void -sc_search(Argument* argument) -{ - g_static_mutex_lock(&(Zathura.Lock.search_lock)); - if(Zathura.Thread.search_thread_running) - { - Zathura.Thread.search_thread_running = FALSE; - g_static_mutex_unlock(&(Zathura.Lock.search_lock)); - gdk_threads_leave(); - g_thread_join(Zathura.Thread.search_thread); - gdk_threads_enter(); - g_static_mutex_lock(&(Zathura.Lock.search_lock)); - } - - Argument* newarg = g_malloc0(sizeof(Argument)); - newarg->n = argument->n; - newarg->data = argument->data ? g_strdup(argument->data) : NULL; - Zathura.Thread.search_thread_running = TRUE; - Zathura.Thread.search_thread = g_thread_create(search, (gpointer) newarg, TRUE, NULL); - g_static_mutex_unlock(&(Zathura.Lock.search_lock)); -} - -void -sc_switch_goto_mode(Argument* argument) -{ - switch(Zathura.Global.goto_mode) - { - case GOTO_LABELS: - Zathura.Global.goto_mode = GOTO_OFFSET; - break; - case GOTO_OFFSET: - Zathura.Global.goto_mode = GOTO_DEFAULT; - break; - default: - if(Zathura.Global.enable_labelmode) - Zathura.Global.goto_mode = GOTO_LABELS; - else - Zathura.Global.goto_mode = GOTO_OFFSET; - break; - } - - update_status(); -} - -gboolean cb_index_row_activated(GtkTreeView* treeview, GtkTreePath* path, - GtkTreeViewColumn* column, gpointer user_data) -{ - GtkTreeModel *model; - GtkTreeIter iter; - - g_object_get(treeview, "model", &model, NULL); - - if(gtk_tree_model_get_iter(model, &iter, path)) - { - PopplerAction* action; - PopplerDest* destination; - - gtk_tree_model_get(model, &iter, 1, &action, -1); - if(!action) - return TRUE; - - if(action->type == POPPLER_ACTION_GOTO_DEST) - { - destination = action->goto_dest.dest; - int page_number = destination->page_num; - - if(action->goto_dest.dest->type == POPPLER_DEST_NAMED) - { - PopplerDest* d = poppler_document_find_dest(Zathura.PDF.document, action->goto_dest.dest->named_dest); - if(d) - { - page_number = d->page_num; - poppler_dest_free(d); - } - } - - set_page(page_number - 1); - update_status(); - Zathura.Global.show_index = FALSE; - gtk_widget_grab_focus(GTK_WIDGET(Zathura.UI.document)); - } - } - - Zathura.Global.mode = NORMAL; - g_object_unref(model); - - return TRUE; -} - -void -sc_navigate_index(Argument* argument) -{ - if(!Zathura.UI.index) - return; - - GtkTreeView *treeview = gtk_container_get_children(GTK_CONTAINER(Zathura.UI.index))->data; - GtkTreePath *path; - - gtk_tree_view_get_cursor(treeview, &path, NULL); - if(!path) - return; - - GtkTreeModel *model = gtk_tree_view_get_model(treeview); - GtkTreeIter iter; - GtkTreeIter child_iter; - - gboolean is_valid_path = TRUE; - - switch(argument->n) - { - case UP: - if(!gtk_tree_path_prev(path)) - is_valid_path = (gtk_tree_path_get_depth(path) > 1) && gtk_tree_path_up(path) ; - else /* row above */ - { - while(gtk_tree_view_row_expanded(treeview, path)) { - gtk_tree_model_get_iter(model, &iter, path); - /* select last child */ - gtk_tree_model_iter_nth_child(model, &child_iter, &iter, - gtk_tree_model_iter_n_children(model, &iter)-1); - gtk_tree_path_free(path); - path = gtk_tree_model_get_path(model, &child_iter); - } - } - break; - case COLLAPSE: - if(!gtk_tree_view_collapse_row(treeview, path) - && gtk_tree_path_get_depth(path) > 1) - { - gtk_tree_path_up(path); - gtk_tree_view_collapse_row(treeview, path); - } - break; - case DOWN: - if(gtk_tree_view_row_expanded(treeview, path)) - gtk_tree_path_down(path); - else - { - do - { - gtk_tree_model_get_iter(model, &iter, path); - if (gtk_tree_model_iter_next(model, &iter)) - { - path = gtk_tree_model_get_path(model, &iter); - break; - } - } - while((is_valid_path = (gtk_tree_path_get_depth(path) > 1)) - && gtk_tree_path_up(path)); - } - break; - case EXPAND: - if(gtk_tree_view_expand_row(treeview, path, FALSE)) - gtk_tree_path_down(path); - break; - case SELECT: - cb_index_row_activated(treeview, path, NULL, NULL); - return; - } - - if (is_valid_path ) - gtk_tree_view_set_cursor(treeview, path, NULL, FALSE); - - gtk_tree_path_free(path); -} - -void -sc_toggle_index(Argument* argument) -{ - if(!Zathura.PDF.document) - return; - - GtkWidget *treeview; - GtkTreeModel *model; - GtkCellRenderer *renderer; - PopplerIndexIter *iter; - - if(!Zathura.UI.index) - { - Zathura.UI.index = gtk_scrolled_window_new (NULL, NULL); - gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(Zathura.UI.index), - GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); - - if((iter = poppler_index_iter_new(Zathura.PDF.document))) - { - model = GTK_TREE_MODEL(gtk_tree_store_new(2, G_TYPE_STRING, G_TYPE_POINTER)); - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - build_index(model, NULL, iter); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - poppler_index_iter_free(iter); - } - else - { - notify(WARNING, "This document does not contain any index"); - Zathura.UI.index = NULL; - return; - } - - treeview = gtk_tree_view_new_with_model (model); - g_object_unref(model); - renderer = gtk_cell_renderer_text_new(); - gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW (treeview), 0, "Title", - renderer, "markup", 0, NULL); - gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), FALSE); - g_object_set(G_OBJECT(renderer), "ellipsize", PANGO_ELLIPSIZE_END, NULL); - g_object_set(G_OBJECT(gtk_tree_view_get_column(GTK_TREE_VIEW(treeview), 0)), "expand", TRUE, NULL); - - gtk_tree_view_set_cursor(GTK_TREE_VIEW(treeview), gtk_tree_path_new_first(), NULL, FALSE); - g_signal_connect(G_OBJECT(treeview), "row-activated", G_CALLBACK(cb_index_row_activated), NULL); - - gtk_container_add (GTK_CONTAINER (Zathura.UI.index), treeview); - gtk_widget_show (treeview); - gtk_widget_show(Zathura.UI.index); - } - - if(!Zathura.Global.show_index) - { - switch_view(Zathura.UI.index); - Zathura.Global.mode = INDEX; - } - else - { - switch_view(Zathura.UI.document); - Zathura.Global.mode = NORMAL; - } - - Zathura.Global.show_index = !Zathura.Global.show_index; -} - -void -sc_toggle_inputbar(Argument* argument) -{ - if(GTK_WIDGET_VISIBLE(GTK_WIDGET(Zathura.UI.inputbar))) - gtk_widget_hide(GTK_WIDGET(Zathura.UI.inputbar)); - else - gtk_widget_show(GTK_WIDGET(Zathura.UI.inputbar)); -} - -void -sc_toggle_fullscreen(Argument* argument) -{ - static gboolean fs = TRUE; - - if(fs) - { - gtk_window_fullscreen(GTK_WINDOW(Zathura.UI.window)); - gtk_widget_hide(GTK_WIDGET(Zathura.UI.inputbar)); - gtk_widget_hide(GTK_WIDGET(Zathura.UI.statusbar)); - - Argument arg; - arg.n = ADJUST_BESTFIT; - sc_adjust_window(&arg); - - Zathura.Global.mode = FULLSCREEN; - fs = FALSE; - } - else - { - gtk_window_unfullscreen(GTK_WINDOW(Zathura.UI.window)); - gtk_widget_show(GTK_WIDGET(Zathura.UI.inputbar)); - gtk_widget_show(GTK_WIDGET(Zathura.UI.statusbar)); - - Zathura.Global.mode = NORMAL; - fs = TRUE; - } - isc_abort(NULL); -} - -void -sc_toggle_statusbar(Argument* argument) -{ - if(GTK_WIDGET_VISIBLE(GTK_WIDGET(Zathura.UI.statusbar))) - gtk_widget_hide(GTK_WIDGET(Zathura.UI.statusbar)); - else - gtk_widget_show(GTK_WIDGET(Zathura.UI.statusbar)); -} - -void -sc_quit(Argument* argument) -{ - cb_destroy(NULL, NULL); -} - -void -sc_zoom(Argument* argument) -{ - bcmd_zoom(NULL, argument); -} - -/* inputbar shortcut declarations */ -void -isc_abort(Argument* argument) -{ - Argument arg = { HIDE }; - isc_completion(&arg); - - notify(DEFAULT, ""); - change_mode(NORMAL); - gtk_widget_grab_focus(GTK_WIDGET(Zathura.UI.view)); - - if(!Zathura.Global.show_inputbar) - gtk_widget_hide(GTK_WIDGET(Zathura.UI.inputbar)); - - /* replace default inputbar handler */ - g_signal_handler_disconnect((gpointer) Zathura.UI.inputbar, Zathura.Handler.inputbar_activate); - Zathura.Handler.inputbar_activate = g_signal_connect(G_OBJECT(Zathura.UI.inputbar), "activate", G_CALLBACK(cb_inputbar_activate), NULL); - sc_abort(NULL); -} - -void -isc_command_history(Argument* argument) -{ - static int current = 0; - int length = g_list_length(Zathura.Global.history); - - if(length > 0) - { - if(argument->n == NEXT) - current = (length + current + 1) % length; - else - current = (length + current - 1) % length; - - gchar* command = (gchar*) g_list_nth_data(Zathura.Global.history, current); - notify(DEFAULT, command); - gtk_widget_grab_focus(GTK_WIDGET(Zathura.UI.inputbar)); - gtk_editable_set_position(GTK_EDITABLE(Zathura.UI.inputbar), -1); - } -} - -void -isc_completion(Argument* argument) -{ - gchar *input = gtk_editable_get_chars(GTK_EDITABLE(Zathura.UI.inputbar), 1, -1); - gchar *tmp_string = gtk_editable_get_chars(GTK_EDITABLE(Zathura.UI.inputbar), 0, 1); - gchar identifier = tmp_string[0]; - int length = strlen(input); - - if(!input || !tmp_string) - { - if(input) - g_free(input); - if(tmp_string) - g_free(tmp_string); - return; - } - - /* get current information*/ - char* first_space = strstr(input, " "); - char* current_command; - char* current_parameter; - int current_command_length; - - if(!first_space) - { - current_command = g_strdup(input); - current_command_length = length; - current_parameter = NULL; - } - else - { - int offset = abs(input - first_space); - current_command = g_strndup(input, offset); - current_command_length = strlen(current_command); - current_parameter = input + offset + 1; - } - - /* if the identifier does not match the command sign and - * the completion should not be hidden, leave this function */ - if((identifier != ':') && (argument->n != HIDE)) - { - if(current_command) - g_free(current_command); - if(input) - g_free(input); - if(tmp_string) - g_free(tmp_string); - return; - } - - /* static elements */ - static GtkBox *results = NULL; - static CompletionRow *rows = NULL; - - static int current_item = 0; - static int n_items = 0; - - static char *previous_command = NULL; - static char *previous_parameter = NULL; - static int previous_id = 0; - static int previous_length = 0; - - static gboolean command_mode = TRUE; - - /* delete old list iff - * the completion should be hidden - * the current command differs from the previous one - * the current parameter differs from the previous one - */ - if( (argument->n == HIDE) || - (current_parameter && previous_parameter && strcmp(current_parameter, previous_parameter)) || - (current_command && previous_command && strcmp(current_command, previous_command)) || - (previous_length != length) - ) - { - if(results) - gtk_widget_destroy(GTK_WIDGET(results)); - - results = NULL; - - if(rows) - { - for(int i = 0; i != n_items; ++i) - { - g_free(rows[i].command); - g_free(rows[i].description); - } - free(rows); - } - - rows = NULL; - current_item = 0; - n_items = 0; - command_mode = TRUE; - - if(argument->n == HIDE) - { - if(current_command) - g_free(current_command); - if(input) - g_free(input); - if(tmp_string) - g_free(tmp_string); - return; - } - } - - /* create new list iff - * there is no current list - * the current command differs from the previous one - * the current parameter differs from the previous one - */ - if( (!results) ) - { - results = GTK_BOX(gtk_vbox_new(FALSE, 0)); - - /* create list based on parameters iff - * there is a current parameter given - * there is an old list with commands - * the current command does not differ from the previous one - * the current command has an completion function - */ - if(strchr(input, ' ')) - { - gboolean search_matching_command = FALSE; - - int i; - for(i = 0; i < LENGTH(commands); i++) - { - int abbr_length = commands[i].abbr ? strlen(commands[i].abbr) : 0; - int cmd_length = commands[i].command ? strlen(commands[i].command) : 0; - - if( ((current_command_length <= cmd_length) && !strncmp(current_command, commands[i].command, current_command_length)) || - ((current_command_length <= abbr_length) && !strncmp(current_command, commands[i].abbr, current_command_length)) - ) - { - if(commands[i].completion) - { - previous_command = current_command; - previous_id = i; - search_matching_command = TRUE; - } - else - { - if(current_command) - g_free(current_command); - if(input) - g_free(input); - if(tmp_string) - g_free(tmp_string); - return; - } - } - } - - if(!search_matching_command) - { - if(current_command) - g_free(current_command); - if(input) - g_free(input); - if(tmp_string) - g_free(tmp_string); - return; - } - - Completion *result = commands[previous_id].completion(current_parameter); - - if(!result || !result->groups) - { - if(current_command) - g_free(current_command); - if(input) - g_free(input); - if(tmp_string) - g_free(tmp_string); - return; - } - - command_mode = FALSE; - CompletionGroup* group = NULL; - CompletionElement* element = NULL; - - rows = malloc(sizeof(CompletionRow)); - if(!rows) - out_of_memory(); - - for(group = result->groups; group != NULL; group = group->next) - { - int group_elements = 0; - - for(element = group->elements; element != NULL; element = element->next) - { - if(element->value) - { - if(group->value && !group_elements) - { - rows = safe_realloc((void**)&rows, n_items + 1, sizeof(CompletionRow)); - if(!rows) - out_of_memory(); - rows[n_items].command = g_strdup(group->value); - rows[n_items].description = NULL; - rows[n_items].command_id = -1; - rows[n_items].is_group = TRUE; - rows[n_items++].row = GTK_WIDGET(create_completion_row(results, group->value, NULL, TRUE)); - } - - rows = safe_realloc((void**)&rows, n_items + 1, sizeof(CompletionRow)); - if(!rows) - out_of_memory(); - rows[n_items].command = g_strdup(element->value); - rows[n_items].description = element->description ? g_strdup(element->description) : NULL; - rows[n_items].command_id = previous_id; - rows[n_items].is_group = FALSE; - rows[n_items++].row = GTK_WIDGET(create_completion_row(results, element->value, element->description, FALSE)); - group_elements++; - } - } - } - - /* clean up */ - completion_free(result); - } - /* create list based on commands */ - else - { - int i = 0; - command_mode = TRUE; - - rows = malloc(LENGTH(commands) * sizeof(CompletionRow)); - if(!rows) - out_of_memory(); - - for(i = 0; i < LENGTH(commands); i++) - { - int abbr_length = commands[i].abbr ? strlen(commands[i].abbr) : 0; - int cmd_length = commands[i].command ? strlen(commands[i].command) : 0; - - /* add command to list iff - * the current command would match the command - * the current command would match the abbreviation - */ - if( ((current_command_length <= cmd_length) && !strncmp(current_command, commands[i].command, current_command_length)) || - ((current_command_length <= abbr_length) && !strncmp(current_command, commands[i].abbr, current_command_length)) - ) - { - rows[n_items].command = g_strdup(commands[i].command); - rows[n_items].description = g_strdup(commands[i].description); - rows[n_items].command_id = i; - rows[n_items].is_group = FALSE; - rows[n_items++].row = GTK_WIDGET(create_completion_row(results, commands[i].command, commands[i].description, FALSE)); - } - } - - rows = safe_realloc((void**)&rows, n_items, sizeof(CompletionRow)); - if(!rows) - out_of_memory(); - } - - gtk_box_pack_start(Zathura.UI.box, GTK_WIDGET(results), FALSE, FALSE, 0); - gtk_widget_show_all(GTK_WIDGET(Zathura.UI.window)); - - current_item = (argument->n == NEXT) ? -1 : 0; - } - - /* update coloring iff - * there is a list with items - */ - if( (results) && (n_items > 0) ) - { - set_completion_row_color(results, NORMAL, current_item); - char* temp; - int i = 0, next_group = 0; - - for(i = 0; i < n_items; i++) - { - if(argument->n == NEXT || argument->n == NEXT_GROUP) - current_item = (current_item + n_items + 1) % n_items; - else if(argument->n == PREVIOUS || argument->n == PREVIOUS_GROUP) - current_item = (current_item + n_items - 1) % n_items; - - if(rows[current_item].is_group) - { - if(!command_mode && (argument->n == NEXT_GROUP || argument->n == PREVIOUS_GROUP)) - next_group = 1; - continue; - } - else - { - if(!command_mode && (next_group == 0) && (argument->n == NEXT_GROUP || argument->n == PREVIOUS_GROUP)) - continue; - break; - } - } - - set_completion_row_color(results, HIGHLIGHT, current_item); - - /* hide other items */ - int uh = ceil(n_completion_items / 2); - int lh = floor(n_completion_items / 2); - - for(i = 0; i < n_items; i++) - { - if((n_items > 1) && ( - (i >= (current_item - lh) && (i <= current_item + uh)) || - (i < n_completion_items && current_item < lh) || - (i >= (n_items - n_completion_items) && (current_item >= (n_items - uh)))) - ) - gtk_widget_show(rows[i].row); - else - gtk_widget_hide(rows[i].row); - } - - if(command_mode) - temp = g_strconcat(":", rows[current_item].command, (n_items == 1) ? " " : NULL, NULL); - else - temp = g_strconcat(":", previous_command, " ", rows[current_item].command, NULL); - - gtk_entry_set_text(Zathura.UI.inputbar, temp); - gtk_editable_set_position(GTK_EDITABLE(Zathura.UI.inputbar), -1); - g_free(temp); - - previous_command = g_strdup((command_mode) ? rows[current_item].command : current_command); - previous_parameter = g_strdup((command_mode) ? current_parameter : rows[current_item].command); - previous_length = strlen(previous_command) + ((command_mode) ? (length - current_command_length) : (strlen(previous_parameter) + 1)); - previous_id = rows[current_item].command_id; - } - - if(current_command) - g_free(current_command); - if(input) - g_free(input); - if(tmp_string) - g_free(tmp_string); -} - -void -isc_string_manipulation(Argument* argument) -{ - gchar *input = gtk_editable_get_chars(GTK_EDITABLE(Zathura.UI.inputbar), 0, -1); - int length = strlen(input); - int pos = gtk_editable_get_position(GTK_EDITABLE(Zathura.UI.inputbar)); - int i; - - switch (argument->n) { - case DELETE_LAST_WORD: - i = pos - 1; - - if(!pos) - return; - - /* remove trailing spaces */ - for(; i >= 0 && input[i] == ' '; i--); - - /* find the beginning of the word */ - while((i > 0) && (input[i] != ' ') && (input[i] != '/')) - i--; - - gtk_editable_delete_text(GTK_EDITABLE(Zathura.UI.inputbar), i, pos); - gtk_editable_set_position(GTK_EDITABLE(Zathura.UI.inputbar), i); - break; - case DELETE_LAST_CHAR: - if((length - 1) <= 0) - isc_abort(NULL); - - gtk_editable_delete_text(GTK_EDITABLE(Zathura.UI.inputbar), pos - 1, pos); - break; - case DELETE_TO_LINE_START: - gtk_editable_delete_text(GTK_EDITABLE(Zathura.UI.inputbar), 1, pos); - break; - case NEXT_CHAR: - gtk_editable_set_position(GTK_EDITABLE(Zathura.UI.inputbar), pos + 1); - break; - case PREVIOUS_CHAR: - gtk_editable_set_position(GTK_EDITABLE(Zathura.UI.inputbar), (pos == 0) ? 0 : pos - 1); - break; - default: /* unreachable */ - break; - } -} - -/* command implementation */ -gboolean -cmd_bookmark(int argc, char** argv) -{ - if(!Zathura.PDF.document || argc < 1) - return TRUE; - - /* get id */ - int i; - GString *id = g_string_new(""); - - for(i = 0; i < argc; i++) - { - if(i != 0) - id = g_string_append_c(id, ' '); - - id = g_string_append(id, argv[i]); - } - - if(strlen(id->str) == 0) - { - notify(WARNING, "Can't set bookmark: bookmark name is empty"); - g_string_free(id, TRUE); - return FALSE; - } - - if(is_reserved_bm_name(id->str)) - { - notify(WARNING, "Can't set bookmark: reserved bookmark name"); - g_string_free(id, TRUE); - return FALSE; - } - - /* reload the bookmark file */ - read_bookmarks_file(); - - /* check for existing bookmark to overwrite */ - for(i = 0; i < Zathura.Bookmarks.number_of_bookmarks; i++) - { - if(!strcmp(id->str, Zathura.Bookmarks.bookmarks[i].id)) - { - Zathura.Bookmarks.bookmarks[i].page = Zathura.PDF.page_number; - g_string_free(id, TRUE); - return TRUE; - } - } - - /* add new bookmark */ - Zathura.Bookmarks.bookmarks = safe_realloc((void**)&Zathura.Bookmarks.bookmarks, - Zathura.Bookmarks.number_of_bookmarks + 1, sizeof(Bookmark)); - if(!Zathura.Bookmarks.bookmarks) - out_of_memory(); - - Zathura.Bookmarks.bookmarks[Zathura.Bookmarks.number_of_bookmarks].id = g_strdup(id->str); - Zathura.Bookmarks.bookmarks[Zathura.Bookmarks.number_of_bookmarks].page = Zathura.PDF.page_number; - Zathura.Bookmarks.number_of_bookmarks++; - - /* write the bookmark file */ - write_bookmarks_file(); - - g_string_free(id, TRUE); - return TRUE; -} - -gboolean -cmd_open_bookmark(int argc, char** argv) -{ - if(!Zathura.PDF.document || argc < 1) - return TRUE; - - /* get id */ - int i; - GString *id = g_string_new(""); - - for(i = 0; i < argc; i++) - { - if(i != 0) - id = g_string_append_c(id, ' '); - - id = g_string_append(id, argv[i]); - } - - /* find bookmark */ - for(i = 0; i < Zathura.Bookmarks.number_of_bookmarks; i++) - { - if(!strcmp(id->str, Zathura.Bookmarks.bookmarks[i].id)) - { - set_page(Zathura.Bookmarks.bookmarks[i].page); - g_string_free(id, TRUE); - return TRUE; - } - } - - notify(WARNING, "No matching bookmark found"); - g_string_free(id, TRUE); - return FALSE; -} - -gboolean -cmd_close(int argc, char** argv) -{ - close_file(FALSE); - - return TRUE; -} - -gboolean -cmd_correct_offset(int argc, char** argv) -{ - if(!Zathura.PDF.document || argc == 0) - return TRUE; - - Zathura.PDF.page_offset = (Zathura.PDF.page_number + 1) - atoi(argv[0]); - - if(Zathura.PDF.page_offset != 0) - Zathura.Global.goto_mode = GOTO_OFFSET; - else - Zathura.Global.goto_mode = GOTO_MODE; - - update_status(); - - return TRUE; -} - -gboolean -cmd_delete_bookmark(int argc, char** argv) -{ - if(!Zathura.PDF.document || argc < 1) - return TRUE; - - /* get id */ - int i; - GString *id = g_string_new(""); - - for(i = 0; i < argc; i++) - { - if(i != 0) - id = g_string_append_c(id, ' '); - - id = g_string_append(id, argv[i]); - } - - /* reload bookmark file */ - read_bookmarks_file(); - - /* check for bookmark to delete */ - for(i = 0; i < Zathura.Bookmarks.number_of_bookmarks; i++) - { - if(!strcmp(id->str, Zathura.Bookmarks.bookmarks[i].id)) - { - /* update key file */ - g_key_file_remove_key(Zathura.Bookmarks.data, Zathura.PDF.file, Zathura.Bookmarks.bookmarks[i].id, NULL); - - g_free(Zathura.Bookmarks.bookmarks[i].id); - /* update bookmarks */ - Zathura.Bookmarks.bookmarks[i].id = Zathura.Bookmarks.bookmarks[Zathura.Bookmarks.number_of_bookmarks - 1].id; - Zathura.Bookmarks.bookmarks[i].page = Zathura.Bookmarks.bookmarks[Zathura.Bookmarks.number_of_bookmarks - 1].page; - Zathura.Bookmarks.bookmarks = safe_realloc((void**)&Zathura.Bookmarks.bookmarks, - Zathura.Bookmarks.number_of_bookmarks, sizeof(Bookmark)); - if(!Zathura.Bookmarks.bookmarks) - out_of_memory(); - - Zathura.Bookmarks.number_of_bookmarks--; - g_string_free(id, TRUE); - - /* write bookmark file */ - write_bookmarks_file(); - - return TRUE; - } - } - - g_string_free(id, TRUE); - return TRUE; -} - -gboolean -cmd_export(int argc, char** argv) -{ - if(argc == 0 || !Zathura.PDF.document) - return TRUE; - - if(argc < 2) - { - notify(WARNING, "No export path specified"); - return FALSE; - } - - /* export images */ - if(!strcmp(argv[0], "images")) - { - int page_number; - for(page_number = 0; page_number < Zathura.PDF.number_of_pages; page_number++) - { - GList *image_list; - GList *images; - cairo_surface_t *image; - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - image_list = poppler_page_get_image_mapping(Zathura.PDF.pages[page_number]->page); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if(!g_list_length(image_list)) - { - notify(WARNING, "This document does not contain any images"); - return FALSE; - } - - for(images = image_list; images; images = g_list_next(images)) - { - PopplerImageMapping *image_mapping; - gint image_id; - char* file; - char* filename; - - image_mapping = (PopplerImageMapping*) images->data; - image_id = image_mapping->image_id; - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - image = poppler_page_get_image(Zathura.PDF.pages[page_number]->page, image_id); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if(!image) - continue; - - filename = g_strdup_printf("%s_p%i_i%i.png", Zathura.PDF.file, page_number + 1, image_id); - - if(argv[1][0] == '~') - { - gchar* home_path = get_home_dir(); - file = g_strdup_printf("%s%s%s", home_path, argv[1] + 1, filename); - g_free(home_path); - } - else - file = g_strdup_printf("%s%s", argv[1], filename); - - cairo_surface_write_to_png(image, file); - - g_free(filename); - g_free(file); - } - } - } - else if(!strcmp(argv[0], "attachments")) - { - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - if(!poppler_document_has_attachments(Zathura.PDF.document)) - { - notify(WARNING, "PDF file has no attachments"); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - return FALSE; - } - - GList *attachment_list = poppler_document_get_attachments(Zathura.PDF.document); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - GList *attachments; - char *file; - - for(attachments = attachment_list; attachments; attachments = g_list_next(attachments)) - { - PopplerAttachment *attachment = (PopplerAttachment*) attachments->data; - - if(argv[1][0] == '~') - { - gchar* home_path = get_home_dir(); - file = g_strdup_printf("%s%s%s", home_path, argv[1] + 1, attachment->name); - g_free(home_path); - } - else - file = g_strdup_printf("%s%s", argv[1], attachment->name); - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_attachment_save(attachment, file, NULL); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - g_free(file); - } - } - - return TRUE; -} - -gboolean -cmd_info(int argc, char** argv) -{ - if(!Zathura.PDF.document) - return TRUE; - - static gboolean visible = FALSE; - - if(!Zathura.UI.information) - { - GtkListStore *list; - GtkTreeIter iter; - GtkCellRenderer *renderer; - GtkTreeSelection *selection; - - list = gtk_list_store_new(2, G_TYPE_STRING, G_TYPE_STRING); - - /* read document information */ - gchar *title, *author; - gchar *subject, *keywords; - gchar *creator, *producer; - GTime creation_date, modification_date; - - g_object_get(Zathura.PDF.document, - "title", &title, - "author", &author, - "subject", &subject, - "keywords", &keywords, - "creator", &creator, - "producer", &producer, - "creation-date", &creation_date, - "mod-date", &modification_date, - NULL); - - /* append information to list */ - gtk_list_store_append(list, &iter); - gtk_list_store_set(list, &iter, 0, "Author", 1, author ? author : "", -1); - gtk_list_store_append(list, &iter); - gtk_list_store_set(list, &iter, 0, "Title", 1, title ? title : "", -1); - gtk_list_store_append(list, &iter); - gtk_list_store_set(list, &iter, 0, "Subject", 1, subject ? subject : "", -1); - gtk_list_store_append(list, &iter); - gtk_list_store_set(list, &iter, 0, "Keywords", 1, keywords ? keywords : "", -1); - gtk_list_store_append(list, &iter); - gtk_list_store_set(list, &iter, 0, "Creator", 1, creator ? creator : "", -1); - gtk_list_store_append(list, &iter); - gtk_list_store_set(list, &iter, 0, "Producer", 1, producer ? producer : "", -1); - - Zathura.UI.information = gtk_tree_view_new_with_model(GTK_TREE_MODEL(list)); - renderer = gtk_cell_renderer_text_new(); - - gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(Zathura.UI.information), -1, - "Name", renderer, "text", 0, NULL); - gtk_tree_view_insert_column_with_attributes(GTK_TREE_VIEW(Zathura.UI.information), -1, - "Value", renderer, "text", 1, NULL); - - selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(Zathura.UI.information)); - gtk_tree_selection_set_mode(selection, GTK_SELECTION_SINGLE); - - gtk_widget_show_all(Zathura.UI.information); - } - - if(!visible) - switch_view(Zathura.UI.information); - else - switch_view(Zathura.UI.document); - - visible = !visible; - - return FALSE; -} - -gboolean -cmd_map(int argc, char** argv) -{ - if(argc < 2) - return TRUE; - - char* ks = argv[0]; - - /* search for the right shortcut function */ - int sc_id = -1; - - int sc_c; - for(sc_c = 0; sc_c < LENGTH(shortcut_names); sc_c++) - { - if(!strcmp(argv[1], shortcut_names[sc_c].name)) - { - sc_id = sc_c; - break; - } - } - - if(sc_id == -1) - { - notify(WARNING, "No such shortcut function exists"); - return FALSE; - } - - /* parse modifier and key */ - int mask = 0; - int key = 0; - int keyl = strlen(ks); - int mode = NORMAL; - - // single key (e.g.: g) - if(keyl == 1) - key = ks[0]; - - // modifier and key (e.g.: - // special key or modifier and key/special key (e.g.: , ) - - else if(keyl >= 3 && ks[0] == '<' && ks[keyl-1] == '>') - { - char* specialkey = NULL; - - /* check for modifier */ - if(keyl >= 5 && ks[2] == '-') - { - /* evaluate modifier */ - switch(ks[1]) - { - case 'S': - mask = GDK_SHIFT_MASK; - break; - case 'C': - mask = GDK_CONTROL_MASK; - break; - } - - /* no valid modifier */ - if(!mask) - { - notify(WARNING, "No valid modifier given."); - return FALSE; - } - - /* modifier and special key */ - if(keyl > 5) - specialkey = g_strndup(ks + 3, keyl - 4); - else - key = ks[3]; - } - else - specialkey = ks; - - /* search special key */ - int g_c; - for(g_c = 0; specialkey && g_c < LENGTH(gdk_keys); g_c++) - { - if(!strcmp(specialkey, gdk_keys[g_c].identifier)) - { - key = gdk_keys[g_c].key; - break; - } - } - - if(specialkey) - g_free(specialkey); - } - - if(!key) - { - notify(WARNING, "No valid key binding given."); - return FALSE; - } - - /* parse argument */ - Argument arg = {0, 0}; - - if(argc >= 3) - { - int arg_id = -1; - - /* compare argument with given argument names... */ - int arg_c; - for(arg_c = 0; arg_c < LENGTH(argument_names); arg_c++) - { - if(!strcmp(argv[2], argument_names[arg_c].name)) - { - arg_id = argument_names[arg_c].argument; - break; - } - } - - /* if not, save it do .data */ - if(arg_id == -1) - arg.data = argv[2]; - else - arg.n = arg_id; - } - - /* parse mode */ - if(argc >= 4) - { - int mode_c; - for(mode_c = 0; mode_c < LENGTH(mode_names); mode_c++) - { - if(!strcmp(argv[3], mode_names[mode_c].name)) - { - mode = mode_names[mode_c].mode; - break; - } - } - } - - /* search for existing binding to overwrite it */ - ShortcutList* sc = Zathura.Bindings.sclist; - while(sc && sc->next != NULL) - { - if(sc->element.key == key && sc->element.mask == mask - && sc->element.mode == mode) - { - sc->element.function = shortcut_names[sc_id].function; - sc->element.argument = arg; - return TRUE; - } - - sc = sc->next; - } - - /* create new entry */ - ShortcutList* entry = malloc(sizeof(ShortcutList)); - if(!entry) - out_of_memory(); - - entry->element.mask = mask; - entry->element.key = key; - entry->element.function = shortcut_names[sc_id].function; - entry->element.mode = mode; - entry->element.argument = arg; - entry->next = NULL; - - /* append to list */ - if(!Zathura.Bindings.sclist) - Zathura.Bindings.sclist = entry; - - if(sc) - sc->next = entry; - - return TRUE; -} - -gboolean -cmd_open(int argc, char** argv) -{ - if(argc == 0 || strlen(argv[0]) == 0) - return TRUE; - - /* assembly the arguments back to one string */ - int i = 0; - GString *filepath = g_string_new(""); - for(i = 0; i < argc; i++) - { - if(i != 0) - filepath = g_string_append_c(filepath, ' '); - - filepath = g_string_append(filepath, argv[i]); - } - - gboolean res = open_file(filepath->str, NULL); - g_string_free(filepath, TRUE); - return res; -} - -gboolean -cmd_print(int argc, char** argv) -{ - if(!Zathura.PDF.document) - return TRUE; - - if(argc == 0) - { - notify(WARNING, "No printer specified"); - return FALSE; - } - - char* printer = argv[0]; - char* sites = (argc >= 2) ? g_strdup(argv[1]) : g_strdup_printf("1-%i", Zathura.PDF.number_of_pages); - GString *addit = g_string_new(""); - - int i; - for(i = 2; i < argc; i++) - { - if(i != 0) - addit = g_string_append_c(addit, ' '); - - addit = g_string_append(addit, argv[i]); - } - - char* escaped_filename = g_shell_quote(Zathura.PDF.file); - char* command = g_strdup_printf(print_command, printer, sites, addit->str, escaped_filename); - system(command); - - g_free(sites); - g_free(escaped_filename); - g_free(command); - g_string_free(addit, TRUE); - - return TRUE; -} - -gboolean -cmd_rotate(int argc, char** argv) -{ - return TRUE; -} - -gboolean -cmd_set(int argc, char** argv) -{ - if(argc <= 0) - return FALSE; - - int i; - for(i = 0; i < LENGTH(settings); i++) - { - if(!strcmp(argv[0], settings[i].name)) - { - /* check var type */ - if(settings[i].type == 'b') - { - gboolean *x = (gboolean*) (settings[i].variable); - *x = !(*x); - - if(argv[1]) - { - if(!strcmp(argv[1], "false") || !strcmp(argv[1], "0")) - *x = FALSE; - else - *x = TRUE; - } - } - else if(settings[i].type == 'i') - { - if(argc != 2) - return FALSE; - - int *x = (int*) (settings[i].variable); - - int id = -1; - int arg_c; - for(arg_c = 0; arg_c < LENGTH(argument_names); arg_c++) - { - if(!strcmp(argv[1], argument_names[arg_c].name)) - { - id = argument_names[arg_c].argument; - break; - } - } - - if(id == -1) - id = atoi(argv[1]); - - *x = id; - } - else if(settings[i].type == 'f') - { - if(argc != 2) - return FALSE; - - float *x = (float*) (settings[i].variable); - if(argv[1]) - *x = atof(argv[1]); - } - else if(settings[i].type == 's') - { - if(argc < 2) - return FALSE; - - /* assembly the arguments back to one string */ - int j; - GString *s = g_string_new(""); - for(j = 1; j < argc; j++) - { - if(j != 1) - s = g_string_append_c(s, ' '); - - s = g_string_append(s, argv[j]); - } - - char **x = (char**) settings[i].variable; - *x = s->str; - } - else if(settings[i].type == 'c') - { - if(argc != 2) - return FALSE; - - char *x = (char*) (settings[i].variable); - if(argv[1]) - *x = argv[1][0]; - } - - /* re-init */ - if(settings[i].reinit) - init_look(); - - /* render */ - if(settings[i].render) - { - if(!Zathura.PDF.document) - return FALSE; - - draw(Zathura.PDF.page_number); - } - } - } - - update_status(); - return TRUE; -} - -gboolean -cmd_quit(int argc, char** argv) -{ - cb_destroy(NULL, NULL); - return TRUE; -} - -gboolean -save_file(int argc, char** argv, gboolean overwrite) -{ - if(argc == 0 || !Zathura.PDF.document) - return TRUE; - - gchar* file_path = NULL; - - if(argv[0][0] == '~') - { - gchar* home_path = get_home_dir(); - file_path = g_build_filename(home_path, argv[0] + 1, NULL); - g_free(home_path); - } - else - file_path = g_strdup(argv[0]); - - if (!overwrite && g_file_test(file_path, G_FILE_TEST_EXISTS)) - { - char* message = g_strdup_printf("File already exists: %s. Use :write! to overwrite it.", file_path); - notify(ERROR, message); - g_free(message); - g_free(file_path); - return FALSE; - } - - char* path = NULL; - if (file_path[0] == '/') - path = g_strdup_printf("file://%s", file_path); - else - { - char* cur = g_get_current_dir(); - path = g_strdup_printf("file://%s/%s", cur, file_path); - g_free(cur); - } - g_free(file_path); - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - /* format path */ - GError* error = NULL; - if (!poppler_document_save(Zathura.PDF.document, path, &error)) - { - g_free(path); - char* message = g_strdup_printf("Can not write file: %s", error->message); - notify(ERROR, message); - g_free(message); - g_error_free(error); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - return FALSE; - } - - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - g_free(path); - - return TRUE; -} - -gboolean -cmd_save(int argc, char** argv) -{ - return save_file(argc, argv, FALSE); -} - -gboolean -cmd_savef(int argc, char** argv) -{ - return save_file(argc, argv, TRUE); -} - -/* completion command implementation */ -Completion* -cc_bookmark(char* input) -{ - Completion* completion = completion_init(); - CompletionGroup* group = completion_group_create(NULL); - - completion_add_group(completion, group); - - int i = 0; - int input_length = input ? strlen(input) : 0; - - for(i = 0; i < Zathura.Bookmarks.number_of_bookmarks; i++) - { - if( (input_length <= strlen(Zathura.Bookmarks.bookmarks[i].id)) && - !strncmp(input, Zathura.Bookmarks.bookmarks[i].id, input_length) ) - { - completion_group_add_element(group, Zathura.Bookmarks.bookmarks[i].id, g_strdup_printf("Page %d", Zathura.Bookmarks.bookmarks[i].page)); - } - } - - return completion; -} - -Completion* -cc_export(char* input) -{ - Completion* completion = completion_init(); - CompletionGroup* group = completion_group_create(NULL); - - completion_add_group(completion, group); - - completion_group_add_element(group, "images", "Export images"); - completion_group_add_element(group, "attachments", "Export attachments"); - - return completion; -} - -Completion* -cc_open(char* input) -{ - Completion* completion = completion_init(); - CompletionGroup* group = completion_group_create(NULL); - - completion_add_group(completion, group); - - /* ~ */ - if(input && input[0] == '~') - { - gchar* home_path = get_home_dir(); - char *file = g_strdup_printf(":open %s/%s", home_path, input + 1); - g_free(home_path); - gtk_entry_set_text(Zathura.UI.inputbar, file); - gtk_editable_set_position(GTK_EDITABLE(Zathura.UI.inputbar), -1); - g_free(file); - completion_free(completion); - return NULL; - } - - /* read dir */ - char* path = g_strdup("/"); - char* file = g_strdup(""); - int file_length = 0; - - /* parse input string */ - if(input && strlen(input) > 0) - { - char* dinput = g_strdup(input); - char* binput = g_strdup(input); - char* path_temp = dirname(dinput); - char* file_temp = basename(binput); - char last_char = input[strlen(input) - 1]; - - if( !strcmp(path_temp, "/") && !strcmp(file_temp, "/") ) - { - g_free(file); - file = g_strdup(""); - } - else if( !strcmp(path_temp, "/") && strcmp(file_temp, "/") && last_char != '/') - { - g_free(file); - file = g_strdup(file_temp); - } - else if( !strcmp(path_temp, "/") && strcmp(file_temp, "/") && last_char == '/') - { - g_free(path); - path = g_strdup_printf("/%s/", file_temp); - } - else if(last_char == '/') - { - g_free(path); - path = g_strdup(input); - } - else - { - g_free(path); - g_free(file); - path = g_strdup_printf("%s/", path_temp); - file = g_strdup(file_temp); - } - - g_free(dinput); - g_free(binput); - } - - file_length = strlen(file); - - /* open directory */ - GDir* dir = g_dir_open(path, 0, NULL); - if(!dir) - { - g_free(path); - g_free(file); - completion_free(completion); - return NULL; - } - - /* create element list */ - char* name = NULL; - - while((name = (char*) g_dir_read_name(dir)) != NULL) - { - char* d_name = g_filename_display_name(name); - int d_length = strlen(d_name); - - if( ((file_length <= d_length) && !strncmp(file, d_name, file_length)) || - (file_length == 0) ) - { - char* d = g_strdup_printf("%s%s", path, d_name); - if(g_file_test(d, G_FILE_TEST_IS_DIR)) - { - gchar *subdir = d; - d = g_strdup_printf("%s/", subdir); - g_free(subdir); - } - completion_group_add_element(group, d, NULL); - g_free(d); - } - g_free(d_name); - } - - g_dir_close(dir); - g_free(file); - g_free(path); - - return completion; -} - -Completion* -cc_print(char* input) -{ - Completion* completion = completion_init(); - CompletionGroup* group = completion_group_create(NULL); - - completion_add_group(completion, group); - - int input_length = input ? strlen(input) : 0; - - /* read printers */ - char *current_line = NULL, current_char; - int count = 0; - FILE *fp; - - fp = popen(list_printer_command, "r"); - - if(!fp) - { - completion_free(completion); - return NULL; - } - - while((current_char = fgetc(fp)) != EOF) - { - if(!current_line) - current_line = malloc(sizeof(char)); - if(!current_line) - out_of_memory(); - - current_line = safe_realloc((void**)¤t_line, count + 1, sizeof(char)); - if(!current_line) - out_of_memory(); - - if(current_char != '\n') - current_line[count++] = current_char; - else - { - current_line[count] = '\0'; - int line_length = strlen(current_line); - - if( (input_length <= line_length) || - (!strncmp(input, current_line, input_length)) ) - { - completion_group_add_element(group, current_line, NULL); - } - - free(current_line); - current_line = NULL; - count = 0; - } - } - - pclose(fp); - - return completion; -} - -Completion* -cc_set(char* input) -{ - Completion* completion = completion_init(); - CompletionGroup* group = completion_group_create(NULL); - - completion_add_group(completion, group); - - int i = 0; - int input_length = input ? strlen(input) : 0; - - for(i = 0; i < LENGTH(settings); i++) - { - if( (input_length <= strlen(settings[i].name)) && - !strncmp(input, settings[i].name, input_length) ) - { - completion_group_add_element(group, settings[i].name, settings[i].description); - } - } - - return completion; -} - -/* buffer command implementation */ -void -bcmd_goto(char* buffer, Argument* argument) -{ - if(!Zathura.PDF.document) - return; - - int b_length = strlen(buffer); - if(b_length < 1) - return; - - if(!strcmp(buffer, "gg")) - set_page(0); - else if(!strcmp(buffer, "G")) - set_page(Zathura.PDF.number_of_pages - 1); - else - { - char* id = g_strndup(buffer, b_length - 1); - int pid = atoi(id); - - if(Zathura.Global.goto_mode == GOTO_LABELS) - { - int i; - for(i = 0; i < Zathura.PDF.number_of_pages; i++) - if(!strcmp(id, Zathura.PDF.pages[i]->label)) - pid = Zathura.PDF.pages[i]->id; - } - else if(Zathura.Global.goto_mode == GOTO_OFFSET) - pid += Zathura.PDF.page_offset; - - set_page(pid - 1); - g_free(id); - } - - update_status(); -} - -gboolean -try_goto(const char* buffer) -{ - char* endptr = NULL; - long page_number = strtol(buffer, &endptr, 10) - 1; - if(*endptr) - /* conversion error */ - return FALSE; - else - { - /* behave like vim: <= 1 => first line, >= #lines => last line */ - page_number = MAX(0, MIN(Zathura.PDF.number_of_pages - 1, page_number)); - set_page(page_number); - update_status(); - return TRUE; - } -} - -void -bcmd_scroll(char* buffer, Argument* argument) -{ - int b_length = strlen(buffer); - if(b_length < 1) - return; - - int percentage = atoi(g_strndup(buffer, b_length - 1)); - percentage = (percentage < 0) ? 0 : ((percentage > 100) ? 100 : percentage); - - GtkAdjustment *adjustment = gtk_scrolled_window_get_vadjustment(Zathura.UI.view); - - gdouble view_size = gtk_adjustment_get_page_size(adjustment); - gdouble max = gtk_adjustment_get_upper(adjustment) - view_size; - gdouble nvalue = (percentage * max) / 100; - - Zathura.State.scroll_percentage = percentage; - gtk_adjustment_set_value(adjustment, nvalue); - update_status(); -} - -void -bcmd_zoom(char* buffer, Argument* argument) -{ - Zathura.Global.adjust_mode = ADJUST_NONE; - - if(argument->n == ZOOM_IN) - { - if((Zathura.PDF.scale + zoom_step) <= zoom_max) - Zathura.PDF.scale += zoom_step; - else - Zathura.PDF.scale = zoom_max; - } - else if(argument->n == ZOOM_OUT) - { - if((Zathura.PDF.scale - zoom_step) >= zoom_min) - Zathura.PDF.scale -= zoom_step; - else - Zathura.PDF.scale = zoom_min; - } - else if(argument->n == ZOOM_SPECIFIC) - { - int b_length = strlen(buffer); - if(b_length < 1) - return; - - int value = atoi(g_strndup(buffer, b_length - 1)); - if(value <= zoom_min) - Zathura.PDF.scale = zoom_min; - else if(value >= zoom_max) - Zathura.PDF.scale = zoom_max; - else - Zathura.PDF.scale = value; - } - else - Zathura.PDF.scale = 100; - - Zathura.Search.draw = TRUE; - draw(Zathura.PDF.page_number); - update_status(); -} - -/* special command implementation */ -gboolean -scmd_search(gchar* input, Argument* argument) -{ - if(!input || !strlen(input)) - return TRUE; - - argument->data = input; - sc_search(argument); - - return TRUE; -} - -/* callback implementation */ -gboolean -cb_destroy(GtkWidget* widget, gpointer data) -{ - pango_font_description_free(Zathura.Style.font); - - if(Zathura.PDF.document) - close_file(FALSE); - - /* clean up bookmarks */ - g_free(Zathura.Bookmarks.file); - if (Zathura.Bookmarks.data) - g_key_file_free(Zathura.Bookmarks.data); - - /* destroy mutexes */ - g_static_mutex_free(&(Zathura.Lock.pdflib_lock)); - g_static_mutex_free(&(Zathura.Lock.search_lock)); - g_static_mutex_free(&(Zathura.Lock.pdf_obj_lock)); - g_static_mutex_free(&(Zathura.Lock.select_lock)); - - /* inotify */ - if(Zathura.FileMonitor.monitor) - g_object_unref(Zathura.FileMonitor.monitor); - if(Zathura.FileMonitor.file) - g_object_unref(Zathura.FileMonitor.file); - - g_list_free(Zathura.Global.history); - - /* clean shortcut list */ - ShortcutList* sc = Zathura.Bindings.sclist; - - while(sc) - { - ShortcutList* ne = sc->next; - free(sc); - sc = ne; - } - - g_free(Zathura.State.filename); - g_free(Zathura.State.pages); - - g_free(Zathura.Config.config_dir); - g_free(Zathura.Config.data_dir); - if (Zathura.StdinSupport.file) - g_unlink(Zathura.StdinSupport.file); - g_free(Zathura.StdinSupport.file); - - gtk_main_quit(); - - return TRUE; -} - -gboolean cb_draw(GtkWidget* widget, GdkEventExpose* expose, gpointer data) -{ - if(!Zathura.PDF.document) - return FALSE; - - int page_id = Zathura.PDF.page_number; - if(page_id < 0 || page_id > Zathura.PDF.number_of_pages) - return FALSE; - - gdk_window_clear(widget->window); - cairo_t *cairo = gdk_cairo_create(widget->window); - - double page_width, page_height, width, height; - double scale = ((double) Zathura.PDF.scale / 100.0); - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_page_get_size(Zathura.PDF.pages[page_id]->page, &page_width, &page_height); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - if(Zathura.PDF.rotate == 0 || Zathura.PDF.rotate == 180) - { - width = page_width * scale; - height = page_height * scale; - } - else - { - width = page_height * scale; - height = page_width * scale; - } - - int window_x, window_y; - gdk_drawable_get_size(widget->window, &window_x, &window_y); - - int offset_x, offset_y; - - if (window_x > width) - offset_x = (window_x - width) / 2; - else - offset_x = 0; - - if (window_y > height) - offset_y = (window_y - height) / 2; - else - offset_y = 0; - - if(Zathura.Search.draw) - { - GList* list; - for(list = Zathura.Search.results; list && list->data; list = g_list_next(list)) - highlight_result(Zathura.Search.page, (PopplerRectangle*) list->data); - Zathura.Search.draw = FALSE; - } - - cairo_set_source_surface(cairo, Zathura.PDF.surface, offset_x, offset_y); - cairo_paint(cairo); - cairo_destroy(cairo); - - return TRUE; -} - -gboolean -cb_inputbar_kb_pressed(GtkWidget *widget, GdkEventKey *event, gpointer data) -{ - int i; - - /* inputbar shortcuts */ - for(i = 0; i < LENGTH(inputbar_shortcuts); i++) - { - if(event->keyval == inputbar_shortcuts[i].key && - (((event->state & inputbar_shortcuts[i].mask) == inputbar_shortcuts[i].mask) - || inputbar_shortcuts[i].mask == 0)) - { - inputbar_shortcuts[i].function(&(inputbar_shortcuts[i].argument)); - return TRUE; - } - } - - /* special commands */ - char* identifier_string = gtk_editable_get_chars(GTK_EDITABLE(Zathura.UI.inputbar), 0, 1); - char identifier = identifier_string[0]; - - for(i = 0; i < LENGTH(special_commands); i++) - { - if((identifier == special_commands[i].identifier) && - (special_commands[i].always == 1)) - { - gchar *input = gtk_editable_get_chars(GTK_EDITABLE(Zathura.UI.inputbar), 1, -1); - guint new_utf_char = gdk_keyval_to_unicode(event->keyval); - - if(new_utf_char != 0) - { - gchar* newchar = g_malloc0(6 * sizeof(gchar)); - if(newchar == NULL) - { - g_free(input); - continue; - } - - gint len = g_unichar_to_utf8(new_utf_char, newchar); - newchar[len] = 0; - gchar* tmp = g_strconcat(input, newchar, NULL); - - g_free(input); - g_free(newchar); - - input = tmp; - } - - // FIXME - if((special_commands[i].function == scmd_search) && (event->keyval == GDK_Return)) - { - Argument argument = { NO_SEARCH, NULL }; - scmd_search(input, &argument); - } - else - { - special_commands[i].function(input, &(special_commands[i].argument)); - } - - g_free(identifier_string); - g_free(input); - return FALSE; - } - } - - g_free(identifier_string); - - return FALSE; -} - -gboolean -cb_inputbar_activate(GtkEntry* entry, gpointer data) -{ - gchar *input = gtk_editable_get_chars(GTK_EDITABLE(entry), 1, -1); - gchar **tokens = g_strsplit(input, " ", -1); - g_free(input); - - gchar *command = tokens[0]; - int length = g_strv_length(tokens); - int i = 0; - gboolean retv = FALSE; - gboolean succ = FALSE; - - /* no input */ - if(length < 1) - { - isc_abort(NULL); - g_strfreev(tokens); - return FALSE; - } - - /* append input to the command history */ - Zathura.Global.history = g_list_append(Zathura.Global.history, g_strdup(gtk_entry_get_text(entry))); - - /* special commands */ - char identifier = gtk_editable_get_chars(GTK_EDITABLE(entry), 0, 1)[0]; - for(i = 0; i < LENGTH(special_commands); i++) - { - if(identifier == special_commands[i].identifier) - { - /* special commands that are evaluated every key change are not - * called here */ - if(special_commands[i].always == 1) - { - isc_abort(NULL); - g_strfreev(tokens); - return TRUE; - } - - retv = special_commands[i].function(input, &(special_commands[i].argument)); - if(retv) isc_abort(NULL); - gtk_widget_grab_focus(GTK_WIDGET(Zathura.UI.view)); - g_strfreev(tokens); - return TRUE; - } - } - - /* search commands */ - for(i = 0; i < LENGTH(commands); i++) - { - if((g_strcmp0(command, commands[i].command) == 0) || - (g_strcmp0(command, commands[i].abbr) == 0)) - { - retv = commands[i].function(length - 1, tokens + 1); - succ = TRUE; - break; - } - } - - if(retv) - isc_abort(NULL); - - if(!succ) { - /* it maybe a goto command */ - if(!try_goto(command)) - notify(ERROR, "Unknown command."); - } - - Argument arg = { HIDE }; - isc_completion(&arg); - gtk_widget_grab_focus(GTK_WIDGET(Zathura.UI.view)); - - g_strfreev(tokens); - return TRUE; -} - -gboolean -cb_inputbar_form_activate(GtkEntry* entry, gpointer data) -{ - if(!Zathura.PDF.document) - return TRUE; - - Page* current_page = Zathura.PDF.pages[Zathura.PDF.page_number]; - int number_of_links = 0, link_id = 1, new_page_id = Zathura.PDF.page_number; - - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - GList *link_list = poppler_page_get_link_mapping(current_page->page); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - link_list = g_list_reverse(link_list); - - if((number_of_links = g_list_length(link_list)) <= 0) - return FALSE; - - /* parse entry */ - gchar *input = gtk_editable_get_chars(GTK_EDITABLE(entry), 1, -1); - gchar *token = input + strlen("Follow hint: ") - 1; - if(!token) - return FALSE; - - int li = atoi(token); - if(li <= 0 || li > number_of_links) - { - set_page(Zathura.PDF.page_number); - isc_abort(NULL); - notify(WARNING, "Invalid hint"); - return TRUE; - } - - /* compare entry */ - GList *links; - for(links = link_list; links; links = g_list_next(links)) - { - PopplerLinkMapping *link_mapping = (PopplerLinkMapping*) links->data; - PopplerAction *action = link_mapping->action; - - /* only handle URI and internal links */ - if(action->type == POPPLER_ACTION_URI) - { - if(li == link_id) - open_uri(action->uri.uri); - } - else if(action->type == POPPLER_ACTION_GOTO_DEST) - { - if(li == link_id) - { - if(action->goto_dest.dest->type == POPPLER_DEST_NAMED) - { - PopplerDest* destination = poppler_document_find_dest(Zathura.PDF.document, action->goto_dest.dest->named_dest); - if(destination) - { - new_page_id = destination->page_num - 1; - poppler_dest_free(destination); - } - } - else - new_page_id = action->goto_dest.dest->page_num - 1; - } - } - else - continue; - - link_id++; - } - - poppler_page_free_link_mapping(link_list); - - /* reset all */ - set_page(new_page_id); - isc_abort(NULL); - - return TRUE; -} - -gboolean -cb_inputbar_password_activate(GtkEntry* entry, gpointer data) -{ - gchar *input = gtk_editable_get_chars(GTK_EDITABLE(entry), 1, -1); - gchar *token = input + strlen("Enter password: ") - 1; - if(!token) - return FALSE; - - if(!open_file(Zathura.PDF.file, token)) - { - enter_password(); - return TRUE; - } - - /* replace default inputbar handler */ - g_signal_handler_disconnect((gpointer) Zathura.UI.inputbar, Zathura.Handler.inputbar_activate); - Zathura.Handler.inputbar_activate = g_signal_connect(G_OBJECT(Zathura.UI.inputbar), "activate", G_CALLBACK(cb_inputbar_activate), NULL); - - isc_abort(NULL); - - return TRUE; -} - -gboolean -cb_view_kb_pressed(GtkWidget *widget, GdkEventKey *event, gpointer data) -{ - ShortcutList* sc = Zathura.Bindings.sclist; - while(sc) - { - if( - event->keyval == sc->element.key - && (CLEAN(event->state) == sc->element.mask || (sc->element.key >= 0x21 - && sc->element.key <= 0x7E && CLEAN(event->state) == GDK_SHIFT_MASK)) - && (Zathura.Global.mode & sc->element.mode || sc->element.mode == ALL) - && sc->element.function - ) - { - if(!(Zathura.Global.buffer && strlen(Zathura.Global.buffer->str)) || (sc->element.mask == GDK_CONTROL_MASK) || - (sc->element.key <= 0x21 || sc->element.key >= 0x7E) - ) - { - sc->element.function(&(sc->element.argument)); - return TRUE; - } - } - - sc = sc->next; - } - - if(Zathura.Global.mode == ADD_MARKER) - { - add_marker(event->keyval); - change_mode(NORMAL); - return TRUE; - } - else if(Zathura.Global.mode == EVAL_MARKER) - { - eval_marker(event->keyval); - change_mode(NORMAL); - return TRUE; - } - - /* append only numbers and characters to buffer */ - if( (event->keyval >= 0x21) && (event->keyval <= 0x7E)) - { - if(!Zathura.Global.buffer) - Zathura.Global.buffer = g_string_new(""); - - Zathura.Global.buffer = g_string_append_c(Zathura.Global.buffer, event->keyval); - gtk_label_set_markup((GtkLabel*) Zathura.Global.status_buffer, Zathura.Global.buffer->str); - } - - /* search buffer commands */ - if(Zathura.Global.buffer) - { - int i; - for(i = 0; i < LENGTH(buffer_commands); i++) - { - regex_t regex; - int status; - - regcomp(®ex, buffer_commands[i].regex, REG_EXTENDED); - status = regexec(®ex, Zathura.Global.buffer->str, (size_t) 0, NULL, 0); - regfree(®ex); - - if(status == 0) - { - buffer_commands[i].function(Zathura.Global.buffer->str, &(buffer_commands[i].argument)); - g_string_free(Zathura.Global.buffer, TRUE); - Zathura.Global.buffer = NULL; - gtk_label_set_markup((GtkLabel*) Zathura.Global.status_buffer, ""); - - return TRUE; - } - } - } - - return FALSE; -} - -gboolean -cb_view_resized(GtkWidget* widget, GtkAllocation* allocation, gpointer data) -{ - Argument arg; - arg.n = Zathura.Global.adjust_mode; - sc_adjust_window(&arg); - - return TRUE; -} - -gboolean -cb_view_button_pressed(GtkWidget* widget, GdkEventButton* event, gpointer data) -{ - if(!Zathura.PDF.document) - return FALSE; - - /* clean page */ - draw(Zathura.PDF.page_number); - g_static_mutex_lock(&(Zathura.Lock.select_lock)); - Zathura.SelectPoint.x = event->x; - Zathura.SelectPoint.y = event->y; - g_static_mutex_unlock(&(Zathura.Lock.select_lock)); - - return TRUE; -} - -gboolean -cb_view_button_release(GtkWidget* widget, GdkEventButton* event, gpointer data) -{ - if(!Zathura.PDF.document) - return FALSE; - - double scale, offset_x, offset_y, page_width, page_height; - PopplerRectangle rectangle; - cairo_t* cairo; - - /* build selection rectangle */ - rectangle.x1 = event->x; - rectangle.y1 = event->y; - - g_static_mutex_lock(&(Zathura.Lock.select_lock)); - rectangle.x2 = Zathura.SelectPoint.x; - rectangle.y2 = Zathura.SelectPoint.y; - g_static_mutex_unlock(&(Zathura.Lock.select_lock)); - - /* calculate offset */ - calculate_offset(widget, &offset_x, &offset_y); - - /* draw selection rectangle */ - cairo = cairo_create(Zathura.PDF.surface); - cairo_set_source_rgba(cairo, Zathura.Style.select_text.red, Zathura.Style.select_text.green, - Zathura.Style.select_text.blue, transparency); - cairo_rectangle(cairo, rectangle.x1 - offset_x, rectangle.y1 - offset_y, - (rectangle.x2 - rectangle.x1), (rectangle.y2 - rectangle.y1)); - cairo_fill(cairo); - cairo_destroy(cairo); - gtk_widget_queue_draw(Zathura.UI.drawing_area); - - /* resize selection rectangle to document page */ - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - poppler_page_get_size(Zathura.PDF.pages[Zathura.PDF.page_number]->page, &page_width, &page_height); - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - scale = ((double) Zathura.PDF.scale / 100.0); - rectangle.x1 = (rectangle.x1 - offset_x) / scale; - rectangle.y1 = (rectangle.y1 - offset_y) / scale; - rectangle.x2 = (rectangle.x2 - offset_x) / scale; - rectangle.y2 = (rectangle.y2 - offset_y) / scale; - - /* rotation */ - int rotate = Zathura.PDF.rotate; - double x1 = rectangle.x1; - double x2 = rectangle.x2; - double y1 = rectangle.y1; - double y2 = rectangle.y2; - - switch(rotate) - { - case 90: - rectangle.x1 = y1; - rectangle.y1 = page_height - x2; - rectangle.x2 = y2; - rectangle.y2 = page_height - x1; - break; - case 180: - rectangle.x1 = (page_height - y1); - rectangle.y1 = (page_width - x2); - rectangle.x2 = (page_height - y2); - rectangle.y2 = (page_width - x1); - break; - case 270: - rectangle.x1 = page_width - y2; - rectangle.y1 = x1; - rectangle.x2 = page_width - y1; - rectangle.y2 = x2; - break; - } - - /* reset points of the rectangle so that p1 is in the top-left corner - * and p2 is in the bottom right corner */ - if(rectangle.x1 > rectangle.x2) - { - double d = rectangle.x1; - rectangle.x1 = rectangle.x2; - rectangle.x2 = d; - } - if(rectangle.y2 > rectangle.y1) - { - double d = rectangle.y1; - rectangle.y1 = rectangle.y2; - rectangle.y2 = d; - } - -#if !POPPLER_CHECK_VERSION(0,15,0) - /* adapt y coordinates */ - rectangle.y1 = page_height - rectangle.y1; - rectangle.y2 = page_height - rectangle.y2; -#endif - - /* get selected text */ - g_static_mutex_lock(&(Zathura.Lock.pdflib_lock)); - char* selected_text = poppler_page_get_selected_text( - Zathura.PDF.pages[Zathura.PDF.page_number]->page,SELECTION_STYLE, - &rectangle); - - if(selected_text) - { - gtk_clipboard_set_text(gtk_clipboard_get(GDK_SELECTION_PRIMARY), selected_text, -1); - g_free(selected_text); - } - g_static_mutex_unlock(&(Zathura.Lock.pdflib_lock)); - - - return TRUE; -} - -gboolean -cb_view_motion_notify(GtkWidget* widget, GdkEventMotion* event, gpointer data) -{ - return TRUE; -} - -gboolean -cb_view_scrolled(GtkWidget* widget, GdkEventScroll* event, gpointer data) -{ - int i; - for(i = 0; i < LENGTH(mouse_scroll_events); i++) - { - if(event->direction == mouse_scroll_events[i].direction) - { - mouse_scroll_events[i].function(&(mouse_scroll_events[i].argument)); - return TRUE; - } - } - - return FALSE; -} - -gboolean -cb_watch_file(GFileMonitor* monitor, GFile* file, GFile* other_file, GFileMonitorEvent event, gpointer data) -{ - if(event != G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT) - return FALSE; - - sc_reload(NULL); - - return TRUE; -} - -/* main function */ -int main(int argc, char* argv[]) -{ - /* embed */ - Zathura.UI.embed = 0; - - Zathura.Config.config_dir = 0; - Zathura.Config.data_dir = 0; - - char* config_dir = 0; - char* data_dir = 0; - GOptionEntry entries[] = - { - { "reparent", 'e', 0, G_OPTION_ARG_INT, &Zathura.UI.embed, "Reparents to window specified by xid", "xid" }, - { "config-dir", 'c', 0, G_OPTION_ARG_FILENAME, &config_dir, "Path to the config directory", "path" }, - { "data-dir", 'd', 0, G_OPTION_ARG_FILENAME, &data_dir, "Path to the data directory", "path" }, - { NULL } +zathura_t* +zathura_init(int argc, char* argv[]) +{ + /* parse command line options */ + GdkNativeWindow embed = 0; + gchar* config_dir = NULL, *data_dir = NULL, *plugin_path = NULL; + GOptionEntry entries[] = + { + { "reparent", 'e', 0, G_OPTION_ARG_INT, &embed, "Reparents to window specified by xid", "xid" }, + { "config-dir", 'c', 0, G_OPTION_ARG_FILENAME, &config_dir, "Path to the config directory", "path" }, + { "data-dir", 'd', 0, G_OPTION_ARG_FILENAME, &data_dir, "Path to the data directory", "path" }, + { "plugins-dir", 'p', 0, G_OPTION_ARG_STRING, &plugin_path, "Path to the directories containing plugins", "path" }, + { NULL, '\0', 0, 0, NULL, NULL, NULL } }; GOptionContext* context = g_option_context_new(" [file] [password]"); g_option_context_add_main_entries(context, entries, NULL); GError* error = NULL; - if(!g_option_context_parse(context, &argc, &argv, &error)) + if (!g_option_context_parse(context, &argc, &argv, &error)) { printf("Error parsing command line arguments: %s\n", error->message); g_option_context_free(context); g_error_free(error); - return 1; + goto error_free; } g_option_context_free(context); - if (config_dir) - Zathura.Config.config_dir = g_strdup(config_dir); - if (data_dir) - Zathura.Config.data_dir = g_strdup(data_dir); + zathura_t* zathura = malloc(sizeof(zathura_t)); - g_thread_init(NULL); - gdk_threads_init(); - - gtk_init(&argc, &argv); - - init_zathura(); - init_directories(); - init_keylist(); - read_configuration(); - init_settings(); - init_bookmarks(); - init_look(); - - if(argc > 1) - { - char* password = (argc == 3) ? argv[2] : NULL; - if (strcmp(argv[1], "-") == 0) - open_stdin(password); - else - open_file(argv[1], password); + if (zathura == NULL) { + return NULL; } - switch_view(Zathura.UI.document); - update_status(); + /* general */ + zathura->document = NULL; - gtk_widget_show_all(GTK_WIDGET(Zathura.UI.window)); - gtk_widget_grab_focus(GTK_WIDGET(Zathura.UI.view)); + /* plugins */ + zathura->plugins.plugins = girara_list_new(); + zathura->plugins.path = girara_list_new(); + girara_list_set_free_function(zathura->plugins.path, g_free); - if(!Zathura.Global.show_inputbar) - gtk_widget_hide(GTK_WIDGET(Zathura.UI.inputbar)); + if (config_dir) { + zathura->config.config_dir = g_strdup(config_dir); + } else { + gchar* path = girara_get_xdg_path(XDG_CONFIG); + zathura->config.config_dir = g_build_filename(path, "zathura", NULL); + g_free(path); + } - if(!Zathura.Global.show_statusbar) - gtk_widget_hide(GTK_WIDGET(Zathura.UI.statusbar)); + if (data_dir) { + zathura->config.data_dir = g_strdup(config_dir); + } else { + gchar* path = girara_get_xdg_path(XDG_DATA); + zathura->config.data_dir = g_build_filename(path, "zathura", NULL); + g_free(path); + } + + if (plugin_path) { + gchar** paths = g_strsplit(plugin_path, ":", 0); + for (unsigned int i = 0; paths[i] != '\0'; ++i) { + girara_list_append(zathura->plugins.path, g_strdup(paths[i])); + } + g_strfreev(paths); + } else { + /* XXX: this shouldn't be hard coded! */ + girara_list_append(zathura->plugins.path, g_strdup("/usr/local/lib/zathura")); + girara_list_append(zathura->plugins.path, g_strdup("/usr/lib/zathura")); + } + + /* UI */ + if ((zathura->ui.session = girara_session_create()) == NULL) { + goto error_out; + } + + zathura->ui.session->global.data = zathura; + zathura->ui.statusbar.file = NULL; + zathura->ui.statusbar.buffer = NULL; + zathura->ui.statusbar.page_number = NULL; + zathura->ui.page_view = NULL; + zathura->ui.index = NULL; + + /* print settings */ + zathura->print.settings = NULL; + zathura->print.page_setup = NULL; + + /* global settings */ + zathura->global.recolor = false; + + /* load plugins */ + zathura_document_plugins_load(zathura); + + /* configuration */ + config_load_default(zathura); + + /* load global configuration files */ + config_load_file(zathura, GLOBAL_RC); + + /* load local configuration files */ + char* configuration_file = g_build_filename(zathura->config.config_dir, ZATHURA_RC, NULL); + config_load_file(zathura, configuration_file); + free(configuration_file); + + /* initialize girara */ + zathura->ui.session->gtk.embed = embed; + if (girara_session_init(zathura->ui.session) == false) { + goto error_out; + } + + /* girara events */ + zathura->ui.session->events.buffer_changed = buffer_changed; + + /* page view */ + zathura->ui.page_view = gtk_table_new(0, 0, TRUE); + if (!zathura->ui.page_view) { + goto error_free; + } + + /* callbacks */ + GtkAdjustment* view_vadjustment = gtk_scrolled_window_get_vadjustment(GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view)); + g_signal_connect(G_OBJECT(view_vadjustment), "value-changed", G_CALLBACK(cb_view_vadjustment_value_changed), zathura); + GtkAdjustment* view_hadjustment = gtk_scrolled_window_get_hadjustment(GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view)); + g_signal_connect(G_OBJECT(view_hadjustment), "value-changed", G_CALLBACK(cb_view_vadjustment_value_changed), zathura); + + gtk_widget_show(zathura->ui.page_view); + + /* statusbar */ + zathura->ui.statusbar.file = girara_statusbar_item_add(zathura->ui.session, TRUE, TRUE, TRUE, NULL); + if (zathura->ui.statusbar.file == NULL) { + goto error_free; + } + + zathura->ui.statusbar.buffer = girara_statusbar_item_add(zathura->ui.session, FALSE, FALSE, FALSE, NULL); + if (zathura->ui.statusbar.buffer == NULL) { + goto error_free; + } + + zathura->ui.statusbar.page_number = girara_statusbar_item_add(zathura->ui.session, FALSE, FALSE, FALSE, NULL); + if (!zathura->ui.statusbar.page_number) { + goto error_free; + } + + girara_statusbar_item_set_text(zathura->ui.session, zathura->ui.statusbar.file, "[No Name]"); + + /* signals */ + g_signal_connect(G_OBJECT(zathura->ui.session->gtk.window), "destroy", G_CALLBACK(cb_destroy), NULL); + + /* save page padding */ + int* page_padding = girara_setting_get(zathura->ui.session, "page-padding"); + zathura->global.page_padding = (page_padding) ? *page_padding : 1; + + gtk_table_set_row_spacings(GTK_TABLE(zathura->ui.page_view), zathura->global.page_padding); + gtk_table_set_col_spacings(GTK_TABLE(zathura->ui.page_view), zathura->global.page_padding); + + /* parse colors */ + char* string_value = girara_setting_get(zathura->ui.session, "recolor-dark-color"); + if (string_value != NULL) { + gdk_color_parse(string_value, &(zathura->ui.colors.recolor_dark_color)); + free(string_value); + } + + string_value = girara_setting_get(zathura->ui.session, "recolor-light-color"); + if (string_value != NULL) { + gdk_color_parse(string_value, &(zathura->ui.colors.recolor_light_color)); + free(string_value); + } + + /* database */ + char* database_path = g_build_filename(zathura->config.data_dir, "bookmarks.sqlite", NULL); + zathura->database = zathura_db_init(database_path); + if (zathura->database == NULL) { + girara_error("Unable to inizialize database. Bookmarks won't be available."); + } + g_free(database_path); + + /* bookmarks */ + zathura->bookmarks.bookmarks = girara_list_new(); + girara_list_set_free_function(zathura->bookmarks.bookmarks, (girara_free_function_t) zathura_bookmark_free); + + /* open document if passed */ + if (argc > 1) { + zathura_document_info_t* document_info = malloc(sizeof(zathura_document_info_t)); + + if (document_info != NULL) { + document_info->zathura = zathura; + document_info->path = argv[1]; + document_info->password = (argc >= 2) ? argv[2] : NULL; + g_idle_add(document_info_open, document_info); + } + } + + return zathura; + +error_free: + + if (zathura->ui.page_view) { + g_object_unref(zathura->ui.page_view); + } + + girara_session_destroy(zathura->ui.session); + girara_list_free(zathura->bookmarks.bookmarks); + zathura_db_free(zathura->database); + +error_out: + + free(zathura); + + return NULL; +} + +void +zathura_free(zathura_t* zathura) +{ + if (zathura == NULL) { + return; + } + + if (zathura->ui.session != NULL) { + girara_session_destroy(zathura->ui.session); + } + + document_close(zathura); + + /* bookmarks */ + girara_list_free(zathura->bookmarks.bookmarks); + + /* database */ + zathura_db_free(zathura->database); + + /* free print settings */ + if(zathura->print.settings != NULL) { + g_object_unref(zathura->print.settings); + } + + if (zathura->print.page_setup != NULL) { + g_object_unref(zathura->print.page_setup); + } + + /* free registered plugins */ + zathura_document_plugins_free(zathura); + girara_list_free(zathura->plugins.plugins); + girara_list_free(zathura->plugins.path); + + /* free config variables */ + g_free(zathura->config.config_dir); + g_free(zathura->config.data_dir); +} + +gboolean +document_info_open(gpointer data) +{ + zathura_document_info_t* document_info = data; + g_return_val_if_fail(document_info != NULL, FALSE); + + if (document_info->zathura == NULL || document_info->path == NULL) { + free(document_info); + return FALSE; + } + + document_open(document_info->zathura, document_info->path, document_info->password); + free(document_info); + + return FALSE; +} + +bool +document_open(zathura_t* zathura, const char* path, const char* password) +{ + if (!path) { + goto error_out; + } + + zathura_document_t* document = zathura_document_open(zathura, path, password); + + if (!document) { + goto error_out; + } + + zathura->document = document; + + /* view mode */ + int* value = girara_setting_get(zathura->ui.session, "pages-per-row"); + int pages_per_row = (value) ? *value : 1; + free(value); + page_view_set_mode(zathura, pages_per_row); + + girara_set_view(zathura->ui.session, zathura->ui.page_view); + + /* threads */ + zathura->sync.render_thread = render_init(zathura); + + if (!zathura->sync.render_thread) { + goto error_free; + } + + /* create blank pages */ + for (unsigned int page_id = 0; page_id < document->number_of_pages; page_id++) { + zathura_page_t* page = document->pages[page_id]; + gtk_widget_realize(page->event_box); + } + + /* bookmarks */ + if (!zathura_bookmarks_load(zathura, zathura->document->file_path)) { + girara_warning("Failed to load bookmarks for %s.\n", zathura->document->file_path); + } + + return true; + +error_free: + + zathura_document_free(document); + +error_out: + + return false; +} + +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); + if (!overwrite && g_file_test(file_path, G_FILE_TEST_EXISTS)) + { + gchar* message = g_strdup_printf("File already exists: %s. Use :write! to overwrite it.", file_path); + girara_error(message); + g_free(message); + return false; + } + + bool res = zathura_document_save_as(zathura->document, file_path); + g_free(file_path); + return res; +} + +static void +remove_page_from_table(GtkWidget* page, gpointer permanent) +{ + if (!permanent) { + g_object_ref(G_OBJECT(page)); + } + + gtk_container_remove(GTK_CONTAINER(page->parent), page); +} + +bool +document_close(zathura_t* zathura) +{ + if (!zathura->document) { + return false; + } + + render_free(zathura->sync.render_thread); + zathura->sync.render_thread = NULL; + + gtk_container_foreach(GTK_CONTAINER(zathura->ui.page_view), remove_page_from_table, (gpointer)1); + + zathura_document_free(zathura->document); + zathura->document = NULL; + + gtk_widget_hide_all(zathura->ui.page_view); + + return true; +} + +bool +page_set(zathura_t* zathura, unsigned int page_id) +{ + if (!zathura->document || !zathura->document->pages) { + goto error_out; + } + + if (page_id >= zathura->document->number_of_pages) { + goto error_out; + } + + /* render page */ + zathura_page_t* page = zathura->document->pages[page_id]; + + if (!page) { + goto error_out; + } + + page_offset_t* offset = page_calculate_offset(page); + if (offset == NULL) { + goto error_out; + } + + GtkAdjustment* view_vadjustment = gtk_scrolled_window_get_vadjustment(GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view)); + GtkAdjustment* view_hadjustment = gtk_scrolled_window_get_hadjustment(GTK_SCROLLED_WINDOW(zathura->ui.session->gtk.view)); + gtk_adjustment_set_value(view_hadjustment, offset->x); + gtk_adjustment_set_value(view_vadjustment, offset->y); + + /* update page number */ + zathura->document->current_page_number = page_id; + statusbar_page_number_update(zathura); + + return true; + +error_out: + + return false; +} + +void +statusbar_page_number_update(zathura_t* zathura) +{ + if (zathura == NULL || zathura->ui.statusbar.page_number == NULL) { + return; + } + + char* page_number_text = g_strdup_printf("[%d/%d]", zathura->document->current_page_number + 1, zathura->document->number_of_pages); + girara_statusbar_item_set_text(zathura->ui.session, zathura->ui.statusbar.page_number, page_number_text); + g_free(page_number_text); +} + +void +page_view_set_mode(zathura_t* zathura, unsigned int pages_per_row) +{ + /* show at least one page */ + if (pages_per_row == 0) { + pages_per_row = 1; + } + + if (zathura->document == NULL) { + return; + } + + gtk_container_foreach(GTK_CONTAINER(zathura->ui.page_view), remove_page_from_table, (gpointer)0); + + gtk_table_resize(GTK_TABLE(zathura->ui.page_view), zathura->document->number_of_pages / pages_per_row + 1, pages_per_row); + for (unsigned int i = 0; i < zathura->document->number_of_pages; i++) + { + int x = i % pages_per_row; + int y = i / pages_per_row; + gtk_table_attach(GTK_TABLE(zathura->ui.page_view), zathura->document->pages[i]->event_box, x, x + 1, y, y + 1, GTK_EXPAND, GTK_EXPAND, 0, 0); + } + + gtk_widget_show_all(zathura->ui.page_view); +} + +/* main function */ +int main(int argc, char* argv[]) +{ + g_thread_init(NULL); + gdk_threads_init(); + gtk_init(&argc, &argv); + + zathura_t* zathura = zathura_init(argc, argv); + if (zathura == NULL) { + printf("error: coult not initialize zathura\n"); + return -1; + } gdk_threads_enter(); gtk_main(); gdk_threads_leave(); + zathura_free(zathura); + return 0; } diff --git a/zathura.h b/zathura.h new file mode 100644 index 0000000..7035542 --- /dev/null +++ b/zathura.h @@ -0,0 +1,180 @@ +/* See LICENSE file for license and copyright information */ + +#ifndef ZATHURA_H +#define ZATHURA_H + +#include +#include + +#ifdef UNUSED +#elif defined(__GNUC__) +# define UNUSED(x) UNUSED_ ## x __attribute__((unused)) +#elif defined(__LCINT__) +# define UNUSED(x) /*@unused@*/ x +#else +# define UNUSED(x) x +#endif + +enum { NEXT, PREVIOUS, LEFT, RIGHT, UP, DOWN, BOTTOM, TOP, HIDE, HIGHLIGHT, + DELETE_LAST_WORD, DELETE_LAST_CHAR, DEFAULT, ERROR, WARNING, NEXT_GROUP, + PREVIOUS_GROUP, ZOOM_IN, ZOOM_OUT, ZOOM_ORIGINAL, ZOOM_SPECIFIC, FORWARD, + BACKWARD, ADJUST_BESTFIT, ADJUST_WIDTH, ADJUST_NONE, CONTINUOUS, DELETE_LAST, + ADD_MARKER, EVAL_MARKER, EXPAND, COLLAPSE, SELECT, GOTO_DEFAULT, GOTO_LABELS, + GOTO_OFFSET, HALF_UP, HALF_DOWN, FULL_UP, FULL_DOWN, NEXT_CHAR, PREVIOUS_CHAR, + DELETE_TO_LINE_START, APPEND_FILEPATH }; + +/* forward declaration for types from document.h */ +struct zathura_document_s; +struct zathura_page_s; +typedef struct zathura_document_s zathura_document_t; +typedef struct zathura_page_s zathura_page_t; + +/* forward declaration for types form database.h */ +struct zathura_database_s; +typedef struct zathura_database_s zathura_database_t; + +/* forward declaration for types from render.h */ +struct render_thread_s; +typedef struct render_thread_s render_thread_t; + +typedef struct zathura_s +{ + struct + { + girara_session_t* session; /**> girara interface session */ + + struct + { + girara_statusbar_item_t* buffer; /**> buffer statusbar entry */ + girara_statusbar_item_t* file; /**> file statusbar entry */ + girara_statusbar_item_t* page_number; /**> page number statusbar entry */ + } statusbar; + + struct + { + GdkColor recolor_dark_color; /**> Dark color for recoloring */ + GdkColor recolor_light_color; /**> Light color for recoloring */ + } colors; + + GtkWidget *page_view; /**> Widget that contains all rendered pages */ + GtkWidget *index; /**> Widget to show the index of the document */ + } ui; + + struct + { + render_thread_t* render_thread; /**> The thread responsible for rendering the pages */ + } sync; + + struct + { + girara_list_t* plugins; /**> List of plugins */ + girara_list_t* path; /**> List of plugin paths */ + } plugins; + + struct + { + gchar* config_dir; /**> Path to the configuration directory */ + gchar* data_dir; /**> Path to the data directory */ + } config; + + struct + { + GtkPrintSettings* settings; /**> Print settings */ + GtkPageSetup* page_setup; /**> Saved page setup */ + } print; + + struct + { + unsigned int page_padding; /**> Padding between the pages */ + bool recolor; /**> Recoloring mode switch */ + } global; + + struct + { + girara_mode_t normal; /**> Normal mode */ + girara_mode_t fullscreen; /**> Fullscreen mode */ + girara_mode_t index; /**> Index mode */ + girara_mode_t insert; /**> Insert mode */ + } modes; + + struct + { + gchar* file; /**> bookmarks file */ + girara_list_t* bookmarks; /**> bookmarks */ + } bookmarks; + + zathura_document_t* document; /**> The current document */ + zathura_database_t* database; /**> The database */ +} zathura_t; + +/** + * Initializes zathura + * + * @param argc Number of arguments + * @param argv Values of arguments + * @return zathura session object or NULL if zathura could not been initialized + */ +zathura_t* zathura_init(int argc, char* argv[]); + +/** + * Free zathura session + * + * @param zathura The zathura session + */ +void zathura_free(zathura_t* zathura); + +/** + * Opens a file + * + * @param zathura The zathura session + * @param path The path to the file + * @param password The password of the file + * + * @return If no error occured true, otherwise false, is returned. + */ +bool document_open(zathura_t* zathura, const char* path, const char* password); + +/** + * Save a open file + * + * @param zathura The zathura session + * @param path The path + * @param overwrite Overwrite existing file + * + * @return If no error occured true, otherwise false, is returned. + */ +bool document_save(zathura_t* zathura, const char* path, bool overwrite); + +/** + * Closes the current opened document + * + * @param zathura The zathura session + * @return If no error occured true, otherwise false, is returned. + */ +bool document_close(zathura_t* zathura); + +/** + * Opens the page with the given number + * + * @param zathura The zathura session + * @return If no error occured true, otherwise false, is returned. + */ +bool page_set(zathura_t* zathura, unsigned int page_id); + +/** + * Builds the box structure to show the rendered pages + * + * @param zathura The zathura session + * @param pages_per_row Number of shown pages per row + */ +void page_view_set_mode(zathura_t* zathura, unsigned int pages_per_row); + +/** + * Updates the page number in the statusbar. Note that 1 will be added to the + * displayed number + * + * @param zathura The zathura session + */ +void statusbar_page_number_update(zathura_t* zathura); + +#endif // ZATHURA_H diff --git a/zathura.pc.in b/zathura.pc.in new file mode 100644 index 0000000..6e36e4f --- /dev/null +++ b/zathura.pc.in @@ -0,0 +1,9 @@ + +INC_PATH=-I${includedir} + +Name: ${project} +Description: document viewer +Version: ${version} +URL: http://pwmt.org/projects/zathura +Cflags: ${INC_PATH} +Libs: