[latexila/wip/latexila-next: 38/55] LatexilaSynctex
- From: Sébastien Wilmet <swilmet src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [latexila/wip/latexila-next: 38/55] LatexilaSynctex
- Date: Sun, 28 Sep 2014 15:47:44 +0000 (UTC)
commit 91acbd87b441b60ff547d9424451df5b0afed70e
Author: Sébastien Wilmet <swilmet gnome org>
Date: Sat May 17 01:25:32 2014 +0200
LatexilaSynctex
It is needed for the build tools for opening a PDF file with synctex
support with evince. Since I don't want to use the Vala code in C, a
rewrite of synctex.vala in C was needed.
synctex.vala uses the synchronous D-Bus API. The rewrite in C uses the
asynchronous API (better).
TODO: send a signal for the sync_source (PDF -> .tex)
configure.ac | 2 +
docs/reference/latexila-docs.xml | 1 +
docs/reference/latexila-sections.txt | 23 ++
src/Makefile.am | 2 +-
src/evince/Makefile.am | 22 ++
src/evince/evince-gdbus.xml | 30 ++
src/liblatexila/Makefile.am | 34 ++-
src/liblatexila/latexila-build-tool.c | 19 +-
src/liblatexila/latexila-synctex.c | 633 +++++++++++++++++++++++++++++++++
src/liblatexila/latexila-synctex.h | 72 ++++
src/liblatexila/latexila-types.h | 1 +
src/liblatexila/latexila-utils.c | 53 +++
src/liblatexila/latexila-utils.h | 10 +-
src/liblatexila/latexila.h | 1 +
14 files changed, 878 insertions(+), 25 deletions(-)
---
diff --git a/configure.ac b/configure.ac
index d4cb31f..58b6bd1 100644
--- a/configure.ac
+++ b/configure.ac
@@ -75,6 +75,7 @@ AC_PROG_INSTALL
AM_PROG_CC_C_O
AC_PATH_PROG([GLIB_COMPILE_RESOURCES], [glib-compile-resources])
AC_PATH_PROG([GLIB_MKENUMS], [glib-mkenums])
+AC_PATH_PROG([GDBUS_CODEGEN], [gdbus-codegen])
AM_PROG_VALAC([${VALA_REQUIRED_VERSION}],
[found_vala=true]
[found_vala=false])
@@ -168,6 +169,7 @@ AC_CONFIG_FILES([Makefile
man/Makefile
po/Makefile.in
src/Makefile
+ src/evince/Makefile
src/gedit/Makefile
src/liblatexila/Makefile
src/ui/Makefile
diff --git a/docs/reference/latexila-docs.xml b/docs/reference/latexila-docs.xml
index 064f80a..2e09e5f 100644
--- a/docs/reference/latexila-docs.xml
+++ b/docs/reference/latexila-docs.xml
@@ -17,6 +17,7 @@
<xi:include href="xml/build-tools-default.xml"/>
<xi:include href="xml/build-tools-personal.xml"/>
<xi:include href="xml/build-view.xml"/>
+ <xi:include href="xml/synctex.xml"/>
<xi:include href="xml/utils.xml"/>
</chapter>
diff --git a/docs/reference/latexila-sections.txt b/docs/reference/latexila-sections.txt
index 6730ec5..eaa205a 100644
--- a/docs/reference/latexila-sections.txt
+++ b/docs/reference/latexila-sections.txt
@@ -131,10 +131,33 @@ latexila_build_view_get_type
</SECTION>
<SECTION>
+<FILE>synctex</FILE>
+<TITLE>LatexilaSynctex</TITLE>
+LatexilaSynctex
+latexila_synctex_get_instance
+latexila_synctex_connect_evince_window
+latexila_synctex_connect_evince_window_async
+latexila_synctex_connect_evince_window_finish
+latexila_synctex_forward_search
+<SUBSECTION Standard>
+LATEXILA_IS_SYNCTEX
+LATEXILA_IS_SYNCTEX_CLASS
+LATEXILA_SYNCTEX
+LATEXILA_SYNCTEX_CLASS
+LATEXILA_SYNCTEX_GET_CLASS
+LATEXILA_TYPE_SYNCTEX
+LatexilaSynctexClass
+LatexilaSynctexPrivate
+latexila_synctex_get_type
+</SECTION>
+
+<SECTION>
<FILE>utils</FILE>
<TITLE>LatexilaUtils</TITLE>
latexila_utils_get_shortname
latexila_utils_replace_home_dir_with_tilde
latexila_utils_register_icons
latexila_utils_str_replace
+latexila_utils_file_query_exists_async
+latexila_utils_file_query_exists_finish
</SECTION>
diff --git a/src/Makefile.am b/src/Makefile.am
index 508fd87..92f905e 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -1,4 +1,4 @@
-SUBDIRS = gedit liblatexila ui
+SUBDIRS = evince gedit liblatexila ui
bin_PROGRAMS = latexila
diff --git a/src/evince/Makefile.am b/src/evince/Makefile.am
new file mode 100644
index 0000000..b787b83
--- /dev/null
+++ b/src/evince/Makefile.am
@@ -0,0 +1,22 @@
+noinst_LTLIBRARIES = libevince.la
+
+libevince_la_CFLAGS = $(WARN_CFLAGS)
+
+libevince_built_sources = \
+ evince-gdbus-generated.c \
+ evince-gdbus-generated.h
+
+nodist_libevince_la_SOURCES = $(libevince_built_sources)
+
+BUILT_SOURCES = $(libevince_built_sources)
+
+evince-gdbus-generated.c evince-gdbus-generated.h: evince-gdbus.xml Makefile
+ $(AM_V_GEN) $(GDBUS_CODEGEN) \
+ --interface-prefix=org.gnome.evince. \
+ --c-namespace=Evince \
+ --generate-c-code evince-gdbus-generated \
+ evince-gdbus.xml
+
+EXTRA_DIST = evince-gdbus.xml
+
+-include $(top_srcdir)/git.mk
diff --git a/src/evince/evince-gdbus.xml b/src/evince/evince-gdbus.xml
new file mode 100644
index 0000000..0da995f
--- /dev/null
+++ b/src/evince/evince-gdbus.xml
@@ -0,0 +1,30 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE node PUBLIC "-//freedesktop//DTD D-BUS Introspection 0.1//EN"
+ "http://www.freedesktop.org/software/dbus/introspection.dtd">
+<node>
+ <interface name="org.gnome.evince.Daemon">
+ <method name="FindDocument">
+ <arg type="s" name="uri" direction="in"/>
+ <arg type="b" name="spawn" direction="in"/>
+ <arg type="s" name="owner" direction="out"/>
+ </method>
+ </interface>
+ <interface name='org.gnome.evince.Application'>
+ <method name='GetWindowList'>
+ <arg type='ao' name='window_list' direction='out'/>
+ </method>
+ </interface>
+ <interface name='org.gnome.evince.Window'>
+ <method name='SyncView'>
+ <arg type='s' name='source_file' direction='in'/>
+ <arg type='(ii)' name='source_point' direction='in'/>
+ <arg type='u' name='timestamp' direction='in'/>
+ </method>
+ <signal name='SyncSource'>
+ <arg type='s' name='source_file' direction='out'/>
+ <arg type='(ii)' name='source_point' direction='out'/>
+ <arg type='u' name='timestamp' direction='out'/>
+ </signal>
+ <signal name='Closed'/>
+ </interface>
+</node>
diff --git a/src/liblatexila/Makefile.am b/src/liblatexila/Makefile.am
index 538c4e5..1beb3cc 100644
--- a/src/liblatexila/Makefile.am
+++ b/src/liblatexila/Makefile.am
@@ -2,10 +2,16 @@
noinst_LTLIBRARIES = liblatexila.la
-liblatexila_la_CFLAGS = $(WARN_CFLAGS) $(CODE_COVERAGE_CFLAGS)
+liblatexila_la_CFLAGS = \
+ $(WARN_CFLAGS) \
+ $(CODE_COVERAGE_CFLAGS) \
+ -I$(top_builddir)/src/evince
+
liblatexila_la_LDFLAGS = $(CODE_COVERAGE_LDFLAGS)
+liblatexila_la_LIBADD = ../evince/libevince.la
liblatexila_headers = \
+ latexila.h \
latexila-build-job.h \
latexila-build-tool.h \
latexila-build-tools.h \
@@ -14,14 +20,11 @@ liblatexila_headers = \
latexila-build-view.h \
latexila-post-processor.h \
latexila-post-processor-all-output.h \
+ latexila-synctex.h \
latexila-types.h \
latexila-utils.h
-BUILT_SOURCES = \
- latexila-enum-types.c \
- latexila-enum-types.h
-
-liblatexila_la_SOURCES = \
+liblatexila_sources = \
latexila-build-job.c \
latexila-build-tool.c \
latexila-build-tools.c \
@@ -30,9 +33,20 @@ liblatexila_la_SOURCES = \
latexila-build-view.c \
latexila-post-processor.c \
latexila-post-processor-all-output.c \
- latexila-utils.c \
- $(liblatexila_headers) \
- $(BUILT_SOURCES)
+ latexila-synctex.c \
+ latexila-utils.c
+
+liblatexila_built_sources = \
+ latexila-enum-types.c \
+ latexila-enum-types.h
+
+liblatexila_la_SOURCES = \
+ $(liblatexila_headers) \
+ $(liblatexila_sources)
+
+nodist_liblatexila_la_SOURCES = $(liblatexila_built_sources)
+
+BUILT_SOURCES = $(liblatexila_built_sources)
ENUM_TYPES = $(liblatexila_headers)
@@ -64,7 +78,7 @@ Latexila.gir: liblatexila.la
Latexila_gir_NAMESPACE = Latexila
Latexila_gir_INCLUDES = Gtk-3.0
Latexila_gir_LIBS = liblatexila.la
-Latexila_gir_FILES = $(liblatexila_la_SOURCES)
+Latexila_gir_FILES = $(liblatexila_la_SOURCES) $(nodist_liblatexila_la_SOURCES)
noinst_DATA += Latexila.gir Latexila.typelib
CLEANFILES += Latexila.gir Latexila.typelib
diff --git a/src/liblatexila/latexila-build-tool.c b/src/liblatexila/latexila-build-tool.c
index f954e68..a45ff02 100644
--- a/src/liblatexila/latexila-build-tool.c
+++ b/src/liblatexila/latexila-build-tool.c
@@ -495,13 +495,12 @@ query_exists_cb (GFile *file,
GAsyncResult *result,
LatexilaBuildTool *build_tool)
{
- GFileInfo *info;
- GCancellable *cancellable;
gboolean file_exists;
+ GCancellable *cancellable;
gchar *uri = NULL;
GError *error = NULL;
- info = g_file_query_info_finish (file, result, NULL);
+ file_exists = latexila_utils_file_query_exists_finish (file, result);
cancellable = g_task_get_cancellable (build_tool->priv->task);
if (g_cancellable_is_cancelled (cancellable))
@@ -513,9 +512,6 @@ query_exists_cb (GFile *file,
goto out;
}
- file_exists = info != NULL;
- g_clear_object (&info);
-
uri = g_file_get_uri (file);
if (!file_exists)
@@ -660,13 +656,10 @@ open_file (LatexilaBuildTool *build_tool)
file = g_file_new_for_uri (uri);
- g_file_query_info_async (file,
- G_FILE_ATTRIBUTE_STANDARD_TYPE,
- G_FILE_QUERY_INFO_NONE,
- G_PRIORITY_DEFAULT,
- g_task_get_cancellable (build_tool->priv->task),
- (GAsyncReadyCallback) query_exists_cb,
- build_tool);
+ latexila_utils_file_query_exists_async (file,
+ g_task_get_cancellable (build_tool->priv->task),
+ (GAsyncReadyCallback) query_exists_cb,
+ build_tool);
g_free (filename);
g_free (shortname);
diff --git a/src/liblatexila/latexila-synctex.c b/src/liblatexila/latexila-synctex.c
new file mode 100644
index 0000000..b8c501f
--- /dev/null
+++ b/src/liblatexila/latexila-synctex.c
@@ -0,0 +1,633 @@
+/*
+ * This file is part of LaTeXila.
+ *
+ * Copyright (C) 2014 - Sébastien Wilmet <swilmet gnome org>
+ *
+ * LaTeXila is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * LaTeXila is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with LaTeXila. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+/**
+ * SECTION:synctex
+ * @title: LatexilaSynctex
+ * @short_description: SyncTeX support between LaTeXila and Evince
+ *
+ * The #LatexilaSynctex class (a singleton) implements the support of SyncTeX
+ * between LaTeXila and the Evince PDF viewer. It is used to switch between the
+ * source file(s) and the PDF, at the same position in the document. It is called
+ * the forward search: source file -> PDF. And backward search: PDF -> source
+ * file.
+ *
+ * D-Bus is used to communicate between LaTeXila and Evince. The implementation
+ * uses the asynchronous gdbus generated functions.
+ */
+
+#include "latexila-synctex.h"
+#include <glib/gi18n.h>
+#include "evince-gdbus-generated.h"
+#include "latexila-utils.h"
+
+static LatexilaSynctex *instance = NULL;
+
+struct _LatexilaSynctexPrivate
+{
+ /* PDF URI -> EvinceWindow object */
+ GHashTable *evince_windows;
+};
+
+typedef struct
+{
+ GtkTextBuffer *buffer;
+ GFile *buffer_location;
+ gchar *pdf_uri;
+} ForwardSearchData;
+
+typedef struct
+{
+ gchar *pdf_uri;
+ gchar *owner;
+} ConnectEvinceWindowData;
+
+G_DEFINE_TYPE_WITH_PRIVATE (LatexilaSynctex, latexila_synctex, G_TYPE_OBJECT)
+
+static ForwardSearchData *
+forward_search_data_new (void)
+{
+ return g_slice_new0 (ForwardSearchData);
+}
+
+static void
+forward_search_data_free (ForwardSearchData *data)
+{
+ if (data != NULL)
+ {
+ g_clear_object (&data->buffer);
+ g_clear_object (&data->buffer_location);
+ g_free (data->pdf_uri);
+ g_slice_free (ForwardSearchData, data);
+ }
+}
+
+static ConnectEvinceWindowData *
+connect_evince_window_data_new (void)
+{
+ return g_slice_new0 (ConnectEvinceWindowData);
+}
+
+static void
+connect_evince_window_data_free (ConnectEvinceWindowData *data)
+{
+ if (data != NULL)
+ {
+ g_free (data->pdf_uri);
+ g_free (data->owner);
+ g_slice_free (ConnectEvinceWindowData, data);
+ }
+}
+
+static void
+latexila_synctex_dispose (GObject *object)
+{
+ LatexilaSynctex *synctex = LATEXILA_SYNCTEX (object);
+
+ if (synctex->priv->evince_windows != NULL)
+ {
+ g_hash_table_unref (synctex->priv->evince_windows);
+ synctex->priv->evince_windows = NULL;
+ }
+
+ G_OBJECT_CLASS (latexila_synctex_parent_class)->dispose (object);
+}
+
+static void
+latexila_synctex_class_init (LatexilaSynctexClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->dispose = latexila_synctex_dispose;
+}
+
+static void
+latexila_synctex_init (LatexilaSynctex *synctex)
+{
+ synctex->priv = latexila_synctex_get_instance_private (synctex);
+
+ synctex->priv->evince_windows = g_hash_table_new_full (g_str_hash,
+ g_str_equal,
+ g_free,
+ g_object_unref);
+}
+
+/**
+ * latexila_synctex_get_instance:
+ *
+ * Returns: (transfer none): the #LatexilaSynctex singleton instance.
+ */
+LatexilaSynctex *
+latexila_synctex_get_instance (void)
+{
+ if (instance == NULL)
+ {
+ instance = g_object_new (LATEXILA_TYPE_SYNCTEX, NULL);
+ }
+
+ return instance;
+}
+
+static void
+show_warning (const gchar *message)
+{
+ GtkApplication *app;
+ GtkWindow *parent;
+ GtkWidget *dialog;
+
+ app = GTK_APPLICATION (g_application_get_default ());
+ parent = gtk_application_get_active_window (app);
+
+ dialog = gtk_message_dialog_new (parent,
+ GTK_DIALOG_DESTROY_WITH_PARENT,
+ GTK_MESSAGE_ERROR,
+ GTK_BUTTONS_OK,
+ "%s", _("Impossible to do the forward search."));
+
+ gtk_message_dialog_format_secondary_text (GTK_MESSAGE_DIALOG (dialog),
+ "%s", message);
+
+ gtk_dialog_run (GTK_DIALOG (dialog));
+ gtk_widget_destroy (dialog);
+}
+
+static gchar *
+get_pdf_uri (GFile *main_tex_file)
+{
+ gchar *tex_uri;
+ gchar *short_uri;
+ gchar *pdf_uri;
+
+ tex_uri = g_file_get_uri (main_tex_file);
+ short_uri = latexila_utils_get_shortname (tex_uri);
+ pdf_uri = g_strdup_printf ("%s.pdf", short_uri);
+
+ g_free (tex_uri);
+ g_free (short_uri);
+ return pdf_uri;
+}
+
+static GVariant *
+get_buffer_position (GtkTextBuffer *buffer)
+{
+ GtkTextIter iter;
+ gint line;
+ gint column;
+
+ gtk_text_buffer_get_iter_at_mark (buffer,
+ &iter,
+ gtk_text_buffer_get_insert (buffer));
+
+ line = gtk_text_iter_get_line (&iter) + 1;
+
+ /* Ignore the column, it gives a better result. */
+ column = -1;
+
+ return g_variant_new ("(ii)", line, column);
+}
+
+static void
+window_closed_cb (EvinceWindow *window,
+ const gchar *pdf_uri)
+{
+ g_hash_table_remove (instance->priv->evince_windows, pdf_uri);
+}
+
+static void
+sync_source_cb (EvinceWindow *window,
+ const gchar *tex_uri,
+ GVariant *pos,
+ guint timestamp,
+ LatexilaSynctex *synctex)
+{
+}
+
+static void
+window_proxy_cb (GObject *object,
+ GAsyncResult *result,
+ GTask *task)
+{
+ EvinceWindow *window;
+ ConnectEvinceWindowData *data;
+ GError *error = NULL;
+
+ window = evince_window_proxy_new_for_bus_finish (result, &error);
+
+ if (error != NULL)
+ {
+ g_warning ("SyncTeX: can not connect to evince window: %s", error->message);
+ g_task_return_boolean (task, FALSE);
+ g_object_unref (task);
+ g_error_free (error);
+ return;
+ }
+
+ data = g_task_get_task_data (task);
+
+ g_hash_table_insert (instance->priv->evince_windows, data->pdf_uri, window);
+
+ g_signal_connect (window,
+ "closed",
+ G_CALLBACK (window_closed_cb),
+ data->pdf_uri);
+
+ g_signal_connect (window,
+ "sync-source",
+ G_CALLBACK (sync_source_cb),
+ instance);
+
+ data->pdf_uri = NULL;
+
+ /* latexila_synctex_connect_evince_window_async() is finally finished! */
+ g_task_return_boolean (task, TRUE);
+ g_object_unref (task);
+}
+
+static void
+get_window_list_cb (EvinceApplication *app,
+ GAsyncResult *result,
+ GTask *task)
+{
+ ConnectEvinceWindowData *data;
+ gchar **window_list;
+ gchar *window_path;
+ GError *error = NULL;
+
+ evince_application_call_get_window_list_finish (app, &window_list, result, &error);
+ g_object_unref (app);
+
+ if (error != NULL)
+ {
+ g_warning ("SyncTeX: can not get window list: %s", error->message);
+ g_task_return_boolean (task, FALSE);
+ g_object_unref (task);
+ g_error_free (error);
+ return;
+ }
+
+ if (window_list == NULL || window_list[0] == NULL)
+ {
+ g_warning ("SyncTeX: the window list is empty.");
+ g_task_return_boolean (task, FALSE);
+ g_object_unref (task);
+ g_strfreev (window_list);
+ return;
+ }
+
+ data = g_task_get_task_data (task);
+
+ /* There is normally only one window. */
+ window_path = window_list[0];
+
+ evince_window_proxy_new_for_bus (G_BUS_TYPE_SESSION,
+ G_DBUS_PROXY_FLAGS_NONE,
+ data->owner,
+ window_path,
+ NULL,
+ (GAsyncReadyCallback) window_proxy_cb,
+ task);
+
+ g_strfreev (window_list);
+}
+
+static void
+application_proxy_cb (GObject *object,
+ GAsyncResult *result,
+ GTask *task)
+{
+ EvinceApplication *app;
+ GError *error = NULL;
+
+ app = evince_application_proxy_new_for_bus_finish (result, &error);
+
+ if (error != NULL)
+ {
+ g_warning ("SyncTeX: can not connect to evince application: %s", error->message);
+ g_task_return_boolean (task, FALSE);
+ g_object_unref (task);
+ g_error_free (error);
+ return;
+ }
+
+ evince_application_call_get_window_list (app,
+ NULL,
+ (GAsyncReadyCallback) get_window_list_cb,
+ task);
+}
+
+static void
+find_document_cb (EvinceDaemon *daemon,
+ GAsyncResult *result,
+ GTask *task)
+{
+ ConnectEvinceWindowData *data;
+ GError *error = NULL;
+
+ data = g_task_get_task_data (task);
+
+ evince_daemon_call_find_document_finish (daemon, &data->owner, result, &error);
+ g_object_unref (daemon);
+
+ if (error != NULL)
+ {
+ g_warning ("SyncTeX: find document: %s", error->message);
+ g_task_return_boolean (task, FALSE);
+ g_object_unref (task);
+ g_error_free (error);
+ return;
+ }
+
+ evince_application_proxy_new_for_bus (G_BUS_TYPE_SESSION,
+ G_DBUS_PROXY_FLAGS_NONE,
+ data->owner,
+ "/org/gnome/evince/Evince",
+ NULL,
+ (GAsyncReadyCallback) application_proxy_cb,
+ task);
+}
+
+static void
+daemon_proxy_cb (GObject *object,
+ GAsyncResult *result,
+ GTask *task)
+{
+ EvinceDaemon *daemon;
+ GError *error = NULL;
+ ConnectEvinceWindowData *data;
+
+ daemon = evince_daemon_proxy_new_for_bus_finish (result, &error);
+
+ if (error != NULL)
+ {
+ g_warning ("SyncTeX: can not connect to the evince daemon: %s", error->message);
+ g_task_return_boolean (task, FALSE);
+ g_object_unref (task);
+ g_error_free (error);
+ return;
+ }
+
+ data = g_task_get_task_data (task);
+
+ evince_daemon_call_find_document (daemon, data->pdf_uri, TRUE, NULL,
+ (GAsyncReadyCallback) find_document_cb,
+ task);
+}
+
+/**
+ * latexila_synctex_connect_evince_window_async:
+ * @synctex: the #LatexilaSynctex instance.
+ * @pdf_uri: the PDF URI
+ * @callback: the callback to call when the operation is finished.
+ * @user_data: the data to pass to the callback function.
+ *
+ * Connects asynchronously the evince window for @pdf_uri. LaTeXila will then
+ * listen the signals emitted by the evince window when the user wants to switch
+ * from the PDF to the corresponding *.tex file.
+ *
+ * The callback will be called when the operation is finished. You can then call
+ * latexila_synctex_connect_evince_window_finish().
+ */
+void
+latexila_synctex_connect_evince_window_async (LatexilaSynctex *synctex,
+ const gchar *pdf_uri,
+ GAsyncReadyCallback callback,
+ gpointer user_data)
+{
+ GTask *task;
+ ConnectEvinceWindowData *data;
+
+ g_return_if_fail (LATEXILA_IS_SYNCTEX (synctex));
+ g_return_if_fail (pdf_uri != NULL);
+
+ task = g_task_new (synctex, NULL, callback, user_data);
+
+ if (g_hash_table_contains (synctex->priv->evince_windows, pdf_uri))
+ {
+ g_task_return_boolean (task, TRUE);
+ g_object_unref (task);
+ return;
+ }
+
+ data = connect_evince_window_data_new ();
+ data->pdf_uri = g_strdup (pdf_uri);
+
+ g_task_set_task_data (task, data, (GDestroyNotify) connect_evince_window_data_free);
+
+ evince_daemon_proxy_new_for_bus (G_BUS_TYPE_SESSION,
+ G_DBUS_PROXY_FLAGS_NONE,
+ "org.gnome.evince.Daemon",
+ "/org/gnome/evince/Daemon",
+ NULL,
+ (GAsyncReadyCallback) daemon_proxy_cb,
+ task);
+}
+
+/**
+ * latexila_synctex_connect_evince_window_finish:
+ * @synctex: the #LatexilaSynctex instance.
+ * @result: a #GAsyncResult.
+ *
+ * Finishes the operation started with
+ * latexila_synctex_connect_evince_window_async().
+ */
+void
+latexila_synctex_connect_evince_window_finish (LatexilaSynctex *synctex,
+ GAsyncResult *result)
+{
+ g_return_if_fail (g_task_is_valid (result, synctex));
+
+ g_task_propagate_boolean (G_TASK (result), NULL);
+}
+
+/**
+ * latexila_synctex_connect_evince_window:
+ * @synctex: the #LatexilaSynctex instance.
+ * @pdf_uri: the PDF URI.
+ *
+ * Simple version of latexila_synctex_connect_evince_window_async().
+ * This function is also asynchronous, but without callback.
+ */
+void
+latexila_synctex_connect_evince_window (LatexilaSynctex *synctex,
+ const gchar *pdf_uri)
+{
+ latexila_synctex_connect_evince_window_async (synctex,
+ pdf_uri,
+ (GAsyncReadyCallback)
latexila_synctex_connect_evince_window_finish,
+ NULL);
+}
+
+static void
+sync_view_cb (EvinceWindow *evince_window,
+ GAsyncResult *result,
+ ForwardSearchData *data)
+{
+ GError *error = NULL;
+
+ evince_window_call_sync_view_finish (evince_window, result, &error);
+
+ if (error != NULL)
+ {
+ g_warning ("SyncTeX: can not sync view: %s", error->message);
+ g_error_free (error);
+ }
+
+ /* latexila_synctex_forward_search() finished! */
+ forward_search_data_free (data);
+}
+
+static void
+connect_evince_window_cb (LatexilaSynctex *synctex,
+ GAsyncResult *result,
+ ForwardSearchData *data)
+{
+ EvinceWindow *evince_window;
+ gchar *buffer_path;
+
+ latexila_synctex_connect_evince_window_finish (synctex, result);
+
+ evince_window = g_hash_table_lookup (synctex->priv->evince_windows, data->pdf_uri);
+
+ if (evince_window == NULL)
+ {
+ show_warning (_("Can not communicate with evince."));
+ forward_search_data_free (data);
+ return;
+ }
+
+ buffer_path = g_file_get_path (data->buffer_location);
+
+ evince_window_call_sync_view (evince_window,
+ buffer_path,
+ get_buffer_position (data->buffer),
+ GDK_CURRENT_TIME,
+ NULL,
+ (GAsyncReadyCallback) sync_view_cb,
+ data);
+
+ g_free (buffer_path);
+}
+
+static void
+synctex_file_query_exists_cb (GFile *synctex_file,
+ GAsyncResult *result,
+ ForwardSearchData *data)
+{
+ gboolean synctex_file_exists;
+
+ synctex_file_exists = latexila_utils_file_query_exists_finish (synctex_file, result);
+
+ if (!synctex_file_exists)
+ {
+ gchar *basename = g_file_get_basename (synctex_file);
+ gchar *message = g_strdup_printf (_("The file \"%s\" doesn't exist."), basename);
+
+ show_warning (message);
+
+ g_free (basename);
+ g_free (message);
+ g_object_unref (synctex_file);
+ forward_search_data_free (data);
+ return;
+ }
+
+ latexila_synctex_connect_evince_window_async (instance,
+ data->pdf_uri,
+ (GAsyncReadyCallback) connect_evince_window_cb,
+ data);
+
+ g_object_unref (synctex_file);
+}
+
+static void
+pdf_file_query_exists_cb (GFile *pdf_file,
+ GAsyncResult *result,
+ ForwardSearchData *data)
+{
+ gboolean pdf_file_exists;
+ gchar *short_uri;
+ gchar *synctex_uri;
+ GFile *synctex_file;
+
+ pdf_file_exists = latexila_utils_file_query_exists_finish (pdf_file, result);
+ g_object_unref (pdf_file);
+
+ if (!pdf_file_exists)
+ {
+ show_warning (_("The PDF file doesn't exist."));
+ forward_search_data_free (data);
+ return;
+ }
+
+ short_uri = latexila_utils_get_shortname (data->pdf_uri);
+ synctex_uri = g_strdup_printf ("%s.synctex.gz", short_uri);
+ synctex_file = g_file_new_for_uri (synctex_uri);
+ g_free (short_uri);
+ g_free (synctex_uri);
+
+ latexila_utils_file_query_exists_async (synctex_file,
+ NULL,
+ (GAsyncReadyCallback) synctex_file_query_exists_cb,
+ data);
+}
+
+/**
+ * latexila_synctex_forward_search:
+ * @synctex: the #LatexilaSynctex instance.
+ * @buffer: a #GtkTextBuffer.
+ * @buffer_location: the *.tex file of @buffer.
+ * @main_tex_file: the main *.tex file of @buffer.
+ *
+ * Does a forward search, i.e. switch from the *.tex file to the PDF file at the
+ * same position as the cursor position in @buffer.
+ */
+void
+latexila_synctex_forward_search (LatexilaSynctex *synctex,
+ GtkTextBuffer *buffer,
+ GFile *buffer_location,
+ GFile *main_tex_file)
+{
+ ForwardSearchData *data;
+ GFile *pdf_file;
+
+ g_return_if_fail (LATEXILA_IS_SYNCTEX (synctex));
+ g_return_if_fail (GTK_IS_TEXT_BUFFER (buffer));
+ g_return_if_fail (buffer_location == NULL || G_IS_FILE (buffer_location));
+ g_return_if_fail (main_tex_file == NULL || G_IS_FILE (main_tex_file));
+
+ if (buffer_location == NULL)
+ {
+ show_warning (_("The document is not saved."));
+ return;
+ }
+
+ g_return_if_fail (G_IS_FILE (main_tex_file));
+
+ data = forward_search_data_new ();
+ data->buffer = g_object_ref (buffer);
+ data->buffer_location = g_object_ref (buffer_location);
+ data->pdf_uri = get_pdf_uri (main_tex_file);
+
+ pdf_file = g_file_new_for_uri (data->pdf_uri);
+
+ latexila_utils_file_query_exists_async (pdf_file,
+ NULL,
+ (GAsyncReadyCallback) pdf_file_query_exists_cb,
+ data);
+}
diff --git a/src/liblatexila/latexila-synctex.h b/src/liblatexila/latexila-synctex.h
new file mode 100644
index 0000000..3b20130
--- /dev/null
+++ b/src/liblatexila/latexila-synctex.h
@@ -0,0 +1,72 @@
+/*
+ * This file is part of LaTeXila.
+ *
+ * Copyright (C) 2014 - Sébastien Wilmet <swilmet gnome org>
+ *
+ * LaTeXila is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * LaTeXila is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with LaTeXila. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef __LATEXILA_SYNCTEX_H__
+#define __LATEXILA_SYNCTEX_H__
+
+#include "latexila-types.h"
+#include <gtk/gtk.h>
+
+G_BEGIN_DECLS
+
+#define LATEXILA_TYPE_SYNCTEX (latexila_synctex_get_type ())
+#define LATEXILA_SYNCTEX(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), LATEXILA_TYPE_SYNCTEX,
LatexilaSynctex))
+#define LATEXILA_SYNCTEX_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), LATEXILA_TYPE_SYNCTEX,
LatexilaSynctexClass))
+#define LATEXILA_IS_SYNCTEX(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), LATEXILA_TYPE_SYNCTEX))
+#define LATEXILA_IS_SYNCTEX_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), LATEXILA_TYPE_SYNCTEX))
+#define LATEXILA_SYNCTEX_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), LATEXILA_TYPE_SYNCTEX,
LatexilaSynctexClass))
+
+typedef struct _LatexilaSynctexClass LatexilaSynctexClass;
+typedef struct _LatexilaSynctexPrivate LatexilaSynctexPrivate;
+
+struct _LatexilaSynctex
+{
+ GObject parent;
+
+ LatexilaSynctexPrivate *priv;
+};
+
+struct _LatexilaSynctexClass
+{
+ GObjectClass parent_class;
+};
+
+GType latexila_synctex_get_type (void) G_GNUC_CONST;
+
+LatexilaSynctex * latexila_synctex_get_instance (void);
+
+void latexila_synctex_connect_evince_window_async (LatexilaSynctex *synctex,
+ const gchar *pdf_uri,
+ GAsyncReadyCallback callback,
+ gpointer user_data);
+
+void latexila_synctex_connect_evince_window_finish (LatexilaSynctex *synctex,
+ GAsyncResult *result);
+
+void latexila_synctex_connect_evince_window (LatexilaSynctex *synctex,
+ const gchar *pdf_uri);
+
+void latexila_synctex_forward_search (LatexilaSynctex *synctex,
+ GtkTextBuffer *buffer,
+ GFile *buffer_location,
+ GFile *main_tex_file);
+
+G_END_DECLS
+
+#endif /* __LATEXILA_SYNCTEX_H__ */
diff --git a/src/liblatexila/latexila-types.h b/src/liblatexila/latexila-types.h
index 9ca5d99..1dd9da4 100644
--- a/src/liblatexila/latexila-types.h
+++ b/src/liblatexila/latexila-types.h
@@ -32,6 +32,7 @@ typedef struct _LatexilaBuildToolsPersonal LatexilaBuildToolsPersonal;
typedef struct _LatexilaBuildView LatexilaBuildView;
typedef struct _LatexilaPostProcessor LatexilaPostProcessor;
typedef struct _LatexilaPostProcessorAllOutput LatexilaPostProcessorAllOutput;
+typedef struct _LatexilaSynctex LatexilaSynctex;
G_END_DECLS
diff --git a/src/liblatexila/latexila-utils.c b/src/liblatexila/latexila-utils.c
index 8c06b8a..00f4bcf 100644
--- a/src/liblatexila/latexila-utils.c
+++ b/src/liblatexila/latexila-utils.c
@@ -237,3 +237,56 @@ latexila_utils_str_replace (const gchar *string,
g_strfreev (chunks);
return ret;
}
+
+/**
+ * latexila_utils_file_query_exists_async:
+ * @file: a #GFile.
+ * @cancellable: a #GCancellable.
+ * @callback: the callback to call when the operation is finished.
+ * @user_data: the data to pass to the callback function.
+ *
+ * The asynchronous version of g_file_query_exists(). When the operation is
+ * finished, @callback will be called. You can then call
+ * latexila_utils_file_query_exists_finish() to get the result of the operation.
+ */
+void
+latexila_utils_file_query_exists_async (GFile *file,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data)
+{
+ g_file_query_info_async (file,
+ G_FILE_ATTRIBUTE_STANDARD_TYPE,
+ G_FILE_QUERY_INFO_NONE,
+ G_PRIORITY_DEFAULT,
+ cancellable,
+ callback,
+ user_data);
+}
+
+/**
+ * latexila_utils_file_query_exists_finish:
+ * @file: a #GFile.
+ * @result: a #GAsyncResult.
+ *
+ * Finishes the operation started with latexila_utils_file_query_exists_async().
+ * There is no output #GError parameter, so you should check if the operation
+ * has been cancelled (in which case %FALSE will be returned).
+ *
+ * Returns: %TRUE if the file exists and the operation hasn't been cancelled,
+ * %FALSE otherwise.
+ */
+gboolean
+latexila_utils_file_query_exists_finish (GFile *file,
+ GAsyncResult *result)
+{
+ GFileInfo *info = g_file_query_info_finish (file, result, NULL);
+
+ if (info != NULL)
+ {
+ g_object_unref (info);
+ return TRUE;
+ }
+
+ return FALSE;
+}
diff --git a/src/liblatexila/latexila-utils.h b/src/liblatexila/latexila-utils.h
index 852a6f0..c1b44a9 100644
--- a/src/liblatexila/latexila-utils.h
+++ b/src/liblatexila/latexila-utils.h
@@ -20,7 +20,7 @@
#ifndef __LATEXILA_UTILS_H__
#define __LATEXILA_UTILS_H__
-#include <glib.h>
+#include <gio/gio.h>
G_BEGIN_DECLS
@@ -34,6 +34,14 @@ gchar * latexila_utils_str_replace (const gchar *st
const gchar *search,
const gchar *replacement);
+void latexila_utils_file_query_exists_async (GFile *file,
+ GCancellable *cancellable,
+ GAsyncReadyCallback callback,
+ gpointer user_data);
+
+gboolean latexila_utils_file_query_exists_finish (GFile *file,
+ GAsyncResult *result);
+
G_END_DECLS
#endif /* __LATEXILA_UTILS_H__ */
diff --git a/src/liblatexila/latexila.h b/src/liblatexila/latexila.h
index 0d799f0..a200ab9 100644
--- a/src/liblatexila/latexila.h
+++ b/src/liblatexila/latexila.h
@@ -31,6 +31,7 @@
#include "latexila-build-view.h"
#include "latexila-post-processor.h"
#include "latexila-post-processor-all-output.h"
+#include "latexila-synctex.h"
#include "latexila-utils.h"
#endif /* __LATEXILA_H__ */
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]