Send rectangles not on the main page too

Signed-off-by: Sebastian Ramacher <sebastian+dev@ramacher.at>
This commit is contained in:
Sebastian Ramacher 2014-01-14 15:05:09 +01:00
parent 7e6112d883
commit 74fd28b4b2
5 changed files with 114 additions and 24 deletions

View file

@ -22,6 +22,7 @@
<method name='HighlightRects'> <method name='HighlightRects'>
<arg type='i' name='page' direction='in' /> <arg type='i' name='page' direction='in' />
<arg type='a(dddd)' name='rectangles' direction='in' /> <arg type='a(dddd)' name='rectangles' direction='in' />
<arg type='a(idddd)' name='secondary_rectangles' direction='in' />
<arg type='b' name='return' direction='out' /> <arg type='b' name='return' direction='out' />
</method> </method>
<property type='s' name='filename' access='read' /> <property type='s' name='filename' access='read' />

View file

@ -5,6 +5,8 @@
#include "macros.h" #include "macros.h"
#include "zathura.h" #include "zathura.h"
#include "document.h" #include "document.h"
#include "utils.h"
#include <girara/utils.h> #include <girara/utils.h>
#include <gio/gio.h> #include <gio/gio.h>
#include <sys/types.h> #include <sys/types.h>
@ -149,6 +151,28 @@ zathura_dbus_new(zathura_t* zathura)
/* D-Bus handler */ /* D-Bus handler */
static void
highlight_rects(zathura_t* zathura, unsigned int page,
girara_list_t** rectangles)
{
const unsigned int number_of_pages = zathura_document_get_number_of_pages(zathura->document);
for (unsigned int p = 0; p != number_of_pages; ++p) {
GObject* widget = G_OBJECT(priv->zathura->pages[p]);
g_object_set(widget, "draw-links", FALSE, "search-results", rectangles[p],
NULL);
if (p == page) {
g_object_set(widget, "search-current", 0, NULL);
} else {
g_object_set(widget, "search-current", -1, NULL);
}
}
page_set(zathura, page);
document_draw_search_results(zathura, true);
}
static void static void
handle_method_call(GDBusConnection* UNUSED(connection), handle_method_call(GDBusConnection* UNUSED(connection),
const gchar* UNUSED(sender), const gchar* object_path, const gchar* UNUSED(sender), const gchar* object_path,
@ -211,29 +235,52 @@ handle_method_call(GDBusConnection* UNUSED(connection),
} else if (g_strcmp0(method_name, "HighlightRects") == 0) { } else if (g_strcmp0(method_name, "HighlightRects") == 0) {
gint page = ZATHURA_PAGE_NUMBER_UNSPECIFIED; gint page = ZATHURA_PAGE_NUMBER_UNSPECIFIED;
GVariantIter* iter = NULL; GVariantIter* iter = NULL;
g_variant_get(parameters, "(ia(dddd))", &page, &iter); GVariantIter* secondary_iter = NULL;
g_variant_get(parameters, "(ia(dddd)a(idddd))", &page, &iter,
&secondary_iter);
if (page < 1 || (unsigned int)page >= number_of_pages) { if (page < 1 || (unsigned int)page >= number_of_pages) {
GVariant* result = g_variant_new("(b)", false); GVariant* result = g_variant_new("(b)", false);
g_variant_iter_free(iter);
g_variant_iter_free(secondary_iter);
g_dbus_method_invocation_return_value(invocation, result); g_dbus_method_invocation_return_value(invocation, result);
return;
} }
/* get rectangles */ /* get rectangles */
girara_list_t* rectangles = girara_list_new2(g_free); girara_list_t** rectangles = g_malloc0(number_of_pages * sizeof(girara_list_t*));
rectangles[page - 1] = girara_list_new2(g_free);
zathura_rectangle_t temp_rect; zathura_rectangle_t temp_rect;
while (g_variant_iter_loop(iter, "(dddd)", &temp_rect.x1, &temp_rect.x2, while (g_variant_iter_loop(iter, "(dddd)", &temp_rect.x1, &temp_rect.x2,
&temp_rect.y1, &temp_rect.y2)) { &temp_rect.y1, &temp_rect.y2)) {
zathura_rectangle_t* rect = g_malloc0(sizeof(zathura_rectangle_t)); zathura_rectangle_t* rect = g_malloc0(sizeof(zathura_rectangle_t));
memcpy(rect, &temp_rect, sizeof(zathura_rectangle_t)); memcpy(rect, &temp_rect, sizeof(zathura_rectangle_t));
girara_list_append(rectangles, rect); girara_list_append(rectangles[page - 1], rect);
} }
g_variant_iter_free(iter); g_variant_iter_free(iter);
page_set(priv->zathura, page - 1); /* get secondary rectangles */
int temp_page = ZATHURA_PAGE_NUMBER_UNSPECIFIED;
while (g_variant_iter_loop(secondary_iter, "(idddd)", &temp_page,
&temp_rect.x1, &temp_rect.x2, &temp_rect.y1, &temp_rect.y2)) {
if (temp_page < 0 || (unsigned int)temp_page >= number_of_pages) {
/* error out here? */
continue;
}
GObject* widget = G_OBJECT(priv->zathura->pages[page - 1]); if (rectangles[temp_page] == NULL) {
g_object_set(widget, "draw-links", FALSE, "search-results", rectangles, rectangles[temp_page] = girara_list_new2(g_free);
"search-current", 0, "draw-search-results", TRUE, NULL); }
zathura_rectangle_t* rect = g_malloc0(sizeof(zathura_rectangle_t));
memcpy(rect, &temp_rect, sizeof(zathura_rectangle_t));
girara_list_append(rectangles[temp_page], rect);
}
g_variant_iter_free(secondary_iter);
highlight_rects(priv->zathura, page - 1, rectangles);
g_free(rectangles);
} }
} }
@ -268,7 +315,7 @@ static const unsigned int TIMEOUT = 3000;
bool bool
zathura_dbus_goto_page_and_highlight(const char* filename, int page, zathura_dbus_goto_page_and_highlight(const char* filename, int page,
girara_list_t* rectangles) girara_list_t* rectangles, girara_list_t* secondary_rects)
{ {
/* note: page is [1, number_of_pages] here */ /* note: page is [1, number_of_pages] here */
@ -343,9 +390,17 @@ zathura_dbus_goto_page_and_highlight(const char* filename, int page,
GIRARA_LIST_FOREACH_END(rectangles, zathura_rectangle_t*, iter, rect); GIRARA_LIST_FOREACH_END(rectangles, zathura_rectangle_t*, iter, rect);
} }
GVariantBuilder* second_builder = g_variant_builder_new(G_VARIANT_TYPE("a(idddd)"));
if (secondary_rects != NULL) {
GIRARA_LIST_FOREACH(secondary_rects, synctex_page_rect_t*, iter, rect)
g_variant_builder_add(builder, "(idddd)", rect->page, rect->rect.x1,
rect->rect.x2, rect->rect.y1, rect->rect.y2);
GIRARA_LIST_FOREACH_END(secondary_rects, synctex_page_rect_t*, iter, rect);
}
GVariant* ret = g_dbus_connection_call_sync(connection, GVariant* ret = g_dbus_connection_call_sync(connection,
name, DBUS_OBJPATH, DBUS_INTERFACE, "HighlightRects", name, DBUS_OBJPATH, DBUS_INTERFACE, "HighlightRects",
g_variant_new("(ia(dddd))", page, builder), g_variant_new("(ia(dddd)a(idddd))", page, builder, second_builder),
G_VARIANT_TYPE("(b)"), G_DBUS_CALL_FLAGS_NONE, TIMEOUT, NULL, &error); G_VARIANT_TYPE("(b)"), G_DBUS_CALL_FLAGS_NONE, TIMEOUT, NULL, &error);
g_variant_builder_unref(builder); g_variant_builder_unref(builder);
if (ret == NULL) { if (ret == NULL) {
@ -370,12 +425,16 @@ zathura_dbus_synctex_position(const char* filename, const char* position)
} }
int page = ZATHURA_PAGE_NUMBER_UNSPECIFIED; int page = ZATHURA_PAGE_NUMBER_UNSPECIFIED;
girara_list_t* rectangles = synctex_rectangles_from_position(filename, position, &page); girara_list_t* secondary_rects = NULL;
girara_list_t* rectangles = synctex_rectangles_from_position(filename,
position, &page, &secondary_rects);
if (rectangles == NULL) { if (rectangles == NULL) {
return false; return false;
} }
bool ret = zathura_dbus_goto_page_and_highlight(filename, page, rectangles); const bool ret = zathura_dbus_goto_page_and_highlight(filename, page,
rectangles, secondary_rects);
girara_list_free(rectangles); girara_list_free(rectangles);
girara_list_free(secondary_rects);
return ret; return ret;
} }

