From 9f8ec2d921de1c31da212c8486ab2bda85d51bf9 Mon Sep 17 00:00:00 2001
From: Marco Pesenti Gritti <mpg@redhat.com>
Date: Wed, 7 Sep 2005 12:36:54 +0000
Subject: [PATCH] Implement xyz links, still not functional. Links are a total
 mess, will

2005-09-07  Marco Pesenti Gritti  <mpg@redhat.com>

        * backend/ev-document.h:
        * backend/ev-link.c: (ev_link_type_get_type), (ev_link_get_top),
        (ev_link_get_left), (ev_link_get_zoom), (ev_link_get_property),
        (ev_link_set_property), (ev_window_dispose), (ev_link_class_init),
        (ev_link_new_page), (ev_link_new_page_xyz):
        * backend/ev-link.h:
        * pdf/ev-poppler.cc:
        * shell/ev-view.c: (doc_point_to_view_point), (scroll_to_xyz_link),
        (go_to_link), (tip_from_link):

        Implement xyz links, still not functional. Links are a total
        mess, will need to refactor and bugfix a lot :(
---
 ChangeLog             |  15 ++++
 backend/ev-document.h |   5 ++
 backend/ev-link.c     | 172 +++++++++++++++++++++++++++---------------
 backend/ev-link.h     |  18 +++--
 pdf/ev-poppler.cc     |  88 +++++++++++++++++++--
 shell/ev-view.c       |  59 ++++++++++++++-
 6 files changed, 281 insertions(+), 76 deletions(-)

diff --git a/ChangeLog b/ChangeLog
index bf36d1d7..d1765274 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,18 @@
+2005-09-07  Marco Pesenti Gritti  <mpg@redhat.com>
+
+	* backend/ev-document.h:
+	* backend/ev-link.c: (ev_link_type_get_type), (ev_link_get_top),
+	(ev_link_get_left), (ev_link_get_zoom), (ev_link_get_property),
+	(ev_link_set_property), (ev_window_dispose), (ev_link_class_init),
+	(ev_link_new_page), (ev_link_new_page_xyz):
+	* backend/ev-link.h:
+	* pdf/ev-poppler.cc:
+	* shell/ev-view.c: (doc_point_to_view_point), (scroll_to_xyz_link),
+	(go_to_link), (tip_from_link):
+
+	Implement xyz links, still not functional. Links are a total
+	mess, will need to refactor and bugfix a lot :(
+
 2005-09-05  Alessio Frusciante  <algol@firenze.linux.it>
 
 	* configure.ac: Added "it" (Italian) to ALL_LINGUAS.
diff --git a/backend/ev-document.h b/backend/ev-document.h
index 2629a47f..2cea42f3 100644
--- a/backend/ev-document.h
+++ b/backend/ev-document.h
@@ -56,6 +56,11 @@ typedef enum
 	EV_DOCUMENT_ERROR_ENCRYPTED
 } EvDocumentError;
 
+typedef struct {
+	double x;
+	double y;
+} EvPoint;
+
 typedef struct {
 	double x1;
 	double y1;
diff --git a/backend/ev-link.c b/backend/ev-link.c
index ee54b285..9016ec0d 100644
--- a/backend/ev-link.c
+++ b/backend/ev-link.c
@@ -29,7 +29,10 @@ enum {
 	PROP_TITLE,
 	PROP_TYPE,
 	PROP_PAGE,
-	PROP_URI
+	PROP_URI,
+	PROP_LEFT,
+	PROP_TOP,
+	PROP_ZOOM
 };
 
 
@@ -47,6 +50,9 @@ struct _EvLinkPrivate {
 	char *uri;
 	EvLinkType type;
 	int page;
+	double top;
+	double left;
+	double zoom;
 };
 
 G_DEFINE_TYPE (EvLink, ev_link, G_TYPE_OBJECT)
@@ -63,6 +69,7 @@ ev_link_type_get_type (void)
 		static const GEnumValue values[] = {
 			{ EV_LINK_TYPE_TITLE, "EV_LINK_TYPE_TITLE", "title" },
 			{ EV_LINK_TYPE_PAGE, "EV_LINK_TYPE_PAGE", "page" },
+			{ EV_LINK_TYPE_PAGE_XYZ, "EV_LINK_TYPE_PAGE_XYZ", "page-xyz" },
 			{ EV_LINK_TYPE_EXTERNAL_URI, "EV_LINK_TYPE_EXTERNAL_URI", "external" },
 			{ 0, NULL, NULL }
                 };
@@ -81,22 +88,6 @@ ev_link_get_title (EvLink *self)
 	return self->priv->title;
 }
 
-void
-ev_link_set_title (EvLink* self, const char *title)
-{
-	g_assert (EV_IS_LINK (self));
-
-	if (self->priv->title != NULL) {
-		g_free (self->priv->title);
-	}
-	if (title)
-		self->priv->title = g_strdup (title);
-	else
-		self->priv->title = NULL;
-
-	g_object_notify (G_OBJECT (self), "title");
-}
-
 const char *
 ev_link_get_uri (EvLink *self)
 {
@@ -105,21 +96,6 @@ ev_link_get_uri (EvLink *self)
 	return self->priv->uri;
 }
 
-void
-ev_link_set_uri (EvLink* self, const char *uri)
-{
-	g_assert (EV_IS_LINK (self));
-	g_assert (uri != NULL);
-
-	if (self->priv->uri != NULL) {
-		g_free (self->priv->uri);
-	}
-
-	self->priv->uri = g_strdup (uri);
-
-	g_object_notify (G_OBJECT (self), "uri");
-}
-
 EvLinkType
 ev_link_get_link_type (EvLink *self)
 {
@@ -128,16 +104,6 @@ ev_link_get_link_type (EvLink *self)
 	return self->priv->type;
 }
 
-void
-ev_link_set_link_type (EvLink* self, EvLinkType type)
-{
-	g_assert (EV_IS_LINK (self));
-
-	self->priv->type = type;
-
-	g_object_notify (G_OBJECT (self), "type");
-}
-
 int
 ev_link_get_page (EvLink *self)
 {
@@ -146,14 +112,28 @@ ev_link_get_page (EvLink *self)
 	return self->priv->page;
 }
 
-void
-ev_link_set_page (EvLink* self, int page)
+double
+ev_link_get_top (EvLink *self)
 {
-	g_assert (EV_IS_LINK (self));
+	g_return_val_if_fail (EV_IS_LINK (self), 0);
+	
+	return self->priv->top;
+}
 
-	self->priv->page = page;
+double
+ev_link_get_left (EvLink *self)
+{
+	g_return_val_if_fail (EV_IS_LINK (self), 0);
+	
+	return self->priv->left;
+}
 
-	g_object_notify (G_OBJECT (self), "page");
+double
+ev_link_get_zoom (EvLink *self)
+{
+	g_return_val_if_fail (EV_IS_LINK (self), 0);
+	
+	return self->priv->zoom;
 }
 
 static void
@@ -177,6 +157,15 @@ ev_link_get_property (GObject *object, guint prop_id, GValue *value,
 	case PROP_PAGE:
 		g_value_set_int (value, self->priv->page);
 		break;
+	case PROP_TOP:
+		g_value_set_double (value, self->priv->top);
+		break;
+	case PROP_LEFT:
+		g_value_set_double (value, self->priv->left);
+		break;
+	case PROP_ZOOM:
+		g_value_set_double (value, self->priv->zoom);
+		break;
 	default:
 		G_OBJECT_WARN_INVALID_PROPERTY_ID (object,
 						   prop_id,
@@ -189,23 +178,31 @@ static void
 ev_link_set_property (GObject *object, guint prop_id, const GValue *value,
 		      GParamSpec *param_spec)
 {
-	EvLink *self;
-	
-	self = EV_LINK (object);
+	EvLink *link = EV_LINK (object);
 	
 	switch (prop_id) {
 	case PROP_TITLE:
-		ev_link_set_title (self, g_value_get_string (value));
+		link->priv->title = g_strdup (g_value_get_string (value));	
 		break;
 	case PROP_URI:
-		ev_link_set_uri (self, g_value_get_string (value));
+		link->priv->uri = g_strdup (g_value_get_string (value));
 		break;
 	case PROP_TYPE:
-		ev_link_set_link_type (self, g_value_get_enum (value));
+		link->priv->type = g_value_get_enum (value);
 		break;
 	case PROP_PAGE:
-		ev_link_set_page (self, g_value_get_int (value));
+		link->priv->page = g_value_get_int (value);
+		break;
+	case PROP_TOP:
+		link->priv->top = g_value_get_double (value);
+		break;
+	case PROP_LEFT:
+		link->priv->left = g_value_get_double (value);
+		break;
+	case PROP_ZOOM:
+		link->priv->zoom = g_value_get_double (value);
 		break;
+
 	default:
 		G_OBJECT_WARN_INVALID_PROPERTY_ID (object,
 						   prop_id,
@@ -228,6 +225,11 @@ ev_window_dispose (GObject *object)
 		priv->title = NULL;
 	}
 
+	if (priv->uri) {
+		g_free (priv->uri);
+		priv->uri = NULL;
+	}
+
 	G_OBJECT_CLASS (ev_link_parent_class)->dispose (object);
 }
 
@@ -257,16 +259,16 @@ ev_link_class_init (EvLinkClass *ev_window_class)
 				     	 		      "Link Title",
 				     			      "The link title",
 							      NULL,
-				     			      G_PARAM_READWRITE));
-
+							      G_PARAM_READWRITE |
+				     			      G_PARAM_CONSTRUCT_ONLY));
 	g_object_class_install_property (g_object_class,
 					 PROP_URI,
 					 g_param_spec_string ("uri",
 				     	 		      "Link URI",
 				     			      "The link URI",
 							      NULL,
-				     			      G_PARAM_READWRITE));
-
+							      G_PARAM_READWRITE |
+				     			      G_PARAM_CONSTRUCT_ONLY));
 	g_object_class_install_property (g_object_class,
 					 PROP_TYPE,
 			 		 g_param_spec_enum  ("type",
@@ -274,8 +276,8 @@ ev_link_class_init (EvLinkClass *ev_window_class)
 							     "The link type",
 							     EV_TYPE_LINK_TYPE,
 							     EV_LINK_TYPE_TITLE,
-							     G_PARAM_READWRITE));
-
+							     G_PARAM_READWRITE |
+							     G_PARAM_CONSTRUCT_ONLY));
 	g_object_class_install_property (g_object_class,
 					 PROP_PAGE,
 					 g_param_spec_int ("page",
@@ -284,7 +286,38 @@ ev_link_class_init (EvLinkClass *ev_window_class)
 							    -1,
 							    G_MAXINT,
 							    0,
-							    G_PARAM_READWRITE));
+							    G_PARAM_READWRITE |
+							    G_PARAM_CONSTRUCT_ONLY));
+	g_object_class_install_property (g_object_class,
+					 PROP_LEFT,
+					 g_param_spec_double ("left",
+							      "Left coordinate",
+							      "The left coordinate",
+							      0,
+							      G_MAXDOUBLE,
+							      0,
+							      G_PARAM_READWRITE |
+							      G_PARAM_CONSTRUCT_ONLY));
+	g_object_class_install_property (g_object_class,
+					 PROP_TOP,
+					 g_param_spec_double ("top",
+							      "Top coordinate",
+							      "The top coordinate",
+							      0,
+							      G_MAXDOUBLE,
+							      0,
+							      G_PARAM_READWRITE |
+							      G_PARAM_CONSTRUCT_ONLY));
+	g_object_class_install_property (g_object_class,
+					 PROP_ZOOM,
+					 g_param_spec_double ("zoom",
+							      "Zoom",
+							      "Zoom",
+							      0,
+							      G_MAXDOUBLE,
+							      0,
+							      G_PARAM_READWRITE |
+							      G_PARAM_CONSTRUCT_ONLY));
 }
 
 EvLink *
