/*
* Copyright (C) 2003 Marco Pesenti Gritti
* Copyright (C) 2003 Christian Persch
*
* This program 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 2, or (at your option)
* any later version.
*
* This program 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 this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Id$
*/
#include "ephy-location-action.h"
#include "ephy-location-entry.h"
#include "ephy-shell.h"
#include "ephy-debug.h"
#define EPHY_LOCATION_ACTION_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_LOCATION_ACTION, EphyLocationActionPrivate))
struct _EphyLocationActionPrivate
{
char *address;
};
static void ephy_location_action_init (EphyLocationAction *action);
static void ephy_location_action_class_init (EphyLocationActionClass *class);
static void ephy_location_action_finalize (GObject *object);
static void user_changed_cb (GtkWidget *proxy,
EphyLocationAction *action);
static void sync_address (GtkAction *action,
GParamSpec *pspec,
GtkWidget *proxy);
enum
{
PROP_0,
PROP_ADDRESS
};
enum
{
GO_LOCATION,
LAST_SIGNAL
};
static GObjectClass *parent_class = NULL;
static guint signals[LAST_SIGNAL] = { 0 };
GType
ephy_location_action_get_type (void)
{
static GType type = 0;
if (!type)
{
static const GTypeInfo type_info =
{
sizeof (EphyLocationActionClass),
(GBaseInitFunc) NULL,
(GBaseFinalizeFunc) NULL,
(GClassInitFunc) ephy_location_action_class_init,
(GClassFinalizeFunc) NULL,
NULL,
sizeof (EphyLocationAction),
0, /* n_preallocs */
(GInstanceInitFunc) ephy_location_action_init,
};
type = g_type_register_static (GTK_TYPE_ACTION,
"EphyLocationAction",
&type_info, 0);
}
return type;
}
static void
location_url_activate_cb (EphyLocationEntry *entry,
const char *content,
const char *target,
EphyLocationAction *action)
{
EphyBookmarks *bookmarks;
LOG ("Location url activated, content %s target %s", content, target)
bookmarks = ephy_shell_get_bookmarks (ephy_shell);
if (!content)
{
LOG ("Go to %s", target);
g_signal_emit (action, signals[GO_LOCATION], 0, target);
}
else
{
char *url;
url = ephy_bookmarks_solve_smart_url
(bookmarks, target, content);
g_return_if_fail (url != NULL);
LOG ("Go to %s", url);
g_signal_emit (action, signals[GO_LOCATION], 0, url);
g_free (url);
}
}
static void
user_changed_cb (GtkWidget *proxy, EphyLocationAction *action)
{
const char *address;
address = ephy_location_entry_get_location (EPHY_LOCATION_ENTRY (proxy));
LOG ("user_changed_cb, new address %s", address)
g_signal_handlers_block_by_func (action, G_CALLBACK (sync_address), proxy);
ephy_location_action_set_address (action, address);
g_signal_handlers_unblock_by_func (action, G_CALLBACK (sync_address), proxy);
}
static void
sync_address (GtkAction *act, GParamSpec *pspec, GtkWidget *proxy)
{
EphyLocationAction *action = EPHY_LOCATION_ACTION (act);
LOG ("sync_address")
g_return_if_fail (EPHY_IS_LOCATION_ENTRY (proxy));
g_signal_handlers_block_by_func (proxy, G_CALLBACK (user_changed_cb), action);
ephy_location_entry_set_location (EPHY_LOCATION_ENTRY (proxy),
action->priv->address);
g_signal_handlers_unblock_by_func (proxy, G_CALLBACK (user_changed_cb), action);
}
static void
connect_proxy (GtkAction *action, GtkWidget *proxy)
{
LOG ("Connect proxy")
if (EPHY_IS_LOCATION_ENTRY (proxy))
{
EphyAutocompletion *ac;
ac = EPHY_AUTOCOMPLETION (ephy_shell_get_autocompletion (ephy_shell));
ephy_location_entry_set_autocompletion (EPHY_LOCATION_ENTRY (proxy), ac);
sync_address (action, NULL, proxy);
g_signal_connect_object (action, "notify::address",
G_CALLBACK (sync_address), proxy, 0);
g_signal_connect_object (proxy, "activated",
G_CALLBACK (location_url_activate_cb),
action, 0);
g_signal_connect_object (proxy, "user_changed",
G_CALLBACK (user_changed_cb), action, 0);
}
(* GTK_ACTION_CLASS (parent_class)->connect_proxy) (action, proxy);
}
static void
disconnect_proxy (GtkAction *action, GtkWidget *proxy)
{
LOG ("Disconnect proxy")
if (EPHY_IS_LOCATION_ENTRY (proxy))
{
g_signal_handlers_disconnect_by_func
(action, G_CALLBACK (sync_address), proxy);
g_signal_handlers_disconnect_by_func
(proxy, G_CALLBACK (location_url_activate_cb), action);
g_signal_handlers_disconnect_by_func
(proxy, G_CALLBACK (user_changed_cb), action);
}
(* GTK_ACTION_CLASS (parent_class)->disconnect_proxy) (action, proxy);
}
static void
ephy_location_action_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
EphyLocationAction *action = EPHY_LOCATION_ACTION (object);
switch (prop_id)
{
case PROP_ADDRESS:
ephy_location_action_set_address (action, g_value_get_string (value));
break;
}
}
static void
ephy_location_action_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
EphyLocationAction *action = EPHY_LOCATION_ACTION (object);
switch (prop_id)
{
case PROP_ADDRESS:
g_value_set_string (value, ephy_location_action_get_address (action));
break;
}
}
static void
ephy_location_action_activate (GtkAction *action)
{
GSList *proxies;
/* Note: this makes sense only for a single proxy */
proxies = gtk_action_get_proxies (action);
if (proxies)
{
ephy_location_entry_activate (EPHY_LOCATION_ENTRY (proxies->data));
}
}
static void
ephy_location_action_class_init (EphyLocationActionClass *class)
{
GObjectClass *object_class = G_OBJECT_CLASS (class);
GtkActionClass *action_class = GTK_ACTION_CLASS (class);
parent_class = g_type_class_peek_parent (class);
object_class->finalize = ephy_location_action_finalize;
object_class->get_property = ephy_location_action_get_property;
object_class->set_property = ephy_location_action_set_property;
action_class->toolbar_item_type = EPHY_TYPE_LOCATION_ENTRY;
action_class->connect_proxy = connect_proxy;
action_class->disconnect_proxy = disconnect_proxy;
action_class->activate = ephy_location_action_activate;
signals[GO_LOCATION] =
g_signal_new ("go_location",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EphyLocationActionClass, go_location),
NULL, NULL,
g_cclosure_marshal_VOID__STRING,
G_TYPE_NONE,
1,
G_TYPE_STRING);
g_object_class_install_property (object_class,
PROP_ADDRESS,
g_param_spec_string ("address",
"Address",
"The address",
"",
G_PARAM_READWRITE));
g_type_class_add_private (object_class, sizeof (EphyLocationActionPrivate));
}
static void
ephy_location_action_init (EphyLocationAction *action)
{
action->priv = EPHY_LOCATION_ACTION_GET_PRIVATE (action);
action->priv->address = g_strdup ("");
}
static void
ephy_location_action_finalize (GObject *object)
{
EphyLocationAction *action = EPHY_LOCATION_ACTION (object);
g_free (action->priv->address);
}
const char *
ephy_location_action_get_address (EphyLocationAction *action)
{
g_return_val_if_fail (EPHY_IS_LOCATION_ACTION (action), "");
return action->priv->address;
}
void
ephy_location_action_set_address (EphyLocationAction *action,
const char *address)
{
g_return_if_fail (EPHY_IS_LOCATION_ACTION (action));
LOG ("set_address %s", address)
g_free (action->priv->address);
action->priv->address = g_strdup (address ? address : "");
g_object_notify (G_OBJECT (action), "address");
}
static void
clear_history (GtkWidget *proxy, gpointer user_data)
{
if (EPHY_IS_LOCATION_ENTRY (proxy))
{
ephy_location_entry_clear_history (EPHY_LOCATION_ENTRY (proxy));
}
}
void
ephy_location_action_clear_history (EphyLocationAction *action)
{
GSList *proxies;
proxies = gtk_action_get_proxies (GTK_ACTION (action));
g_slist_foreach (proxies, (GFunc) clear_history, NULL);
}