View file

@ -47,11 +47,13 @@ ZathuraDbus* zathura_dbus_new(zathura_t* zathura);
* @param filename filename * @param filename filename
* @param page page number * @param page page number
* @param rectangles list of rectangles to highlight * @param rectangles list of rectangles to highlight
* @param secondary_rects list of synctex_page_rect_ts for rectangles not on the
* page given by page
* @returns true if a instance was found that has the given filename open, false * @returns true if a instance was found that has the given filename open, false
* otherwise * otherwise
*/ */
bool zathura_dbus_goto_page_and_highlight(const char* filename, int page, bool zathura_dbus_goto_page_and_highlight(const char* filename, int page,
girara_list_t* rectangles); girara_list_t* rectangles, girara_list_t* secondary_rects);
bool zathura_dbus_synctex_position(const char* filename, const char* position); bool zathura_dbus_synctex_position(const char* filename, const char* position);

View file

@ -1,6 +1,7 @@
/* See LICENSE file for license and copyright information */ /* See LICENSE file for license and copyright information */
#include <glib.h> #include <glib.h>
#include <string.h>
#include "synctex.h" #include "synctex.h"
@ -69,7 +70,8 @@ synctex_edit(zathura_t* zathura, zathura_page_t* page, int x, int y)
argv[0] = g_strdup("synctex"); argv[0] = g_strdup("synctex");
argv[1] = g_strdup("edit"); argv[1] = g_strdup("edit");
argv[2] = g_strdup("-o"); argv[2] = g_strdup("-o");
argv[3] = g_strdup_printf("%d:%d:%d:%s", zathura_page_get_index(page) + 1, x, y, filename); argv[3] = g_strdup_printf("%d:%d:%d:%s", zathura_page_get_index(page) + 1, x,
y, filename);
if (zathura->synctex.editor != NULL) { if (zathura->synctex.editor != NULL) {
argv[4] = g_strdup("-x"); argv[4] = g_strdup("-x");
argv[5] = g_strdup(zathura->synctex.editor); argv[5] = g_strdup(zathura->synctex.editor);
@ -93,9 +95,10 @@ scan_float(GScanner* scanner)
} }
girara_list_t* girara_list_t*
synctex_rectangles_from_position(const char* filename, const char* position, int* page) synctex_rectangles_from_position(const char* filename, const char* position,
int* page, girara_list_t** secondary_rects)
{ {
if (filename == NULL || position == NULL || page == NULL) { if (filename == NULL || position == NULL || page) {
return NULL; return NULL;
} }
@ -107,7 +110,7 @@ synctex_rectangles_from_position(const char* filename, const char* position, int
argv[4] = g_strdup("-o"); argv[4] = g_strdup("-o");
argv[5] = g_strdup(filename); argv[5] = g_strdup(filename);
gint output; gint output = -1;
bool ret = g_spawn_async_with_pipes(NULL, argv, NULL, bool ret = g_spawn_async_with_pipes(NULL, argv, NULL,
G_SPAWN_SEARCH_PATH | G_SPAWN_STDERR_TO_DEV_NULL, NULL, NULL, NULL, NULL, G_SPAWN_SEARCH_PATH | G_SPAWN_STDERR_TO_DEV_NULL, NULL, NULL, NULL, NULL,
&output, NULL, NULL); &output, NULL, NULL);
@ -147,9 +150,10 @@ synctex_rectangles_from_position(const char* filename, const char* position, int
} }
} }
*page = ZATHURA_PAGE_NUMBER_UNSPECIFIED; int rpage = ZATHURA_PAGE_NUMBER_UNSPECIFIED;
int current_page; int current_page = ZATHURA_PAGE_NUMBER_UNSPECIFIED;
girara_list_t* hitlist = girara_list_new2(g_free);; girara_list_t* hitlist = girara_list_new2(g_free);
girara_list_t* other_rects = girara_list_new2(g_free);
zathura_rectangle_t* rectangle = NULL; zathura_rectangle_t* rectangle = NULL;
while (found_end == false) { while (found_end == false) {
@ -167,13 +171,18 @@ synctex_rectangles_from_position(const char* filename, const char* position, int
case SYNCTEX_PROP_PAGE: case SYNCTEX_PROP_PAGE:
if (g_scanner_get_next_token(scanner) == G_TOKEN_INT) { if (g_scanner_get_next_token(scanner) == G_TOKEN_INT) {
current_page = g_scanner_cur_value(scanner).v_int; current_page = g_scanner_cur_value(scanner).v_int;
if (*page == ZATHURA_PAGE_NUMBER_UNSPECIFIED) { if (rpage == ZATHURA_PAGE_NUMBER_UNSPECIFIED) {
*page = current_page; rpage = current_page;
} }
if (*page == current_page && rectangle != NULL) { if (*page == current_page && rectangle != NULL) {
girara_list_append(hitlist, rectangle); girara_list_append(hitlist, rectangle);
rectangle = NULL; rectangle = NULL;
} else if (rectangle != NULL) {
synctex_page_rect_t* page_rect = g_malloc0(sizeof(synctex_page_rect_t));
page_rect->page = current_page;
memcpy(&page_rect->rect, rectangle, sizeof(zathura_rectangle_t));
girara_list_append(other_rects, page_rect);
} }
g_free(rectangle); g_free(rectangle);
@ -205,9 +214,13 @@ synctex_rectangles_from_position(const char* filename, const char* position, int
} }
if (rectangle != NULL) { if (rectangle != NULL) {
if (current_page == *page) { if (current_page == rpage) {
girara_list_append(hitlist, rectangle); girara_list_append(hitlist, rectangle);
} else { } else {
synctex_page_rect_t* page_rect = g_malloc0(sizeof(synctex_page_rect_t));
page_rect->page = current_page;
memcpy(&page_rect->rect, rectangle, sizeof(zathura_rectangle_t));
girara_list_append(other_rects, page_rect);
g_free(rectangle); g_free(rectangle);
} }
} }
@ -215,5 +228,14 @@ synctex_rectangles_from_position(const char* filename, const char* position, int
g_scanner_destroy(scanner); g_scanner_destroy(scanner);
close(output); close(output);
if (page != NULL) {
*page = rpage;
}
if (secondary_rects != NULL) {
*secondary_rects = other_rects;
} else {
girara_list_free(other_rects);
}
return hitlist; return hitlist;
} }

View file

@ -5,7 +5,13 @@
#include "types.h" #include "types.h"
typedef struct synctex_page_rect_s {
int page;
zathura_rectangle_t rect;
} synctex_page_rect_t;
void synctex_edit(zathura_t* zathura, zathura_page_t* page, int x, int y); void synctex_edit(zathura_t* zathura, zathura_page_t* page, int x, int y);
girara_list_t* synctex_rectangles_from_position(const char* filename, const char* position, int* page); girara_list_t* synctex_rectangles_from_position(const char* filename,
const char* position, int* page, girara_list_t** secondary_rects);
#endif #endif