@@ -306,6 +339,23 @@ ev_link_new_page (const char *title, int page)
 				      NULL));
 }
 
+EvLink *
+ev_link_new_page_xyz (const char *title,
+		      int         page,
+		      double      top,
+		      double      left,
+		      double      zoom)
+{
+	return EV_LINK (g_object_new (EV_TYPE_LINK,
+				      "title", title,
+				      "page", page,
+				      "type", EV_LINK_TYPE_PAGE_XYZ,
+				      "left", left,
+				      "top", top,
+				      "zoom", zoom,
+				      NULL));
+}
+
 EvLink *
 ev_link_new_external (const char *title, const char *uri)
 {
diff --git a/backend/ev-link.h b/backend/ev-link.h
index b79e2fa4..8ee84294 100644
--- a/backend/ev-link.h
+++ b/backend/ev-link.h
@@ -43,6 +43,7 @@ typedef enum
 {
 	EV_LINK_TYPE_TITLE,
 	EV_LINK_TYPE_PAGE,
+	EV_LINK_TYPE_PAGE_XYZ,
 	EV_LINK_TYPE_EXTERNAL_URI,
 	/* We'll probably fill this in more as we support the other types of
 	 * links */
@@ -54,20 +55,21 @@ GType		ev_link_get_type	(void);
 EvLink	       *ev_link_new_title	(const char     *title);
 EvLink	       *ev_link_new_page	(const char     *title,
 					 int             page);
+EvLink	       *ev_link_new_page_xyz	(const char     *title,
+					 int             page,
+					 double          top,
+					 double          left,
+					 double          zoom);
 EvLink	       *ev_link_new_external	(const char     *title,
 					 const char     *uri);
+
 const char     *ev_link_get_title	(EvLink     *link);
-void		ev_link_set_title	(EvLink     *link,
-					 const char     *title);
 const char     *ev_link_get_uri		(EvLink     *link);
-void		ev_link_set_uri		(EvLink     *link,
-					 const char     *uri);
 EvLinkType	ev_link_get_link_type	(EvLink     *link);
-void		ev_link_set_link_type	(EvLink     *link,
-					 EvLinkType  type);
 int		ev_link_get_page	(EvLink     *link);
-void		ev_link_set_page	(EvLink     *link,
-					 int             page);
+double		ev_link_get_top		(EvLink     *link);
+double		ev_link_get_left	(EvLink     *link);
+double		ev_link_get_zoom	(EvLink     *link);
 
 /* Link Mapping stuff */
 
diff --git a/pdf/ev-poppler.cc b/pdf/ev-poppler.cc
index bbcb9781..769c78a7 100644
--- a/pdf/ev-poppler.cc
+++ b/pdf/ev-poppler.cc
@@ -703,18 +703,94 @@ pdf_document_links_has_document_links (EvDocumentLinks *document_links)
 }
 
 static EvLink *
-ev_link_from_action (PopplerAction *action)
+ev_link_from_dest (PopplerAction *action)
 {
 	EvLink *link;
+	const char *unimplemented_dest = NULL;
+
+	switch (action->goto_dest.dest->type) {
+	case POPPLER_DEST_UNKNOWN:
+		unimplemented_dest = "POPPLER_DEST_UNKNOWN";
+		break;
+	case POPPLER_DEST_XYZ:
+		link = ev_link_new_page_xyz (action->any.title,
+					     action->goto_dest.dest->page_num - 1,
+					     action->goto_dest.dest->left,
+					     action->goto_dest.dest->top,
+					     action->goto_dest.dest->zoom);
+		break;
+	case POPPLER_DEST_FIT:
+		unimplemented_dest = "POPPLER_DEST_FIT";
+		break;
+	case POPPLER_DEST_FITH:
+		unimplemented_dest = "POPPLER_DEST_FITH";
+		break;
+	case POPPLER_DEST_FITV:
+		unimplemented_dest = "POPPLER_DEST_FITV";
+		break;
+	case POPPLER_DEST_FITR:
+		unimplemented_dest = "POPPLER_DEST_FITR";
+		break;
+	case POPPLER_DEST_FITB:
+		unimplemented_dest = "POPPLER_DEST_FITB";
+		break;
+	case POPPLER_DEST_FITBH:
+		unimplemented_dest = "POPPLER_DEST_FITBH";
+		break;
+	case POPPLER_DEST_FITBV:
+		unimplemented_dest = "POPPLER_DEST_FITBV";
+		break;
+	}
+
+	if (unimplemented_dest) {
+		g_warning ("Unimplemented destination: %s, please post a bug report with a testcase.",
+			   unimplemented_dest);
+	}
+
+	link = ev_link_new_page (action->any.title, action->goto_dest.dest->page_num - 1);
+
+	return link;
+}
+
+static EvLink *
+ev_link_from_action (PopplerAction *action)
+{
+	EvLink *link = NULL;
 	const char *title;
+	const char *unimplemented_action = NULL;
 
 	title = action->any.title;
-	
-	if (action->type == POPPLER_ACTION_GOTO_DEST) {
-		link = ev_link_new_page (title, action->goto_dest.dest->page_num - 1);
-	} else if (action->type == POPPLER_ACTION_URI) {
+
+	switch (action->type) {
+	case POPPLER_ACTION_UNKNOWN:
+		g_warning ("Unknown action"); 
+		break;
+	case POPPLER_ACTION_GOTO_DEST:
+		link = ev_link_from_dest (action);
+		break;
+	case POPPLER_ACTION_GOTO_REMOTE:
+		unimplemented_action = "POPPLER_ACTION_GOTO_REMOTE";
+		break;
+	case POPPLER_ACTION_LAUNCH:
+		unimplemented_action = "POPPLER_ACTION_LAUNCH";
+		break;
+	case POPPLER_ACTION_URI:
 		link = ev_link_new_external (title, action->uri.uri);
-	} else {
+		break;
+	case POPPLER_ACTION_NAMED:
+		unimplemented_action = "POPPLER_ACTION_NAMED";
+		break;
+	case POPPLER_ACTION_MOVIE:
+		unimplemented_action = "POPPLER_ACTION_MOVIE";
+		break;
+	}
+
+	if (unimplemented_action) {
+		g_warning ("Unimplemented action: %s, please post a bug report with a testcase.",
+			   unimplemented_action);
+	}
+
+	if (link == NULL) {
 		link = ev_link_new_title (title);
 	}
 
diff --git a/shell/ev-view.c b/shell/ev-view.c
index 2edcd858..052529eb 100644
--- a/shell/ev-view.c
+++ b/shell/ev-view.c
@@ -922,6 +922,44 @@ view_rect_to_doc_rect (EvView *view,
 	doc_rect->y2 = doc_rect->y1 + (double) view_rect->height / view->scale;
 }
 
+static void
+doc_point_to_view_point (EvView       *view,
+                         int           page,
+		         EvPoint      *doc_point,
+		         GdkPoint     *view_point)
+{
+	GdkRectangle page_area;
+	GtkBorder border;
+	double x, y;
+	int width, height;
+
+	ev_page_cache_get_size (view->page_cache, page,
+				view->rotation,
+				1.0,
+				&width, &height);
+
+	if (view->rotation == 0) {
+		x = doc_point->x;
+		y = doc_point->y;
+	} else if (view->rotation == 90) {
+		x = width - doc_point->y;
+		y = doc_point->x;
+	} else if (view->rotation == 180) {
+		x = width - doc_point->x;
+		y = height - doc_point->y;
+	} else if (view->rotation == 270) {
+		x = doc_point->y;
+		y = height - doc_point->x;
+	} else {
+		g_assert_not_reached ();
+	}
+
+	get_page_extents (view, page, &page_area, &border);
+
+	view_point->x = x * view->scale + page_area.x;
+	view_point->y = y * view->scale + page_area.y;
+}
+
 static void
 doc_rect_to_view_rect (EvView       *view,
                        int           page,
@@ -1053,7 +1091,22 @@ get_link_at_location (EvView  *view,
 		return NULL;
 }
 
-/* FIXME: standardize this sometime */
+static void
+scroll_to_xyz_link (EvView *view, EvLink *link)
+{
+	GdkPoint view_point;
+	EvPoint doc_point;
+
+	doc_point.x = ev_link_get_left (link);
+	doc_point.y = ev_link_get_top (link);
+
+	doc_point_to_view_point (view, ev_link_get_page (link),
+				 &doc_point, &view_point);
+
+	gtk_adjustment_set_value (view->hadjustment, view_point.x);
+	gtk_adjustment_set_value (view->vadjustment, view_point.y);
+}
+
 static void
 go_to_link (EvView *view, EvLink *link)
 {
@@ -1070,6 +1123,9 @@ go_to_link (EvView *view, EvLink *link)
 			page = ev_link_get_page (link);
 			ev_page_cache_set_current_page (view->page_cache, page);
 			break;
+		case EV_LINK_TYPE_PAGE_XYZ:
+			scroll_to_xyz_link (view, link);
+			break;
 		case EV_LINK_TYPE_EXTERNAL_URI:
 			uri = ev_link_get_uri (link);
 			gnome_vfs_url_show (uri);
@@ -1092,6 +1148,7 @@ tip_from_link (EvView *view, EvLink *link)
 				msg = g_strdup (ev_link_get_title (link));
 			break;
 		case EV_LINK_TYPE_PAGE:
+		case EV_LINK_TYPE_PAGE_XYZ:
 			page_label = ev_page_cache_get_page_label (view->page_cache, ev_link_get_page (link));
 			msg = g_strdup_printf (_("Go to page %s"), page_label);
 			g_free (page_label);
-- 
2.43.5