/* * Copyright (C) 2002 Jorn Baayen * * 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 of the License, 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. * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include "ephy-new-bookmark.h" #include "ephy-state.h" #include "ephy-topics-selector.h" #include "ephy-debug.h" #include "ephy-stock-icons.h" #include "ephy-gui.h" static void ephy_new_bookmark_class_init (EphyNewBookmarkClass *klass); static void ephy_new_bookmark_init (EphyNewBookmark *editor); static void ephy_new_bookmark_finalize (GObject *object); static void ephy_new_bookmark_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void ephy_new_bookmark_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); #define EPHY_NEW_BOOKMARK_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_NEW_BOOKMARK, EphyNewBookmarkPrivate)) struct EphyNewBookmarkPrivate { EphyBookmarks *bookmarks; char *location; char *icon; gulong id; GtkWidget *title_entry; GtkWidget *topics_selector; }; enum { PROP_0, PROP_BOOKMARKS, PROP_LOCATION }; static GObjectClass *parent_class = NULL; GType ephy_new_bookmark_get_type (void) { static GType ephy_new_bookmark_type = 0; if (ephy_new_bookmark_type == 0) { static const GTypeInfo our_info = { sizeof (EphyNewBookmarkClass), NULL, NULL, (GClassInitFunc) ephy_new_bookmark_class_init, NULL, NULL, sizeof (EphyNewBookmark), 0, (GInstanceInitFunc) ephy_new_bookmark_init }; ephy_new_bookmark_type = g_type_register_static (GTK_TYPE_DIALOG, "EphyNewBookmark", &our_info, 0); } return ephy_new_bookmark_type; } static void ephy_new_bookmark_class_init (EphyNewBookmarkClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); parent_class = g_type_class_peek_parent (klass); object_class->finalize = ephy_new_bookmark_finalize; object_class->set_property = ephy_new_bookmark_set_property; object_class->get_property = ephy_new_bookmark_get_property; g_object_class_install_property (object_class, PROP_BOOKMARKS, g_param_spec_object ("bookmarks", "Bookmarks set", "Bookmarks set", EPHY_TYPE_BOOKMARKS, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (object_class, PROP_LOCATION, g_param_spec_string ("location", "Bookmark location", "Bookmark location", "", G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_type_class_add_private (object_class, sizeof(EphyNewBookmarkPrivate)); } static void ephy_new_bookmark_finalize (GObject *object) { EphyNewBookmark *editor = EPHY_NEW_BOOKMARK (object); g_free (editor->priv->location); g_free (editor->priv->icon); G_OBJECT_CLASS (parent_class)->finalize (object); } static void ephy_new_bookmark_add (EphyNewBookmark *new_bookmark) { char *title; EphyNode *node; EphyTopicsSelector *selector; selector = EPHY_TOPICS_SELECTOR (new_bookmark->priv->topics_selector); title = gtk_editable_get_chars (GTK_EDITABLE (new_bookmark->priv->title_entry), 0, -1); node = ephy_bookmarks_add (new_bookmark->priv->bookmarks, title, new_bookmark->priv->location); new_bookmark->priv->id = ephy_node_get_id (node); ephy_topics_selector_set_bookmark (selector, node); ephy_topics_selector_apply (selector); if (new_bookmark->priv->icon) { ephy_bookmarks_set_icon (new_bookmark->priv->bookmarks, new_bookmark->priv->location, new_bookmark->priv->icon); } g_free (title); } static void response_cb (EphyNewBookmark *new_bookmark, int response_id, gpointer user_data) { switch (response_id) { case GTK_RESPONSE_HELP: ephy_gui_help (GTK_WINDOW (new_bookmark), "epiphany", "to-create-new-bookmark"); break; /* For both OK and Cancel we want to destroy the dialog */ case GTK_RESPONSE_OK: ephy_new_bookmark_add (new_bookmark); case GTK_RESPONSE_CANCEL: default: gtk_widget_destroy (GTK_WIDGET (new_bookmark)); break; } } static GtkWidget * build_editing_table (EphyNewBookmark *editor) { GtkWidget *table, *label, *entry, *topics_selector, *scrolled_window; char *str; table = gtk_table_new (2, 2, FALSE); gtk_table_set_row_spacings (GTK_TABLE (table), 6); gtk_table_set_col_spacings (GTK_TABLE (table), 12); gtk_container_set_border_width (GTK_CONTAINER (GTK_TABLE (table)), 5); gtk_widget_show (table); entry = gtk_entry_new (); gtk_entry_set_activates_default (GTK_ENTRY (entry), TRUE); editor->priv->title_entry = entry; gtk_widget_set_size_request (entry, 200, -1); gtk_widget_show (entry); label = gtk_label_new (NULL); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); str = g_strconcat ("", _("_Title:"), "", NULL); gtk_label_set_markup_with_mnemonic (GTK_LABEL (label), str); g_free (str); gtk_label_set_mnemonic_widget (GTK_LABEL (label), entry); gtk_widget_show (label); gtk_table_attach (GTK_TABLE (table), label, 0, 1, 0, 1, GTK_FILL, 0, 0, 0); gtk_table_attach (GTK_TABLE (table), entry, 1, 2, 0, 1, GTK_FILL, 0, 0, 0); topics_selector = ephy_topics_selector_new (editor->priv->bookmarks, NULL); gtk_widget_show (topics_selector); scrolled_window = g_object_new (GTK_TYPE_SCROLLED_WINDOW, "hadjustment", NULL, "vadjustment", NULL, "hscrollbar_policy", GTK_POLICY_AUTOMATIC, "vscrollbar_policy", GTK_POLICY_AUTOMATIC, "shadow_type", GTK_SHADOW_IN, NULL); gtk_widget_show (scrolled_window); gtk_container_add (GTK_CONTAINER (scrolled_window), topics_selector); editor->priv->topics_selector = topics_selector; label = gtk_label_new (NULL); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.0); str = g_strconcat ("", _("To_pics:"), "", NULL); gtk_label_set_markup_with_mnemonic (GTK_LABEL (label), str); g_free (str); gtk_label_set_mnemonic_widget (GTK_LABEL (label), topics_selector); gtk_widget_show (label); gtk_table_attach (GTK_TABLE (table), label, 0, 1, 1, 2, GTK_FILL, GTK_FILL, 0, 0); gtk_table_attach (GTK_TABLE (table), scrolled_window, 1, 2, 1, 2, GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0); return table; } static void ephy_new_bookmark_construct (EphyNewBookmark *editor) { GdkPixbuf *icon; ephy_state_add_window (GTK_WIDGET(editor), "new_bookmark", 280, 240, EPHY_STATE_WINDOW_SAVE_SIZE); gtk_window_set_title (GTK_WINDOW (editor), _("New Bookmark")); icon = gtk_widget_render_icon (GTK_WIDGET (editor), EPHY_STOCK_BOOKMARK_PAGE, GTK_ICON_SIZE_MENU, NULL); gtk_window_set_icon (GTK_WINDOW (editor), icon); g_object_unref(icon); gtk_dialog_set_has_separator (GTK_DIALOG (editor), FALSE); gtk_container_set_border_width (GTK_CONTAINER (editor), 5); g_signal_connect (G_OBJECT (editor), "response", G_CALLBACK (response_cb), editor); gtk_box_set_spacing (GTK_BOX (GTK_DIALOG (editor)->vbox), 2); gtk_box_pack_start (GTK_BOX (GTK_DIALOG (editor)->vbox), build_editing_table (editor), TRUE, TRUE, 0); gtk_dialog_add_button (GTK_DIALOG (editor), GTK_STOCK_HELP, GTK_RESPONSE_HELP); gtk_dialog_add_button (GTK_DIALOG (editor), GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL); gtk_dialog_add_button (GTK_DIALOG (editor), GTK_STOCK_OK, GTK_RESPONSE_OK); gtk_dialog_set_default_response (GTK_DIALOG (editor), GTK_RESPONSE_OK); } static GtkWidget* duplicate_dialog_construct (GtkWindow *parent, const char *title) { GtkWidget *dialog; GtkWidget *hbox, *vbox, *label, *image; char *str, *tmp_str, *tmp_title; /* FIXME: We "should" use gtk_message dialog here * but it doesn't support markup of text yet * so we build our own. See bug 65501. */ dialog = gtk_dialog_new_with_buttons (_("Duplicated Bookmark"), GTK_WINDOW (parent), GTK_DIALOG_NO_SEPARATOR, GTK_STOCK_OK, GTK_RESPONSE_OK, NULL); gtk_window_set_resizable (GTK_WINDOW (dialog), FALSE); gtk_container_set_border_width (GTK_CONTAINER (dialog), 5); gtk_box_set_spacing (GTK_BOX (GTK_DIALOG (dialog)->vbox), 14); hbox = gtk_hbox_new (FALSE, 6); gtk_widget_show (hbox); gtk_box_pack_start (GTK_BOX (GTK_DIALOG (dialog)->vbox), hbox, TRUE, TRUE, 0); image = gtk_image_new_from_stock (GTK_STOCK_DIALOG_INFO, GTK_ICON_SIZE_DIALOG); gtk_misc_set_alignment (GTK_MISC (image), 0.5, 0.0); gtk_widget_show (image); gtk_box_pack_start (GTK_BOX (hbox), image, TRUE, TRUE, 0); vbox = gtk_vbox_new (FALSE, 6); gtk_widget_show (vbox); gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0); label = gtk_label_new (NULL); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_label_set_line_wrap (GTK_LABEL (label), TRUE); tmp_title = g_strconcat ("", title, "", NULL); tmp_str = g_strdup_printf (_("A bookmark titled %s already exists for this page."), tmp_title); str = g_strconcat ("", tmp_str, "", NULL); gtk_label_set_markup (GTK_LABEL (label), str); g_free (tmp_title); g_free (tmp_str); g_free (str); gtk_box_pack_start (GTK_BOX (vbox), label, TRUE, TRUE, 0); gtk_widget_show (label); return dialog; } static void duplicate_bookmark_response_cb (EphyNewBookmark *new_bookmark, int response_id, gpointer user_data) { switch (response_id) { case GTK_RESPONSE_OK: gtk_widget_destroy (GTK_WIDGET (new_bookmark)); break; } } gboolean ephy_new_bookmark_is_unique (EphyBookmarks *bookmarks, GtkWindow *parent, const char *address) { EphyNode *node; node = ephy_bookmarks_find_bookmark (bookmarks, address); if (node) { GtkWidget *dialog; const char *title; title = ephy_node_get_property_string (node, EPHY_NODE_BMK_PROP_TITLE); dialog = duplicate_dialog_construct (parent, title); gtk_window_set_transient_for (GTK_WINDOW (dialog), parent); g_signal_connect (G_OBJECT (dialog), "response", G_CALLBACK (duplicate_bookmark_response_cb), dialog); gtk_widget_show (GTK_WIDGET (dialog)); return FALSE; } return TRUE; } GtkWidget * ephy_new_bookmark_new (EphyBookmarks *bookmarks, GtkWindow *parent, const char *location) { EphyNewBookmark *editor; g_assert (bookmarks != NULL); editor = EPHY_NEW_BOOKMARK (g_object_new (EPHY_TYPE_NEW_BOOKMARK, "bookmarks", bookmarks, "location", location, NULL)); if (parent) { gtk_window_set_transient_for (GTK_WINDOW (editor), parent); } ephy_new_bookmark_construct (editor); return GTK_WIDGET (editor); } static void ephy_new_bookmark_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { EphyNewBookmark *editor = EPHY_NEW_BOOKMARK (object); switch (prop_id) { case PROP_BOOKMARKS: editor->priv->bookmarks = g_value_get_object (value); break; case PROP_LOCATION: g_free (editor->priv->location); editor->priv->location = g_strdup (g_value_get_string (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void ephy_new_bookmark_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { EphyNewBookmark *editor = EPHY_NEW_BOOKMARK (object); switch (prop_id) { case PROP_LOCATION: g_value_set_string (value, editor->priv->location); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void ephy_new_bookmark_init (EphyNewBookmark *editor) { editor->priv = EPHY_NEW_BOOKMARK_GET_PRIVATE (editor); editor->priv->location = NULL; editor->priv->icon = NULL; editor->priv->id = 0; } void ephy_new_bookmark_set_title (EphyNewBookmark *bookmark, const char *title) { const char *real_title; LOG ("Setting new bookmark title to: \"%s\"", title) if (title == NULL || strlen (title) == 0) { real_title = bookmark->priv->location; } else { real_title = title; } gtk_entry_set_text (GTK_ENTRY (bookmark->priv->title_entry), real_title); } void ephy_new_bookmark_set_icon (EphyNewBookmark *bookmark, const char *icon) { g_free (bookmark->priv->icon); bookmark->priv->icon = icon ? g_strdup (icon) : NULL; } gulong ephy_new_bookmark_get_id (EphyNewBookmark *bookmark) { return bookmark->priv->id; }