/* Copyright (C) 2007-2011 Christian Dywan Copyright (C) 2008 Dale Whittaker Copyright (C) 2009 Jérôme Geulfucci This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. See the file COPYING for the full license text. */ #include "midori-browser.h" #include "midori-app.h" #include "midori-array.h" #include "midori-view.h" #include "midori-preferences.h" #include "midori-panel.h" #include "midori-locationaction.h" #include "midori-searchaction.h" #include "midori-findbar.h" #include "midori-transferbar.h" #include "midori-platform.h" #include "midori-core.h" #include "marshal.h" #include #include #include #include #include #ifdef HAVE_GRANITE #include #endif #ifdef HAVE_ZEITGEIST #include #endif #ifdef HAVE_UNISTD_H #include #endif #ifdef HAVE_HILDON_2_2 #include #include #include #define MCE_SIGNAL_MATCH "type='signal'," \ "sender='" MCE_SERVICE "'," \ "path='" MCE_SIGNAL_PATH "'," \ "interface='" MCE_SIGNAL_IF "'" #include #include #endif #include struct _MidoriBrowser { #if HAVE_HILDON HildonWindow parent_instance; #else GtkWindow parent_instance; #endif GtkActionGroup* action_group; GtkWidget* menubar; GtkWidget* throbber; GtkWidget* navigationbar; GtkWidget* bookmarkbar; GtkWidget* panel; GtkWidget* notebook; GtkWidget* inspector; GtkWidget* inspector_view; GtkWidget* find; GtkWidget* statusbar; GtkWidget* statusbar_contents; GtkWidget* transferbar; gchar* statusbar_text; gint last_window_width, last_window_height; guint alloc_timeout; gint last_tab_size; guint panel_timeout; gint clear_private_data; MidoriWebSettings* settings; KatzeArray* proxy_array; KatzeArray* bookmarks; KatzeArray* trash; KatzeArray* search_engines; KatzeArray* history; MidoriSpeedDial* dial; gboolean show_tabs; gboolean show_navigationbar; gboolean show_statusbar; guint maximum_history_age; gchar* location_entry_search; guint last_web_search; gchar* news_aggregator; }; #if HAVE_HILDON G_DEFINE_TYPE (MidoriBrowser, midori_browser, HILDON_TYPE_WINDOW) #else G_DEFINE_TYPE (MidoriBrowser, midori_browser, GTK_TYPE_WINDOW) #endif enum { PROP_0, PROP_MENUBAR, PROP_NAVIGATIONBAR, PROP_NOTEBOOK, PROP_PANEL, PROP_URI, PROP_TAB, PROP_LOAD_STATUS, PROP_STATUSBAR, PROP_STATUSBAR_TEXT, PROP_SETTINGS, PROP_BOOKMARKS, PROP_TRASH, PROP_SEARCH_ENGINES, PROP_HISTORY, PROP_SPEED_DIAL, PROP_SHOW_TABS, }; enum { NEW_WINDOW, ADD_TAB, REMOVE_TAB, MOVE_TAB, ACTIVATE_ACTION, ADD_DOWNLOAD, SEND_NOTIFICATION, POPULATE_TOOL_MENU, POPULATE_TOOLBAR_MENU, QUIT, SHOW_PREFERENCES, LAST_SIGNAL }; static guint signals[LAST_SIGNAL]; static void midori_browser_dispose (GObject* object); static void midori_browser_finalize (GObject* object); static void midori_browser_set_property (GObject* object, guint prop_id, const GValue* value, GParamSpec* pspec); static void midori_browser_get_property (GObject* object, guint prop_id, GValue* value, GParamSpec* pspec); void midori_bookmarks_import_array_db (sqlite3* db, KatzeArray* array, gint64 parentid); gboolean midori_bookmarks_update_item_db (sqlite3* db, KatzeItem* item); void midori_browser_open_bookmark (MidoriBrowser* browser, KatzeItem* item); static void midori_bookmarkbar_populate (MidoriBrowser* browser); static void midori_bookmarkbar_clear (GtkWidget* toolbar); static void midori_browser_new_history_item (MidoriBrowser* browser, KatzeItem* item); static void _midori_browser_set_toolbar_style (MidoriBrowser* browser, MidoriToolbarStyle toolbar_style); GtkWidget* midori_panel_construct_menu_item (MidoriPanel* panel, MidoriViewable* viewable, gboolean popup); static void midori_browser_settings_notify (MidoriWebSettings* web_settings, GParamSpec* pspec, MidoriBrowser* browser); void midori_panel_set_toolbar_style (MidoriPanel* panel, GtkToolbarStyle style); static void midori_browser_set_bookmarks (MidoriBrowser* browser, KatzeArray* bookmarks); static void midori_browser_add_speed_dial (MidoriBrowser* browser); gboolean midori_transferbar_confirm_delete (MidoriTransferbar* transferbar); static void midori_browser_notebook_size_allocate_cb (GtkWidget* notebook, GdkRectangle* allocation, MidoriBrowser* browser); void midori_transferbar_add_download_item (MidoriTransferbar* transferbar, WebKitDownload* download); void midori_transferbar_check_size (GtkWidget* statusbar, MidoriTransferbar* transferbar); #define _action_by_name(brwsr, nme) \ gtk_action_group_get_action (brwsr->action_group, nme) #define _action_set_sensitive(brwsr, nme, snstv) \ gtk_action_set_sensitive (_action_by_name (brwsr, nme), snstv); #define _action_set_visible(brwsr, nme, vsbl) \ gtk_action_set_visible (_action_by_name (brwsr, nme), vsbl); #define _action_set_active(brwsr, nme, actv) \ gtk_toggle_action_set_active (GTK_TOGGLE_ACTION ( \ _action_by_name (brwsr, nme)), actv); static gboolean midori_browser_is_fullscreen (MidoriBrowser* browser) { GdkWindow* window = gtk_widget_get_window (GTK_WIDGET (browser)); GdkWindowState state = window ? gdk_window_get_state (window) : 0; return state & GDK_WINDOW_STATE_FULLSCREEN; } static gboolean _toggle_tabbar_smartly (MidoriBrowser* browser, gboolean ignore_fullscreen) { gboolean has_tabs = midori_browser_get_n_pages (browser) > 1; #ifdef HAVE_GRANITE gboolean show_tabs = !midori_browser_is_fullscreen (browser) || ignore_fullscreen; granite_widgets_dynamic_notebook_set_show_tabs ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), show_tabs); #else gboolean show_tabs = browser->show_tabs && (!midori_browser_is_fullscreen (browser) || ignore_fullscreen) && (has_tabs || katze_object_get_boolean (browser->settings, "always-show-tabbar")); gtk_notebook_set_show_tabs (GTK_NOTEBOOK (browser->notebook), show_tabs); gtk_notebook_set_show_border (GTK_NOTEBOOK (browser->notebook), show_tabs); #endif return has_tabs; } static void _midori_browser_update_actions (MidoriBrowser* browser) { gboolean has_tabs = _toggle_tabbar_smartly (browser, FALSE); _action_set_sensitive (browser, "TabPrevious", has_tabs); _action_set_sensitive (browser, "TabNext", has_tabs); if (browser->trash) { gboolean trash_empty = katze_array_is_empty (browser->trash); _action_set_sensitive (browser, "UndoTabClose", !trash_empty); _action_set_sensitive (browser, "Trash", !trash_empty); } } static void midori_browser_update_secondary_icon (MidoriBrowser* browser, MidoriView* view, GtkAction* action) { if (g_object_get_data (G_OBJECT (view), "news-feeds")) { midori_location_action_set_secondary_icon ( MIDORI_LOCATION_ACTION (action), STOCK_NEWS_FEED); _action_set_sensitive (browser, "AddNewsFeed", TRUE); } else { midori_location_action_set_secondary_icon ( MIDORI_LOCATION_ACTION (action), NULL); _action_set_sensitive (browser, "AddNewsFeed", FALSE); } } static void _midori_browser_update_interface (MidoriBrowser* browser, MidoriView* view) { GtkAction* action; _action_set_sensitive (browser, "Back", midori_view_can_go_back (view)); _action_set_sensitive (browser, "Forward", midori_view_can_go_forward (view)); _action_set_sensitive (browser, "Previous", midori_view_get_previous_page (view) != NULL); _action_set_sensitive (browser, "Next", midori_view_get_next_page (view) != NULL); _action_set_sensitive (browser, "AddSpeedDial", !midori_view_is_blank (view)); _action_set_sensitive (browser, "BookmarkAdd", !midori_view_is_blank (view)); _action_set_sensitive (browser, "SaveAs", midori_view_can_save (view)); _action_set_sensitive (browser, "ZoomIn", midori_view_can_zoom_in (view)); _action_set_sensitive (browser, "ZoomOut", midori_view_can_zoom_out (view)); _action_set_sensitive (browser, "ZoomNormal", midori_view_get_zoom_level (view) != 1.0f); _action_set_sensitive (browser, "Encoding", midori_view_can_view_source (view)); _action_set_sensitive (browser, "SourceView", midori_view_can_view_source (view)); action = _action_by_name (browser, "NextForward"); if (midori_view_can_go_forward (view)) { g_object_set (action, "stock-id", GTK_STOCK_GO_FORWARD, "tooltip", _("Go forward to the next page"), "sensitive", TRUE, NULL); } else { g_object_set (action, "stock-id", GTK_STOCK_MEDIA_NEXT, "tooltip", _("Go to the next sub-page"), "sensitive", midori_view_get_next_page (view) != NULL, NULL); } action = _action_by_name (browser, "Location"); midori_location_action_set_security_hint ( MIDORI_LOCATION_ACTION (action), midori_view_get_security (view)); midori_browser_update_secondary_icon (browser, view, action); } static void _midori_browser_set_statusbar_text (MidoriBrowser* browser, MidoriView* view, const gchar* text) { #if GTK_CHECK_VERSION (3, 2, 0) gboolean is_location = FALSE; #else GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); gboolean is_location = widget && GTK_IS_ENTRY (widget) && GTK_IS_ALIGNMENT (gtk_widget_get_parent (widget)); #endif katze_assign (browser->statusbar_text, midori_uri_format_for_display (text)); if (view == NULL) return; if (!gtk_widget_get_visible (browser->statusbar) && !is_location && text && *text) { #if GTK_CHECK_VERSION (3, 2, 0) midori_view_set_overlay_text (view, browser->statusbar_text); #else GtkAction* action = _action_by_name (browser, "Location"); MidoriLocationAction* location_action = MIDORI_LOCATION_ACTION (action); midori_location_action_set_text (location_action, browser->statusbar_text); midori_location_action_set_secondary_icon (location_action, NULL); #endif } else if (!gtk_widget_get_visible (browser->statusbar) && !is_location) { #if GTK_CHECK_VERSION (3, 2, 0) midori_view_set_overlay_text (view, NULL); #else GtkAction* action = _action_by_name (browser, "Location"); MidoriLocationAction* location_action = MIDORI_LOCATION_ACTION (action); midori_browser_update_secondary_icon (browser, view, action); midori_location_action_set_text (location_action, midori_view_get_display_uri (view)); #endif } else { gtk_statusbar_pop (GTK_STATUSBAR (browser->statusbar), 1); gtk_statusbar_push (GTK_STATUSBAR (browser->statusbar), 1, katze_str_non_null (browser->statusbar_text)); } } void midori_browser_set_current_page_smartly (MidoriBrowser* browser, gint n) { if (!katze_object_get_boolean (browser->settings, "open-tabs-in-the-background")) midori_browser_set_current_page (browser, n); } static void _midori_browser_update_progress (MidoriBrowser* browser, MidoriView* view) { GtkAction* action; gdouble progress = midori_view_get_progress (view); gboolean loading = progress > 0.0; action = _action_by_name (browser, "Location"); midori_location_action_set_progress (MIDORI_LOCATION_ACTION (action), progress); _action_set_sensitive (browser, "Reload", !loading); _action_set_sensitive (browser, "Stop", loading); action = _action_by_name (browser, "ReloadStop"); if (!loading) { g_object_set (action, "stock-id", GTK_STOCK_REFRESH, "tooltip", _("Reload the current page"), NULL); katze_item_set_meta_integer (midori_view_get_proxy_item (view), "dont-write-history", -1); } else { g_object_set (action, "stock-id", GTK_STOCK_STOP, "tooltip", _("Stop loading the current page"), NULL); } gtk_widget_set_sensitive (browser->throbber, loading); katze_throbber_set_animated (KATZE_THROBBER (browser->throbber), loading); } /** * midori_browser_update_history: * @item: a #KatzeItem * @type: "website", "bookmark" or "download" * @event: "access", "leave", "modify", "delete" * * Since: 0.4.7 **/ void midori_browser_update_history (KatzeItem* item, const gchar* type, const gchar* event) { #ifdef HAVE_ZEITGEIST const gchar* inter; if (strstr (event, "access")) inter = ZEITGEIST_ZG_ACCESS_EVENT; else if (strstr (event, "leave")) inter = ZEITGEIST_ZG_LEAVE_EVENT; else if (strstr (event, "modify")) inter = ZEITGEIST_ZG_MODIFY_EVENT; else if (strstr (event, "create")) inter = ZEITGEIST_ZG_CREATE_EVENT; else if (strstr (event, "delete")) inter = ZEITGEIST_ZG_DELETE_EVENT; else g_assert_not_reached (); zeitgeist_log_insert_events_no_reply (zeitgeist_log_get_default (), zeitgeist_event_new_full (inter, ZEITGEIST_ZG_USER_ACTIVITY, "application://midori.desktop", zeitgeist_subject_new_full ( item->uri, strstr (type, "bookmark") ? ZEITGEIST_NFO_BOOKMARK : ZEITGEIST_NFO_WEBSITE, zeitgeist_manifestation_for_uri (item->uri), katze_item_get_meta_string (item, "mime-type"), NULL, item->name, NULL), NULL), NULL); #endif } static void midori_browser_update_history_title (MidoriBrowser* browser, KatzeItem* item) { sqlite3* db; static sqlite3_stmt* stmt = NULL; g_return_if_fail (katze_item_get_uri (item) != NULL); db = g_object_get_data (G_OBJECT (browser->history), "db"); g_return_if_fail (db != NULL); if (!stmt) { const gchar* sqlcmd; sqlcmd = "UPDATE history SET title=? WHERE uri = ? and date=?"; sqlite3_prepare_v2 (db, sqlcmd, -1, &stmt, NULL); } sqlite3_bind_text (stmt, 1, katze_item_get_name (item), -1, 0); sqlite3_bind_text (stmt, 2, katze_item_get_uri (item), -1, 0); sqlite3_bind_int64 (stmt, 3, katze_item_get_added (item)); if (sqlite3_step (stmt) != SQLITE_DONE) g_printerr (_("Failed to update title: %s\n"), sqlite3_errmsg (db)); sqlite3_reset (stmt); sqlite3_clear_bindings (stmt); midori_browser_update_history (item, "website", "access"); } static void _midori_browser_activate_action (MidoriBrowser* browser, const gchar* name) { GtkAction* action = _action_by_name (browser, name); if (action) gtk_action_activate (action); else g_warning (_("Unexpected action '%s'."), name); } static void midori_view_notify_icon_cb (MidoriView* view, GParamSpec* pspec, MidoriBrowser* browser) { if (midori_browser_get_current_tab (browser) != (GtkWidget*)view) return; if (midori_paths_is_readonly () /* APP, PRIVATE */) gtk_window_set_icon (GTK_WINDOW (browser), midori_view_get_icon (view)); } static void midori_view_notify_load_status_cb (GtkWidget* widget, GParamSpec* pspec, MidoriBrowser* browser) { MidoriView* view = MIDORI_VIEW (widget); MidoriLoadStatus load_status = midori_view_get_load_status (view); const gchar* uri; GtkAction* action; uri = midori_view_get_display_uri (view); action = _action_by_name (browser, "Location"); if (widget == midori_browser_get_current_tab (browser)) { if (load_status == MIDORI_LOAD_COMMITTED) { midori_location_action_set_text ( MIDORI_LOCATION_ACTION (action), uri); g_object_notify (G_OBJECT (browser), "uri"); } _midori_browser_update_interface (browser, view); _midori_browser_set_statusbar_text (browser, view, NULL); /* This is a hack to ensure that the address entry is focussed with speed dial open. */ if (midori_view_is_blank (view)) midori_browser_activate_action (browser, "Location"); } g_object_notify (G_OBJECT (browser), "load-status"); } static void midori_view_notify_progress_cb (GtkWidget* view, GParamSpec* pspec, MidoriBrowser* browser) { if (view == midori_browser_get_current_tab (browser)) _midori_browser_update_progress (browser, MIDORI_VIEW (view)); } static void midori_view_notify_uri_cb (GtkWidget* widget, GParamSpec* pspec, MidoriBrowser* browser) { if (widget == midori_browser_get_current_tab (browser)) { MidoriView* view = MIDORI_VIEW (widget); const gchar* uri = midori_view_get_display_uri (view); GtkAction* action = _action_by_name (browser, "Location"); midori_location_action_set_text (MIDORI_LOCATION_ACTION (action), uri); _action_set_sensitive (browser, "Back", midori_view_can_go_back (view)); _action_set_sensitive (browser, "Forward", midori_view_can_go_forward (view)); } } static void midori_browser_set_title (MidoriBrowser* browser, const gchar* title) { if (katze_object_get_boolean (browser->settings, "enable-private-browsing")) { gchar* window_title = g_strdup_printf (_("%s (Private Browsing)"), title); gtk_window_set_title (GTK_WINDOW (browser), window_title); g_free (window_title); } else gtk_window_set_title (GTK_WINDOW (browser), title); } static void midori_view_notify_title_cb (GtkWidget* widget, GParamSpec* pspec, MidoriBrowser* browser) { MidoriView* view = MIDORI_VIEW (widget); if (midori_view_get_load_status (view) == MIDORI_LOAD_COMMITTED) { KatzeItem* proxy; if (browser->history && browser->maximum_history_age) { const gchar* proxy_uri; proxy = midori_view_get_proxy_item (view); proxy_uri = katze_item_get_uri (proxy); if (!midori_uri_is_blank (proxy_uri) && (katze_item_get_meta_integer (proxy, "history-step") == -1)) { if (!katze_item_get_meta_boolean (proxy, "dont-write-history")) { midori_browser_new_history_item (browser, proxy); katze_item_set_meta_integer (proxy, "history-step", 1); } } else if (katze_item_get_name (proxy) && !midori_uri_is_blank (proxy_uri) && (katze_item_get_meta_integer (proxy, "history-step") == 1)) { midori_browser_update_history_title (browser, proxy); katze_item_set_meta_integer (proxy, "history-step", 2); } } } if (widget == midori_browser_get_current_tab (browser)) midori_browser_set_title (browser, midori_view_get_display_title (view)); } static void midori_view_notify_minimized_cb (GtkWidget* widget, GParamSpec* pspec, MidoriBrowser* browser) { if (katze_object_get_boolean (widget, "minimized")) { #ifndef HAVE_GRANITE GtkNotebook* notebook = GTK_NOTEBOOK (browser->notebook); GtkWidget* label = gtk_notebook_get_tab_label (notebook, widget); gtk_widget_set_size_request (label, -1, -1); #endif } else midori_browser_notebook_size_allocate_cb (NULL, NULL, browser); } static void midori_view_notify_zoom_level_cb (GtkWidget* view, GParamSpec* pspec, MidoriBrowser* browser) { if (view == midori_browser_get_current_tab (browser)) _action_set_sensitive (browser, "ZoomNormal", midori_view_get_zoom_level (MIDORI_VIEW (view)) != 1.0f); } static void midori_view_notify_statusbar_text_cb (GtkWidget* view, GParamSpec* pspec, MidoriBrowser* browser) { gchar* text; if (view == midori_browser_get_current_tab (browser)) { g_object_get (view, "statusbar-text", &text, NULL); _midori_browser_set_statusbar_text (browser, MIDORI_VIEW (view), text); g_free (text); } } static GtkWidget* midori_bookmark_folder_button_new (KatzeArray* array, gboolean new_bookmark, gint64 selected, gint64 parentid) { GtkListStore* model; GtkWidget* combo; GtkCellRenderer* renderer; guint n; sqlite3* db; sqlite3_stmt* statement; gint result; const gchar* sqlcmd = "SELECT title, id FROM bookmarks WHERE uri='' ORDER BY title ASC"; model = gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_INT64); combo = gtk_combo_box_new_with_model (GTK_TREE_MODEL (model)); renderer = gtk_cell_renderer_text_new (); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), renderer, TRUE); gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), renderer, "text", 0); gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), renderer, "ellipsize", 1); gtk_list_store_insert_with_values (model, NULL, G_MAXINT, 0, _("Bookmarks"), 1, PANGO_ELLIPSIZE_END, 2, (gint64)0, -1); gtk_combo_box_set_active (GTK_COMBO_BOX (combo), 0); db = g_object_get_data (G_OBJECT (array), "db"); g_return_val_if_fail (db != NULL, NULL); n = 1; result = sqlite3_prepare_v2 (db, sqlcmd, -1, &statement, NULL); while ((result = sqlite3_step (statement)) == SQLITE_ROW) { const unsigned char* name = sqlite3_column_text (statement, 0); gint64 id = sqlite3_column_int64 (statement, 1); /* do not show the folder itself */ if (id != selected) { gtk_list_store_insert_with_values (model, NULL, G_MAXINT, 0, name, 1, PANGO_ELLIPSIZE_END, 2, id, -1); if (!new_bookmark && id == parentid) gtk_combo_box_set_active (GTK_COMBO_BOX (combo), n); n++; } } if (n < 2) gtk_widget_set_sensitive (combo, FALSE); return combo; } static gint64 midori_bookmark_folder_button_get_active (GtkWidget* combo) { gint64 id; GtkTreeIter iter; g_return_val_if_fail (GTK_IS_COMBO_BOX (combo), 0); if (gtk_combo_box_get_active_iter (GTK_COMBO_BOX (combo), &iter)) { gchar* selected = NULL; GtkTreeModel* model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo)); gtk_tree_model_get (GTK_TREE_MODEL (model), &iter, 0, &selected, 2, &id, -1); if (g_str_equal (selected, _("Bookmarks"))) id = 0; g_free (selected); } return id; } static void midori_browser_edit_bookmark_title_changed_cb (GtkEntry* entry, GtkDialog* dialog) { const gchar* title = gtk_entry_get_text (entry); gtk_dialog_set_response_sensitive (dialog, GTK_RESPONSE_ACCEPT, title != NULL && title[0] != '\0'); } static void midori_browser_edit_bookmark_add_speed_dial_cb (GtkWidget* button, KatzeItem* bookmark) { MidoriBrowser* browser = midori_browser_get_for_widget (button); gtk_widget_set_sensitive (button, FALSE); midori_browser_add_speed_dial (browser); } /* Private function, used by MidoriBookmarks and MidoriHistory */ /* static */ gboolean midori_browser_edit_bookmark_dialog_new (MidoriBrowser* browser, KatzeItem* bookmark, gboolean new_bookmark, gboolean is_folder, GtkWidget* proxy) { const gchar* title; GtkWidget* dialog; GtkWidget* content_area; GtkWidget* view; GtkWidget* vbox; GtkWidget* hbox; GtkWidget* label; const gchar* value; GtkWidget* entry_title; GtkWidget* entry_uri; GtkWidget* combo_folder; GtkWidget* check_toolbar; GtkWidget* check_app; gboolean return_status = FALSE; sqlite3* db; if (!browser->bookmarks || !gtk_widget_get_visible (GTK_WIDGET (browser))) return FALSE; db = g_object_get_data (G_OBJECT (browser->bookmarks), "db"); if (!db) return FALSE; if (is_folder) title = new_bookmark ? _("New folder") : _("Edit folder"); else title = new_bookmark ? _("New bookmark") : _("Edit bookmark"); #ifdef HAVE_GRANITE if (proxy != NULL) { /* FIXME: granite: should return GtkWidget* like GTK+ */ dialog = (GtkWidget*)granite_widgets_pop_over_new (); granite_widgets_pop_over_move_to_widget ( GRANITE_WIDGETS_POP_OVER (dialog), proxy, TRUE); } else #endif { dialog = gtk_dialog_new_with_buttons (title, GTK_WINDOW (browser), GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR, NULL, NULL); } gtk_dialog_add_buttons (GTK_DIALOG (dialog), GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, new_bookmark ? GTK_STOCK_ADD : GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT, NULL); content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog)); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), gtk_image_new_from_stock (STOCK_BOOKMARK_ADD, GTK_ICON_SIZE_DIALOG), FALSE, FALSE, 0); vbox = gtk_vbox_new (FALSE, 0); #ifdef HAVE_GRANITE if (proxy != NULL) { gchar* markup = g_strdup_printf ("%s", title); label = gtk_label_new (markup); g_free (markup); gtk_label_set_use_markup (GTK_LABEL (label), TRUE); gtk_box_pack_start (GTK_BOX (vbox), label, FALSE, FALSE, 0); } #endif label = gtk_label_new (_("Type a name for this bookmark and choose where to keep it.")); gtk_box_pack_start (GTK_BOX (vbox), label, FALSE, FALSE, 12); gtk_box_pack_start (GTK_BOX (hbox), vbox, FALSE, FALSE, 12); gtk_box_pack_start (GTK_BOX (content_area), hbox, FALSE, FALSE, 0); gtk_window_set_icon_name (GTK_WINDOW (dialog), new_bookmark ? GTK_STOCK_ADD : GTK_STOCK_REMOVE); gtk_container_set_border_width (GTK_CONTAINER (dialog), 5); gtk_container_set_border_width (GTK_CONTAINER (content_area), 5); if (!bookmark) { view = midori_browser_get_current_tab (browser); if (is_folder) { bookmark = (KatzeItem*)katze_array_new (KATZE_TYPE_ARRAY); katze_item_set_name (bookmark, midori_view_get_display_title (MIDORI_VIEW (view))); } else bookmark = g_object_new (KATZE_TYPE_ITEM, "uri", midori_view_get_display_uri (MIDORI_VIEW (view)), "name", midori_view_get_display_title (MIDORI_VIEW (view)), NULL); } entry_title = gtk_entry_new (); gtk_entry_set_activates_default (GTK_ENTRY (entry_title), TRUE); value = katze_item_get_name (bookmark); gtk_entry_set_text (GTK_ENTRY (entry_title), value ? value : ""); midori_browser_edit_bookmark_title_changed_cb (GTK_ENTRY (entry_title), GTK_DIALOG (dialog)); g_signal_connect (entry_title, "changed", G_CALLBACK (midori_browser_edit_bookmark_title_changed_cb), dialog); gtk_container_add (GTK_CONTAINER (content_area), entry_title); entry_uri = NULL; if (!is_folder) { entry_uri = katze_uri_entry_new ( #if GTK_CHECK_VERSION (2, 20, 0) gtk_dialog_get_widget_for_response (GTK_DIALOG (dialog), GTK_RESPONSE_ACCEPT)); #else NULL); #endif #if HAVE_HILDON HildonGtkInputMode mode = hildon_gtk_entry_get_input_mode (GTK_ENTRY (entry_uri)); mode &= ~HILDON_GTK_INPUT_MODE_AUTOCAP; hildon_gtk_entry_set_input_mode (GTK_ENTRY (entry_uri), mode); #endif gtk_entry_set_activates_default (GTK_ENTRY (entry_uri), TRUE); gtk_entry_set_text (GTK_ENTRY (entry_uri), katze_item_get_uri (bookmark)); gtk_container_add (GTK_CONTAINER (content_area), entry_uri); } combo_folder = midori_bookmark_folder_button_new (browser->bookmarks, new_bookmark, katze_item_get_meta_integer (bookmark, "id"), katze_item_get_meta_integer (bookmark, "parentid")); gtk_container_add (GTK_CONTAINER (content_area), combo_folder); if (new_bookmark && !is_folder) { label = gtk_button_new_with_mnemonic (_("Add to _Speed Dial")); g_signal_connect (label, "clicked", G_CALLBACK (midori_browser_edit_bookmark_add_speed_dial_cb), bookmark); gtk_container_add (GTK_CONTAINER (content_area), label); } check_toolbar = gtk_check_button_new_with_mnemonic (_("Show in the tool_bar")); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (check_toolbar), katze_item_get_meta_boolean (bookmark, "toolbar")); gtk_container_add (GTK_CONTAINER (content_area), check_toolbar); check_app = NULL; if (!is_folder) { check_app = gtk_check_button_new_with_mnemonic (_("Run as _web application")); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (check_app), katze_item_get_meta_boolean (bookmark, "app")); gtk_container_add (GTK_CONTAINER (content_area), check_app); } gtk_widget_show_all (content_area); gtk_dialog_set_default_response (GTK_DIALOG (dialog), GTK_RESPONSE_ACCEPT); if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT) { gint64 selected; katze_item_set_name (bookmark, gtk_entry_get_text (GTK_ENTRY (entry_title))); katze_item_set_meta_integer (bookmark, "toolbar", gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (check_toolbar))); if (!is_folder) { katze_item_set_uri (bookmark, gtk_entry_get_text (GTK_ENTRY (entry_uri))); katze_item_set_meta_integer (bookmark, "app", gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (check_app))); } selected = midori_bookmark_folder_button_get_active (combo_folder); katze_item_set_meta_integer (bookmark, "parentid", selected); if (new_bookmark) katze_array_add_item (browser->bookmarks, bookmark); else midori_bookmarks_update_item_db (db, bookmark); midori_browser_update_history (bookmark, "bookmark", new_bookmark ? "create" : "modify"); if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (check_toolbar))) if (!gtk_widget_get_visible (browser->bookmarkbar)) _action_set_active (browser, "Bookmarkbar", TRUE); return_status = TRUE; } if (gtk_widget_get_visible (browser->bookmarkbar)) midori_bookmarkbar_populate (browser); gtk_widget_destroy (dialog); return return_status; } static gboolean midori_browser_prepare_download (MidoriBrowser* browser, WebKitDownload* download, const gchar* uri) { if (!midori_download_has_enough_space (download, uri)) return FALSE; webkit_download_set_destination_uri (download, uri); g_signal_emit (browser, signals[ADD_DOWNLOAD], 0, download); midori_transferbar_add_download_item (MIDORI_TRANSFERBAR (browser->transferbar), download); midori_transferbar_check_size (browser->statusbar, MIDORI_TRANSFERBAR (browser->transferbar)); return TRUE; } static void midori_browser_save_resources (MidoriView* view, const gchar* folder) { WebKitWebView* web_view = WEBKIT_WEB_VIEW (midori_view_get_web_view (view)); WebKitWebFrame* frame = webkit_web_view_get_main_frame (web_view); WebKitWebDataSource* data_source = webkit_web_frame_get_data_source (frame); GList* resources = webkit_web_data_source_get_subresources (data_source); GList* list; katze_mkdir_with_parents (folder, 0700); for (list = resources; list; list = g_list_next (list)) { WebKitWebResource* resource = WEBKIT_WEB_RESOURCE (list->data); GString* data = webkit_web_resource_get_data (resource); /* Resource could be adblocked, skip it in that case */ if (!g_strcmp0 (webkit_web_resource_get_uri (resource), "about:blank")) continue; gchar* sub_filename = midori_download_get_filename_suggestion_for_uri ( webkit_web_resource_get_mime_type (resource), webkit_web_resource_get_uri (resource)); gchar* sub_path = g_build_filename (folder, sub_filename, NULL); sub_path = midori_download_get_unique_filename (sub_path); if (data) { GError* error = NULL; if (!g_file_set_contents (sub_path, data->str, data->len, &error)) { g_warning ("Failed to save %s: %s", sub_filename, error->message); g_error_free (error); } } else g_warning ("Skipping empty resource %s", sub_filename); g_free (sub_filename); g_free (sub_path); } g_list_free (resources); } static void midori_browser_save_uri (MidoriBrowser* browser, MidoriView* view, const gchar* uri) { static gchar* last_dir = NULL; GtkWidget* dialog; const gchar* title = midori_view_get_display_title (view); gchar* filename; gboolean file_only = TRUE; GtkWidget* checkbox = NULL; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; dialog = sokoke_file_chooser_dialog_new (_("Save file as"), GTK_WINDOW (browser), GTK_FILE_CHOOSER_ACTION_SAVE); gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (dialog), TRUE); if (uri == NULL) uri = midori_view_get_display_uri (view); if (midori_view_can_view_source (view)) { file_only = FALSE; checkbox = gtk_check_button_new_with_mnemonic (_("Save associated _resources")); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (checkbox), TRUE); gtk_file_chooser_set_extra_widget (GTK_FILE_CHOOSER (dialog), checkbox); } if (last_dir && *last_dir) gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), last_dir); else { gchar* dirname = midori_uri_get_folder (uri); if (dirname == NULL) dirname = katze_object_get_string (browser->settings, "download-folder"); gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), dirname); g_free (dirname); } if (!file_only && !g_str_equal (title, uri)) filename = midori_download_clean_filename (title); else { gchar* mime_type = katze_object_get_object (view, "mime-type"); filename = midori_download_get_filename_suggestion_for_uri (mime_type, uri); g_free (mime_type); } gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), filename); g_free (filename); if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_OK) { filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog)); if (checkbox != NULL) file_only = !file_only && !gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (checkbox)); if (!file_only && !g_str_equal (title, uri)) { gchar* fullname = g_strconcat (filename, ".html", NULL); midori_view_save_source (view, uri, fullname); g_free (fullname); } else midori_view_save_source (view, uri, filename); if (!file_only) midori_browser_save_resources (view, filename); katze_assign (last_dir, gtk_file_chooser_get_current_folder (GTK_FILE_CHOOSER (dialog))); } gtk_widget_destroy (dialog); } static void midori_view_save_as_cb (GtkWidget* menuitem, const gchar* uri, GtkWidget* view) { MidoriBrowser* browser = midori_browser_get_for_widget (view); midori_browser_save_uri (browser, MIDORI_VIEW (view), uri); } static void midori_browser_speed_dial_refresh_cb (MidoriSpeedDial* dial, MidoriBrowser* browser) { GList* tabs = midori_browser_get_tabs (browser); for (; tabs != NULL; tabs = g_list_next (tabs)) if (midori_view_is_blank (tabs->data)) midori_view_reload (tabs->data, FALSE); g_list_free (tabs); } static void midori_browser_add_speed_dial (MidoriBrowser* browser) { GdkPixbuf* img; GtkWidget* view = midori_browser_get_current_tab (browser); if ((img = midori_view_get_snapshot (MIDORI_VIEW (view), 240, 160))) { midori_speed_dial_add (browser->dial, midori_view_get_display_uri (MIDORI_VIEW (view)), midori_view_get_display_title (MIDORI_VIEW (view)), img); g_object_unref (img); } } static void midori_view_add_speed_dial_cb (GtkWidget* menuitem, const gchar* uri, GtkWidget* view) { MidoriBrowser* browser; browser = midori_browser_get_for_widget (menuitem); midori_browser_add_speed_dial (browser); } static gboolean midori_browser_tab_leave_notify_event_cb (GtkWidget* widget, GdkEventCrossing* event, MidoriBrowser* browser) { _midori_browser_set_statusbar_text (browser, MIDORI_VIEW (widget), NULL); return TRUE; } static void midori_view_activate_action_cb (GtkWidget* view, const gchar* action, MidoriBrowser* browser) { midori_browser_activate_action (browser, action); } static void midori_view_attach_inspector_cb (GtkWidget* view, GtkWidget* inspector_view, MidoriBrowser* browser) { GtkWidget* toplevel; GtkWidget* scrolled; toplevel = gtk_widget_get_toplevel (inspector_view); gtk_widget_hide (toplevel); scrolled = gtk_widget_get_parent (browser->inspector_view); gtk_widget_destroy (browser->inspector_view); gtk_widget_reparent (inspector_view, scrolled); gtk_widget_show_all (browser->inspector); browser->inspector_view = inspector_view; gtk_widget_destroy (toplevel); if (!katze_object_get_boolean (browser->settings, "last-inspector-attached")) g_object_set (browser->settings, "last-inspector-attached", TRUE, NULL); } static void midori_view_detach_inspector_cb (GtkWidget* view, GtkWidget* inspector_view, MidoriBrowser* browser) { GtkWidget* scrolled = gtk_widget_get_parent (GTK_WIDGET (inspector_view)); GtkWidget* paned = gtk_widget_get_parent (scrolled); browser->inspector_view = gtk_viewport_new (NULL, NULL); gtk_container_remove (GTK_CONTAINER (scrolled), GTK_WIDGET (inspector_view)); gtk_container_add (GTK_CONTAINER (scrolled), browser->inspector_view); gtk_widget_hide (paned); if (katze_object_get_boolean (browser->settings, "last-inspector-attached")) g_object_set (browser->settings, "last-inspector-attached", FALSE, NULL); } static void midori_browser_view_copy_history (GtkWidget* view_to, GtkWidget* view_from, gboolean omit_last) { WebKitWebView* copy_from; WebKitWebBackForwardList* list_from; WebKitWebView* copy_to; WebKitWebBackForwardList* list_to; guint length_from; gint i; copy_from = WEBKIT_WEB_VIEW (midori_view_get_web_view (MIDORI_VIEW (view_from))); list_from = webkit_web_view_get_back_forward_list (copy_from); copy_to = WEBKIT_WEB_VIEW (midori_view_get_web_view (MIDORI_VIEW (view_to))); list_to = webkit_web_view_get_back_forward_list (copy_to); length_from = webkit_web_back_forward_list_get_back_length (list_from); g_return_if_fail (!webkit_web_back_forward_list_get_back_length (list_to)); for (i = -length_from; i <= (omit_last ? -1 : 0); i++) { webkit_web_back_forward_list_add_item (list_to, webkit_web_back_forward_list_get_nth_item (list_from, i)); } } static gboolean midori_browser_notify_new_tab_timeout_cb (MidoriBrowser *browser) { #ifndef G_OS_WIN32 gtk_window_set_opacity (GTK_WINDOW (browser), 1); #endif return G_SOURCE_REMOVE; } static void midori_browser_notify_new_tab (MidoriBrowser* browser) { if (katze_object_get_boolean (browser->settings, "flash-window-on-new-bg-tabs")) { #ifndef G_OS_WIN32 gtk_window_set_opacity (GTK_WINDOW (browser), 0.8); #endif g_timeout_add (100, (GSourceFunc) midori_browser_notify_new_tab_timeout_cb, browser); } } static void midori_view_new_tab_cb (GtkWidget* view, const gchar* uri, gboolean background, MidoriBrowser* browser) { gint n = midori_browser_add_uri (browser, uri); midori_browser_view_copy_history (midori_browser_get_nth_tab (browser, n), view, FALSE); if (!background) midori_browser_set_current_page (browser, n); else midori_browser_notify_new_tab (browser); } static void midori_view_new_window_cb (GtkWidget* view, const gchar* uri, MidoriBrowser* browser) { MidoriBrowser* new_browser; g_signal_emit (browser, signals[NEW_WINDOW], 0, NULL, &new_browser); if (new_browser) midori_browser_add_uri (new_browser, uri); else /* No MidoriApp, so this is app or private mode */ sokoke_spawn_app (uri, TRUE); } static void midori_view_new_view_cb (GtkWidget* view, GtkWidget* new_view, MidoriNewView where, gboolean user_initiated, MidoriBrowser* browser) { midori_browser_view_copy_history (new_view, view, TRUE); if (where == MIDORI_NEW_VIEW_WINDOW) { MidoriBrowser* new_browser; g_signal_emit (browser, signals[NEW_WINDOW], 0, NULL, &new_browser); midori_browser_add_tab (new_browser, new_view); midori_browser_set_current_tab (new_browser, new_view); } else if (gtk_widget_get_parent (new_view) != browser->notebook) { gint n = midori_browser_add_tab (browser, new_view); if (where != MIDORI_NEW_VIEW_BACKGROUND) midori_browser_set_current_page (browser, n); } else midori_browser_notify_new_tab (browser); if (!user_initiated) { GdkWindow* window = gtk_widget_get_window (GTK_WIDGET (browser)); GdkWindowState state = gdk_window_get_state (window); if ((state | GDK_WINDOW_STATE_MAXIMIZED) || (state | GDK_WINDOW_STATE_FULLSCREEN)) { if (where == MIDORI_NEW_VIEW_WINDOW) g_signal_emit (browser, signals[SEND_NOTIFICATION], 0, _("New Window"), _("A new window has been opened")); else if (!browser->show_tabs) g_signal_emit (browser, signals[SEND_NOTIFICATION], 0, _("New Tab"), _("A new tab has been opened")); } } } static void midori_browser_download_status_cb (WebKitDownload* download, GParamSpec* pspec, GtkWidget* widget) { const gchar* uri = webkit_download_get_destination_uri (download); switch (webkit_download_get_status (download)) { case WEBKIT_DOWNLOAD_STATUS_FINISHED: if (!sokoke_show_uri (gtk_widget_get_screen (widget), uri, 0, NULL)) { sokoke_message_dialog (GTK_MESSAGE_ERROR, _("Error opening the image!"), _("Can not open selected image in a default viewer."), FALSE); } break; case WEBKIT_DOWNLOAD_STATUS_ERROR: webkit_download_cancel (download); sokoke_message_dialog (GTK_MESSAGE_ERROR, _("Error downloading the image!"), _("Can not download selected image."), FALSE); break; case WEBKIT_DOWNLOAD_STATUS_CREATED: case WEBKIT_DOWNLOAD_STATUS_STARTED: case WEBKIT_DOWNLOAD_STATUS_CANCELLED: break; } } static gboolean midori_browser_remove_tab_idle (gpointer view) { MidoriBrowser* browser; g_return_val_if_fail (GTK_IS_WIDGET (view), FALSE); browser = midori_browser_get_for_widget (GTK_WIDGET (view)); midori_browser_remove_tab (browser, GTK_WIDGET (view)); return G_SOURCE_REMOVE; } static gboolean midori_view_download_requested_cb (GtkWidget* view, WebKitDownload* download, MidoriBrowser* browser) { MidoriDownloadType type = midori_download_get_type (download); GtkWidget* web_view; WebKitWebFrame* web_frame; WebKitWebDataSource* datasource; gboolean handled; g_return_val_if_fail (MIDORI_IS_VIEW (view), FALSE); handled = TRUE; if (type == MIDORI_DOWNLOAD_CANCEL) { handled = FALSE; } else if (type == MIDORI_DOWNLOAD_OPEN_IN_VIEWER) { gchar* destination_uri = midori_download_prepare_destination_uri (download, NULL); midori_browser_prepare_download (browser, download, destination_uri); g_signal_connect (download, "notify::status", G_CALLBACK (midori_browser_download_status_cb), GTK_WIDGET (browser)); g_free (destination_uri); webkit_download_start (download); } else if (!webkit_download_get_destination_uri (download)) { if (type == MIDORI_DOWNLOAD_SAVE_AS) { static GtkWidget* dialog = NULL; gchar* filename; if (!dialog) { gchar* folder; dialog = sokoke_file_chooser_dialog_new (_("Save file"), GTK_WINDOW (browser), GTK_FILE_CHOOSER_ACTION_SAVE); gtk_file_chooser_set_do_overwrite_confirmation ( GTK_FILE_CHOOSER (dialog), TRUE); gtk_window_set_destroy_with_parent (GTK_WINDOW (dialog), TRUE); folder = midori_uri_get_folder (webkit_download_get_uri (download)); if (folder == NULL) folder = katze_object_get_string (browser->settings, "download-folder"); gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), folder); g_free (folder); g_signal_connect (dialog, "destroy", G_CALLBACK (gtk_widget_destroyed), &dialog); } g_object_set_data (G_OBJECT (dialog), "download", download); filename = midori_download_get_suggested_filename (download); gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), filename); g_free (filename); if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_OK) { gtk_widget_hide (dialog); gchar* uri = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (dialog)); if (!midori_browser_prepare_download (browser, download, uri)) { g_free (uri); return FALSE; } g_free (uri); } else { gtk_widget_hide (dialog); return FALSE; } } else { gchar* folder = type == MIDORI_DOWNLOAD_OPEN ? NULL : katze_object_get_string (browser->settings, "download-folder"); gchar* destination_uri = midori_download_prepare_destination_uri (download, folder); midori_browser_prepare_download (browser, download, destination_uri); g_free (destination_uri); } webkit_download_start (download); } /* Close empty tabs due to download links with a target */ web_view = midori_view_get_web_view (MIDORI_VIEW (view)); web_frame = webkit_web_view_get_main_frame (WEBKIT_WEB_VIEW (web_view)); datasource = webkit_web_frame_get_data_source (web_frame); if (midori_view_is_blank (MIDORI_VIEW (view)) && webkit_web_data_source_get_data (datasource) == NULL) g_idle_add (midori_browser_remove_tab_idle, view); return handled; } static void midori_view_search_text_cb (GtkWidget* view, gboolean found, gchar* typing, MidoriBrowser* browser) { midori_findbar_search_text (MIDORI_FINDBAR (browser->find), view, found, typing); } gint midori_browser_get_n_pages (MidoriBrowser* browser) { #ifdef HAVE_GRANITE return granite_widgets_dynamic_notebook_get_n_tabs ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook)); #else return gtk_notebook_get_n_pages (GTK_NOTEBOOK (browser->notebook)); #endif } static void midori_browser_tab_destroy_cb (MidoriView* view, MidoriBrowser* browser) { if (browser->proxy_array) { KatzeItem* item = midori_view_get_proxy_item (view); if (katze_array_get_item_index (browser->proxy_array, item) != -1) { if (!midori_view_is_blank (view)) { if (browser->trash) katze_array_add_item (browser->trash, item); midori_browser_update_history (item, "website", "leave"); } katze_array_remove_item (browser->proxy_array, item); } /* We don't ever want to be in a situation with no tabs, so just create an empty one if the last one is closed. The only exception is when we are closing the window, which is indicated by the proxy array having been unset. */ if (midori_browser_get_n_pages (browser) == 0) midori_browser_add_uri (browser, ""); } _midori_browser_update_actions (browser); } static void _midori_browser_remove_tab (MidoriBrowser* browser, GtkWidget* widget) { } #ifndef HAVE_GRANITE static void midori_browser_notebook_resize (MidoriBrowser* browser, GdkRectangle* allocation) { gint new_size = 0; gint n = MAX (1, gtk_notebook_get_n_pages (GTK_NOTEBOOK (browser->notebook))); const gint max_size = 150; gint min_size; gint icon_size = 16; GtkAllocation notebook_size; GList* children; if (allocation != NULL) notebook_size.width = allocation->width; else gtk_widget_get_allocation (browser->notebook, ¬ebook_size); new_size = notebook_size.width / n; gtk_icon_size_lookup_for_settings (gtk_widget_get_settings (browser->notebook), GTK_ICON_SIZE_MENU, &icon_size, NULL); min_size = icon_size; if (katze_object_get_boolean (browser->settings, "close-buttons-on-tabs")) min_size += icon_size; if (new_size < min_size) new_size = min_size; if (new_size > max_size) new_size = max_size; if (new_size > browser->last_tab_size - 3 && new_size < browser->last_tab_size + 3) return; browser->last_tab_size = new_size; children = gtk_container_get_children (GTK_CONTAINER (browser->notebook)); for (; children; children = g_list_next (children)) { GtkWidget* view = children->data; GtkWidget* label; label = gtk_notebook_get_tab_label (GTK_NOTEBOOK(browser->notebook), view); /* Don't resize empty bin, which is used for thumbnail tabs */ if (GTK_IS_BIN (label) && gtk_bin_get_child (GTK_BIN (label)) && !katze_object_get_boolean (view, "minimized")) gtk_widget_set_size_request (label, new_size, -1); } } #endif static void midori_browser_notebook_size_allocate_cb (GtkWidget* widget, GdkRectangle* allocation, MidoriBrowser* browser) { #ifndef HAVE_GRANITE if (!gtk_notebook_get_show_tabs (GTK_NOTEBOOK (browser->notebook))) return; midori_browser_notebook_resize (browser, allocation); #endif } static void midori_browser_connect_tab (MidoriBrowser* browser, GtkWidget* view) { KatzeItem* item = midori_view_get_proxy_item (MIDORI_VIEW (view)); katze_array_add_item (browser->proxy_array, item); gtk_widget_set_can_focus (view, TRUE); g_object_connect (view, "signal::notify::icon", midori_view_notify_icon_cb, browser, "signal::notify::load-status", midori_view_notify_load_status_cb, browser, "signal::notify::progress", midori_view_notify_progress_cb, browser, "signal::notify::uri", midori_view_notify_uri_cb, browser, "signal::notify::title", midori_view_notify_title_cb, browser, "signal::notify::minimized", midori_view_notify_minimized_cb, browser, "signal::notify::zoom-level", midori_view_notify_zoom_level_cb, browser, "signal::notify::statusbar-text", midori_view_notify_statusbar_text_cb, browser, "signal::activate-action", midori_view_activate_action_cb, browser, "signal::attach-inspector", midori_view_attach_inspector_cb, browser, "signal::detach-inspector", midori_view_detach_inspector_cb, browser, "signal::new-tab", midori_view_new_tab_cb, browser, "signal::new-window", midori_view_new_window_cb, browser, "signal::new-view", midori_view_new_view_cb, browser, "signal-after::download-requested", midori_view_download_requested_cb, browser, "signal::search-text", midori_view_search_text_cb, browser, "signal::save-as", midori_view_save_as_cb, browser, "signal::add-speed-dial", midori_view_add_speed_dial_cb, browser, "signal::leave-notify-event", midori_browser_tab_leave_notify_event_cb, browser, "signal::destroy", midori_browser_tab_destroy_cb, browser, NULL); } static void _midori_browser_add_tab (MidoriBrowser* browser, GtkWidget* view) { GtkWidget* notebook = browser->notebook; KatzeItem* item = midori_view_get_proxy_item (MIDORI_VIEW (view)); #ifndef HAVE_GRANITE GtkWidget* tab_label; #endif guint n; midori_browser_connect_tab (browser, view); if (!katze_item_get_meta_boolean (item, "append") && katze_object_get_boolean (browser->settings, "open-tabs-next-to-current")) { n = midori_browser_get_current_page (browser) + 1; katze_array_move_item (browser->proxy_array, item, n); } else n = midori_browser_get_n_pages (browser); katze_item_set_meta_integer (item, "append", -1); #ifdef HAVE_GRANITE granite_widgets_dynamic_notebook_insert_tab ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (notebook), midori_view_get_tab (MIDORI_VIEW (view)), n); #else tab_label = midori_view_get_proxy_tab_label (MIDORI_VIEW (view)); /* Don't resize empty bin, which is used for thumbnail tabs */ if (GTK_IS_BIN (tab_label) && gtk_bin_get_child (GTK_BIN (tab_label)) && !katze_object_get_boolean (view, "minimized")) gtk_widget_set_size_request (tab_label, browser->last_tab_size, -1); gtk_notebook_insert_page (GTK_NOTEBOOK (notebook), view, tab_label, n); gtk_notebook_set_tab_reorderable (GTK_NOTEBOOK (notebook), view, TRUE); gtk_notebook_set_tab_detachable (GTK_NOTEBOOK (notebook), view, TRUE); midori_browser_notebook_size_allocate_cb (browser->notebook, NULL, browser); #endif _midori_browser_update_actions (browser); } /** * midori_browser_foreach: * @browser: a #MidoriBrowser * @callback: a #GtkCallback * @callback_data: custom data * * Calls the specified callback for each view contained * in the browser. * * Since: 0.1.7 **/ void midori_browser_foreach (MidoriBrowser* browser, GtkCallback callback, gpointer callback_data) { g_return_if_fail (MIDORI_IS_BROWSER (browser)); #ifdef HAVE_GRANITE /* FIXME */ if (GTK_IS_BIN (browser->notebook)) gtk_container_foreach (GTK_CONTAINER (gtk_bin_get_child (GTK_BIN ( browser->notebook))), callback, callback_data); else #endif gtk_container_foreach (GTK_CONTAINER (browser->notebook), callback, callback_data); } static void _midori_browser_quit (MidoriBrowser* browser) { /* Nothing to do */ } static gboolean midori_browser_key_press_event (GtkWidget* widget, GdkEventKey* event) { GtkWindow* window = GTK_WINDOW (widget); MidoriBrowser* browser = MIDORI_BROWSER (widget); GtkWidgetClass* widget_class; guint clean_state; GtkWidget* focus; /* Interpret Ctrl(+Shift)+Tab as tab switching for compatibility */ if (midori_browser_get_nth_tab (browser, 1) != NULL && event->keyval == GDK_KEY_Tab && (event->state & GDK_CONTROL_MASK)) { midori_browser_activate_action (browser, "TabNext"); return TRUE; } else if (event->keyval == GDK_KEY_ISO_Left_Tab && (event->state & GDK_CONTROL_MASK) && (event->state & GDK_SHIFT_MASK)) { midori_browser_activate_action (browser, "TabPrevious"); return TRUE; } /* Interpret Ctrl+= as Zoom In for compatibility */ else if ((event->keyval == GDK_KEY_KP_Equal || event->keyval == GDK_KEY_equal) && (event->state & GDK_CONTROL_MASK)) { midori_browser_activate_action (browser, "ZoomIn"); return TRUE; } /* Interpret F5 as reloading for compatibility */ else if (event->keyval == GDK_KEY_F5) { midori_browser_activate_action (browser, "Reload"); return TRUE; } focus = gtk_window_get_focus (GTK_WINDOW (widget)); if (focus == NULL) gtk_widget_grab_focus (midori_browser_get_current_tab (MIDORI_BROWSER (widget))); else if (G_OBJECT_TYPE (focus) == WEBKIT_TYPE_WEB_VIEW && event->keyval == GDK_KEY_space && (!(event->state & GDK_SHIFT_MASK)) && !webkit_web_view_can_cut_clipboard (WEBKIT_WEB_VIEW (focus)) && !webkit_web_view_can_paste_clipboard (WEBKIT_WEB_VIEW (focus))) { /* Space at the bottom of the page: Go to next page */ MidoriView* view = midori_view_get_for_widget (focus); GtkScrolledWindow* scrolled = GTK_SCROLLED_WINDOW (gtk_widget_get_parent (focus)); GtkAdjustment* vadjust = gtk_scrolled_window_get_vadjustment (scrolled); if (gtk_adjustment_get_value (vadjust) == (gtk_adjustment_get_upper (vadjust) - gtk_adjustment_get_page_size (vadjust))) { /* Duplicate here because the URI pointer might change */ gchar* uri = g_strdup (midori_view_get_next_page (view)); if (uri != NULL) { midori_view_set_uri (view, uri); g_free (uri); return TRUE; } } } if (event->state & (GDK_CONTROL_MASK | GDK_MOD1_MASK)) if (sokoke_window_activate_key (window, event)) return TRUE; clean_state = event->state & gtk_accelerator_get_default_mod_mask(); if (!clean_state && gtk_window_propagate_key_event (window, event)) return TRUE; if (!(event->state & (GDK_CONTROL_MASK | GDK_MOD1_MASK))) if (sokoke_window_activate_key (window, event)) return TRUE; if (event->state && gtk_window_propagate_key_event (window, event)) return TRUE; /* Interpret (Shift+)Backspace as going back (forward) for compatibility */ if ((event->keyval == GDK_KEY_BackSpace) && (event->state & GDK_SHIFT_MASK)) { midori_browser_activate_action (browser, "Forward"); return TRUE; } else if (event->keyval == GDK_KEY_BackSpace) { midori_browser_activate_action (browser, "Back"); return TRUE; } widget_class = g_type_class_peek_static (g_type_parent (GTK_TYPE_WINDOW)); return widget_class->key_press_event (widget, event); } static gboolean midori_browser_delete_event (GtkWidget* widget, GdkEventAny* event) { MidoriBrowser* browser = MIDORI_BROWSER (widget); return midori_transferbar_confirm_delete (MIDORI_TRANSFERBAR (browser->transferbar)); } static void midori_browser_class_init (MidoriBrowserClass* class) { GtkWidgetClass* gtkwidget_class; GObjectClass* gobject_class; GParamFlags flags; /** * MidoriBrowser::new-window: * @browser: the object on which the signal is emitted * @window: a new browser window, or %NULL * * Emitted when a new browser window was created. * * Note: Before 0.1.7 the second argument was an URI string. * * Note: Since 0.2.1 the return value is a #MidoriBrowser * * Return value: a new #MidoriBrowser */ signals[NEW_WINDOW] = g_signal_new ( "new-window", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST), G_STRUCT_OFFSET (MidoriBrowserClass, new_window), 0, NULL, midori_cclosure_marshal_OBJECT__OBJECT, MIDORI_TYPE_BROWSER, 1, MIDORI_TYPE_BROWSER); signals[ADD_TAB] = g_signal_new ( "add-tab", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION), G_STRUCT_OFFSET (MidoriBrowserClass, add_tab), 0, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, GTK_TYPE_WIDGET); signals[REMOVE_TAB] = g_signal_new ( "remove-tab", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION), G_STRUCT_OFFSET (MidoriBrowserClass, remove_tab), 0, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, GTK_TYPE_WIDGET); /** * MidoriBrowser::move-tab: * @browser: the object on which the signal is emitted * @notebook: the notebook containing the tabs * @cur_pos: the current position of the tab * @new_pos: the new position of the tab * * Emitted when a tab is moved. * * Since: 0.3.3 */ signals[MOVE_TAB] = g_signal_new ( "move-tab", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION), 0, 0, NULL, midori_cclosure_marshal_VOID__OBJECT_INT_INT, G_TYPE_NONE, 3, GTK_TYPE_NOTEBOOK, G_TYPE_INT, G_TYPE_INT); signals[ACTIVATE_ACTION] = g_signal_new ( "activate-action", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION), G_STRUCT_OFFSET (MidoriBrowserClass, activate_action), 0, NULL, g_cclosure_marshal_VOID__STRING, G_TYPE_NONE, 1, G_TYPE_STRING); /** * MidoriBrowser::add-download: * @browser: the object on which the signal is emitted * @download: a new download * * Emitted when a new download was accepted and is * about to start, before the browser adds items * to the transferbar. * * Emitting this signal manually is equal to a * user initiating and confirming a download * * Note: This requires WebKitGTK+ 1.1.3. * * Since: 0.1.5 */ signals[ADD_DOWNLOAD] = g_signal_new ( "add-download", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST), 0, 0, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, G_TYPE_OBJECT); /** * MidoriBrowser::send-notification: * @browser: the object on which the signal is emitted * @title: the title for the notification * @message: the message for the notification * * Emitted when a browser wants to display a notification message, * e.g. when a download has been completed or a new tab was opened. * * Since: 0.1.7 */ signals[SEND_NOTIFICATION] = g_signal_new ( "send-notification", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST), 0, 0, NULL, midori_cclosure_marshal_VOID__STRING_STRING, G_TYPE_NONE, 2, G_TYPE_STRING, G_TYPE_STRING); /** * MidoriBrowser::populate-tool-menu: * @browser: the object on which the signal is emitted * @menu: the #GtkMenu to populate * * Emitted when a Tool menu is displayed, such as the * toplevel Tools in the menubar or the compact menu. * * Since: 0.1.9 */ signals[POPULATE_TOOL_MENU] = g_signal_new ( "populate-tool-menu", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST), 0, 0, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, GTK_TYPE_MENU); /** * MidoriBrowser::populate-toolbar-menu: * @browser: the object on which the signal is emitted * @menu: the #GtkMenu to populate * * Emitted when a toolbar menu is displayed on right-click. * * Since: 0.3.4 */ signals[POPULATE_TOOLBAR_MENU] = g_signal_new ( "populate-toolbar-menu", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST), 0, 0, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, GTK_TYPE_MENU); signals[QUIT] = g_signal_new ( "quit", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION), G_STRUCT_OFFSET (MidoriBrowserClass, quit), 0, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); /** * MidoriBrowser::show-preferences: * @browser: the object on which the signal is emitted * @preferences: the #KatzePreferences to populate * * Emitted when a preference dialogue displayed, to allow * adding of a new page, to be used sparingly. * * Since: 0.3.4 */ signals[SHOW_PREFERENCES] = g_signal_new ( "show-preferences", G_TYPE_FROM_CLASS (class), (GSignalFlags)(G_SIGNAL_RUN_LAST), 0, 0, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, KATZE_TYPE_PREFERENCES); class->add_tab = _midori_browser_add_tab; class->remove_tab = _midori_browser_remove_tab; class->activate_action = _midori_browser_activate_action; class->quit = _midori_browser_quit; gtkwidget_class = GTK_WIDGET_CLASS (class); gtkwidget_class->key_press_event = midori_browser_key_press_event; gtkwidget_class->delete_event = midori_browser_delete_event; gobject_class = G_OBJECT_CLASS (class); gobject_class->dispose = midori_browser_dispose; gobject_class->finalize = midori_browser_finalize; gobject_class->set_property = midori_browser_set_property; gobject_class->get_property = midori_browser_get_property; flags = G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS; g_object_class_install_property (gobject_class, PROP_MENUBAR, g_param_spec_object ( "menubar", "Menubar", "The menubar", GTK_TYPE_MENU_BAR, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_NAVIGATIONBAR, g_param_spec_object ( "navigationbar", "Navigationbar", "The navigationbar", GTK_TYPE_TOOLBAR, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_NOTEBOOK, g_param_spec_object ( "notebook", "Notebook", "The notebook containing the views", #ifdef HAVE_GRANITE GRANITE_WIDGETS_TYPE_DYNAMIC_NOTEBOOK, #else GTK_TYPE_NOTEBOOK, #endif G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_PANEL, g_param_spec_object ( "panel", "Panel", "The side panel embedded in the browser", MIDORI_TYPE_PANEL, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_URI, g_param_spec_string ( "uri", "URI", "The current URI", "", G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_TAB, g_param_spec_object ( "tab", "Tab", "The current tab", GTK_TYPE_WIDGET, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_LOAD_STATUS, g_param_spec_enum ( "load-status", "Load Status", "The current load status", MIDORI_TYPE_LOAD_STATUS, MIDORI_LOAD_FINISHED, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); /** * MidoriBrowser:statusbar: * * The widget representing the statusbar contents. This is * not an actual #GtkStatusbar but rather a #GtkBox. */ g_object_class_install_property (gobject_class, PROP_STATUSBAR, g_param_spec_object ( "statusbar", "Statusbar", "The statusbar", GTK_TYPE_BOX, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); /** * MidoriBrowser:statusbar-text: * * The text that is displayed in the statusbar. * * This value reflects changes to the text visible in the statusbar, such * as the uri of a hyperlink the mouse hovers over or the description of * a menuitem. * * Setting this value changes the displayed text until the next change. */ g_object_class_install_property (gobject_class, PROP_STATUSBAR_TEXT, g_param_spec_string ( "statusbar-text", "Statusbar Text", "The text that is displayed in the statusbar", "", flags)); /** * MidoriBrowser:settings: * * An associated settings instance that is shared among all web views. * * Setting this value is propagated to every present web view. Also * every newly created web view will use this instance automatically. * * If no settings are specified a default will be used. */ g_object_class_install_property (gobject_class, PROP_SETTINGS, g_param_spec_object ( "settings", "Settings", "The associated settings", MIDORI_TYPE_WEB_SETTINGS, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); /** * MidoriBrowser:bookmarks: * * The bookmarks folder, containing all bookmarks. * * This is actually a reference to a bookmarks instance, * so if bookmarks should be used it must be initially set. */ g_object_class_install_property (gobject_class, PROP_BOOKMARKS, g_param_spec_object ( "bookmarks", "Bookmarks", "The bookmarks folder, containing all bookmarks", KATZE_TYPE_ARRAY, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); /** * MidoriBrowser:trash: * * The trash, that collects all closed tabs and windows. * * This is actually a reference to a trash instance, so if a trash should * be used it must be initially set. * * Note: In the future the trash might collect other types of items. */ g_object_class_install_property (gobject_class, PROP_TRASH, g_param_spec_object ( "trash", "Trash", "The trash, collecting recently closed tabs and windows", KATZE_TYPE_ARRAY, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); /** * MidoriBrowser:search-engines: * * The list of search engines to be used for web search. * * This is actually a reference to a search engines instance, * so if search engines should be used it must be initially set. */ g_object_class_install_property (gobject_class, PROP_SEARCH_ENGINES, g_param_spec_object ( "search-engines", "Search Engines", "The list of search engines to be used for web search", KATZE_TYPE_ARRAY, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); /** * MidoriBrowser:history: * * The list of history items. * * This is actually a reference to a history instance, * so if history should be used it must be initially set. */ g_object_class_install_property (gobject_class, PROP_HISTORY, g_param_spec_object ( "history", "History", "The list of history items", KATZE_TYPE_ARRAY, flags)); /** * MidoriBrowser:speed-dial: * * The speed dial configuration file. * * Since: 0.3.4 * Since 0.4.7 this is a Midori.SpeedDial instance. */ g_object_class_install_property (gobject_class, PROP_SPEED_DIAL, g_param_spec_pointer ( "speed-dial", "Speeddial", "Speed dial", G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); /** * MidoriBrowser:show-tabs: * * Whether or not to show tabs. * * If disabled, no tab labels are shown. This is intended for * extensions that want to provide alternative tab labels. * * Since 0.1.8 */ g_object_class_install_property (gobject_class, PROP_SHOW_TABS, g_param_spec_boolean ( "show-tabs", "Show Tabs", "Whether or not to show tabs", TRUE, flags)); #if !GTK_CHECK_VERSION (3, 0, 0) /* Add 2px space between tool buttons */ gtk_rc_parse_string ( "style \"tool-button-style\"\n {\n" "GtkToolButton::icon-spacing = 2\n }\n" "widget \"MidoriBrowser.*.MidoriBookmarkbar.Gtk*ToolButton\" " "style \"tool-button-style\"\n" "widget \"MidoriBrowser.*.MidoriFindbar.Gtk*ToolButton\" " "style \"tool-button-style\"\n"); #endif } static void _action_window_new_activate (GtkAction* action, MidoriBrowser* browser) { midori_view_new_window_cb (NULL, "", browser); } static void _action_tab_new_activate (GtkAction* action, MidoriBrowser* browser) { gint n = midori_browser_add_uri (browser, ""); midori_browser_set_current_page (browser, n); } static void _action_private_browsing_activate (GtkAction* action, MidoriBrowser* browser) { sokoke_spawn_app ("about:private", TRUE); } static void _action_open_activate (GtkAction* action, MidoriBrowser* browser) { #if !GTK_CHECK_VERSION (3, 1, 10) static gchar* last_dir = NULL; gboolean folder_set = FALSE; #endif gchar* uri = NULL; GtkWidget* dialog; GtkWidget* view; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; dialog = sokoke_file_chooser_dialog_new (_("Open file"), GTK_WINDOW (browser), GTK_FILE_CHOOSER_ACTION_OPEN); /* base the start folder on the current view's uri if it is local */ view = midori_browser_get_current_tab (browser); if ((uri = (gchar*)midori_view_get_display_uri (MIDORI_VIEW (view)))) { gchar* filename = g_filename_from_uri (uri, NULL, NULL); if (filename) { gchar* dirname = g_path_get_dirname (filename); if (dirname && g_file_test (dirname, G_FILE_TEST_IS_DIR)) { gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), dirname); #if !GTK_CHECK_VERSION (3, 1, 10) folder_set = TRUE; #endif } g_free (dirname); g_free (filename); } } #if !GTK_CHECK_VERSION (3, 1, 10) if (!folder_set && last_dir && *last_dir) gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), last_dir); #endif if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_OK) { #if !GTK_CHECK_VERSION (3, 1, 10) gchar* folder; folder = gtk_file_chooser_get_current_folder (GTK_FILE_CHOOSER (dialog)); katze_assign (last_dir, folder); #endif uri = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (dialog)); midori_browser_set_current_uri (browser, uri); g_free (uri); } gtk_widget_destroy (dialog); } static void _action_save_as_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); midori_browser_save_uri (browser, MIDORI_VIEW (view), NULL); } static void _action_add_speed_dial_activate (GtkAction* action, MidoriBrowser* browser) { midori_browser_add_speed_dial (browser); } static void _action_add_desktop_shortcut_activate (GtkAction* action, MidoriBrowser* browser) { #if HAVE_HILDON /* TODO: Implement */ #elif defined (GDK_WINDOWING_X11) GtkWidget* tab = midori_browser_get_current_tab (browser); KatzeItem* item = midori_view_get_proxy_item (MIDORI_VIEW (tab)); const gchar* app_name = katze_item_get_name (item); gchar* app_exec = g_strconcat ("midori -a ", katze_item_get_uri (item), NULL); const gchar* icon_uri = midori_view_get_icon_uri (MIDORI_VIEW (tab)); gchar* app_icon; GKeyFile* keyfile = g_key_file_new (); gchar* filename = g_strconcat (app_name, ".desktop", NULL); gchar* app_dir; int i = 0; while (filename[i] != '\0') { if (filename[i] == '/') filename[i] = '_'; i++; } app_dir = g_build_filename (g_get_user_data_dir (), "applications", filename, NULL); app_icon = katze_net_get_cached_path (NULL, icon_uri, "icons"); if (!g_file_test (app_icon, G_FILE_TEST_EXISTS)) katze_assign (app_icon, g_strdup (STOCK_WEB_BROWSER)); g_key_file_set_string (keyfile, "Desktop Entry", "Version", "1.0"); g_key_file_set_string (keyfile, "Desktop Entry", "Type", "Application"); g_key_file_set_string (keyfile, "Desktop Entry", "Name", app_name); g_key_file_set_string (keyfile, "Desktop Entry", "Exec", app_exec); g_key_file_set_string (keyfile, "Desktop Entry", "TryExec", "midori"); g_key_file_set_string (keyfile, "Desktop Entry", "Icon", app_icon); g_key_file_set_string (keyfile, "Desktop Entry", "Categories", "Network;"); sokoke_key_file_save_to_file (keyfile, app_dir, NULL); g_free (app_dir); g_free (filename); g_free (app_exec); g_key_file_free (keyfile); #elif defined(GDK_WINDOWING_QUARTZ) /* TODO: Implement */ #elif defined (GDK_WINDOWING_WIN32) /* TODO: Implement */ #endif } static void midori_browser_subscribe_to_news_feed (MidoriBrowser* browser, const gchar* uri) { if (browser->news_aggregator && *browser->news_aggregator) { /* Thunderbird only accepts feed://, Liferea doesn't mind */ gchar* feed = g_strdup (uri); if (g_str_has_prefix (feed, "http://")) { feed[0] = 'f'; feed[1] = 'e'; feed[2] = 'e'; feed[3] = 'd'; } /* Special-case Liferea because a helper script may be required */ if (g_str_equal (browser->news_aggregator, "liferea") && g_find_program_in_path ("liferea-add-feed")) sokoke_spawn_program ("liferea-add-feed", FALSE, feed, TRUE); else sokoke_spawn_program (browser->news_aggregator, TRUE, feed, TRUE); g_free (feed); } else { gchar* description = g_strdup_printf ("%s\n\n%s", uri, _("To use the above URI open a news aggregator. " "There is usually a menu or button \"New Subscription\", " "\"New News Feed\" or similar.\n" "Alternatively go to Preferences, Applications in Midori, " "and select a News Aggregator. Next time you click the " "news feed icon, it will be added automatically.")); sokoke_message_dialog (GTK_MESSAGE_INFO, _("New feed"), description, FALSE); g_free (description); } } static void _action_add_news_feed_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view; const gchar* uri; if (!(view = midori_browser_get_current_tab (browser))) return; if (!(uri = g_object_get_data (G_OBJECT (view), "news-feeds"))) return; midori_browser_subscribe_to_news_feed (browser, uri); } static void _action_compact_add_response_cb (GtkWidget* dialog, gint response, gpointer data) { gtk_widget_destroy (dialog); } static void _action_compact_add_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* dialog; GtkBox* box; const gchar* actions[] = { "BookmarkAdd", "AddSpeedDial", "AddDesktopShortcut", "AddNewsFeed" }; guint i; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; dialog = g_object_new (GTK_TYPE_DIALOG, "transient-for", browser, "title", _("Add a new bookmark"), NULL); box = GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))); for (i = 0; i < G_N_ELEMENTS (actions); i++) { gchar* label; GtkWidget* button; action = _action_by_name (browser, actions[i]); label = katze_object_get_string (action, "label"); button = gtk_button_new_with_mnemonic (label); g_free (label); gtk_widget_set_name (button, "GtkButton-thumb"); gtk_box_pack_start (box, button, TRUE, TRUE, 4); gtk_activatable_set_related_action (GTK_ACTIVATABLE (button), action); g_signal_connect_swapped (button, "clicked", G_CALLBACK (gtk_widget_destroy), dialog); } gtk_widget_show (dialog); g_signal_connect (dialog, "response", G_CALLBACK (_action_compact_add_response_cb), NULL); } static void _action_tab_close_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = midori_browser_get_current_tab (browser); MidoriView* view = MIDORI_VIEW (widget); gboolean last_tab = midori_browser_get_n_pages (browser) == 1; if (last_tab && midori_paths_is_readonly () /* APP, PRIVATE */) { gtk_widget_destroy (GTK_WIDGET (browser)); return; } if (last_tab && midori_view_is_blank (view)) return; midori_browser_remove_tab (browser, widget); } static void _action_window_close_activate (GtkAction* action, MidoriBrowser* browser) { gtk_widget_destroy (GTK_WIDGET (browser)); } static void _action_print_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; if (!(view = midori_browser_get_current_tab (browser))) return; #ifdef HAVE_GRANITE /* FIXME: Blacklist/ custom contract doesn't work gchar* blacklisted_contracts[] = { "print", NULL }; */ /* FIXME: granite: should return GtkWidget* like GTK+ */ GtkWidget* dialog = (GtkWidget*)granite_widgets_pop_over_new (); GtkWidget* content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog)); /* FIXME: granite: should return GtkWidget* like GTK+ */ gchar* filename = midori_view_save_source (MIDORI_VIEW (view), NULL, NULL); const gchar* mime_type = katze_item_get_meta_string ( midori_view_get_proxy_item (MIDORI_VIEW (view)), "mime-type"); GtkWidget* contractor = (GtkWidget*)granite_widgets_contractor_view_new ( filename, mime_type, 32, TRUE); /* granite_widgets_contractor_view_add_item (GRANITE_WIDGETS_CONTRACTOR_VIEW ( contractor), _("_Print"), _("Send document to the printer"), "document-print", 32, G_MAXINT, midori_view_print, view); granite_widgets_contractor_view_name_blacklist (GRANITE_WIDGETS_CONTRACTOR_VIEW ( contractor), blacklisted_contracts, -1); */ g_free (filename); gtk_container_add (GTK_CONTAINER (content_area), contractor); gtk_widget_show (contractor); gtk_widget_show (dialog); if (gtk_widget_get_visible (browser->navigationbar)) granite_widgets_pop_over_move_to_widget ( GRANITE_WIDGETS_POP_OVER (dialog), browser->navigationbar, TRUE); #else midori_view_print (MIDORI_VIEW (view)); #endif } static void _action_quit_activate (GtkAction* action, MidoriBrowser* browser) { g_signal_emit (browser, signals[QUIT], 0); } static void _action_edit_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); gboolean can_undo = FALSE, can_redo = FALSE; gboolean can_cut = FALSE, can_copy = FALSE, can_paste = FALSE; gboolean has_selection, can_select_all = FALSE; if (WEBKIT_IS_WEB_VIEW (widget)) { WebKitWebView* view = WEBKIT_WEB_VIEW (widget); can_undo = webkit_web_view_can_undo (view); can_redo = webkit_web_view_can_redo (view); can_cut = webkit_web_view_can_cut_clipboard (view); can_copy = webkit_web_view_can_copy_clipboard (view); can_paste = webkit_web_view_can_paste_clipboard (view); can_select_all = TRUE; } else if (GTK_IS_EDITABLE (widget)) { GtkEditable* editable = GTK_EDITABLE (widget); has_selection = gtk_editable_get_selection_bounds (editable, NULL, NULL); can_cut = has_selection && gtk_editable_get_editable (editable); can_copy = has_selection; can_paste = gtk_editable_get_editable (editable); can_select_all = TRUE; } else if (GTK_IS_TEXT_VIEW (widget)) { GtkTextView* text_view = GTK_TEXT_VIEW (widget); GtkTextBuffer* buffer = gtk_text_view_get_buffer (text_view); has_selection = gtk_text_buffer_get_has_selection (buffer); can_cut = gtk_text_view_get_editable (text_view); can_copy = has_selection; can_paste = gtk_text_view_get_editable (text_view) && has_selection; can_select_all = TRUE; } _action_set_sensitive (browser, "Undo", can_undo); _action_set_sensitive (browser, "Redo", can_redo); _action_set_sensitive (browser, "Cut", can_cut); _action_set_sensitive (browser, "Copy", can_copy); _action_set_sensitive (browser, "Paste", can_paste); _action_set_sensitive (browser, "Delete", can_cut); _action_set_sensitive (browser, "SelectAll", can_select_all); } static void _action_undo_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); if (WEBKIT_IS_WEB_VIEW (widget)) webkit_web_view_undo (WEBKIT_WEB_VIEW (widget)); } static void _action_redo_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); if (WEBKIT_IS_WEB_VIEW (widget)) webkit_web_view_redo (WEBKIT_WEB_VIEW (widget)); } static void _action_cut_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); if (G_LIKELY (widget) && g_signal_lookup ("cut-clipboard", G_OBJECT_TYPE (widget))) g_signal_emit_by_name (widget, "cut-clipboard"); } static void _action_copy_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); #if !WEBKIT_CHECK_VERSION (1, 4, 3) /* Work around broken clipboard handling for the sake of the user */ if (WEBKIT_IS_WEB_VIEW (widget)) { GtkWidget* scrolled = gtk_widget_get_parent (widget); GtkWidget* view = gtk_widget_get_parent (scrolled); const gchar* selected = midori_view_get_selected_text (MIDORI_VIEW (view)); sokoke_widget_copy_clipboard (widget, selected); return; } #endif if (G_LIKELY (widget) && g_signal_lookup ("copy-clipboard", G_OBJECT_TYPE (widget))) g_signal_emit_by_name (widget, "copy-clipboard"); } static void _action_paste_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); if (G_LIKELY (widget) && g_signal_lookup ("paste-clipboard", G_OBJECT_TYPE (widget))) g_signal_emit_by_name (widget, "paste-clipboard"); } static void _action_delete_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); if (G_LIKELY (widget)) { if (WEBKIT_IS_WEB_VIEW (widget)) webkit_web_view_delete_selection (WEBKIT_WEB_VIEW (widget)); else if (GTK_IS_EDITABLE (widget)) gtk_editable_delete_selection (GTK_EDITABLE (widget)); else if (GTK_IS_TEXT_VIEW (widget)) gtk_text_buffer_delete_selection ( gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)), TRUE, FALSE); } } static void _action_select_all_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* widget = gtk_window_get_focus (GTK_WINDOW (browser)); if (G_LIKELY (widget)) { if (GTK_IS_EDITABLE (widget)) gtk_editable_select_region (GTK_EDITABLE (widget), 0, -1); else if (g_signal_lookup ("select-all", G_OBJECT_TYPE (widget))) { if (GTK_IS_TEXT_VIEW (widget)) g_signal_emit_by_name (widget, "select-all", TRUE); else if (GTK_IS_TREE_VIEW (widget)) { gboolean dummy; g_signal_emit_by_name (widget, "select-all", &dummy); } else g_signal_emit_by_name (widget, "select-all"); } } } static void _action_find_activate (GtkAction* action, MidoriBrowser* browser) { midori_findbar_invoke (MIDORI_FINDBAR (browser->find)); } static void _action_find_next_activate (GtkAction* action, MidoriBrowser* browser) { midori_findbar_find_text (MIDORI_FINDBAR (browser->find), NULL, TRUE); } static void _action_find_previous_activate (GtkAction* action, MidoriBrowser* browser) { midori_findbar_find_text (MIDORI_FINDBAR (browser->find), NULL, FALSE); } static void midori_browser_navigationbar_notify_style_cb (GObject* object, GParamSpec* arg1, MidoriBrowser* browser) { MidoriToolbarStyle toolbar_style; g_object_get (browser->settings, "toolbar-style", &toolbar_style, NULL); _midori_browser_set_toolbar_style (browser, toolbar_style); } static gboolean midori_browser_toolbar_item_button_press_event_cb (GtkWidget* toolitem, GdkEventButton* event, MidoriBrowser* browser); /** * midori_browser_get_toolbar_actions: * * Retrieves a list of actions which are suitable for use in a toolbar. * * Return value: a NULL-terminated array of strings with actions * * Since: 0.1.8 **/ const gchar** midori_browser_get_toolbar_actions (MidoriBrowser* browser) { static const gchar* actions[] = { "WindowNew", "TabNew", "Open", "SaveAs", "Print", "Find", "Fullscreen", "Preferences", "Window", "Bookmarks", "ReloadStop", "ZoomIn", "TabClose", "NextForward", "ZoomOut", "Separator", "Back", "Forward", "Homepage", "Panel", "Trash", "Search", "BookmarkAdd", "Previous", "Next", NULL }; return actions; } /** * midori_browser_get_settings: * * Retrieves the settings instance of the browser. * * Return value: a #MidoriWebSettings instance * * Since: 0.2.5 **/ MidoriWebSettings* midori_browser_get_settings (MidoriBrowser* browser) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); return browser->settings; } static gboolean midori_browser_toolbar_popup_context_menu_cb (GtkWidget* widget, gint x, gint y, gint button, MidoriBrowser* browser) { GtkWidget* menu; GtkWidget* menuitem; menu = gtk_menu_new (); menuitem = sokoke_action_create_popup_menu_item ( _action_by_name (browser, "Menubar")); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); menuitem = sokoke_action_create_popup_menu_item ( _action_by_name (browser, "Navigationbar")); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); menuitem = sokoke_action_create_popup_menu_item ( _action_by_name (browser, "Bookmarkbar")); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); menuitem = sokoke_action_create_popup_menu_item ( _action_by_name (browser, "Statusbar")); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); g_signal_emit (browser, signals[POPULATE_TOOLBAR_MENU], 0, menu); katze_widget_popup (widget, GTK_MENU (menu), NULL, button == -1 ? KATZE_MENU_POSITION_LEFT : KATZE_MENU_POSITION_CURSOR); return TRUE; } static gboolean midori_bookmarkbar_activate_item_alt (GtkAction* action, KatzeItem* item, guint button, MidoriBrowser* browser) { if (MIDORI_EVENT_NEW_TAB (gtk_get_current_event ())) { gint n = midori_browser_add_uri (browser, katze_item_get_uri (item)); midori_browser_set_current_page_smartly (browser, n); } else if (button == 1) { midori_browser_open_bookmark (browser, item); } return TRUE; } static void _action_trash_populate_popup (GtkAction* action, GtkMenu* menu, MidoriBrowser* browser) { GtkWidget* menuitem; menuitem = gtk_separator_menu_item_new (); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); menuitem = gtk_action_create_menu_item ( _action_by_name (browser, "TrashEmpty")); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); } static gint midori_browser_restore_tab (MidoriBrowser* browser, KatzeItem* item) { gint n; g_object_ref (item); katze_array_remove_item (browser->trash, item); n = midori_browser_add_item (browser, item); g_object_unref (item); return n; } static gboolean _action_trash_activate_item_alt (GtkAction* action, KatzeItem* item, guint button, MidoriBrowser* browser) { if (MIDORI_EVENT_NEW_TAB (gtk_get_current_event ())) { midori_browser_set_current_page_smartly (browser, midori_browser_restore_tab (browser, item)); } else if (button == 1) { midori_browser_set_current_page (browser, midori_browser_restore_tab (browser, item)); } return TRUE; } /* static */ void midori_browser_open_bookmark (MidoriBrowser* browser, KatzeItem* item) { const gchar* uri = katze_item_get_uri (item); gchar* uri_fixed; if (!(uri && *uri)) return; /* Imported bookmarks may lack a protocol */ uri_fixed = sokoke_magic_uri (uri); if (!uri_fixed) uri_fixed = g_strdup (uri); if (katze_item_get_meta_boolean (item, "app")) sokoke_spawn_app (uri_fixed, FALSE); else { midori_browser_set_current_uri (browser, uri_fixed); gtk_widget_grab_focus (midori_browser_get_current_tab (browser)); } g_free (uri_fixed); } static void _action_tools_populate_popup (GtkAction* action, GtkMenu* menu, MidoriBrowser* browser) { static const GtkActionEntry actions[] = { { "ManageSearchEngines" }, { "ClearPrivateData" }, { "InspectPage" }, { "-" }, { NULL }, { "p" }, #ifdef G_OS_WIN32 { NULL }, { "Preferences" }, #endif }; guint i; for (i = 0; i < G_N_ELEMENTS (actions); i++) { GtkWidget* menuitem; if (actions[i].name != NULL) { if (actions[i].name[0] == '-') { g_signal_emit (browser, signals[POPULATE_TOOL_MENU], 0, menu); continue; } else if (actions[i].name[0] == 'p') { MidoriPanel* panel; gsize j; GtkWidget* widget; panel = MIDORI_PANEL (browser->panel); j = 0; while ((widget = midori_panel_get_nth_page (panel, j++))) { menuitem = midori_panel_construct_menu_item (panel, MIDORI_VIEWABLE (widget), FALSE); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); } continue; } menuitem = sokoke_action_create_popup_menu_item ( _action_by_name (browser, actions[i].name)); } else menuitem = gtk_separator_menu_item_new (); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); } } static void midori_browser_bookmark_popup (GtkWidget* widget, GdkEventButton* event, KatzeItem* item, MidoriBrowser* browser); static gboolean _action_bookmarks_populate_folder (GtkAction* action, GtkMenuShell* menu, KatzeArray* folder, MidoriBrowser* browser) { gint64 id; KatzeArray* bookmarks; GtkWidget* menuitem; id = katze_item_get_meta_integer (KATZE_ITEM (folder), "id"); if (id == -1) { if (!(bookmarks = midori_array_query (browser->bookmarks, "id, title, parentid, uri, app, pos_panel, pos_bar", "parentid is NULL", NULL))) { g_warning ("midori_array_query returned NULL)"); return FALSE; } } else { gchar *parentid = g_strdup_printf ("%" G_GINT64_FORMAT, id); if (!(bookmarks = midori_array_query (browser->bookmarks, "id, title, parentid, uri, app, pos_panel, pos_bar", "parentid = %q", parentid))) { g_warning ("midori_array_query returned NULL (id='%s')", parentid); g_free (parentid); return FALSE; } g_free (parentid); } /* Clear items from dummy array here */ gtk_container_foreach (GTK_CONTAINER (menu), (GtkCallback)(gtk_widget_destroy), NULL); if (katze_array_is_empty (bookmarks)) { menuitem = gtk_image_menu_item_new_with_label (_("Empty")); gtk_widget_set_sensitive (menuitem, FALSE); gtk_menu_shell_append (menu, menuitem); gtk_widget_show (menuitem); return TRUE; } katze_array_action_generate_menu (KATZE_ARRAY_ACTION (action), bookmarks, menu, GTK_WIDGET (browser)); return TRUE; } static void _action_window_populate_popup (GtkAction* action, GtkMenu* menu, MidoriBrowser* browser) { GtkWidget* menuitem; menuitem = gtk_separator_menu_item_new (); gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); menuitem = gtk_action_create_menu_item ( _action_by_name (browser, "LastSession")); gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); menuitem = gtk_action_create_menu_item ( _action_by_name (browser, "TabCurrent")); gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); menuitem = gtk_action_create_menu_item ( _action_by_name (browser, "NextView")); gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); menuitem = gtk_action_create_menu_item ( _action_by_name (browser, "TabNext")); gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); menuitem = gtk_action_create_menu_item ( _action_by_name (browser, "TabPrevious")); gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); } static void _action_window_activate_item_alt (GtkAction* action, KatzeItem* item, gint button, MidoriBrowser* browser) { guint i; guint n = katze_array_get_length (browser->proxy_array); for (i = 0; i < n; i++) { GtkWidget* view; view = midori_browser_get_nth_tab (browser, i); if (midori_view_get_proxy_item (MIDORI_VIEW (view)) == item) midori_browser_set_current_page (browser, i); } } static void _action_compact_menu_populate_popup (GtkAction* action, GtkWidget* menu, MidoriBrowser* browser) { static const GtkActionEntry actions[] = { { "TabNew" }, { "WindowNew" }, { "PrivateBrowsing" }, { NULL }, { "Find" }, #if !HAVE_HILDON { "Print" }, #endif { "Fullscreen" }, { NULL }, #if !HAVE_HILDON { "p" }, { NULL }, { "BookmarksImport" }, { "BookmarksExport" }, { "ClearPrivateData" }, #if defined (GDK_WINDOWING_X11) { "AddDesktopShortcut" }, #endif { "-" }, #endif { NULL }, #if !HAVE_HILDON #ifndef HAVE_GRANITE { "HelpFAQ" }, { "HelpBugs"}, #endif #endif { "About" }, { "Preferences" }, #if HAVE_HILDON { NULL }, { "auto-load-images" }, { "enable-scripts" }, { "enable-plugins" }, #endif }; guint i; for (i = 0; i < G_N_ELEMENTS (actions); i++) { #ifdef HAVE_HILDON_2_2 GtkAction* _action; gchar* label; GtkWidget* button; if (!actions[i].name) continue; _action = _action_by_name (browser, actions[i].name); if (_action) { label = katze_object_get_string (_action, "label"); button = hildon_gtk_button_new (HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH); gtk_button_set_label (GTK_BUTTON (button), label); gtk_button_set_use_underline (GTK_BUTTON (button), TRUE); g_free (label); g_signal_connect_swapped (button, "clicked", G_CALLBACK (gtk_action_activate), _action); } else button = katze_property_proxy (browser->settings, actions[i].name, NULL); gtk_widget_show (button); hildon_app_menu_append (HILDON_APP_MENU (menu), GTK_BUTTON (button)); #else GtkWidget* menuitem; if (actions[i].name != NULL) { if (actions[i].name[0] == '-') { g_signal_emit (browser, signals[POPULATE_TOOL_MENU], 0, menu); continue; } else if (actions[i].name[0] == 'p') { MidoriPanel* panel; gsize j; GtkWidget* widget; panel = MIDORI_PANEL (browser->panel); j = 0; while ((widget = midori_panel_get_nth_page (panel, j++))) { menuitem = midori_panel_construct_menu_item (panel, MIDORI_VIEWABLE (widget), TRUE); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); } continue; } menuitem = sokoke_action_create_popup_menu_item ( _action_by_name (browser, actions[i].name)); } else { menuitem = gtk_separator_menu_item_new (); gtk_widget_show (menuitem); } gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); #endif } } static void midori_preferences_response_help_cb (GtkWidget* preferences, gint response, MidoriBrowser* browser) { if (response == GTK_RESPONSE_HELP) midori_browser_activate_action (browser, "HelpFAQ"); } static void _action_preferences_activate (GtkAction* action, MidoriBrowser* browser) { static GtkWidget* dialog = NULL; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; if (!dialog) { dialog = midori_preferences_new (GTK_WINDOW (browser), browser->settings); g_signal_emit (browser, signals[SHOW_PREFERENCES], 0, dialog); g_signal_connect (dialog, "response", G_CALLBACK (midori_preferences_response_help_cb), browser); g_signal_connect (dialog, "destroy", G_CALLBACK (gtk_widget_destroyed), &dialog); gtk_widget_show (dialog); } else gtk_window_present (GTK_WINDOW (dialog)); } static gboolean midori_browser_has_native_menubar (void) { #if HAVE_HILDON return TRUE; #else static const gchar* ubuntu_menuproxy = NULL; if (ubuntu_menuproxy == NULL) ubuntu_menuproxy = g_getenv ("UBUNTU_MENUPROXY"); return ubuntu_menuproxy && strstr (ubuntu_menuproxy, ".so") != NULL; #endif } static void _action_menubar_activate (GtkToggleAction* menubar_action, MidoriBrowser* browser) { gboolean active = gtk_toggle_action_get_active (menubar_action); GtkAction* menu_action = _action_by_name (browser, "CompactMenu"); GString* toolbar_items; GList* children; gchar* items; if (midori_browser_has_native_menubar ()) active = FALSE; toolbar_items = g_string_new (NULL); children = gtk_container_get_children (GTK_CONTAINER (browser->navigationbar)); for (; children != NULL; children = g_list_next (children)) { GtkAction* action = gtk_activatable_get_related_action ( GTK_ACTIVATABLE (children->data)); if (!action) continue; if (action == ((GtkAction*)menu_action)) { if (active) { gtk_container_remove (GTK_CONTAINER (browser->navigationbar), GTK_WIDGET (children->data)); } continue; } else if (MIDORI_IS_PANED_ACTION (action)) { MidoriPanedAction* paned_action = MIDORI_PANED_ACTION (action); g_string_append_printf (toolbar_items, "%s,%s", midori_paned_action_get_child1_name (paned_action), midori_paned_action_get_child2_name (paned_action)); } else g_string_append (toolbar_items, gtk_action_get_name (action)); g_string_append_c (toolbar_items, ','); } g_list_free (children); if (katze_object_get_boolean (browser->settings, "show-menubar") != active) g_object_set (browser->settings, "show-menubar", active, NULL); items = g_string_free (toolbar_items, FALSE); g_object_set (browser->settings, "toolbar-items", items, NULL); g_free (items); sokoke_widget_set_visible (browser->menubar, active); g_object_set_data (G_OBJECT (browser), "midori-toolbars-visible", gtk_widget_get_visible (browser->menubar) || gtk_widget_get_visible (browser->navigationbar) ? (void*)0xdeadbeef : NULL); } static void _action_navigationbar_activate (GtkToggleAction* action, MidoriBrowser* browser) { gboolean active = gtk_toggle_action_get_active (action); g_object_set (browser->settings, "show-navigationbar", active, NULL); sokoke_widget_set_visible (browser->navigationbar, active); g_object_set_data (G_OBJECT (browser), "midori-toolbars-visible", gtk_widget_get_visible (browser->menubar) || gtk_widget_get_visible (browser->navigationbar) ? (void*)0xdeadbeef : NULL); } static void _action_bookmarkbar_activate (GtkToggleAction* action, MidoriBrowser* browser) { gboolean active = gtk_toggle_action_get_active (action); g_object_set (browser->settings, "show-bookmarkbar", active, NULL); sokoke_widget_set_visible (browser->bookmarkbar, active); } static void _action_statusbar_activate (GtkToggleAction* action, MidoriBrowser* browser) { gboolean active = gtk_toggle_action_get_active (action); g_object_set (browser->settings, "show-statusbar", active, NULL); sokoke_widget_set_visible (browser->statusbar, active); } static void _action_reload_stop_activate (GtkAction* action, MidoriBrowser* browser) { gchar* stock_id; GtkWidget* view; if (!(view = midori_browser_get_current_tab (browser))) return; g_object_get (action, "stock-id", &stock_id, NULL); /* Refresh or stop, depending on the stock id */ if (!strcmp (stock_id, GTK_STOCK_REFRESH)) { GdkModifierType state = (GdkModifierType)0; gint x, y; GdkWindow* window; gboolean from_cache = TRUE; if (!strcmp (gtk_action_get_name (action), "ReloadUncached")) from_cache = FALSE; else if ((window = gtk_widget_get_window (GTK_WIDGET (browser)))) { gdk_window_get_pointer (window, &x, &y, &state); if (state & GDK_SHIFT_MASK) from_cache = FALSE; } midori_view_reload (MIDORI_VIEW (view), from_cache); } else midori_view_stop_loading (MIDORI_VIEW (view)); g_free (stock_id); } static void _action_zoom_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); if (!view) return; if (g_str_equal (gtk_action_get_name (action), "ZoomIn")) midori_view_set_zoom_level (MIDORI_VIEW (view), midori_view_get_zoom_level (MIDORI_VIEW (view)) + 0.10f); else if (g_str_equal (gtk_action_get_name (action), "ZoomOut")) midori_view_set_zoom_level (MIDORI_VIEW (view), midori_view_get_zoom_level (MIDORI_VIEW (view)) - 0.10f); else midori_view_set_zoom_level (MIDORI_VIEW (view), 1.0f); } static void _action_view_encoding_activate (GtkAction* action, GtkAction* current, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); if (view) { const gchar* name; GtkWidget* web_view; name = gtk_action_get_name (current); web_view = midori_view_get_web_view (MIDORI_VIEW (view)); if (!strcmp (name, "EncodingAutomatic")) g_object_set (web_view, "custom-encoding", NULL, NULL); else { const gchar* encoding; if (!strcmp (name, "EncodingChinese")) encoding = "BIG5"; else if (!strcmp (name, "EncodingChineseSimplified")) encoding = "GB18030"; else if (!strcmp (name, "EncodingJapanese")) encoding = "SHIFT_JIS"; else if (!strcmp (name, "EncodingKorean")) encoding = "EUC-KR"; else if (!strcmp (name, "EncodingRussian")) encoding = "KOI8-R"; else if (!strcmp (name, "EncodingUnicode")) encoding = "UTF-8"; else if (!strcmp (name, "EncodingWestern")) encoding = "ISO-8859-1"; else g_assert_not_reached (); g_object_set (web_view, "custom-encoding", encoding, NULL); } } } static void _action_source_view_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view; gchar* text_editor; gchar* filename = NULL; if (!(view = midori_browser_get_current_tab (browser))) return; filename = midori_view_save_source (MIDORI_VIEW (view), NULL, NULL); g_object_get (browser->settings, "text-editor", &text_editor, NULL); if (!(text_editor && *text_editor)) { GtkWidget* source; GtkWidget* source_view; gchar* source_uri; source_uri = g_filename_to_uri (filename, NULL, NULL); g_free (filename); source = midori_view_new_with_item (NULL, browser->settings); source_view = midori_view_get_web_view (MIDORI_VIEW (source)); webkit_web_view_set_view_source_mode (WEBKIT_WEB_VIEW (source_view), TRUE); webkit_web_view_load_uri (WEBKIT_WEB_VIEW (source_view), source_uri); midori_browser_add_tab (browser, source); } else { sokoke_spawn_program (text_editor, TRUE, filename, TRUE); g_free (filename); } g_free (text_editor); } static void _action_caret_browsing_activate (GtkAction* action, MidoriBrowser* browser) { gint response; GtkWidget* dialog; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; if (!katze_object_get_boolean (browser->settings, "enable-caret-browsing")) { dialog = gtk_message_dialog_new (GTK_WINDOW (browser), GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR, GTK_MESSAGE_QUESTION, GTK_BUTTONS_NONE, _("Toggle text cursor navigation")); gtk_window_set_title (GTK_WINDOW (dialog), _("Toggle text cursor navigation")); gtk_message_dialog_format_secondary_text (GTK_MESSAGE_DIALOG (dialog), _("Pressing F7 toggles Caret Browsing. When active, a text cursor appears in all websites.")); gtk_dialog_add_buttons (GTK_DIALOG (dialog), GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, _("_Enable Caret Browsing"), GTK_RESPONSE_ACCEPT, NULL); response = gtk_dialog_run (GTK_DIALOG (dialog)); gtk_widget_destroy (dialog); if (response != GTK_RESPONSE_ACCEPT) return; } g_object_set (browser->settings, "enable-caret-browsing", !katze_object_get_boolean (browser->settings, "enable-caret-browsing"), NULL); } static void _action_fullscreen_activate (GtkAction* action, MidoriBrowser* browser) { GdkWindowState state; if (!gtk_widget_get_window (GTK_WIDGET (browser))) return; state = gdk_window_get_state (gtk_widget_get_window (GTK_WIDGET (browser))); if (state & GDK_WINDOW_STATE_FULLSCREEN) { if (katze_object_get_boolean (G_OBJECT (browser->settings), "show-menubar")) gtk_widget_show (browser->menubar); if (katze_object_get_boolean (G_OBJECT (browser->settings), "show-panel")) gtk_widget_show (browser->panel); if (katze_object_get_boolean (G_OBJECT (browser->settings), "show-bookmarkbar")) gtk_widget_show (browser->bookmarkbar); if (browser->show_navigationbar) gtk_widget_show (browser->navigationbar); if (browser->show_statusbar) gtk_widget_show (browser->statusbar); _toggle_tabbar_smartly (browser, TRUE); gtk_window_unfullscreen (GTK_WINDOW (browser)); } else { gtk_widget_hide (browser->menubar); gtk_widget_hide (browser->panel); gtk_widget_hide (browser->bookmarkbar); gtk_widget_hide (browser->navigationbar); gtk_widget_hide (browser->statusbar); #ifdef HAVE_GRANITE granite_widgets_dynamic_notebook_set_show_tabs ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), FALSE); #else gtk_notebook_set_show_tabs (GTK_NOTEBOOK (browser->notebook), FALSE); gtk_notebook_set_show_border (GTK_NOTEBOOK (browser->notebook), FALSE); #endif gtk_window_fullscreen (GTK_WINDOW (browser)); } } static void _action_scroll_somewhere_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view; WebKitWebView* web_view; const gchar* name; view = midori_browser_get_current_tab (browser); if (!view) return; web_view = WEBKIT_WEB_VIEW (midori_view_get_web_view (MIDORI_VIEW (view))); name = gtk_action_get_name (action); if (g_str_equal (name, "ScrollLeft")) webkit_web_view_move_cursor (web_view, GTK_MOVEMENT_VISUAL_POSITIONS, -1); else if (g_str_equal (name, "ScrollDown")) webkit_web_view_move_cursor (web_view, GTK_MOVEMENT_DISPLAY_LINES, 1); else if (g_str_equal (name, "ScrollUp")) webkit_web_view_move_cursor (web_view, GTK_MOVEMENT_DISPLAY_LINES, -1); else if (g_str_equal (name, "ScrollRight")) webkit_web_view_move_cursor (web_view, GTK_MOVEMENT_VISUAL_POSITIONS, 1); } static void _action_readable_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); gchar* filename; gchar* stylesheet; gint i; gchar* script; gchar* exception; if (!view) return; filename = midori_paths_get_res_filename ("faq.css"); stylesheet = NULL; if (!g_file_get_contents (filename, &stylesheet, NULL, NULL)) { katze_assign (filename, midori_paths_get_data_filename ("doc/midori/faq.css", FALSE)); g_file_get_contents (filename, &stylesheet, NULL, NULL); } if (!(stylesheet && *stylesheet)) { g_free (filename); g_free (stylesheet); midori_view_add_info_bar (MIDORI_VIEW (view), GTK_MESSAGE_ERROR, "Stylesheet faq.css not found", NULL, view, GTK_STOCK_OK, GTK_RESPONSE_ACCEPT, NULL); return; } i = 0; while (stylesheet[i]) { /* Replace line breaks with spaces */ if (stylesheet[i] == '\n' || stylesheet[i] == '\r') stylesheet[i] = ' '; /* Change all single quotes to double quotes */ else if (stylesheet[i] == '\'') stylesheet[i] = '\"'; i++; } script = g_strdup_printf ( "(function () {" "var style = document.createElement ('style');" "style.setAttribute ('type', 'text/css');" "style.appendChild (document.createTextNode ('%s'));" "var head = document.getElementsByTagName ('head')[0];" "if (head) head.appendChild (style);" "else document.documentElement.insertBefore" "(style, document.documentElement.firstChild);" "}) ();", stylesheet); g_free (stylesheet); exception = NULL; if (!midori_view_execute_script (MIDORI_VIEW (view), script, &exception)) { midori_view_add_info_bar (MIDORI_VIEW (view), GTK_MESSAGE_ERROR, exception, NULL, view, GTK_STOCK_OK, GTK_RESPONSE_ACCEPT, NULL); g_free (exception); } g_free (script); } static gboolean _action_navigation_activate (GtkAction* action, MidoriBrowser* browser) { MidoriView* view; GtkWidget* tab; gchar* uri; const gchar* name; gboolean middle_click; g_assert (GTK_IS_ACTION (action)); if (GPOINTER_TO_INT (g_object_get_data (G_OBJECT (action), "midori-middle-click"))) { middle_click = TRUE; g_object_set_data (G_OBJECT (action), "midori-middle-click", GINT_TO_POINTER(0)); } else middle_click = FALSE; tab = midori_browser_get_current_tab (browser); if (!tab) return FALSE; view = MIDORI_VIEW (tab); name = gtk_action_get_name (action); if (!strcmp (name, "NextForward")) name = midori_view_can_go_forward (view) ? "Forward" : "Next"; if (g_str_equal (name, "Back")) { if (middle_click) { GtkWidget* web_view; WebKitWebBackForwardList* back_forward_list; WebKitWebHistoryItem* back_item; const gchar* back_uri; gint n; web_view = midori_view_get_web_view (view); back_forward_list = webkit_web_view_get_back_forward_list (WEBKIT_WEB_VIEW (web_view)); back_item = webkit_web_back_forward_list_get_back_item (back_forward_list); back_uri = webkit_web_history_item_get_uri (back_item); n = midori_browser_add_uri (browser, back_uri); midori_browser_set_current_page_smartly (browser, n); } else midori_view_go_back (view); return TRUE; } else if (g_str_equal (name, "Forward")) { if (middle_click) { GtkWidget* web_view; WebKitWebBackForwardList* back_forward_list; WebKitWebHistoryItem* forward_item; const gchar* forward_uri; gint n; web_view = midori_view_get_web_view (view); back_forward_list = webkit_web_view_get_back_forward_list (WEBKIT_WEB_VIEW (web_view)); forward_item = webkit_web_back_forward_list_get_forward_item (back_forward_list); forward_uri = webkit_web_history_item_get_uri (forward_item); n = midori_browser_add_uri (browser, forward_uri); midori_browser_set_current_page_smartly (browser, n); } else midori_view_go_forward (view); return TRUE; } else if (g_str_equal (name, "Previous")) { /* Duplicate here because the URI pointer might change */ uri = g_strdup (midori_view_get_previous_page (view)); if (middle_click) { gint n; n = midori_browser_add_uri (browser, uri); midori_browser_set_current_page_smartly (browser, n); } else midori_view_set_uri (view, uri); g_free (uri); return TRUE; } else if (g_str_equal (name, "Next")) { /* Duplicate here because the URI pointer might change */ uri = g_strdup (midori_view_get_next_page (view)); if (middle_click) { gint n; n = midori_browser_add_uri (browser, uri); midori_browser_set_current_page_smartly (browser, n); } else midori_view_set_uri (view, uri); g_free (uri); return TRUE; } else if (g_str_equal (name, "Homepage")) { g_object_get (browser->settings, "homepage", &uri, NULL); if (middle_click) { gint n; n = midori_browser_add_uri (browser, uri); midori_browser_set_current_page_smartly (browser, n); } else midori_view_set_uri (view, uri); g_free (uri); return TRUE; } return FALSE; } static void _action_location_activate (GtkAction* action, MidoriBrowser* browser) { if (!gtk_widget_get_visible (browser->navigationbar)) gtk_widget_show (browser->navigationbar); } static void _action_location_focus_in (GtkAction* action, MidoriBrowser* browser) { GdkScreen* screen = gtk_widget_get_screen (browser->notebook); GtkIconTheme* icon_theme = gtk_icon_theme_get_for_screen (screen); if (gtk_icon_theme_has_icon (icon_theme, "go-jump-symbolic")) midori_location_action_set_secondary_icon ( MIDORI_LOCATION_ACTION (action), "go-jump-symbolic"); else midori_location_action_set_secondary_icon ( MIDORI_LOCATION_ACTION (action), GTK_STOCK_JUMP_TO); } static void _action_location_focus_out (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); if (!browser->show_navigationbar || midori_browser_is_fullscreen (browser)) gtk_widget_hide (browser->navigationbar); midori_browser_update_secondary_icon (browser, MIDORI_VIEW (view), action); } static void _action_location_reset_uri (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view; if ((view = midori_browser_get_current_tab (browser))) { midori_location_action_set_text (MIDORI_LOCATION_ACTION (action), midori_view_get_display_uri (MIDORI_VIEW (view))); } } static void _action_location_submit_uri (GtkAction* action, const gchar* uri, gboolean new_tab, MidoriBrowser* browser) { gchar* new_uri; gint n; uri = katze_skip_whitespace (uri); new_uri = sokoke_magic_uri (uri); if (!new_uri) { const gchar* keywords = NULL; const gchar* search_uri = NULL; KatzeItem* item; /* Do we have a keyword and a string? */ if (browser->search_engines && (item = katze_array_find_token (browser->search_engines, uri))) { keywords = strchr (uri, ' '); if (keywords != NULL) keywords++; else keywords = ""; search_uri = katze_item_get_uri (item); } if (keywords == NULL) { keywords = uri; search_uri = browser->location_entry_search; } new_uri = midori_uri_for_search (search_uri, keywords); if (browser->history != NULL) { time_t now = time (NULL); gint64 day = sokoke_time_t_to_julian (&now); sqlite3* db = g_object_get_data (G_OBJECT (browser->history), "db"); static sqlite3_stmt* statement = NULL; if (!statement) { const gchar* sqlcmd; sqlcmd = "INSERT INTO search (keywords, uri, day) VALUES (?,?,?)"; sqlite3_prepare_v2 (db, sqlcmd, strlen (sqlcmd) + 1, &statement, NULL); } sqlite3_bind_text (statement, 1, keywords, -1, 0); sqlite3_bind_text (statement, 2, search_uri, -1, 0); sqlite3_bind_int64 (statement, 3, day); if (sqlite3_step (statement) != SQLITE_DONE) g_printerr (_("Failed to insert new history item: %s\n"), sqlite3_errmsg (db)); sqlite3_reset (statement); if (sqlite3_step (statement) == SQLITE_DONE) sqlite3_clear_bindings (statement); } } if (new_tab) { n = midori_browser_add_uri (browser, new_uri); midori_browser_set_current_page (browser, n); } else midori_browser_set_current_uri (browser, new_uri); g_free (new_uri); gtk_widget_grab_focus (midori_browser_get_current_tab (browser)); } static void midori_browser_news_feed_clicked_cb (GtkWidget* menuitem, MidoriBrowser* browser) { gchar* uri = g_object_get_data (G_OBJECT (menuitem), "uri"); midori_browser_subscribe_to_news_feed (browser, uri); } static gboolean _action_location_secondary_icon_released (GtkAction* action, GtkWidget* widget, MidoriBrowser* browser) { GtkWidget* view; if ((view = midori_browser_get_current_tab (browser))) { const gchar* uri = midori_view_get_display_uri (MIDORI_VIEW (view)); const gchar* feed; /* Clicking icon on blank is equal to Paste and Proceed */ if (midori_view_is_blank (MIDORI_VIEW (view))) { GtkClipboard* clipboard = gtk_clipboard_get_for_display ( gtk_widget_get_display (view), GDK_SELECTION_CLIPBOARD); gchar* text = gtk_clipboard_wait_for_text (clipboard); if (text != NULL) { _action_location_submit_uri (action, text, FALSE, browser); g_free (text); } } else if (gtk_window_get_focus (GTK_WINDOW (browser)) == widget) { const gchar* text = gtk_entry_get_text (GTK_ENTRY (widget)); _action_location_submit_uri (action, text, FALSE, browser); } else if ((feed = g_object_get_data (G_OBJECT (view), "news-feeds"))) { KatzeArray* news_feeds; KatzeItem* item; KatzeItem* itemm; news_feeds = katze_object_get_object (G_OBJECT (view), "news-feeds"); item = katze_array_get_nth_item (news_feeds, 0); if ((itemm = katze_array_get_nth_item (news_feeds, 1))) { guint i; GtkWidget* menu; GtkWidget* menuitem; menu = gtk_menu_new (); menuitem = gtk_menu_item_new_with_label (katze_item_get_name (item)); g_object_set_data_full (G_OBJECT (menuitem), "uri", g_strdup (katze_item_get_uri (item)), (GDestroyNotify)g_free); g_signal_connect (menuitem, "activate", G_CALLBACK (midori_browser_news_feed_clicked_cb), browser); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); menuitem = gtk_menu_item_new_with_label (katze_item_get_name (itemm)); g_object_set_data_full (G_OBJECT (menuitem), "uri", g_strdup (katze_item_get_uri (itemm)), (GDestroyNotify)g_free); g_signal_connect (menuitem, "activate", G_CALLBACK (midori_browser_news_feed_clicked_cb), browser); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); i = 2; while ((itemm = katze_array_get_nth_item (news_feeds, i++))) { menuitem = gtk_menu_item_new_with_label ( katze_item_get_name (itemm)); g_object_set_data_full (G_OBJECT (menuitem), "uri", g_strdup (katze_item_get_uri (itemm)), (GDestroyNotify)g_free); g_signal_connect (menuitem, "activate", G_CALLBACK (midori_browser_news_feed_clicked_cb), browser); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); } gtk_container_foreach (GTK_CONTAINER (menu), (GtkCallback)(gtk_widget_show_all), NULL); katze_widget_popup (widget, GTK_MENU (menu), NULL, KATZE_MENU_POSITION_RIGHT); } else midori_browser_subscribe_to_news_feed (browser, feed); g_object_unref (news_feeds); } else _action_location_submit_uri (action, uri, FALSE, browser); return TRUE; } return FALSE; } static void _action_search_submit (GtkAction* action, const gchar* keywords, gboolean new_tab, MidoriBrowser* browser) { KatzeItem* item; const gchar* url; gchar* search; item = katze_array_get_nth_item (browser->search_engines, browser->last_web_search); if (item) url = katze_item_get_uri (item); else /* The location entry search is our fallback */ url = browser->location_entry_search; search = midori_uri_for_search (url, keywords); if (new_tab) { int n = midori_browser_add_uri (browser, search); midori_browser_set_current_page_smartly (browser, n); } else midori_browser_set_current_uri (browser, search); g_free (search); } static void _action_search_activate (GtkAction* action, MidoriBrowser* browser) { GSList* proxies = gtk_action_get_proxies (action); const gchar* uri; gchar* search; for (; proxies != NULL; proxies = g_slist_next (proxies)) if (GTK_IS_TOOL_ITEM (proxies->data)) { if (!gtk_widget_get_visible (browser->navigationbar)) gtk_widget_show (browser->navigationbar); return; } /* Load default search engine in current tab */ uri = browser->location_entry_search; search = midori_uri_for_search (uri ? uri : "", ""); midori_browser_set_current_uri (browser, search); gtk_widget_grab_focus (midori_browser_get_current_tab (browser)); g_free (search); } static void _action_search_notify_current_item (GtkAction* action, GParamSpec* pspec, MidoriBrowser* browser) { MidoriSearchAction* search_action; KatzeItem* item; guint idx; search_action = MIDORI_SEARCH_ACTION (action); item = midori_search_action_get_current_item (search_action); if (item) idx = katze_array_get_item_index (browser->search_engines, item); else idx = 0; g_object_set (browser->settings, "last-web-search", idx, NULL); browser->last_web_search = idx; } static void _action_search_notify_default_item (GtkAction* action, GParamSpec* pspec, MidoriBrowser* browser) { MidoriSearchAction* search_action; KatzeItem* item; search_action = MIDORI_SEARCH_ACTION (action); item = midori_search_action_get_default_item (search_action); if (item) g_object_set (browser->settings, "location-entry-search", katze_item_get_uri (item), NULL); } static void _action_search_focus_out (GtkAction* action, MidoriBrowser* browser) { if ((gtk_widget_get_visible (browser->statusbar) && !browser->show_navigationbar) || midori_browser_is_fullscreen (browser)) { gtk_widget_hide (browser->navigationbar); } } static void midori_browser_bookmark_popup_item (GtkWidget* menu, const gchar* stock_id, const gchar* label, KatzeItem* item, gpointer callback, MidoriBrowser* browser) { const gchar* uri; GtkWidget* menuitem; uri = katze_item_get_uri (item); menuitem = gtk_image_menu_item_new_from_stock (stock_id, NULL); if (label) gtk_label_set_text_with_mnemonic (GTK_LABEL (gtk_bin_get_child ( GTK_BIN (menuitem))), label); if (!strcmp (stock_id, GTK_STOCK_EDIT)) gtk_widget_set_sensitive (menuitem, KATZE_IS_ARRAY (item) || uri != NULL); else if (!KATZE_IS_ARRAY (item) && strcmp (stock_id, GTK_STOCK_DELETE)) gtk_widget_set_sensitive (menuitem, uri != NULL); g_object_set_data (G_OBJECT (menuitem), "KatzeItem", item); g_signal_connect (menuitem, "activate", G_CALLBACK (callback), browser); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); } static void midori_browser_bookmark_open_activate_cb (GtkWidget* menuitem, MidoriBrowser* browser) { KatzeItem* item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem"); midori_browser_open_bookmark (browser, item); } static void midori_browser_bookmark_open_in_tab_activate_cb (GtkWidget* menuitem, MidoriBrowser* browser) { KatzeItem* item; const gchar* uri; guint n; item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem"); if (KATZE_IS_ARRAY (item)) { KatzeItem* child; KATZE_ARRAY_FOREACH_ITEM (child, KATZE_ARRAY (item)) { if ((uri = katze_item_get_uri (child)) && *uri) { n = midori_browser_add_item (browser, child); midori_browser_set_current_page_smartly (browser, n); } } } else { if ((uri = katze_item_get_uri (item)) && *uri) { n = midori_browser_add_item (browser, item); midori_browser_set_current_page_smartly (browser, n); } } } static void midori_browser_bookmark_open_in_window_activate_cb (GtkWidget* menuitem, MidoriBrowser* browser) { KatzeItem* item; const gchar* uri; item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem"); uri = katze_item_get_uri (item); midori_view_new_window_cb (NULL, uri, browser); } static void midori_browser_bookmark_edit_activate_cb (GtkWidget* menuitem, MidoriBrowser* browser) { KatzeItem* item; item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem"); if (KATZE_ITEM_IS_BOOKMARK (item)) midori_browser_edit_bookmark_dialog_new (browser, item, FALSE, FALSE, NULL); else midori_browser_edit_bookmark_dialog_new (browser, item, FALSE, TRUE, NULL); } static void midori_browser_bookmark_delete_activate_cb (GtkWidget* menuitem, MidoriBrowser* browser) { KatzeItem* item; item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem"); katze_array_remove_item (browser->bookmarks, item); } static void midori_browser_bookmark_popup (GtkWidget* widget, GdkEventButton* event, KatzeItem* item, MidoriBrowser* browser) { GtkWidget* menu; GtkWidget* menuitem; menu = gtk_menu_new (); if (!katze_item_get_uri (item)) midori_browser_bookmark_popup_item (menu, STOCK_TAB_NEW, _("Open all in _Tabs"), item, midori_browser_bookmark_open_in_tab_activate_cb, browser); else { midori_browser_bookmark_popup_item (menu, GTK_STOCK_OPEN, NULL, item, midori_browser_bookmark_open_activate_cb, browser); midori_browser_bookmark_popup_item (menu, STOCK_TAB_NEW, _("Open in New _Tab"), item, midori_browser_bookmark_open_in_tab_activate_cb, browser); midori_browser_bookmark_popup_item (menu, STOCK_WINDOW_NEW, _("Open in New _Window"), item, midori_browser_bookmark_open_in_window_activate_cb, browser); } menuitem = gtk_separator_menu_item_new (); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); gtk_widget_show (menuitem); midori_browser_bookmark_popup_item (menu, GTK_STOCK_EDIT, NULL, item, midori_browser_bookmark_edit_activate_cb, browser); midori_browser_bookmark_popup_item (menu, GTK_STOCK_DELETE, NULL, item, midori_browser_bookmark_delete_activate_cb, browser); katze_widget_popup (widget, GTK_MENU (menu), event, KATZE_MENU_POSITION_CURSOR); } static gboolean midori_browser_menu_button_press_event_cb (GtkWidget* toolitem, GdkEventButton* event, MidoriBrowser* browser) { if (event->button != 3) return FALSE; /* GtkMenuBar catches button events on children with submenus, so we need to see if the actual widget is the menubar, and if it is an item, we forward it to the actual widget. */ if ((GTK_IS_BOX (toolitem) || GTK_IS_MENU_BAR (toolitem))) { midori_browser_toolbar_popup_context_menu_cb ( GTK_IS_BIN (toolitem) && gtk_bin_get_child (GTK_BIN (toolitem)) ? gtk_widget_get_parent (toolitem) : toolitem, event->x, event->y, event->button, browser); return TRUE; } else if (GTK_IS_MENU_ITEM (toolitem)) { gboolean handled; g_signal_emit_by_name (toolitem, "button-press-event", event, &handled); return handled; } return FALSE; } static gboolean midori_browser_menu_item_middle_click_event_cb (GtkWidget* toolitem, GdkEventButton* event, MidoriBrowser* browser) { if (MIDORI_EVENT_NEW_TAB (event)) { GtkAction* action; action = gtk_activatable_get_related_action (GTK_ACTIVATABLE (toolitem)); g_object_set_data (G_OBJECT (action), "midori-middle-click", GINT_TO_POINTER (1)); return _action_navigation_activate (action, browser); } return FALSE; } static void _action_bookmark_add_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* proxy = NULL; GSList* proxies = gtk_action_get_proxies (action); for (; proxies != NULL; proxies = g_slist_next (proxies)) if (GTK_IS_TOOL_ITEM (proxies->data)) { proxy = proxies->data; break; } if (g_str_equal (gtk_action_get_name (action), "BookmarkFolderAdd")) midori_browser_edit_bookmark_dialog_new (browser, NULL, TRUE, TRUE, proxy); else midori_browser_edit_bookmark_dialog_new (browser, NULL, TRUE, FALSE, proxy); } static void _action_bookmarks_import_activate (GtkAction* action, MidoriBrowser* browser) { typedef struct { const gchar* path; const gchar* name; const gchar* icon; } BookmarkClient; static const BookmarkClient bookmark_clients[] = { { ".local/share/data/Arora/bookmarks.xbel", N_("Arora"), "arora" }, { ".kazehakase/bookmarks.xml", N_("Kazehakase"), "kazehakase-icon" }, { ".opera/bookmarks.adr", N_("Opera"), "opera" }, { ".kde/share/apps/konqueror/bookmarks.xml", N_("Konqueror"), "konqueror" }, { ".gnome2/epiphany/bookmarks.rdf", N_("Epiphany"), "epiphany" }, { ".mozilla/firefox/*/bookmarks.html", N_("Firefox (%s)"), "firefox" }, { ".config/midori/bookmarks.xbel", N_("Midori 0.2.6"), "midori" }, }; GtkWidget* dialog; GtkWidget* content_area; GtkSizeGroup* sizegroup; GtkWidget* hbox; GtkWidget* label; GtkWidget* combo; GtkComboBox* combobox; GtkListStore* model; GtkCellRenderer* renderer; GtkWidget* combobox_folder; gint icon_width = 16; guint i; KatzeArray* bookmarks; if (!browser->bookmarks || !gtk_widget_get_visible (GTK_WIDGET (browser))) return; dialog = gtk_dialog_new_with_buttons ( _("Import bookmarks…"), GTK_WINDOW (browser), GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, _("_Import bookmarks"), GTK_RESPONSE_ACCEPT, NULL); content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog)); gtk_window_set_icon_name (GTK_WINDOW (dialog), STOCK_BOOKMARKS); gtk_container_set_border_width (GTK_CONTAINER (dialog), 5); gtk_container_set_border_width (GTK_CONTAINER (content_area), 5); sizegroup = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL); hbox = gtk_hbox_new (FALSE, 8); gtk_container_set_border_width (GTK_CONTAINER (hbox), 5); label = gtk_label_new_with_mnemonic (_("_Application:")); gtk_size_group_add_widget (sizegroup, label); gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0); model = gtk_list_store_new (4, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT); combo = gtk_combo_box_new_with_model (GTK_TREE_MODEL (model)); renderer = gtk_cell_renderer_pixbuf_new (); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), renderer, FALSE); gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), renderer, "icon-name", 1); gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), renderer, "width", 3); renderer = gtk_cell_renderer_text_new (); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), renderer, TRUE); gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), renderer, "text", 0); combobox = GTK_COMBO_BOX (combo); gtk_icon_size_lookup_for_settings (gtk_widget_get_settings (GTK_WIDGET (browser)), GTK_ICON_SIZE_MENU, &icon_width, NULL); for (i = 0; i < G_N_ELEMENTS (bookmark_clients); i++) { const gchar* location = bookmark_clients[i].path; const gchar* client = bookmark_clients[i].name; gchar* path = NULL; /* Interpret * as 'any folder' */ if (strchr (location, '*') != NULL) { gchar** parts = g_strsplit (location, "*", 2); GDir* dir; path = g_build_filename (g_get_home_dir (), parts[0], NULL); if ((dir = g_dir_open (path, 0, NULL))) { const gchar* name; while ((name = g_dir_read_name (dir))) { gchar* file = g_build_filename (path, name, parts[1], NULL); if (g_access (file, F_OK) == 0) { /* If name is XYZ.Name, we use Name only */ gchar* real_name = strchr (name, '.'); gchar* display = strstr (_(client), "%s") ? g_strdup_printf (_(client), real_name ? real_name + 1 : name) : g_strdup (_(client)); gtk_list_store_insert_with_values (model, NULL, G_MAXINT, 0, display, 1, bookmark_clients[i].icon, 2, file, 3, icon_width, -1); g_free (display); } g_free (file); } g_dir_close (dir); } g_free (path); g_strfreev (parts); continue; } path = g_build_filename (g_get_home_dir (), location, NULL); if (g_access (path, F_OK) == 0) gtk_list_store_insert_with_values (model, NULL, G_MAXINT, 0, _(client), 1, bookmark_clients[i].icon, 2, path, 3, icon_width, -1); g_free (path); } gtk_list_store_insert_with_values (model, NULL, G_MAXINT, 0, _("Import from XBEL or HTML file"), 1, NULL, 2, NULL, 3, icon_width, -1); gtk_combo_box_set_active (combobox, 0); gtk_box_pack_start (GTK_BOX (hbox), combo, TRUE, TRUE, 0); gtk_container_add (GTK_CONTAINER (content_area), hbox); gtk_widget_show_all (hbox); combobox_folder = midori_bookmark_folder_button_new (browser->bookmarks, FALSE, 0, 0); gtk_container_add (GTK_CONTAINER (content_area), combobox_folder); gtk_dialog_set_default_response (GTK_DIALOG (dialog), GTK_RESPONSE_ACCEPT); if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT) { GtkTreeIter iter; gchar* path = NULL; gint64 selected; GError* error; sqlite3* db = g_object_get_data (G_OBJECT (browser->bookmarks), "db"); if (gtk_combo_box_get_active_iter (combobox, &iter)) gtk_tree_model_get (GTK_TREE_MODEL (model), &iter, 2, &path, -1); selected = midori_bookmark_folder_button_get_active (combobox_folder); gtk_widget_destroy (dialog); if (!path) { GtkWidget* file_dialog; file_dialog = sokoke_file_chooser_dialog_new (_("Import from a file"), GTK_WINDOW (browser), GTK_FILE_CHOOSER_ACTION_OPEN); if (gtk_dialog_run (GTK_DIALOG (file_dialog)) == GTK_RESPONSE_OK) path = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (file_dialog)); gtk_widget_destroy (file_dialog); } error = NULL; bookmarks = katze_array_new (KATZE_TYPE_ARRAY); if (path && !midori_array_from_file (bookmarks, path, NULL, &error)) { sokoke_message_dialog (GTK_MESSAGE_ERROR, _("Failed to import bookmarks"), error ? error->message : "", FALSE); if (error) g_error_free (error); } midori_bookmarks_import_array_db (db, bookmarks, selected); katze_array_update (browser->bookmarks); g_object_unref (bookmarks); g_free (path); } else gtk_widget_destroy (dialog); } static void _action_bookmarks_export_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* file_dialog; GtkFileFilter* filter; const gchar* format; gchar* path = NULL; GError* error; KatzeArray* bookmarks; if (!browser->bookmarks || !gtk_widget_get_visible (GTK_WIDGET (browser))) return; wrong_format: file_dialog = sokoke_file_chooser_dialog_new (_("Save file as"), GTK_WINDOW (browser), GTK_FILE_CHOOSER_ACTION_SAVE); gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (file_dialog), "bookmarks.xbel"); filter = gtk_file_filter_new (); gtk_file_filter_set_name (filter, _("XBEL Bookmarks")); gtk_file_filter_add_mime_type (filter, "application/xml"); gtk_file_filter_add_pattern (filter, "*.xbel"); gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (file_dialog), filter); filter = gtk_file_filter_new (); gtk_file_filter_set_name (filter, _("Netscape Bookmarks")); gtk_file_filter_add_mime_type (filter, "text/html"); gtk_file_filter_add_pattern (filter, "*.html"); gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (file_dialog), filter); if (gtk_dialog_run (GTK_DIALOG (file_dialog)) == GTK_RESPONSE_OK) path = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (file_dialog)); gtk_widget_destroy (file_dialog); if (g_str_has_suffix (path, ".xbel")) format = "xbel"; else if (g_str_has_suffix (path, ".html")) format = "netscape"; else if (path != NULL) { sokoke_message_dialog (GTK_MESSAGE_ERROR, _("Midori can only export to XBEL (*.xbel) and Netscape (*.html)"), "", TRUE); katze_assign (path, NULL); goto wrong_format; } if (path == NULL) return; error = NULL; bookmarks = midori_array_query_recursive (browser->bookmarks, "*", "parentid IS NULL", NULL, TRUE); if (!midori_array_to_file (bookmarks, path, format, &error)) { sokoke_message_dialog (GTK_MESSAGE_ERROR, _("Failed to export bookmarks"), error ? error->message : "", FALSE); if (error) g_error_free (error); } g_object_unref (bookmarks); g_free (path); } static void _action_manage_search_engines_activate (GtkAction* action, MidoriBrowser* browser) { static GtkWidget* dialog = NULL; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; if (!dialog) { dialog = midori_search_action_get_dialog ( MIDORI_SEARCH_ACTION (_action_by_name (browser, "Search"))); g_signal_connect (dialog, "destroy", G_CALLBACK (gtk_widget_destroyed), &dialog); gtk_widget_show (dialog); } else gtk_window_present (GTK_WINDOW (dialog)); } static void midori_browser_clear_private_data_response_cb (GtkWidget* dialog, gint response_id, MidoriBrowser* browser) { if (response_id == GTK_RESPONSE_ACCEPT) { GtkToggleButton* button; gint clear_prefs = MIDORI_CLEAR_NONE; gint saved_prefs = MIDORI_CLEAR_NONE; GList* data_items = sokoke_register_privacy_item (NULL, NULL, NULL); GString* clear_data = g_string_new (NULL); g_object_get (browser->settings, "clear-private-data", &saved_prefs, NULL); button = g_object_get_data (G_OBJECT (dialog), "session"); if (gtk_toggle_button_get_active (button)) { GList* tabs = gtk_container_get_children (GTK_CONTAINER (browser->notebook)); for (; tabs != NULL; tabs = g_list_next (tabs)) gtk_widget_destroy (tabs->data); g_list_free (tabs); clear_prefs |= MIDORI_CLEAR_SESSION; } button = g_object_get_data (G_OBJECT (dialog), "history"); if (gtk_toggle_button_get_active (button)) { katze_array_clear (browser->history); katze_array_clear (browser->trash); _midori_browser_update_actions (browser); clear_prefs |= MIDORI_CLEAR_HISTORY; clear_prefs |= MIDORI_CLEAR_TRASH; /* For backward-compatibility */ } if (clear_prefs != saved_prefs) { clear_prefs |= (saved_prefs & MIDORI_CLEAR_ON_QUIT); g_object_set (browser->settings, "clear-private-data", clear_prefs, NULL); } for (; data_items != NULL; data_items = g_list_next (data_items)) { SokokePrivacyItem* privacy = data_items->data; button = g_object_get_data (G_OBJECT (dialog), privacy->name); g_return_if_fail (button != NULL && GTK_IS_TOGGLE_BUTTON (button)); if (gtk_toggle_button_get_active (button)) { privacy->clear (); g_string_append (clear_data, privacy->name); g_string_append_c (clear_data, ','); } } g_object_set (browser->settings, "clear-data", clear_data->str, NULL); g_string_free (clear_data, TRUE); } if (response_id != GTK_RESPONSE_DELETE_EVENT) gtk_widget_destroy (dialog); } static void midori_browser_clear_on_quit_toggled_cb (GtkToggleButton* button, MidoriWebSettings* settings) { gint clear_prefs = MIDORI_CLEAR_NONE; g_object_get (settings, "clear-private-data", &clear_prefs, NULL); clear_prefs ^= MIDORI_CLEAR_ON_QUIT; g_object_set (settings, "clear-private-data", clear_prefs, NULL); } static void _action_clear_private_data_activate (GtkAction* action, MidoriBrowser* browser) { static GtkWidget* dialog = NULL; if (!gtk_widget_get_visible (GTK_WIDGET (browser))) return; if (!dialog) { GtkWidget* content_area; GdkScreen* screen; GtkSizeGroup* sizegroup; GtkWidget* hbox; GtkWidget* alignment; GtkWidget* vbox; GtkWidget* icon; GtkWidget* label; GtkWidget* button; GList* data_items; gchar* clear_data = katze_object_get_string (browser->settings, "clear-data"); gint clear_prefs = MIDORI_CLEAR_NONE; g_object_get (browser->settings, "clear-private-data", &clear_prefs, NULL); /* i18n: Dialog: Clear Private Data, in the Tools menu */ dialog = gtk_dialog_new_with_buttons (_("Clear Private Data"), GTK_WINDOW (browser), GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, _("_Clear private data"), GTK_RESPONSE_ACCEPT, NULL); katze_widget_add_class (gtk_dialog_get_widget_for_response ( GTK_DIALOG (dialog), GTK_RESPONSE_ACCEPT), "noundo"); content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog)); gtk_window_set_skip_taskbar_hint (GTK_WINDOW (dialog), FALSE); screen = gtk_widget_get_screen (GTK_WIDGET (browser)); if (screen) { gtk_window_set_icon_name (GTK_WINDOW (dialog), GTK_STOCK_CLEAR); } sizegroup = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL); hbox = gtk_hbox_new (FALSE, 4); icon = gtk_image_new_from_stock (GTK_STOCK_CLEAR, GTK_ICON_SIZE_DIALOG); gtk_size_group_add_widget (sizegroup, icon); gtk_box_pack_start (GTK_BOX (hbox), icon, FALSE, FALSE, 0); label = gtk_label_new (_("Clear the following data:")); gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0); gtk_box_pack_start (GTK_BOX (content_area), hbox, FALSE, FALSE, 0); hbox = gtk_hbox_new (FALSE, 4); icon = gtk_image_new (); gtk_size_group_add_widget (sizegroup, icon); gtk_box_pack_start (GTK_BOX (hbox), icon, FALSE, FALSE, 0); vbox = gtk_vbox_new (TRUE, 4); alignment = gtk_alignment_new (0, 0, 1, 1); gtk_alignment_set_padding (GTK_ALIGNMENT (alignment), 0, 6, 12, 0); button = gtk_check_button_new_with_mnemonic (_("Last open _tabs")); if ((clear_prefs & MIDORI_CLEAR_SESSION) == MIDORI_CLEAR_SESSION) gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), TRUE); g_object_set_data (G_OBJECT (dialog), "session", button); gtk_box_pack_start (GTK_BOX (vbox), button, TRUE, TRUE, 0); /* i18n: Browsing history, visited web pages, closed tabs */ button = gtk_check_button_new_with_mnemonic (_("_History")); if ((clear_prefs & MIDORI_CLEAR_HISTORY) == MIDORI_CLEAR_HISTORY) gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), TRUE); g_object_set_data (G_OBJECT (dialog), "history", button); gtk_box_pack_start (GTK_BOX (vbox), button, TRUE, TRUE, 0); data_items = sokoke_register_privacy_item (NULL, NULL, NULL); for (; data_items != NULL; data_items = g_list_next (data_items)) { SokokePrivacyItem* privacy = data_items->data; button = gtk_check_button_new_with_mnemonic (privacy->label); gtk_box_pack_start (GTK_BOX (vbox), button, TRUE, TRUE, 0); g_object_set_data (G_OBJECT (dialog), privacy->name, button); if (clear_data && strstr (clear_data, privacy->name)) gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), TRUE); } g_free (clear_data); gtk_container_add (GTK_CONTAINER (alignment), vbox); gtk_box_pack_start (GTK_BOX (hbox), alignment, TRUE, TRUE, 0); gtk_box_pack_start (GTK_BOX (content_area), hbox, FALSE, FALSE, 0); button = gtk_check_button_new_with_mnemonic (_("Clear private data when _quitting Midori")); if ((clear_prefs & MIDORI_CLEAR_ON_QUIT) == MIDORI_CLEAR_ON_QUIT) gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), TRUE); g_signal_connect (button, "toggled", G_CALLBACK (midori_browser_clear_on_quit_toggled_cb), browser->settings); alignment = gtk_alignment_new (0, 0, 1, 1); gtk_alignment_set_padding (GTK_ALIGNMENT (alignment), 0, 0, 2, 0); gtk_container_add (GTK_CONTAINER (alignment), button); gtk_box_pack_start (GTK_BOX (content_area), alignment, FALSE, FALSE, 0); gtk_widget_show_all (content_area); g_signal_connect (dialog, "response", G_CALLBACK (midori_browser_clear_private_data_response_cb), browser); g_signal_connect (dialog, "destroy", G_CALLBACK (gtk_widget_destroyed), &dialog); gtk_widget_show (dialog); gtk_dialog_set_default_response (GTK_DIALOG (dialog), GTK_RESPONSE_ACCEPT); } else gtk_window_present (GTK_WINDOW (dialog)); } static void _action_inspect_page_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); WebKitWebView* web_view = WEBKIT_WEB_VIEW (midori_view_get_web_view (MIDORI_VIEW (view))); WebKitWebInspector* inspector = webkit_web_view_get_inspector (web_view); webkit_web_inspector_show (inspector); } static void _action_tab_move_backward_activate (GtkAction* action, MidoriBrowser* browser) { gint new_pos; gint cur_pos = midori_browser_get_current_page (browser); GtkWidget* widget = midori_browser_get_nth_tab (browser, cur_pos); if (cur_pos > 0) new_pos = cur_pos - 1; else new_pos = midori_browser_get_n_pages (browser) - 1; #ifdef HAVE_GRANITE granite_widgets_dynamic_notebook_set_tab_position ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), midori_view_get_tab (MIDORI_VIEW (widget)), new_pos); #else gtk_notebook_reorder_child (GTK_NOTEBOOK (browser->notebook), widget, new_pos); #endif g_signal_emit (browser, signals[MOVE_TAB], 0, browser->notebook, cur_pos, new_pos); } static void _action_tab_move_forward_activate (GtkAction* action, MidoriBrowser* browser) { gint new_pos; gint cur_pos = midori_browser_get_current_page (browser); GtkWidget* widget = midori_browser_get_nth_tab (browser, cur_pos); if (cur_pos == (midori_browser_get_n_pages (browser) - 1)) new_pos = 0; else new_pos = cur_pos + 1; #ifdef HAVE_GRANITE granite_widgets_dynamic_notebook_set_tab_position ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), midori_view_get_tab (MIDORI_VIEW (widget)), new_pos); #else gtk_notebook_reorder_child (GTK_NOTEBOOK (browser->notebook), widget, new_pos); #endif g_signal_emit (browser, signals[MOVE_TAB], 0, browser->notebook, cur_pos, new_pos); } static void _action_tab_previous_activate (GtkAction* action, MidoriBrowser* browser) { gint n = midori_browser_get_current_page (browser); midori_browser_set_current_page (browser, n - 1); } static void _action_tab_next_activate (GtkAction* action, MidoriBrowser* browser) { /* Advance one tab or jump to the first one if we are at the last one */ gint n = midori_browser_get_current_page (browser); if (n == midori_browser_get_n_pages (browser) - 1) n = -1; midori_browser_set_current_page (browser, n + 1); } static void _action_tab_current_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); gtk_widget_grab_focus (view); } static void _action_next_view_activate (GtkAction* action, MidoriBrowser* browser) { gtk_widget_grab_focus (midori_browser_get_current_tab (browser)); } static void _action_tab_minimize_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); g_object_set (view, "minimized", !katze_object_get_boolean (view, "minimized"), NULL); } static void _action_tab_duplicate_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); MidoriNewView where = MIDORI_NEW_VIEW_TAB; GtkWidget* new_view = midori_view_new_with_item ( midori_view_get_proxy_item (MIDORI_VIEW (view)), browser->settings); const gchar* uri = midori_view_get_display_uri (MIDORI_VIEW (view)); g_signal_emit_by_name (view, "new-view", new_view, where, TRUE); midori_view_set_uri (MIDORI_VIEW (new_view), uri); } static void midori_browser_close_other_tabs_cb (GtkWidget* view, gpointer data) { GtkWidget* remaining_view = data; if (view != remaining_view) gtk_widget_destroy (view); } static void _action_tab_close_other_activate (GtkAction* action, MidoriBrowser* browser) { GtkWidget* view = midori_browser_get_current_tab (browser); midori_browser_foreach (browser, midori_browser_close_other_tabs_cb, view); } static const gchar* credits_authors[] = { "Christian Dywan ", NULL }; static const gchar* credits_documenters[] = { "Christian Dywan ", NULL }; static const gchar* credits_artists[] = { "Nancy Runge ", NULL }; #if !GTK_CHECK_VERSION (2, 24, 0) static void _action_about_activate_link (GtkAboutDialog* about, const gchar* uri, gpointer user_data) { MidoriBrowser* browser; gint n; browser = MIDORI_BROWSER (user_data); n = midori_browser_add_uri (browser, uri); midori_browser_set_current_page (browser, n); } static void _action_about_activate_email (GtkAboutDialog* about, const gchar* uri, gpointer user_data) { /* Some email clients need the 'mailto' to function properly */ gchar* newuri = NULL; if (!g_str_has_prefix (uri, "mailto:")) newuri = g_strconcat ("mailto:", uri, NULL); sokoke_show_uri (NULL, newuri ? newuri : uri, GDK_CURRENT_TIME, NULL); g_free (newuri); } #endif static gchar* midori_browser_get_docs (gboolean error) { #ifdef G_OS_WIN32 gchar* path = midori_paths_get_data_filename ("doc/midori/faq.html", FALSE); if (g_access (path, F_OK) == 0) return g_filename_to_uri (path, NULL, NULL); else { #ifdef DOCDIR if (g_access (DOCDIR "/faq.html", F_OK) == 0) return g_strdup ("file://" DOCDIR "/faq.html"); else #endif return error ? g_strdup ("error:nodocs share/doc/midori/faq.html") : NULL; } g_free (path); #else #ifdef DOCDIR if (g_access (DOCDIR "/faq.html", F_OK) == 0) return g_strdup ("file://" DOCDIR "/faq.html"); else #endif return error ? g_strdup ("error:nodocs " DOCDIR "/faq.html") : NULL; #endif } static void _action_about_activate (GtkAction* action, MidoriBrowser* browser) { gchar* comments = g_strdup_printf ("%s\n%s", _("A lightweight web browser."), _("See about:version for version info.")); const gchar* license = _("This library is free software; you can redistribute it and/or " "modify it under the terms of the GNU Lesser General Public " "License as published by the Free Software Foundation; either " "version 2.1 of the License, or (at your option) any later version."); #if !GTK_CHECK_VERSION (2, 24, 0) gtk_about_dialog_set_email_hook (_action_about_activate_email, NULL, NULL); gtk_about_dialog_set_url_hook (_action_about_activate_link, browser, NULL); #endif #ifdef HAVE_GRANITE gchar* docs = midori_browser_get_docs (FALSE); granite_widgets_show_about_dialog (GTK_WINDOW (browser), "translate", "https://translations.xfce.org/projects/p/midori/", "bug", PACKAGE_BUGREPORT, "help", docs, "copyright", "2007-2012 Christian Dywan", #else gtk_show_about_dialog (GTK_WINDOW (browser), "wrap-license", TRUE, "copyright", "Copyright © 2007-2012 Christian Dywan", #endif "logo-icon-name", gtk_window_get_icon_name (GTK_WINDOW (browser)), "program-name", PACKAGE_NAME, "version", PACKAGE_VERSION, "comments", comments, "website", "http://www.midori-browser.org", "authors", credits_authors, "documenters", credits_documenters, "artists", credits_artists, "license", license, "translator-credits", _("translator-credits"), NULL); g_free (comments); #ifdef HAVE_GRANITE g_free (docs); #endif } static void _action_help_link_activate (GtkAction* action, MidoriBrowser* browser) { const gchar* action_name = gtk_action_get_name (action); gchar* uri = NULL; if (!strncmp ("HelpFAQ", action_name, 7)) uri = midori_browser_get_docs (TRUE); else if (!strncmp ("HelpBugs", action_name, 8)) { if (!g_spawn_command_line_async ("ubuntu-bug " PACKAGE_NAME, NULL)) uri = g_strdup (PACKAGE_BUGREPORT); } if (uri) { gint n = midori_browser_add_uri (browser, uri); midori_browser_set_current_page (browser, n); g_free (uri); } } static void _action_panel_activate (GtkToggleAction* action, MidoriBrowser* browser) { gboolean active = gtk_toggle_action_get_active (action); g_object_set (browser->settings, "show-panel", active, NULL); sokoke_widget_set_visible (browser->panel, active); } static gboolean midori_browser_panel_timeout (GtkWidget* hpaned) { gint position = gtk_paned_get_position (GTK_PANED (hpaned)); MidoriBrowser* browser = midori_browser_get_for_widget (hpaned); g_object_set (browser->settings, "last-panel-position", position, NULL); browser->panel_timeout = 0; return FALSE; } static void midori_panel_notify_position_cb (GObject* hpaned, GParamSpec* pspec, MidoriBrowser* browser) { if (!browser->panel_timeout) browser->panel_timeout = g_timeout_add_full (G_PRIORITY_LOW, 5000, (GSourceFunc)midori_browser_panel_timeout, hpaned, NULL); } static gboolean midori_panel_cycle_child_focus_cb (GtkWidget* hpaned, gboolean reversed, MidoriBrowser* browser) { /* Default cycle goes between all GtkPaned widgets. If focus is in the panel, focus the location as if it's a paned. If nothing is focussed, simply go to the location. Be sure to suppress the default because the signal can recurse. */ GtkWidget* focus = gtk_window_get_focus (GTK_WINDOW (browser)); if (gtk_widget_get_ancestor (focus, MIDORI_TYPE_PANEL) || !gtk_widget_get_ancestor (focus, GTK_TYPE_PANED)) { g_signal_stop_emission_by_name (hpaned, "cycle-child-focus"); midori_browser_activate_action (browser, "Location"); return TRUE; } return FALSE; } static void midori_panel_notify_page_cb (MidoriPanel* panel, GParamSpec* pspec, MidoriBrowser* browser) { gint page = midori_panel_get_current_page (panel); if (page > -1) g_object_set (browser->settings, "last-panel-page", page, NULL); } static void midori_panel_notify_show_titles_cb (MidoriPanel* panel, GParamSpec* pspec, MidoriBrowser* browser) { gboolean show_titles = katze_object_get_boolean (panel, "show-titles"); g_signal_handlers_block_by_func (browser->settings, midori_browser_settings_notify, browser); g_object_set (browser->settings, "compact-sidepanel", !show_titles, NULL); g_signal_handlers_unblock_by_func (browser->settings, midori_browser_settings_notify, browser); } static void midori_panel_notify_right_aligned_cb (MidoriPanel* panel, GParamSpec* pspec, MidoriBrowser* browser) { gboolean right_aligned = katze_object_get_boolean (panel, "right-aligned"); GtkWidget* hpaned = gtk_widget_get_parent (browser->panel); GtkWidget* vpaned = gtk_widget_get_parent (browser->notebook); gint paned_position = gtk_paned_get_position (GTK_PANED (hpaned)); GtkAllocation allocation; gint paned_size; gtk_widget_get_allocation (hpaned, &allocation); paned_size = allocation.width; g_object_set (browser->settings, "right-align-sidepanel", right_aligned, NULL); g_object_ref (browser->panel); g_object_ref (vpaned); gtk_container_remove (GTK_CONTAINER (hpaned), browser->panel); gtk_container_remove (GTK_CONTAINER (hpaned), vpaned); if (right_aligned) { gtk_paned_pack1 (GTK_PANED (hpaned), vpaned, TRUE, FALSE); gtk_paned_pack2 (GTK_PANED (hpaned), browser->panel, FALSE, FALSE); } else { gtk_paned_pack1 (GTK_PANED (hpaned), browser->panel, FALSE, FALSE); gtk_paned_pack2 (GTK_PANED (hpaned), vpaned, TRUE, FALSE); } gtk_paned_set_position (GTK_PANED (hpaned), paned_size - paned_position); g_object_unref (browser->panel); g_object_unref (vpaned); } static gboolean midori_panel_close_cb (MidoriPanel* panel, MidoriBrowser* browser) { _action_set_active (browser, "Panel", FALSE); return FALSE; } static void midori_browser_switched_tab (MidoriBrowser* browser, GtkWidget* old_widget, MidoriView* new_view, gint new_page) { GtkAction* action; const gchar* text; const gchar* uri; if (old_widget != NULL) { action = _action_by_name (browser, "Location"); text = midori_location_action_get_text (MIDORI_LOCATION_ACTION (action)); g_object_set_data_full (G_OBJECT (old_widget), "midori-browser-typed-text", g_strdup (text), g_free); } if (new_view == NULL) return; g_return_if_fail (MIDORI_IS_VIEW (new_view)); uri = g_object_get_data (G_OBJECT (new_view), "midori-browser-typed-text"); if (!uri) uri = midori_view_get_display_uri (new_view); midori_browser_set_title (browser, midori_view_get_display_title (new_view)); action = _action_by_name (browser, "Location"); midori_location_action_set_text (MIDORI_LOCATION_ACTION (action), uri); if (midori_paths_is_readonly () /* APP, PRIVATE */) gtk_window_set_icon (GTK_WINDOW (browser), midori_view_get_icon (new_view)); if (browser->proxy_array) katze_item_set_meta_integer (KATZE_ITEM (browser->proxy_array), "current", new_page); g_object_notify (G_OBJECT (browser), "tab"); _midori_browser_set_statusbar_text (browser, new_view, NULL); _midori_browser_update_interface (browser, new_view); _midori_browser_update_progress (browser, new_view); } static void midori_browser_notebook_page_reordered_cb (GtkWidget* notebook, MidoriView* view, guint page_num, MidoriBrowser* browser) { KatzeItem* item = midori_view_get_proxy_item (view); katze_array_move_item (browser->proxy_array, item, page_num); g_object_notify (G_OBJECT (browser), "tab"); } static GtkWidget* midori_browser_notebook_create_window_cb (GtkWidget* notebook, GtkWidget* view, gint x, gint y, MidoriBrowser* browser) { MidoriBrowser* new_browser; g_signal_emit (browser, signals[NEW_WINDOW], 0, NULL, &new_browser); if (new_browser) { GtkWidget* new_notebook = new_browser->notebook; gtk_window_move (GTK_WINDOW (new_browser), x, y); return new_notebook; } else /* No MidoriApp, so this is app or private mode */ return NULL; } #ifdef HAVE_GRANITE static void midori_browser_notebook_tab_added_cb (GtkWidget* notebook, GraniteWidgetsTab* tab, MidoriBrowser* browser) { GtkWidget* view = midori_view_new_with_item (NULL, browser->settings); midori_view_set_tab (MIDORI_VIEW (view), tab); midori_browser_connect_tab (browser, view); /* FIXME: signal add-tab */ _midori_browser_update_actions (browser); midori_view_set_uri (MIDORI_VIEW (view), ""); } static gboolean midori_browser_notebook_tab_removed_cb (GtkWidget* notebook, GraniteWidgetsTab* tab, MidoriBrowser* browser) { MidoriView* view = MIDORI_VIEW (granite_widgets_tab_get_page (tab)); g_signal_emit (browser, signals[REMOVE_TAB], 0, view); gtk_widget_destroy (view); return TRUE; } static void midori_browser_notebook_tab_switched_cb (GraniteWidgetsDynamicNotebook* notebook, GraniteWidgetsTab* old_tab, GraniteWidgetsTab* new_tab, MidoriBrowser* browser) { gint new_pos = granite_widgets_dynamic_notebook_get_tab_position (notebook, new_tab); midori_browser_switched_tab (browser, old_tab ? granite_widgets_tab_get_page (old_tab) : NULL, MIDORI_VIEW (granite_widgets_tab_get_page (new_tab)), new_pos); } static void midori_browser_notebook_tab_moved_cb (GtkWidget* notebook, GraniteWidgetsTab* tab, gint old_pos, gboolean new_window, gint x, gint y, MidoriBrowser* browser) { GtkWidget* view = granite_widgets_tab_get_page (tab); if (new_window) { GtkWidget* notebook = midori_browser_notebook_create_window_cb ( browser->notebook, view, x, y, browser); if (notebook != NULL) { g_object_ref (tab); granite_widgets_dynamic_notebook_remove_tab ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), tab); granite_widgets_dynamic_notebook_insert_tab ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (notebook), tab, 0); g_object_unref (tab); } } else { gint new_pos = granite_widgets_dynamic_notebook_get_tab_position ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (notebook), tab); midori_browser_notebook_page_reordered_cb (notebook, MIDORI_VIEW (view), new_pos, browser); } } #else static void midori_browser_notebook_switch_page_cb (GtkWidget* notebook, gpointer page, guint page_num, MidoriBrowser* browser) { midori_browser_switched_tab (browser, midori_browser_get_current_tab (browser), MIDORI_VIEW (midori_browser_get_nth_tab (browser, page_num)), page_num); } static void midori_browser_notebook_page_removed_cb (GtkWidget* notebook, GtkWidget* view, guint page_num, MidoriBrowser* browser) { _midori_browser_remove_tab (browser, view); midori_browser_notebook_size_allocate_cb (browser->notebook, NULL, browser); } static gboolean midori_browser_notebook_reorder_tab_cb (GtkNotebook* notebook, GtkDirectionType arg1, gboolean arg2, gpointer user_data) { g_signal_stop_emission_by_name (notebook, "reorder-tab"); return TRUE; } static void midori_browser_menu_item_switch_tab_cb (GtkWidget* menuitem, MidoriBrowser* browser) { gint page = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (menuitem), "index")); midori_browser_set_current_page (browser, page); } static gboolean midori_browser_notebook_button_press_event_after_cb (GtkNotebook* notebook, GdkEventButton* event, MidoriBrowser* browser) { #if !GTK_CHECK_VERSION(3,0,0) /* TODO */ if (event->window != notebook->event_window) return FALSE; #endif /* FIXME: Handle double click only when it wasn't handled by GtkNotebook */ /* Open a new tab on double click or middle mouse click */ if (/*(event->type == GDK_2BUTTON_PRESS && event->button == 1) || */(event->type == GDK_BUTTON_PRESS && MIDORI_EVENT_NEW_TAB (event))) { gint n = midori_browser_add_uri (browser, ""); midori_browser_set_current_page (browser, n); return TRUE; } else if (event->type == GDK_BUTTON_PRESS && MIDORI_EVENT_CONTEXT_MENU (event)) { GtkWidget* menu = gtk_menu_new (); GList* tabs = gtk_container_get_children (GTK_CONTAINER (notebook)); GtkWidget* menuitem = sokoke_action_create_popup_menu_item ( gtk_action_group_get_action (browser->action_group, "TabNew")); gint i = 0; gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); menuitem = sokoke_action_create_popup_menu_item ( gtk_action_group_get_action (browser->action_group, "UndoTabClose")); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); menuitem = gtk_separator_menu_item_new (); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); for (; tabs != NULL; tabs = g_list_next (tabs)) { const gchar* title = midori_view_get_display_title (tabs->data); menuitem = katze_image_menu_item_new_ellipsized (title); gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (menuitem), gtk_image_new_from_pixbuf (midori_view_get_icon (tabs->data))); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem); g_object_set_data (G_OBJECT (menuitem), "index", GINT_TO_POINTER (i)); g_signal_connect (menuitem, "activate", G_CALLBACK (midori_browser_menu_item_switch_tab_cb), browser); i++; } g_list_free (tabs); gtk_widget_show_all (menu); katze_widget_popup (GTK_WIDGET (notebook), GTK_MENU (menu), NULL, KATZE_MENU_POSITION_CURSOR); } return FALSE; } #endif static void _action_undo_tab_close_activate (GtkAction* action, MidoriBrowser* browser) { guint last; KatzeItem* item; if (!browser->trash) return; /* Reopen the most recent trash item */ last = katze_array_get_length (browser->trash) - 1; item = katze_array_get_nth_item (browser->trash, last); midori_browser_set_current_page (browser, midori_browser_restore_tab (browser, item)); } static void _action_trash_empty_activate (GtkAction* action, MidoriBrowser* browser) { if (browser->trash) { katze_array_clear (browser->trash); _midori_browser_update_actions (browser); } } static const GtkActionEntry entries[] = { { "File", NULL, N_("_File") }, { "WindowNew", STOCK_WINDOW_NEW, N_("New _Window"), "n", N_("Open a new window"), G_CALLBACK (_action_window_new_activate) }, { "TabNew", STOCK_TAB_NEW, NULL, "t", N_("Open a new tab"), G_CALLBACK (_action_tab_new_activate) }, { "PrivateBrowsing", NULL, N_("New P_rivate Browsing Window"), "n", NULL, G_CALLBACK (_action_private_browsing_activate), }, { "Open", GTK_STOCK_OPEN, NULL, "o", N_("Open a file"), G_CALLBACK (_action_open_activate) }, { "SaveAs", GTK_STOCK_SAVE_AS, N_("_Save Page As…"), "s", N_("Save to a file"), G_CALLBACK (_action_save_as_activate) }, { "AddSpeedDial", NULL, N_("Add to Speed _dial"), "h", NULL, G_CALLBACK (_action_add_speed_dial_activate) }, { "AddDesktopShortcut", NULL, #if HAVE_HILDON N_("Add Shortcut to the _desktop"), "", #else N_("Create _Launcher"), "", #endif NULL, G_CALLBACK (_action_add_desktop_shortcut_activate) }, { "AddNewsFeed", NULL, N_("Subscribe to News _feed"), NULL, NULL, G_CALLBACK (_action_add_news_feed_activate) }, { "CompactAdd", GTK_STOCK_ADD, NULL, NULL, NULL, G_CALLBACK (_action_compact_add_activate) }, { "TabClose", GTK_STOCK_CLOSE, N_("_Close Tab"), "w", N_("Close the current tab"), G_CALLBACK (_action_tab_close_activate) }, { "WindowClose", NULL, N_("C_lose Window"), "w", NULL, G_CALLBACK (_action_window_close_activate) }, #ifdef HAVE_GRANITE { "Print", "document-export", N_("_Share"), "p", NULL, G_CALLBACK (_action_print_activate) }, #else { "Print", GTK_STOCK_PRINT, NULL, "p", N_("Print the current page"), G_CALLBACK (_action_print_activate) }, #endif { "Quit", GTK_STOCK_QUIT, N_("Close a_ll Windows"), "q", NULL, G_CALLBACK (_action_quit_activate) }, { "Edit", NULL, N_("_Edit"), NULL, NULL, G_CALLBACK (_action_edit_activate) }, { "Undo", GTK_STOCK_UNDO, NULL, "z", NULL, G_CALLBACK (_action_undo_activate) }, { "Redo", GTK_STOCK_REDO, NULL, "z", NULL, G_CALLBACK (_action_redo_activate) }, { "Cut", GTK_STOCK_CUT, NULL, "x", NULL, G_CALLBACK (_action_cut_activate) }, { "Copy", GTK_STOCK_COPY, NULL, "c", NULL, G_CALLBACK (_action_copy_activate) }, { "Paste", GTK_STOCK_PASTE, NULL, "v", NULL, G_CALLBACK (_action_paste_activate) }, { "Delete", GTK_STOCK_DELETE, NULL, NULL, NULL, G_CALLBACK (_action_delete_activate) }, { "SelectAll", GTK_STOCK_SELECT_ALL, NULL, "a", NULL, G_CALLBACK (_action_select_all_activate) }, { "Find", GTK_STOCK_FIND, N_("_Find…"), "f", N_("Find a word or phrase in the page"), G_CALLBACK (_action_find_activate) }, { "FindNext", GTK_STOCK_GO_FORWARD, N_("Find _Next"), "g", NULL, G_CALLBACK (_action_find_next_activate) }, { "FindPrevious", GTK_STOCK_GO_BACK, N_("Find _Previous"), "g", NULL, G_CALLBACK (_action_find_previous_activate) }, { "Preferences", GTK_STOCK_PREFERENCES, NULL, "p", N_("Configure the application preferences"), G_CALLBACK (_action_preferences_activate) }, { "View", NULL, N_("_View") }, { "Toolbars", NULL, N_("_Toolbars") }, { "Reload", GTK_STOCK_REFRESH, NULL, "r", N_("Reload the current page"), G_CALLBACK (_action_reload_stop_activate) }, { "ReloadUncached", GTK_STOCK_REFRESH, N_("Reload page without caching"), "r", NULL, G_CALLBACK (_action_reload_stop_activate) }, { "Stop", GTK_STOCK_STOP, NULL, "Escape", N_("Stop loading the current page"), G_CALLBACK (_action_reload_stop_activate) }, { "ReloadStop", GTK_STOCK_STOP, NULL, "", N_("Reload the current page"), G_CALLBACK (_action_reload_stop_activate) }, { "ZoomIn", GTK_STOCK_ZOOM_IN, NULL, "plus", N_("Increase the zoom level"), G_CALLBACK (_action_zoom_activate) }, { "ZoomOut", GTK_STOCK_ZOOM_OUT, NULL, "minus", N_("Decrease the zoom level"), G_CALLBACK (_action_zoom_activate) }, { "ZoomNormal", GTK_STOCK_ZOOM_100, NULL, "0", NULL, G_CALLBACK (_action_zoom_activate) }, { "Encoding", NULL, N_("_Encoding") }, { "SourceView", NULL, N_("View So_urce"), "U", NULL, G_CALLBACK (_action_source_view_activate) }, { "CaretBrowsing", NULL, N_("Ca_ret Browsing"), "F7", NULL, G_CALLBACK (_action_caret_browsing_activate) }, { "Fullscreen", GTK_STOCK_FULLSCREEN, NULL, "F11", N_("Toggle fullscreen view"), G_CALLBACK (_action_fullscreen_activate) }, { "ScrollLeft", NULL, N_("Scroll _Left"), "h", NULL, G_CALLBACK (_action_scroll_somewhere_activate) }, { "ScrollDown", NULL, N_("Scroll _Down"), "j", NULL, G_CALLBACK (_action_scroll_somewhere_activate) }, { "ScrollUp", NULL, N_("Scroll _Up"), "k", NULL, G_CALLBACK (_action_scroll_somewhere_activate) }, { "ScrollRight", NULL, N_("Scroll _Right"), "l", NULL, G_CALLBACK (_action_scroll_somewhere_activate) }, { "Readable", NULL, N_("_Readable"), "R", NULL, G_CALLBACK (_action_readable_activate) }, { "Go", NULL, N_("_Go") }, { "Back", GTK_STOCK_GO_BACK, NULL, "Left", N_("Go back to the previous page"), G_CALLBACK (_action_navigation_activate) }, { "Forward", GTK_STOCK_GO_FORWARD, NULL, "Right", N_("Go forward to the next page"), G_CALLBACK (_action_navigation_activate) }, { "Previous", GTK_STOCK_MEDIA_PREVIOUS, NULL, "Left", /* i18n: Visit the previous logical page, ie. in a forum or blog */ N_("Go to the previous sub-page"), G_CALLBACK (_action_navigation_activate) }, { "Next", GTK_STOCK_MEDIA_NEXT, NULL, "Right", /* i18n: Visit the following logical page, ie. in a forum or blog */ N_("Go to the next sub-page"), G_CALLBACK (_action_navigation_activate) }, { "NextForward", GTK_STOCK_MEDIA_NEXT, NULL, "", N_("Go to the next sub-page"), G_CALLBACK (_action_navigation_activate) }, { "Homepage", GTK_STOCK_HOME, N_("_Homepage"), "Home", N_("Go to your homepage"), G_CALLBACK (_action_navigation_activate) }, { "TrashEmpty", GTK_STOCK_CLEAR, N_("Empty Trash"), "", NULL, G_CALLBACK (_action_trash_empty_activate) }, { "UndoTabClose", GTK_STOCK_UNDELETE, N_("Undo _Close Tab"), "t", NULL, G_CALLBACK (_action_undo_tab_close_activate) }, { "BookmarkAdd", STOCK_BOOKMARK_ADD, NULL, "d", N_("Add a new bookmark"), G_CALLBACK (_action_bookmark_add_activate) }, { "BookmarkFolderAdd", NULL, N_("Add a new _folder"), "", NULL, G_CALLBACK (_action_bookmark_add_activate) }, { "BookmarksImport", NULL, N_("_Import bookmarks"), "", NULL, G_CALLBACK (_action_bookmarks_import_activate) }, { "BookmarksExport", NULL, N_("_Export bookmarks"), "", NULL, G_CALLBACK (_action_bookmarks_export_activate) }, { "ManageSearchEngines", GTK_STOCK_PROPERTIES, N_("_Manage Search Engines"), "s", NULL, G_CALLBACK (_action_manage_search_engines_activate) }, { "ClearPrivateData", NULL, N_("_Clear Private Data"), "Delete", NULL, G_CALLBACK (_action_clear_private_data_activate) }, { "InspectPage", NULL, N_("_Inspect Page"), "i", NULL, G_CALLBACK (_action_inspect_page_activate) }, { "TabPrevious", GTK_STOCK_GO_BACK, N_("_Previous Tab"), "Page_Up", NULL, G_CALLBACK (_action_tab_previous_activate) }, { "TabNext", GTK_STOCK_GO_FORWARD, N_("_Next Tab"), "Page_Down", NULL, G_CALLBACK (_action_tab_next_activate) }, { "TabMoveBackward", NULL, N_("Move Tab _Backward"), "Page_Up", NULL, G_CALLBACK (_action_tab_move_backward_activate) }, { "TabMoveForward", NULL, N_("_Move Tab Forward"), "Page_Down", NULL, G_CALLBACK (_action_tab_move_forward_activate) }, { "TabCurrent", NULL, N_("Focus _Current Tab"), "Home", NULL, G_CALLBACK (_action_tab_current_activate) }, { "NextView", NULL, N_("Focus _Next view"), "F6", NULL, G_CALLBACK (_action_next_view_activate) }, { "TabMinimize", NULL, N_("Only show the Icon of the _Current Tab"), "", NULL, G_CALLBACK (_action_tab_minimize_activate) }, { "TabDuplicate", NULL, N_("_Duplicate Current Tab"), "", NULL, G_CALLBACK (_action_tab_duplicate_activate) }, { "TabCloseOther", NULL, N_("Close Ot_her Tabs"), "", NULL, G_CALLBACK (_action_tab_close_other_activate) }, { "LastSession", NULL, N_("Open last _session"), NULL, NULL, NULL }, { "Help", NULL, N_("_Help") }, { "HelpFAQ", GTK_STOCK_HELP, N_("_Frequent Questions"), "F1", NULL, G_CALLBACK (_action_help_link_activate) }, { "HelpBugs", NULL, N_("_Report a Problem…"), NULL, NULL, G_CALLBACK (_action_help_link_activate) }, { "About", GTK_STOCK_ABOUT, NULL, "", NULL, G_CALLBACK (_action_about_activate) }, { "Dummy", NULL, N_("_Tools") }, }; static const guint entries_n = G_N_ELEMENTS (entries); static const GtkToggleActionEntry toggle_entries[] = { { "Menubar", NULL, N_("_Menubar"), "", NULL, G_CALLBACK (_action_menubar_activate), FALSE }, { "Navigationbar", NULL, N_("_Navigationbar"), "", NULL, G_CALLBACK (_action_navigationbar_activate), FALSE }, { "Panel", GTK_STOCK_INDENT, N_("Side_panel"), "F9", N_("Sidepanel"), G_CALLBACK (_action_panel_activate), FALSE }, { "Bookmarkbar", NULL, N_("_Bookmarkbar"), "", NULL, G_CALLBACK (_action_bookmarkbar_activate), FALSE }, { "Statusbar", NULL, N_("_Statusbar"), "j", NULL, G_CALLBACK (_action_statusbar_activate), FALSE }, }; static const guint toggle_entries_n = G_N_ELEMENTS (toggle_entries); static const GtkRadioActionEntry encoding_entries[] = { { "EncodingAutomatic", NULL, N_("_Automatic"), "", NULL, 1 }, { "EncodingChinese", NULL, N_("Chinese Traditional (BIG5)"), "", NULL, 1 }, { "EncodingChineseSimplified", NULL, N_("Chinese Simplified (GB18030)"), "", NULL, 1 }, { "EncodingJapanese", NULL, /* i18n: A double underscore "__" is used to prevent the mnemonic */ N_("Japanese (SHIFT__JIS)"), "", NULL, 1 }, { "EncodingKorean", NULL, N_("Korean (EUC-KR)"), "", NULL, 1 }, { "EncodingRussian", NULL, N_("Russian (KOI8-R)"), "", NULL, 1 }, { "EncodingUnicode", NULL, N_("Unicode (UTF-8)"), "", NULL, 1 }, { "EncodingWestern", NULL, N_("Western (ISO-8859-1)"), "", NULL, 1 }, { "EncodingCustom", NULL, N_("Custom…"), "", NULL, 1 }, }; static const guint encoding_entries_n = G_N_ELEMENTS (encoding_entries); static void midori_browser_window_state_event_cb (MidoriBrowser* browser, GdkEventWindowState* event) { MidoriWindowState window_state = MIDORI_WINDOW_NORMAL; if (event->new_window_state & GDK_WINDOW_STATE_ICONIFIED) window_state = MIDORI_WINDOW_MINIMIZED; else if (event->new_window_state & GDK_WINDOW_STATE_MAXIMIZED) window_state = MIDORI_WINDOW_MAXIMIZED; else if (event->new_window_state & GDK_WINDOW_STATE_FULLSCREEN) window_state = MIDORI_WINDOW_FULLSCREEN; g_object_set (browser->settings, "last-window-state", window_state, NULL); } static gboolean midori_browser_alloc_timeout (MidoriBrowser* browser) { GtkWidget* widget = GTK_WIDGET (browser); GdkWindowState state = gdk_window_get_state (gtk_widget_get_window (widget)); if (!(state & (GDK_WINDOW_STATE_MAXIMIZED | GDK_WINDOW_STATE_FULLSCREEN))) { GtkAllocation allocation; gtk_widget_get_allocation (widget, &allocation); if (allocation.width != browser->last_window_width) { browser->last_window_width = allocation.width; g_object_set (browser->settings, "last-window-width", browser->last_window_width, NULL); } if (allocation.height != browser->last_window_height) { browser->last_window_height = allocation.height; g_object_set (browser->settings, "last-window-height", allocation.height, NULL); } } browser->alloc_timeout = 0; return FALSE; } static void midori_browser_size_allocate_cb (MidoriBrowser* browser, GtkAllocation* allocation) { GtkWidget* widget = GTK_WIDGET (browser); if (!browser->alloc_timeout && gtk_widget_get_realized (widget)) { gpointer last_page; if ((last_page = g_object_get_data (G_OBJECT (browser), "last-page"))) { midori_panel_set_current_page (MIDORI_PANEL (browser->panel), GPOINTER_TO_INT (last_page)); g_object_set_data (G_OBJECT (browser), "last-page", NULL); } browser->alloc_timeout = g_timeout_add_full (G_PRIORITY_LOW, 5000, (GSourceFunc)midori_browser_alloc_timeout, browser, NULL); } } static void midori_browser_destroy_cb (MidoriBrowser* browser) { g_object_set_data (G_OBJECT (browser), "midori-browser-destroyed", (void*)1); if (G_UNLIKELY (browser->panel_timeout)) g_source_remove (browser->panel_timeout); if (G_UNLIKELY (browser->alloc_timeout)) g_source_remove (browser->alloc_timeout); /* Destroy panel first, so panels don't need special care */ gtk_widget_destroy (browser->panel); #ifndef HAVE_GRANITE g_signal_handlers_disconnect_by_func (browser->notebook, midori_browser_notebook_reorder_tab_cb, NULL); g_signal_handlers_disconnect_by_func (browser->notebook, midori_browser_notebook_size_allocate_cb, browser); #endif /* Destroy tabs second, so child widgets don't need special care */ gtk_container_foreach (GTK_CONTAINER (browser->notebook), (GtkCallback) gtk_widget_destroy, NULL); } static const gchar* ui_markup = "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" #ifndef G_OS_WIN32 "" "" #endif "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" /* For accelerators to work all actions need to be used *somewhere* in the UI definition */ /* These also show up in Unity's HUD */ "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" "" ""; static void midori_browser_realize_cb (GtkStyle* style, MidoriBrowser* browser) { GdkScreen* screen; GtkIconTheme* icon_theme; #ifdef HAVE_HILDON_2_2 /* hildon_gtk_window_enable_zoom_keys */ guint32 set = 1; gdk_property_change (gtk_widget_get_window (GTK_WIDGET (browser)), gdk_atom_intern ("_HILDON_ZOOM_KEY_ATOM", FALSE), gdk_x11_xatom_to_atom (XA_INTEGER), 32, GDK_PROP_MODE_REPLACE, (const guchar *) &set, 1); #endif screen = gtk_widget_get_screen (GTK_WIDGET (browser)); if (screen) { icon_theme = gtk_icon_theme_get_for_screen (screen); if (gtk_icon_theme_has_icon (icon_theme, "midori")) gtk_window_set_icon_name (GTK_WINDOW (browser), "midori"); else gtk_window_set_icon_name (GTK_WINDOW (browser), STOCK_WEB_BROWSER); } } static void midori_browser_new_history_item (MidoriBrowser* browser, KatzeItem* item) { time_t now; gint64 day; sqlite3* db; static sqlite3_stmt* stmt = NULL; g_return_if_fail (katze_item_get_uri (item) != NULL); now = time (NULL); katze_item_set_added (item, now); day = sokoke_time_t_to_julian (&now); db = g_object_get_data (G_OBJECT (browser->history), "db"); g_return_if_fail (db != NULL); if (!stmt) { const gchar* sqlcmd; sqlcmd = "INSERT INTO history (uri, title, date, day) VALUES (?,?,?,?)"; sqlite3_prepare_v2 (db, sqlcmd, -1, &stmt, NULL); } sqlite3_bind_text (stmt, 1, katze_item_get_uri (item), -1, 0); sqlite3_bind_text (stmt, 2, katze_item_get_name (item), -1, 0); sqlite3_bind_int64 (stmt, 3, katze_item_get_added (item)); sqlite3_bind_int64 (stmt, 4, day); if (sqlite3_step (stmt) != SQLITE_DONE) g_printerr (_("Failed to insert new history item: %s\n"), sqlite3_errmsg (db)); sqlite3_reset (stmt); sqlite3_clear_bindings (stmt); /* FIXME: Workaround for the lack of a database interface */ katze_array_add_item (browser->history, item); katze_array_remove_item (browser->history, item); } static void midori_browser_set_history (MidoriBrowser* browser, KatzeArray* history) { if (browser->history == history) return; if (history) g_object_ref (history); katze_object_assign (browser->history, history); if (!history) return; g_object_set (_action_by_name (browser, "Location"), "history", browser->history, NULL); } static void midori_browser_accel_switch_tab_activate_cb (GtkAccelGroup* accel_group, GObject* acceleratable, guint keyval, GdkModifierType modifiers) { GtkAccelGroupEntry* entry; if ((entry = gtk_accel_group_query (accel_group, keyval, modifiers, NULL))) { gint n; MidoriBrowser* browser; GtkWidget* view; /* Switch to n-th tab. 9 and 0 go to the last tab. */ n = keyval - GDK_KEY_0; browser = g_object_get_data (G_OBJECT (accel_group), "midori-browser"); if ((view = midori_browser_get_nth_tab (browser, n < 9 ? n - 1 : -1))) midori_browser_set_current_tab (browser, view); } } #ifdef HAVE_HILDON_2_2 static void midori_browser_set_portrait_mode (MidoriBrowser* browser, gboolean portrait) { if (portrait) hildon_gtk_window_set_portrait_flags (GTK_WINDOW (browser), HILDON_PORTRAIT_MODE_REQUEST); else hildon_gtk_window_set_portrait_flags (GTK_WINDOW (browser), ~HILDON_PORTRAIT_MODE_REQUEST); _action_set_visible (browser, "Tools", !portrait); _action_set_visible (browser, "CompactAdd", !portrait); _action_set_visible (browser, "Back", !portrait); _action_set_visible (browser, "SourceView", !portrait); _action_set_visible (browser, "Fullscreen", !portrait); } static DBusHandlerResult midori_browser_mce_filter_cb (DBusConnection* connection, DBusMessage* message, gpointer data) { if (dbus_message_is_signal (message, MCE_SIGNAL_IF, MCE_DEVICE_ORIENTATION_SIG)) { DBusError error; char *rotation, *stand, *face; int x, y, z; dbus_error_init (&error); if (dbus_message_get_args (message, &error, DBUS_TYPE_STRING, &rotation, DBUS_TYPE_STRING, &stand, DBUS_TYPE_STRING, &face, DBUS_TYPE_INT32, &x, DBUS_TYPE_INT32, &y, DBUS_TYPE_INT32, &z, DBUS_TYPE_INVALID)) { gboolean portrait = !strcmp (rotation, MCE_ORIENTATION_PORTRAIT); midori_browser_set_portrait_mode (MIDORI_BROWSER (data), portrait); } else { g_warning ("%s: %s\n", error.name, error.message); dbus_error_free (&error); } } return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; } #endif static void midori_browser_add_actions (MidoriBrowser* browser) { /* 0,053 versus 0,002 compared to gtk_action_group_add_ API */ guint i; GSList* group = NULL; for (i = 0; i < G_N_ELEMENTS (entries); i++) { GtkActionEntry entry = entries[i]; GtkAction* action = gtk_action_new (entry.name, _(entry.label), _(entry.tooltip), entry.stock_id); if (entry.callback) g_signal_connect (action, "activate", entry.callback, browser); gtk_action_group_add_action_with_accel (browser->action_group, GTK_ACTION (action), entry.accelerator); } for (i = 0; i < G_N_ELEMENTS (toggle_entries); i++) { GtkToggleActionEntry entry = toggle_entries[i]; GtkToggleAction* action = gtk_toggle_action_new (entry.name, _(entry.label), _(entry.tooltip), entry.stock_id); if (entry.is_active) gtk_toggle_action_set_active (action, TRUE); if (entry.callback) g_signal_connect (action, "activate", entry.callback, browser); gtk_action_group_add_action_with_accel (browser->action_group, GTK_ACTION (action), entry.accelerator); } for (i = 0; i < G_N_ELEMENTS (encoding_entries); i++) { GtkRadioActionEntry entry = encoding_entries[i]; GtkRadioAction* action = gtk_radio_action_new (entry.name, _(entry.label), _(entry.tooltip), entry.stock_id, entry.value); if (i == 0) { group = gtk_radio_action_get_group (action); gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action), TRUE); g_signal_connect (action, "changed", G_CALLBACK (_action_view_encoding_activate), browser); } else { gtk_radio_action_set_group (action, group); group = gtk_radio_action_get_group (action); } gtk_action_group_add_action_with_accel (browser->action_group, GTK_ACTION (action), entry.accelerator); } } static void midori_browser_init (MidoriBrowser* browser) { GtkWidget* vbox; GtkUIManager* ui_manager; GtkAccelGroup* accel_group; guint i; GClosure* accel_closure; GError* error; GtkAction* action; #if !HAVE_HILDON GtkWidget* menuitem; #endif GtkWidget* homepage; GtkWidget* back; GtkWidget* forward; GtkSettings* gtk_settings; GtkWidget* hpaned; GtkWidget* vpaned; GtkWidget* scrolled; KatzeArray* dummy_array; browser->settings = midori_web_settings_new (); browser->proxy_array = katze_array_new (KATZE_TYPE_ARRAY); browser->bookmarks = NULL; browser->trash = NULL; browser->search_engines = NULL; browser->dial = NULL; /* Setup the window metrics */ g_signal_connect (browser, "realize", G_CALLBACK (midori_browser_realize_cb), browser); g_signal_connect (browser, "window-state-event", G_CALLBACK (midori_browser_window_state_event_cb), NULL); g_signal_connect (browser, "size-allocate", G_CALLBACK (midori_browser_size_allocate_cb), NULL); g_signal_connect (browser, "destroy", G_CALLBACK (midori_browser_destroy_cb), NULL); gtk_window_set_role (GTK_WINDOW (browser), "browser"); gtk_window_set_icon_name (GTK_WINDOW (browser), STOCK_WEB_BROWSER); #if GTK_CHECK_VERSION (3, 4, 0) gtk_window_set_hide_titlebar_when_maximized (GTK_WINDOW (browser), TRUE); #endif vbox = gtk_vbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (browser), vbox); gtk_widget_show (vbox); /* Let us see some ui manager magic */ browser->action_group = gtk_action_group_new ("Browser"); gtk_action_group_set_translation_domain (browser->action_group, GETTEXT_PACKAGE); midori_browser_add_actions (browser); ui_manager = gtk_ui_manager_new (); accel_group = gtk_ui_manager_get_accel_group (ui_manager); gtk_window_add_accel_group (GTK_WINDOW (browser), accel_group); gtk_ui_manager_insert_action_group (ui_manager, browser->action_group, 0); g_object_set_data (G_OBJECT (accel_group), "midori-browser", browser); accel_closure = g_cclosure_new (G_CALLBACK ( midori_browser_accel_switch_tab_activate_cb), browser, NULL); for (i = 0; i < 10; i++) { gchar* accel_path = g_strdup_printf ("/Browser/SwitchTab%d", i); gtk_accel_map_add_entry (accel_path, GDK_KEY_0 + i, GDK_MOD1_MASK); gtk_accel_group_connect_by_path (accel_group, accel_path, accel_closure); g_free (accel_path); } g_closure_unref (accel_closure); error = NULL; if (!gtk_ui_manager_add_ui_from_string (ui_manager, ui_markup, -1, &error)) { g_message ("User interface couldn't be created: %s", error->message); g_error_free (error); } /* Hide the 'Dummy' which only holds otherwise unused actions */ _action_set_visible (browser, "Dummy", FALSE); action = g_object_new (KATZE_TYPE_SEPARATOR_ACTION, "name", "Separator", "label", _("_Separator"), NULL); gtk_action_group_add_action (browser->action_group, action); g_object_unref (action); action = g_object_new (MIDORI_TYPE_LOCATION_ACTION, "name", "Location", "label", _("_Location…"), "stock-id", GTK_STOCK_JUMP_TO, "tooltip", _("Open a particular location"), NULL); g_object_connect (action, "signal::activate", _action_location_activate, browser, "signal::focus-in", _action_location_focus_in, browser, "signal::focus-out", _action_location_focus_out, browser, "signal::reset-uri", _action_location_reset_uri, browser, "signal::submit-uri", _action_location_submit_uri, browser, "signal-after::secondary-icon-released", _action_location_secondary_icon_released, browser, NULL); gtk_action_group_add_action_with_accel (browser->action_group, action, "L"); g_object_unref (action); action = g_object_new (MIDORI_TYPE_SEARCH_ACTION, "name", "Search", "label", _("_Web Search…"), "stock-id", GTK_STOCK_FIND, "tooltip", _("Run a web search"), NULL); g_object_connect (action, "signal::activate", _action_search_activate, browser, "signal::submit", _action_search_submit, browser, "signal::focus-out", _action_search_focus_out, browser, "signal::notify::current-item", _action_search_notify_current_item, browser, "signal::notify::default-item", _action_search_notify_default_item, browser, NULL); gtk_action_group_add_action_with_accel (browser->action_group, action, "K"); g_object_unref (action); action = g_object_new (MIDORI_TYPE_PANED_ACTION, "name", "LocationSearch", NULL); gtk_action_group_add_action (browser->action_group, action); g_object_unref (action); action = g_object_new (KATZE_TYPE_ARRAY_ACTION, "name", "Trash", "stock-id", STOCK_USER_TRASH, "tooltip", _("Reopen a previously closed tab or window"), NULL); g_object_connect (action, "signal::populate-popup", _action_trash_populate_popup, browser, "signal::activate-item-alt", _action_trash_activate_item_alt, browser, NULL); gtk_action_group_add_action_with_accel (browser->action_group, action, ""); g_object_unref (action); dummy_array = katze_array_new (KATZE_TYPE_ARRAY); katze_array_update (dummy_array); action = g_object_new (KATZE_TYPE_ARRAY_ACTION, "name", "Bookmarks", "label", _("_Bookmarks"), "stock-id", STOCK_BOOKMARKS, "tooltip", _("Show the saved bookmarks"), "array", dummy_array /* updated, unique */, NULL); g_object_connect (action, "signal::populate-folder", _action_bookmarks_populate_folder, browser, "signal::activate-item-alt", midori_bookmarkbar_activate_item_alt, browser, NULL); gtk_action_group_add_action_with_accel (browser->action_group, action, ""); g_object_unref (action); g_object_unref (dummy_array); dummy_array = katze_array_new (KATZE_TYPE_ITEM); katze_array_update (dummy_array); action = g_object_new (KATZE_TYPE_ARRAY_ACTION, "name", "Tools", "label", _("_Tools"), "stock-id", GTK_STOCK_PREFERENCES, "array", dummy_array /* updated, unique */, NULL); g_object_connect (action, "signal::populate-popup", _action_tools_populate_popup, browser, "signal::activate-item-alt", midori_bookmarkbar_activate_item_alt, browser, NULL); gtk_action_group_add_action (browser->action_group, action); g_object_unref (action); g_object_unref (dummy_array); action = g_object_new (KATZE_TYPE_ARRAY_ACTION, "name", "Window", "label", _("_Window"), "stock-id", GTK_STOCK_INDEX, "tooltip", _("Show a list of all open tabs"), "array", browser->proxy_array, NULL); g_object_connect (action, "signal::populate-popup", _action_window_populate_popup, browser, "signal::activate-item-alt", _action_window_activate_item_alt, browser, NULL); gtk_action_group_add_action_with_accel (browser->action_group, action, ""); g_object_unref (action); action = g_object_new (KATZE_TYPE_ARRAY_ACTION, "name", "CompactMenu", "label", _("_Menu"), "stock-id", GTK_STOCK_PROPERTIES, "tooltip", _("Menu"), "array", katze_array_new (KATZE_TYPE_ITEM), NULL); g_object_connect (action, "signal::populate-popup", _action_compact_menu_populate_popup, browser, NULL); gtk_action_group_add_action (browser->action_group, action); g_object_unref (action); /* Create the menubar */ browser->menubar = gtk_ui_manager_get_widget (ui_manager, "/menubar"); gtk_box_pack_start (GTK_BOX (vbox), browser->menubar, FALSE, FALSE, 0); gtk_widget_hide (browser->menubar); _action_set_visible (browser, "Menubar", !midori_browser_has_native_menubar ()); #if HAVE_HILDON #if HILDON_CHECK_VERSION (2, 2, 0) browser->menubar = hildon_app_menu_new (); _action_compact_menu_populate_popup (NULL, browser->menubar, browser); hildon_window_set_app_menu (HILDON_WINDOW (browser), HILDON_APP_MENU (browser->menubar)); #else browser->menubar = gtk_menu_new (); _action_compact_menu_populate_popup (NULL, browser->menubar, browser); hildon_window_set_menu (HILDON_WINDOW (browser), GTK_MENU (browser->menubar)); #endif hildon_program_add_window (hildon_program_get_instance (), HILDON_WINDOW (browser)); #else g_signal_connect (browser->menubar, "button-press-event", G_CALLBACK (midori_browser_menu_button_press_event_cb), browser); menuitem = gtk_menu_item_new (); gtk_widget_show (menuitem); browser->throbber = katze_throbber_new (); gtk_widget_show (browser->throbber); gtk_container_add (GTK_CONTAINER (menuitem), browser->throbber); gtk_widget_set_sensitive (menuitem, FALSE); #if GTK_CHECK_VERSION (3, 2, 0) /* FIXME: Doesn't work */ gtk_widget_set_hexpand (menuitem, TRUE); gtk_widget_set_halign (menuitem, GTK_ALIGN_END); #else gtk_menu_item_set_right_justified (GTK_MENU_ITEM (menuitem), TRUE); #endif gtk_menu_shell_append (GTK_MENU_SHELL (browser->menubar), menuitem); #endif gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM ( gtk_ui_manager_get_widget (ui_manager, "/menubar/File/WindowNew")), NULL); gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM ( gtk_ui_manager_get_widget (ui_manager, "/menubar/Go/Location")), NULL); homepage = gtk_ui_manager_get_widget (ui_manager, "/menubar/Go/Homepage"); g_signal_connect (homepage, "button-press-event", G_CALLBACK (midori_browser_menu_item_middle_click_event_cb), browser); back = gtk_ui_manager_get_widget (ui_manager, "/menubar/Go/Back"); g_signal_connect (back, "button-press-event", G_CALLBACK (midori_browser_menu_item_middle_click_event_cb), browser); forward = gtk_ui_manager_get_widget (ui_manager, "/menubar/Go/Forward"); g_signal_connect (forward, "button-press-event", G_CALLBACK (midori_browser_menu_item_middle_click_event_cb), browser); forward = gtk_ui_manager_get_widget (ui_manager, "/menubar/Go/Previous"); g_signal_connect (forward, "button-press-event", G_CALLBACK (midori_browser_menu_item_middle_click_event_cb), browser); forward = gtk_ui_manager_get_widget (ui_manager, "/menubar/Go/Next"); g_signal_connect (forward, "button-press-event", G_CALLBACK (midori_browser_menu_item_middle_click_event_cb), browser); _action_set_sensitive (browser, "EncodingCustom", FALSE); _action_set_visible (browser, "LastSession", FALSE); #if !HAVE_HILDON && !defined (GDK_WINDOWING_X11) _action_set_visible (browser, "AddDesktopShortcut", FALSE); #endif _action_set_visible (browser, "Bookmarks", browser->bookmarks != NULL); _action_set_visible (browser, "BookmarkAdd", browser->bookmarks != NULL); _action_set_visible (browser, "BookmarksImport", browser->bookmarks != NULL); _action_set_visible (browser, "BookmarksExport", browser->bookmarks != NULL); _action_set_visible (browser, "Bookmarkbar", browser->bookmarks != NULL); _action_set_visible (browser, "Trash", browser->trash != NULL); _action_set_visible (browser, "UndoTabClose", browser->trash != NULL); /* Create the navigationbar */ browser->navigationbar = gtk_ui_manager_get_widget ( ui_manager, "/toolbar_navigation"); katze_widget_add_class (browser->navigationbar, "primary-toolbar"); /* FIXME: Settings should be connected with screen changes */ gtk_settings = gtk_widget_get_settings (GTK_WIDGET (browser)); if (gtk_settings) g_signal_connect (gtk_settings, "notify::gtk-toolbar-style", G_CALLBACK (midori_browser_navigationbar_notify_style_cb), browser); gtk_toolbar_set_show_arrow (GTK_TOOLBAR (browser->navigationbar), TRUE); g_object_set (_action_by_name (browser, "Back"), "is-important", TRUE, NULL); gtk_widget_hide (browser->navigationbar); g_signal_connect (browser->navigationbar, "popup-context-menu", G_CALLBACK (midori_browser_toolbar_popup_context_menu_cb), browser); #if HAVE_HILDON hildon_window_add_toolbar (HILDON_WINDOW (browser), GTK_TOOLBAR (browser->navigationbar)); #else gtk_box_pack_start (GTK_BOX (vbox), browser->navigationbar, FALSE, FALSE, 0); #endif #ifdef HAVE_HILDON_2_2 DBusConnection* system_bus = dbus_bus_get (DBUS_BUS_SYSTEM, NULL); if (system_bus) { dbus_bus_add_match (system_bus, MCE_SIGNAL_MATCH, NULL); dbus_connection_add_filter (system_bus, midori_browser_mce_filter_cb, browser, NULL); hildon_gtk_window_set_portrait_flags (GTK_WINDOW (browser), HILDON_PORTRAIT_MODE_SUPPORT); } #endif /* Bookmarkbar */ browser->bookmarkbar = gtk_toolbar_new (); katze_widget_add_class (browser->bookmarkbar, "secondary-toolbar"); gtk_widget_set_name (browser->bookmarkbar, "MidoriBookmarkbar"); gtk_toolbar_set_icon_size (GTK_TOOLBAR (browser->bookmarkbar), GTK_ICON_SIZE_MENU); gtk_toolbar_set_style (GTK_TOOLBAR (browser->bookmarkbar), GTK_TOOLBAR_BOTH_HORIZ); #if HAVE_HILDON hildon_window_add_toolbar (HILDON_WINDOW (browser), GTK_TOOLBAR (browser->bookmarkbar)); #else gtk_box_pack_start (GTK_BOX (vbox), browser->bookmarkbar, FALSE, FALSE, 0); #endif g_signal_connect (browser->bookmarkbar, "popup-context-menu", G_CALLBACK (midori_browser_toolbar_popup_context_menu_cb), browser); /* Create the panel */ hpaned = gtk_hpaned_new (); g_signal_connect (hpaned, "notify::position", G_CALLBACK (midori_panel_notify_position_cb), browser); g_signal_connect (hpaned, "cycle-child-focus", G_CALLBACK (midori_panel_cycle_child_focus_cb), browser); gtk_box_pack_start (GTK_BOX (vbox), hpaned, TRUE, TRUE, 0); gtk_widget_show (hpaned); browser->panel = g_object_new (MIDORI_TYPE_PANEL, "action-group", browser->action_group, NULL); g_object_connect (browser->panel, "signal::notify::page", midori_panel_notify_page_cb, browser, "signal::notify::show-titles", midori_panel_notify_show_titles_cb, browser, "signal::notify::right-aligned", midori_panel_notify_right_aligned_cb, browser, "signal::close", midori_panel_close_cb, browser, NULL); gtk_paned_pack1 (GTK_PANED (hpaned), browser->panel, FALSE, FALSE); /* Notebook, containing all views */ vpaned = gtk_vpaned_new (); gtk_paned_pack2 (GTK_PANED (hpaned), vpaned, TRUE, FALSE); gtk_widget_show (vpaned); #ifdef HAVE_GRANITE /* FIXME: granite: should return GtkWidget* like GTK+ */ browser->notebook = (GtkWidget*)granite_widgets_dynamic_notebook_new (); granite_widgets_dynamic_notebook_set_allow_new_window ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), TRUE); /* FIXME: work-around a bug */ gtk_widget_show_all (browser->notebook); #else browser->notebook = gtk_notebook_new (); gtk_notebook_set_scrollable (GTK_NOTEBOOK (browser->notebook), TRUE); #endif #if !GTK_CHECK_VERSION (3, 0, 0) { /* Remove the inner border between scrollbars and the window border */ GtkRcStyle* rcstyle = gtk_rc_style_new (); rcstyle->xthickness = 0; gtk_widget_modify_style (browser->notebook, rcstyle); g_object_unref (rcstyle); } #endif gtk_paned_pack1 (GTK_PANED (vpaned), browser->notebook, FALSE, FALSE); #ifdef HAVE_GRANITE /* FIXME menu items */ g_signal_connect (browser->notebook, "tab-added", G_CALLBACK (midori_browser_notebook_tab_added_cb), browser); g_signal_connect (browser->notebook, "tab-removed", G_CALLBACK (midori_browser_notebook_tab_removed_cb), browser); g_signal_connect (browser->notebook, "tab-switched", G_CALLBACK (midori_browser_notebook_tab_switched_cb), browser); g_signal_connect (browser->notebook, "tab-moved", G_CALLBACK (midori_browser_notebook_tab_moved_cb), browser); #else g_signal_connect (browser->notebook, "switch-page", G_CALLBACK (midori_browser_notebook_switch_page_cb), browser); g_signal_connect (browser->notebook, "page-reordered", G_CALLBACK (midori_browser_notebook_page_reordered_cb), browser); g_signal_connect (browser->notebook, "page-removed", G_CALLBACK (midori_browser_notebook_page_removed_cb), browser); g_signal_connect (browser->notebook, "size-allocate", G_CALLBACK (midori_browser_notebook_size_allocate_cb), browser); g_signal_connect_after (browser->notebook, "button-press-event", G_CALLBACK (midori_browser_notebook_button_press_event_after_cb), browser); g_signal_connect (browser->notebook, "reorder-tab", G_CALLBACK (midori_browser_notebook_reorder_tab_cb), NULL); g_signal_connect (browser->notebook, "create-window", G_CALLBACK (midori_browser_notebook_create_window_cb), browser); #endif gtk_widget_show (browser->notebook); /* Inspector container */ browser->inspector = gtk_vbox_new (FALSE, 0); gtk_paned_pack2 (GTK_PANED (vpaned), browser->inspector, FALSE, FALSE); scrolled = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_widget_set_can_focus (scrolled, TRUE); gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scrolled), GTK_SHADOW_ETCHED_IN); gtk_box_pack_start (GTK_BOX (browser->inspector), scrolled, TRUE, TRUE, 0); browser->inspector_view = gtk_viewport_new (NULL, NULL); gtk_container_add (GTK_CONTAINER (scrolled), browser->inspector_view); /* Incremental findbar */ browser->find = g_object_new (MIDORI_TYPE_FINDBAR, NULL); #if HAVE_HILDON hildon_window_add_toolbar (HILDON_WINDOW (browser), GTK_TOOLBAR (browser->find)); #else gtk_box_pack_start (GTK_BOX (vbox), browser->find, FALSE, FALSE, 0); #endif /* Statusbar */ browser->statusbar = gtk_statusbar_new (); #if GTK_CHECK_VERSION (2, 20, 0) browser->statusbar_contents = gtk_statusbar_get_message_area (GTK_STATUSBAR (browser->statusbar)); #else /* Rearrange the statusbar packing. This is necessariy to keep themes happy while there is no support from GtkStatusbar. */ forward = GTK_STATUSBAR (browser->statusbar)->label; if (GTK_IS_BOX (gtk_widget_get_parent (forward))) browser->statusbar_contents = gtk_widget_get_parent (forward); else { browser->statusbar_contents = gtk_hbox_new (FALSE, 4); gtk_widget_show (browser->statusbar_contents); g_object_ref (GTK_STATUSBAR (browser->statusbar)->label); gtk_container_remove ( GTK_CONTAINER (GTK_STATUSBAR (browser->statusbar)->frame), forward); gtk_box_pack_start (GTK_BOX (browser->statusbar_contents), forward, TRUE, TRUE, 0); g_object_unref (forward); gtk_container_add (GTK_CONTAINER (GTK_STATUSBAR (browser->statusbar)->frame), browser->statusbar_contents); } #endif gtk_box_pack_start (GTK_BOX (vbox), browser->statusbar, FALSE, FALSE, 0); browser->transferbar = g_object_new (MIDORI_TYPE_TRANSFERBAR, NULL); #if GTK_CHECK_VERSION (3, 0, 0) /* FIXME: Transfers should go between text and statusbar features like GTK+2 */ gtk_box_pack_end (GTK_BOX (browser->statusbar_contents), browser->transferbar, FALSE, FALSE, 3); #else gtk_box_pack_start (GTK_BOX (browser->statusbar_contents), browser->transferbar, FALSE, FALSE, 3); #endif gtk_toolbar_set_show_arrow (GTK_TOOLBAR (browser->transferbar), FALSE); gtk_widget_show (browser->transferbar); g_signal_connect (browser->statusbar, "button-press-event", G_CALLBACK (midori_browser_menu_button_press_event_cb), browser); g_object_unref (ui_manager); } static void midori_browser_dispose (GObject* object) { MidoriBrowser* browser = MIDORI_BROWSER (object); /* We are done, the session mustn't change anymore */ katze_object_assign (browser->proxy_array, NULL); g_signal_handlers_disconnect_by_func (browser->settings, midori_browser_settings_notify, browser); midori_browser_set_bookmarks (browser, NULL); G_OBJECT_CLASS (midori_browser_parent_class)->dispose (object); } static void midori_browser_finalize (GObject* object) { MidoriBrowser* browser = MIDORI_BROWSER (object); katze_assign (browser->statusbar_text, NULL); katze_object_assign (browser->settings, NULL); katze_object_assign (browser->trash, NULL); katze_object_assign (browser->search_engines, NULL); katze_object_assign (browser->history, NULL); browser->dial = NULL; katze_assign (browser->news_aggregator, NULL); G_OBJECT_CLASS (midori_browser_parent_class)->finalize (object); } static void _midori_browser_set_toolbar_style (MidoriBrowser* browser, MidoriToolbarStyle toolbar_style) { GtkToolbarStyle gtk_toolbar_style; GtkIconSize icon_size; GtkSettings* gtk_settings = gtk_widget_get_settings (GTK_WIDGET (browser)); g_object_get (gtk_settings, "gtk-toolbar-icon-size", &icon_size, NULL); if (toolbar_style == MIDORI_TOOLBAR_DEFAULT && gtk_settings) g_object_get (gtk_settings, "gtk-toolbar-style", >k_toolbar_style, NULL); else { switch (toolbar_style) { case MIDORI_TOOLBAR_SMALL_ICONS: icon_size = GTK_ICON_SIZE_SMALL_TOOLBAR; case MIDORI_TOOLBAR_ICONS: gtk_toolbar_style = GTK_TOOLBAR_ICONS; break; case MIDORI_TOOLBAR_TEXT: gtk_toolbar_style = GTK_TOOLBAR_TEXT; break; case MIDORI_TOOLBAR_BOTH: gtk_toolbar_style = GTK_TOOLBAR_BOTH; break; case MIDORI_TOOLBAR_BOTH_HORIZ: case MIDORI_TOOLBAR_DEFAULT: gtk_toolbar_style = GTK_TOOLBAR_BOTH_HORIZ; } } gtk_toolbar_set_style (GTK_TOOLBAR (browser->navigationbar), gtk_toolbar_style); gtk_toolbar_set_icon_size (GTK_TOOLBAR (browser->navigationbar), icon_size); midori_panel_set_toolbar_style (MIDORI_PANEL (browser->panel), gtk_toolbar_style); } static void midori_browser_toolbar_popup_context_menu_history_cb (GtkMenuItem* menu_item, MidoriBrowser* browser) { gint steps = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (menu_item), "steps")); MidoriView* view = MIDORI_VIEW (midori_browser_get_current_tab (browser)); midori_view_go_back_or_forward (view, steps); } static void midori_browser_toolbar_popup_context_menu_history (MidoriBrowser* browser, GtkWidget* widget, gboolean back, gint x, gint y) { const gint step = back ? -1 : 1; gint steps = step; GtkWidget* menu; GtkWidget* menu_item; WebKitWebBackForwardList* list; WebKitWebHistoryItem* current_item; WebKitWebHistoryItem* history_item; WebKitWebHistoryItem* (*history_next)(WebKitWebBackForwardList*); void (*history_action)(WebKitWebBackForwardList*); list = webkit_web_view_get_back_forward_list ( WEBKIT_WEB_VIEW (midori_view_get_web_view ( MIDORI_VIEW (midori_browser_get_current_tab (browser))))); if (!list) return; menu = gtk_menu_new (); history_action = back ? webkit_web_back_forward_list_go_back : webkit_web_back_forward_list_go_forward; history_next = back ? webkit_web_back_forward_list_get_back_item : webkit_web_back_forward_list_get_forward_item; current_item = webkit_web_back_forward_list_get_current_item (list); for (; (history_item = history_next (list)); history_action (list), steps += step) { #if WEBKIT_CHECK_VERSION (1, 3, 13) gchar* icon_uri; gchar* icon_path; GdkPixbuf* pixbuf; GdkPixbuf* pixbuf_scaled = NULL; #endif menu_item = gtk_image_menu_item_new_with_label ( webkit_web_history_item_get_title (history_item)); #if WEBKIT_CHECK_VERSION (1, 3, 13) icon_uri = webkit_icon_database_get_icon_uri (webkit_get_icon_database (), webkit_web_history_item_get_uri (history_item)); icon_path = katze_net_get_cached_path (NULL, icon_uri, "icons"); if ((pixbuf = gdk_pixbuf_new_from_file (icon_path, NULL))) { gint w = 16, h = 16; gtk_icon_size_lookup_for_settings (gtk_widget_get_settings (widget), GTK_ICON_SIZE_MENU, &w, &h); pixbuf_scaled = gdk_pixbuf_scale_simple (pixbuf, w, h, GDK_INTERP_BILINEAR); } gtk_image_menu_item_set_image ( GTK_IMAGE_MENU_ITEM (menu_item), pixbuf_scaled ? gtk_image_new_from_pixbuf (pixbuf_scaled) : gtk_image_new_from_stock (GTK_STOCK_FILE, GTK_ICON_SIZE_MENU)); g_free (icon_uri); g_free (icon_path); #endif g_object_set_data (G_OBJECT (menu_item), "uri", (gpointer) webkit_web_history_item_get_uri (history_item)); g_object_set_data (G_OBJECT (menu_item), "steps", GINT_TO_POINTER (steps)); gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item); g_signal_connect (G_OBJECT (menu_item), "activate", G_CALLBACK (midori_browser_toolbar_popup_context_menu_history_cb), browser); if (steps == (10 - 1)) break; } webkit_web_back_forward_list_go_to_item (list, current_item); gtk_widget_show_all (menu); katze_widget_popup (widget, GTK_MENU (menu), NULL, KATZE_MENU_POSITION_LEFT); } static gboolean midori_browser_toolbar_item_button_press_event_cb (GtkWidget* toolitem, GdkEventButton* event, MidoriBrowser* browser) { if (MIDORI_EVENT_NEW_TAB (event)) { GtkWidget* parent = gtk_widget_get_parent (toolitem); GtkAction* action = gtk_activatable_get_related_action ( GTK_ACTIVATABLE (parent)); g_object_set_data (G_OBJECT (action), "midori-middle-click", GINT_TO_POINTER (1)); return _action_navigation_activate (action, browser); } else if (MIDORI_EVENT_CONTEXT_MENU (event)) { if (g_object_get_data (G_OBJECT (toolitem), "history-back")) { midori_browser_toolbar_popup_context_menu_history ( browser, GTK_IS_BIN (toolitem) && gtk_bin_get_child (GTK_BIN (toolitem)) ? gtk_widget_get_parent (toolitem) : toolitem, TRUE, event->x, event->y); } else if (g_object_get_data (G_OBJECT (toolitem), "history-forward")) { midori_browser_toolbar_popup_context_menu_history ( browser, GTK_IS_BIN (toolitem) && gtk_bin_get_child (GTK_BIN (toolitem)) ? gtk_widget_get_parent (toolitem) : toolitem, FALSE, event->x, event->y); } else { midori_browser_toolbar_popup_context_menu_cb ( GTK_IS_BIN (toolitem) && gtk_bin_get_child (GTK_BIN (toolitem)) ? gtk_widget_get_parent (toolitem) : toolitem, event->x, event->y, event->button, browser); } return TRUE; } return FALSE; } static void _midori_browser_search_item_allocate_cb (GtkWidget* widget, GdkRectangle* allocation, gpointer user_data) { MidoriBrowser* browser = MIDORI_BROWSER (user_data); MidoriWebSettings* settings = browser->settings; g_object_set (settings, "search-width", allocation->width, NULL); } static void _midori_browser_set_toolbar_items (MidoriBrowser* browser, const gchar* items) { gchar** names; gchar** name; GtkAction* action; GtkWidget* toolitem; const char* token_location = g_intern_static_string ("Location"); const char* token_search = g_intern_static_string ("Search"); const char* token_dontcare = g_intern_static_string ("Dontcare"); const char* token_current = token_dontcare; const char* token_last = token_dontcare; gtk_container_foreach (GTK_CONTAINER (browser->navigationbar), (GtkCallback)gtk_widget_destroy, NULL); names = g_strsplit (items ? items : "", ",", 0); name = names; for (; *name; ++name) { action = _action_by_name (browser, *name); if (action && strstr (*name, "CompactMenu") == NULL) { token_last = token_current; /* Decide, what kind of token (item) we got now */ if (name && !g_strcmp0 (*name, "Location")) token_current = token_location; else if (name && !g_strcmp0 (*name, "Search")) token_current = token_search; else token_current = token_dontcare; if ((token_current == token_location || token_current == token_search) && (token_last == token_location || token_last == token_search)) { GtkWidget* toolitem_first = gtk_action_create_tool_item ( _action_by_name (browser, token_last)); GtkWidget* toolitem_second = gtk_action_create_tool_item ( _action_by_name (browser, token_current)); MidoriPanedAction* paned_action = MIDORI_PANED_ACTION ( _action_by_name (browser, "LocationSearch")); MidoriWebSettings* midori_settings = browser->settings; midori_paned_action_set_child1 (paned_action, toolitem_first, token_last, token_last == token_search ? FALSE : TRUE, TRUE); midori_paned_action_set_child2 (paned_action, toolitem_second, token_current, token_current == token_search ? FALSE : TRUE, TRUE); g_signal_connect (G_OBJECT (token_current == token_search ? toolitem_second : toolitem_first), "size-allocate", G_CALLBACK (_midori_browser_search_item_allocate_cb), (gpointer) browser); gtk_widget_set_size_request ( token_last == token_search ? toolitem_first : toolitem_second, katze_object_get_int ((gpointer) midori_settings, "search-width"), -1); toolitem = gtk_action_create_tool_item (GTK_ACTION (paned_action)); token_current = token_dontcare; token_last = token_dontcare; } else if (token_current == token_dontcare && token_last != token_dontcare) { /* There was a location or search item, but was not followed by the other one, that form a couple */ gtk_toolbar_insert (GTK_TOOLBAR (browser->navigationbar), GTK_TOOL_ITEM (gtk_action_create_tool_item ( _action_by_name (browser, token_last))), -1); toolitem = gtk_action_create_tool_item (action); } else if (token_current != token_dontcare && token_last == token_dontcare) continue; #ifdef HAVE_GRANITE /* A "new tab" button is already part of the notebook */ else if (!strcmp (gtk_action_get_name (action), "TabNew")) continue; #endif else toolitem = gtk_action_create_tool_item (action); if (gtk_bin_get_child (GTK_BIN (toolitem))) { if (!g_strcmp0 (*name, "Back")) g_object_set_data (G_OBJECT (gtk_bin_get_child (GTK_BIN (toolitem))), "history-back", (void*) 0xdeadbeef); else if (!g_strcmp0 (*name, "Forward")) g_object_set_data (G_OBJECT (gtk_bin_get_child (GTK_BIN (toolitem))), "history-forward", (void*) 0xdeadbeef); g_signal_connect (gtk_bin_get_child (GTK_BIN (toolitem)), "button-press-event", G_CALLBACK (midori_browser_toolbar_item_button_press_event_cb), browser); } else { gtk_tool_item_set_use_drag_window (GTK_TOOL_ITEM (toolitem), TRUE); g_signal_connect (toolitem, "button-press-event", G_CALLBACK (midori_browser_toolbar_item_button_press_event_cb), browser); } gtk_toolbar_insert (GTK_TOOLBAR (browser->navigationbar), GTK_TOOL_ITEM (toolitem), -1); } } g_strfreev (names); /* There was a last item, which could have formed a couple, but there is no item left, we add that last item to toolbar as is */ if (token_current != token_dontcare) { gtk_toolbar_insert (GTK_TOOLBAR (browser->navigationbar), GTK_TOOL_ITEM (gtk_action_create_tool_item ( _action_by_name (browser, token_current))), -1); } if (!katze_object_get_boolean (browser->settings, "show-menubar")) { toolitem = gtk_action_create_tool_item ( GTK_ACTION (_action_by_name (browser, "CompactMenu"))); gtk_toolbar_insert (GTK_TOOLBAR (browser->navigationbar), GTK_TOOL_ITEM (toolitem), -1); g_signal_connect (gtk_bin_get_child (GTK_BIN (toolitem)), "button-press-event", G_CALLBACK (midori_browser_toolbar_item_button_press_event_cb), browser); } } static void _midori_browser_update_settings (MidoriBrowser* browser) { gboolean remember_last_window_size; MidoriWindowState last_window_state; gboolean compact_sidepanel; gboolean right_align_sidepanel, open_panels_in_windows; gint last_panel_position, last_panel_page; gboolean show_menubar, show_bookmarkbar; gboolean show_panel; MidoriToolbarStyle toolbar_style; gchar* toolbar_items; gboolean close_buttons_on_tabs; KatzeItem* item; g_free (browser->location_entry_search); g_free (browser->news_aggregator); g_object_get (browser->settings, "remember-last-window-size", &remember_last_window_size, "last-window-width", &browser->last_window_width, "last-window-height", &browser->last_window_height, "last-window-state", &last_window_state, "compact-sidepanel", &compact_sidepanel, "right-align-sidepanel", &right_align_sidepanel, "open-panels-in-windows", &open_panels_in_windows, "last-panel-position", &last_panel_position, "last-panel-page", &last_panel_page, "show-menubar", &show_menubar, "show-navigationbar", &browser->show_navigationbar, "show-bookmarkbar", &show_bookmarkbar, "show-panel", &show_panel, "show-statusbar", &browser->show_statusbar, "toolbar-style", &toolbar_style, "toolbar-items", &toolbar_items, "location-entry-search", &browser->location_entry_search, "close-buttons-on-tabs", &close_buttons_on_tabs, "maximum-history-age", &browser->maximum_history_age, "news-aggregator", &browser->news_aggregator, NULL); #ifdef HAVE_GRANITE granite_widgets_dynamic_notebook_set_tabs_closable ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), close_buttons_on_tabs); #endif midori_findbar_set_close_button_left (MIDORI_FINDBAR (browser->find), katze_object_get_boolean (browser->settings, "close-buttons-left")); if (browser->dial != NULL) midori_speed_dial_set_close_buttons_left (browser->dial, katze_object_get_boolean (browser->settings, "close-buttons-left")); if (remember_last_window_size) { if (browser->last_window_width && browser->last_window_height) gtk_window_set_default_size (GTK_WINDOW (browser), browser->last_window_width, browser->last_window_height); else { GdkScreen* screen; GdkRectangle monitor; gint default_width, default_height; screen = gtk_window_get_screen (GTK_WINDOW (browser)); gdk_screen_get_monitor_geometry (screen, 0, &monitor); default_width = monitor.width / 1.7; default_height = monitor.height / 1.7; gtk_window_set_default_size (GTK_WINDOW (browser), default_width, default_height); } switch (last_window_state) { case MIDORI_WINDOW_MINIMIZED: gtk_window_iconify (GTK_WINDOW (browser)); break; case MIDORI_WINDOW_MAXIMIZED: gtk_window_maximize (GTK_WINDOW (browser)); break; case MIDORI_WINDOW_FULLSCREEN: gtk_window_fullscreen (GTK_WINDOW (browser)); break; default: ;/* Do nothing. */ } } _midori_browser_set_toolbar_style (browser, toolbar_style); _toggle_tabbar_smartly (browser, FALSE); _midori_browser_set_toolbar_items (browser, toolbar_items); if (browser->search_engines) { item = katze_array_get_nth_item (browser->search_engines, browser->last_web_search); if (item) midori_search_action_set_current_item (MIDORI_SEARCH_ACTION ( _action_by_name (browser, "Search")), item); KATZE_ARRAY_FOREACH_ITEM (item, browser->search_engines) if (!g_strcmp0 (katze_item_get_uri (item), browser->location_entry_search)) { midori_search_action_set_default_item (MIDORI_SEARCH_ACTION ( _action_by_name (browser, "Search")), item); break; } } g_object_set (browser->panel, "show-titles", !compact_sidepanel, "right-aligned", right_align_sidepanel, "open-panels-in-windows", open_panels_in_windows, NULL); gtk_paned_set_position (GTK_PANED (gtk_widget_get_parent (browser->panel)), last_panel_position); /* The browser may not yet be visible, which means that we can't set the page. So we set it in midori_browser_size_allocate_cb */ if (gtk_widget_get_visible (GTK_WIDGET (browser))) midori_panel_set_current_page (MIDORI_PANEL (browser->panel), last_panel_page); else g_object_set_data (G_OBJECT (browser), "last-page", GINT_TO_POINTER (last_panel_page)); _action_set_active (browser, "Menubar", show_menubar); _action_set_active (browser, "Navigationbar", browser->show_navigationbar); _action_set_active (browser, "Bookmarkbar", show_bookmarkbar && browser->bookmarks != NULL); _action_set_active (browser, "Panel", show_panel); _action_set_active (browser, "Statusbar", browser->show_statusbar); g_free (toolbar_items); } static void midori_browser_settings_notify (MidoriWebSettings* web_settings, GParamSpec* pspec, MidoriBrowser* browser) { const gchar* name; GValue value = {0, }; name = g_intern_string (pspec->name); g_value_init (&value, pspec->value_type); g_object_get_property (G_OBJECT (web_settings), name, &value); if (name == g_intern_string ("toolbar-style")) _midori_browser_set_toolbar_style (browser, g_value_get_enum (&value)); else if (name == g_intern_string ("toolbar-items")) _midori_browser_set_toolbar_items (browser, g_value_get_string (&value)); else if (name == g_intern_string ("compact-sidepanel")) { g_signal_handlers_block_by_func (browser->panel, midori_panel_notify_show_titles_cb, browser); g_object_set (browser->panel, "show-titles", !g_value_get_boolean (&value), NULL); g_signal_handlers_unblock_by_func (browser->panel, midori_panel_notify_show_titles_cb, browser); } else if (name == g_intern_string ("open-panels-in-windows")) g_object_set (browser->panel, "open-panels-in-windows", g_value_get_boolean (&value), NULL); else if (name == g_intern_string ("always-show-tabbar")) _toggle_tabbar_smartly (browser, FALSE); else if (name == g_intern_string ("show-menubar")) { _action_set_active (browser, "Menubar", g_value_get_boolean (&value)); } else if (name == g_intern_string ("show-navigationbar")) { browser->show_navigationbar = g_value_get_boolean (&value); _action_set_active (browser, "Navigationbar", g_value_get_boolean (&value)); } else if (name == g_intern_string ("show-bookmarkbar")) { _action_set_active (browser, "Bookmarkbar", g_value_get_boolean (&value)); } else if (name == g_intern_string ("show-statusbar")) { browser->show_statusbar = g_value_get_boolean (&value); _action_set_active (browser, "Statusbar", g_value_get_boolean (&value)); } else if (name == g_intern_string ("location-entry-search")) { katze_assign (browser->location_entry_search, g_value_dup_string (&value)); } else if (name == g_intern_string ("maximum-history-age")) browser->maximum_history_age = g_value_get_int (&value); else if (name == g_intern_string ("news-aggregator")) { katze_assign (browser->news_aggregator, g_value_dup_string (&value)); } else if (name == g_intern_string ("close-buttons-left")) { midori_findbar_set_close_button_left (MIDORI_FINDBAR (browser->find), g_value_get_boolean (&value)); midori_speed_dial_set_close_buttons_left (browser->dial, katze_object_get_boolean (browser->settings, "close-buttons-left")); } else if (!g_object_class_find_property (G_OBJECT_GET_CLASS (web_settings), name)) g_warning (_("Unexpected setting '%s'"), name); g_value_unset (&value); } static gboolean midori_bookmarkbar_item_button_press_event_cb (GtkWidget* toolitem, GdkEventButton* event, MidoriBrowser* browser) { KatzeItem* item; gint n; item = (KatzeItem*)g_object_get_data (G_OBJECT (toolitem), "KatzeItem"); if (MIDORI_EVENT_NEW_TAB (event)) { if (KATZE_ITEM_IS_BOOKMARK (item)) { n = midori_browser_add_uri (browser, katze_item_get_uri (item)); midori_browser_set_current_page_smartly (browser, n); return TRUE; } } else if (MIDORI_EVENT_CONTEXT_MENU (event)) { midori_browser_bookmark_popup (toolitem, NULL, item, browser); return TRUE; } return FALSE; } static void midori_bookmarkbar_insert_item (GtkWidget* toolbar, KatzeItem* item) { MidoriBrowser* browser = midori_browser_get_for_widget (toolbar); GtkAction* action = _action_by_name (browser, "Tools"); GtkToolItem* toolitem = katze_array_action_create_tool_item_for ( KATZE_ARRAY_ACTION (action), item); g_object_set_data (G_OBJECT (toolitem), "KatzeItem", item); if (KATZE_IS_ITEM (item)) { GtkWidget* child = gtk_bin_get_child (GTK_BIN (toolitem)); g_object_set_data (G_OBJECT (child), "KatzeItem", item); g_signal_connect (child, "button-press-event", G_CALLBACK (midori_bookmarkbar_item_button_press_event_cb), browser); } else /* Separator */ gtk_tool_item_set_use_drag_window (toolitem, TRUE); gtk_widget_show (GTK_WIDGET (toolitem)); gtk_toolbar_insert (GTK_TOOLBAR (toolbar), toolitem, -1); } static void midori_bookmarkbar_remove_item_cb (KatzeArray* bookmarks, KatzeItem* item, MidoriBrowser* browser) { if (gtk_widget_get_visible (browser->bookmarkbar)) midori_bookmarkbar_populate (browser); midori_browser_update_history (item, "bookmark", "delete"); } static void midori_bookmarkbar_populate (MidoriBrowser* browser) { KatzeArray* array; KatzeItem* item; midori_bookmarkbar_clear (browser->bookmarkbar); /* Use a dummy to ensure height of the toolbar */ gtk_toolbar_insert (GTK_TOOLBAR (browser->bookmarkbar), gtk_separator_tool_item_new (), -1); array = midori_array_query (browser->bookmarks, "id, parentid, title, uri, desc, app, toolbar, pos_panel, pos_bar", "toolbar = 1", NULL); if (!array) { g_warning ("midori_array_query returned NULL"); _action_set_sensitive (browser, "BookmarkAdd", FALSE); _action_set_sensitive (browser, "BookmarkFolderAdd", FALSE); return; } KATZE_ARRAY_FOREACH_ITEM (item, array) { if (KATZE_ITEM_IS_BOOKMARK (item)) midori_bookmarkbar_insert_item (browser->bookmarkbar, item); else { gint64 id = katze_item_get_meta_integer (item, "id"); gchar* parentid = g_strdup_printf ("%" G_GINT64_FORMAT, id); KatzeArray* subfolder = midori_array_query (browser->bookmarks, "id, parentid, title, uri, desc, app, toolbar, pos_panel, pos_bar", "parentid = %q AND uri != ''", parentid); if (subfolder) { katze_item_set_name (KATZE_ITEM (subfolder), katze_item_get_name (item)); katze_item_set_meta_integer (KATZE_ITEM (subfolder), "id", id); midori_bookmarkbar_insert_item (browser->bookmarkbar, KATZE_ITEM (subfolder)); } else g_warning ("midori_array_query returned NULL (id='%s')", parentid); g_free (parentid); } } _action_set_sensitive (browser, "BookmarkAdd", TRUE); _action_set_sensitive (browser, "BookmarkFolderAdd", TRUE); } static void midori_bookmarkbar_clear (GtkWidget* toolbar) { GList* children = gtk_container_get_children (GTK_CONTAINER (toolbar)); while (children != NULL) { gtk_widget_destroy (children->data); children = g_list_next (children); } } static void midori_browser_show_bookmarkbar_notify_value_cb (MidoriWebSettings* settings, GParamSpec* pspec, MidoriBrowser* browser) { if (!katze_object_get_boolean (browser->settings, "show-bookmarkbar")) midori_bookmarkbar_clear (browser->bookmarkbar); else midori_bookmarkbar_populate (browser); } static void midori_browser_set_bookmarks (MidoriBrowser* browser, KatzeArray* bookmarks) { MidoriWebSettings* settings; if (browser->bookmarks != NULL) g_signal_handlers_disconnect_by_func (browser->bookmarks, midori_bookmarkbar_remove_item_cb, browser); settings = midori_browser_get_settings (browser); g_signal_handlers_disconnect_by_func (settings, midori_browser_show_bookmarkbar_notify_value_cb, browser); katze_object_assign (browser->bookmarks, bookmarks); _action_set_visible (browser, "Bookmarks", bookmarks != NULL); if (bookmarks != NULL) { /* FIXME: Proxies aren't shown propely. Why? */ GSList* proxies = gtk_action_get_proxies ( _action_by_name (browser, "Bookmarks")); for (; proxies; proxies = g_slist_next (proxies)) gtk_widget_show (proxies->data); } _action_set_visible (browser, "BookmarkAdd", bookmarks != NULL); _action_set_visible (browser, "BookmarksImport", bookmarks != NULL); _action_set_visible (browser, "BookmarksExport", bookmarks != NULL); _action_set_visible (browser, "Bookmarkbar", bookmarks != NULL); if (!bookmarks) return; if (katze_object_get_boolean (browser->settings, "show-bookmarkbar")) _action_set_active (browser, "Bookmarkbar", TRUE); g_object_ref (bookmarks); g_signal_connect (settings, "notify::show-bookmarkbar", G_CALLBACK (midori_browser_show_bookmarkbar_notify_value_cb), browser); g_object_notify (G_OBJECT (settings), "show-bookmarkbar"); g_signal_connect_after (bookmarks, "remove-item", G_CALLBACK (midori_bookmarkbar_remove_item_cb), browser); } static void midori_browser_set_property (GObject* object, guint prop_id, const GValue* value, GParamSpec* pspec) { MidoriBrowser* browser = MIDORI_BROWSER (object); KatzeItem* item; switch (prop_id) { case PROP_URI: midori_browser_set_current_uri (browser, g_value_get_string (value)); break; case PROP_TAB: midori_browser_set_current_tab (browser, g_value_get_object (value)); break; case PROP_STATUSBAR_TEXT: _midori_browser_set_statusbar_text (browser, MIDORI_VIEW (midori_browser_get_current_tab (browser)), g_value_get_string (value)); break; case PROP_SETTINGS: g_signal_handlers_disconnect_by_func (browser->settings, midori_browser_settings_notify, browser); katze_object_assign (browser->settings, g_value_dup_object (value)); if (!browser->settings) browser->settings = midori_web_settings_new (); _midori_browser_update_settings (browser); g_signal_connect (browser->settings, "notify", G_CALLBACK (midori_browser_settings_notify), browser); midori_browser_foreach (browser, (GtkCallback) midori_view_set_settings, browser->settings); break; case PROP_BOOKMARKS: midori_browser_set_bookmarks (browser, g_value_get_object (value)); break; case PROP_TRASH: /* FIXME: Disconnect handlers */ katze_object_assign (browser->trash, g_value_dup_object (value)); g_object_set (_action_by_name (browser, "Trash"), "array", browser->trash, "reversed", TRUE, NULL); _action_set_visible (browser, "Trash", browser->trash != NULL); _action_set_visible (browser, "UndoTabClose", browser->trash != NULL); /* FIXME: Connect to updates */ _midori_browser_update_actions (browser); break; case PROP_SEARCH_ENGINES: { /* FIXME: Disconnect handlers */ katze_object_assign (browser->search_engines, g_value_dup_object (value)); midori_location_action_set_search_engines (MIDORI_LOCATION_ACTION ( _action_by_name (browser, "Location")), browser->search_engines); midori_search_action_set_search_engines (MIDORI_SEARCH_ACTION ( _action_by_name (browser, "Search")), browser->search_engines); /* FIXME: Connect to updates */ if (browser->search_engines) { g_object_get (browser->settings, "last-web-search", &browser->last_web_search, NULL); item = katze_array_get_nth_item (browser->search_engines, browser->last_web_search); midori_search_action_set_current_item (MIDORI_SEARCH_ACTION ( _action_by_name (browser, "Search")), item); KATZE_ARRAY_FOREACH_ITEM (item, browser->search_engines) if (!g_strcmp0 (katze_item_get_uri (item), browser->location_entry_search)) { midori_search_action_set_default_item (MIDORI_SEARCH_ACTION ( _action_by_name (browser, "Search")), item); break; } } break; } case PROP_HISTORY: midori_browser_set_history (browser, g_value_get_object (value)); break; case PROP_SPEED_DIAL: browser->dial = g_value_get_pointer (value); g_signal_connect (browser->dial, "refresh", G_CALLBACK (midori_browser_speed_dial_refresh_cb), browser); break; case PROP_SHOW_TABS: browser->show_tabs = g_value_get_boolean (value); _toggle_tabbar_smartly (browser, FALSE); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void midori_browser_get_property (GObject* object, guint prop_id, GValue* value, GParamSpec* pspec) { MidoriBrowser* browser = MIDORI_BROWSER (object); switch (prop_id) { case PROP_MENUBAR: g_value_set_object (value, browser->menubar); break; case PROP_NAVIGATIONBAR: g_value_set_object (value, browser->navigationbar); break; case PROP_NOTEBOOK: g_value_set_object (value, browser->notebook); break; case PROP_PANEL: g_value_set_object (value, browser->panel); break; case PROP_URI: g_value_set_string (value, midori_browser_get_current_uri (browser)); break; case PROP_TAB: g_value_set_object (value, midori_browser_get_current_tab (browser)); break; case PROP_LOAD_STATUS: { GtkWidget* view = midori_browser_get_current_tab (browser); if (view) g_value_set_enum (value, midori_view_get_load_status (MIDORI_VIEW (view))); else g_value_set_enum (value, MIDORI_LOAD_FINISHED); break; } case PROP_STATUSBAR: g_value_set_object (value, browser->statusbar_contents); break; case PROP_STATUSBAR_TEXT: g_value_set_string (value, browser->statusbar_text); break; case PROP_SETTINGS: g_value_set_object (value, browser->settings); break; case PROP_BOOKMARKS: g_value_set_object (value, browser->bookmarks); break; case PROP_TRASH: g_value_set_object (value, browser->trash); break; case PROP_SEARCH_ENGINES: g_value_set_object (value, browser->search_engines); break; case PROP_HISTORY: g_value_set_object (value, browser->history); break; case PROP_SPEED_DIAL: g_value_set_pointer (value, browser->dial); break; case PROP_SHOW_TABS: g_value_set_boolean (value, browser->show_tabs); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } /** * midori_browser_new: * * Creates a new browser widget. * * A browser is a window with a menubar, toolbars, a notebook, panels * and a statusbar. You should mostly treat it as an opaque widget. * * Return value: a new #MidoriBrowser **/ MidoriBrowser* midori_browser_new (void) { MidoriBrowser* browser = g_object_new (MIDORI_TYPE_BROWSER, NULL); return browser; } /** * midori_browser_add_tab: * @browser: a #MidoriBrowser * @widget: a view * * Appends a view in the form of a new tab and creates an * according item in the Window menu. * * Return value: the index of the new tab, or -1 in case of an error **/ gint midori_browser_add_tab (MidoriBrowser* browser, GtkWidget* view) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), -1); g_return_val_if_fail (GTK_IS_WIDGET (view), -1); if (!g_object_get_data (G_OBJECT (webkit_get_default_session ()), "midori-session-initialized")) g_critical ("midori_load_soup_session was not called!"); g_signal_emit (browser, signals[ADD_TAB], 0, view); return midori_browser_page_num (browser, view); } /** * midori_browser_page_num: * @browser: a #MidoriBrowser * @widget: a widget in the browser * * Retrieves the position of @widget in the browser. * * If there is no page present at all, -1 is returned. * * Return value: the index of the widget, or -1 * * Since: 0.4.5 **/ gint midori_browser_page_num (MidoriBrowser* browser, GtkWidget* view) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), -1); g_return_val_if_fail (MIDORI_IS_VIEW (view), -1); #ifdef HAVE_GRANITE return granite_widgets_dynamic_notebook_get_tab_position ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), midori_view_get_tab (MIDORI_VIEW (view))); #else return gtk_notebook_page_num (GTK_NOTEBOOK (browser->notebook), view); #endif } /** * midori_browser_remove_tab: * @browser: a #MidoriBrowser * @widget: a view * * Removes an existing view from the browser, * including an associated menu item. **/ void midori_browser_remove_tab (MidoriBrowser* browser, GtkWidget* view) { g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_return_if_fail (GTK_IS_WIDGET (view)); g_signal_emit (browser, signals[REMOVE_TAB], 0, view); #ifdef HAVE_GRANITE granite_widgets_dynamic_notebook_remove_tab ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), midori_view_get_tab (MIDORI_VIEW (view))); #else gtk_widget_destroy (view); #endif } /** * midori_browser_add_item: * @browser: a #MidoriBrowser * @item: an item * * Appends a new view as described by @item. * * Return value: the index of the new tab, or -1 in case of an error **/ gint midori_browser_add_item (MidoriBrowser* browser, KatzeItem* item) { const gchar* uri; GtkWidget* view; gint page; g_return_val_if_fail (MIDORI_IS_BROWSER (browser), -1); g_return_val_if_fail (KATZE_IS_ITEM (item), -1); uri = katze_item_get_uri (item); view = midori_view_new_with_item (item, browser->settings); page = midori_browser_add_tab (browser, view); midori_view_set_uri (MIDORI_VIEW (view), uri); return page; } /** * midori_browser_add_uri: * @browser: a #MidoriBrowser * @uri: an URI * * Appends an uri in the form of a new view. * * Return value: the index of the new view, or -1 **/ gint midori_browser_add_uri (MidoriBrowser* browser, const gchar* uri) { KatzeItem* item; g_return_val_if_fail (MIDORI_IS_BROWSER (browser), -1); g_return_val_if_fail (uri != NULL, -1); item = katze_item_new (); item->uri = g_strdup (uri); return midori_browser_add_item (browser, item); } /** * midori_browser_activate_action: * @browser: a #MidoriBrowser * @name: name of the action * * Activates the specified action. **/ void midori_browser_activate_action (MidoriBrowser* browser, const gchar* name) { g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_return_if_fail (name != NULL); g_signal_emit (browser, signals[ACTIVATE_ACTION], 0, name); } void midori_browser_set_action_visible (MidoriBrowser* browser, const gchar* name, gboolean visible) { g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_return_if_fail (name != NULL); _action_set_visible (browser, name, visible); _action_set_sensitive (browser, name, visible); } /** * midori_browser_block_action: * @browser: a #MidoriBrowser * @name: the action to be blocked * * Blocks built-in behavior of the specified action without * disabling it, which gives you a chance to connect your * own signal handling. * Call midori_browser_unblock_action() to undo the effect. * * Since: 0.3.4 **/ void midori_browser_block_action (MidoriBrowser* browser, GtkAction* action) { const gchar* name; guint i; g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_return_if_fail (GTK_IS_ACTION (action)); name = gtk_action_get_name (action); for (i = 0; i < entries_n; i++) if (g_str_equal (entries[i].name, name)) { g_signal_handlers_block_by_func (action, entries[i].callback, browser); return; } g_critical ("%s: Action \"%s\" can't be blocked.", G_STRFUNC, name); } /** * midori_browser_unblock_action: * @browser: a #MidoriBrowser * @name: the action to be unblocked * * Restores built-in behavior of the specified action after * previously blocking it with midori_browser_block_action(). * * Since: 0.3.4 **/ void midori_browser_unblock_action (MidoriBrowser* browser, GtkAction* action) { const gchar* name; guint i; g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_return_if_fail (GTK_IS_ACTION (action)); name = gtk_action_get_name (action); for (i = 0; i < entries_n; i++) if (g_str_equal (entries[i].name, name)) { g_signal_handlers_unblock_by_func (action, entries[i].callback, browser); return; } g_critical ("%s: Action \"%s\" can't be unblocked.", G_STRFUNC, name); } /** * midori_browser_get_action_group: * @browser: a #MidoriBrowser * * Retrieves the action group holding all actions used * by the browser. It allows obtaining individual * actions and adding new actions. * * Return value: the action group of the browser * * Since: 0.1.4 **/ GtkActionGroup* midori_browser_get_action_group (MidoriBrowser* browser) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); return browser->action_group; } /** * midori_browser_set_current_uri: * @browser: a #MidoriBrowser * @uri: an URI * * Loads the specified URI in the current view. * * If the current view is opaque, and cannot load * new pages, it will automatically open a new tab. **/ void midori_browser_set_current_uri (MidoriBrowser* browser, const gchar* uri) { GtkWidget* view; g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_return_if_fail (uri != NULL); if ((view = midori_browser_get_current_tab (browser))) midori_view_set_uri (MIDORI_VIEW (view), uri); else midori_browser_add_uri (browser, uri); } /** * midori_browser_get_current_uri: * @browser: a #MidoriBrowser * * Determines the URI loaded in the current view. * * If there is no view present at all, %NULL is returned. * * Return value: the current URI, or %NULL **/ const gchar* midori_browser_get_current_uri (MidoriBrowser* browser) { GtkWidget* view; g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); if ((view = midori_browser_get_current_tab (browser))) return midori_view_get_display_uri (MIDORI_VIEW (view)); return NULL; } /** * midori_browser_set_current_page: * @browser: a #MidoriBrowser * @n: the index of a page * * Switches to the page with the index @n. * * The widget will also grab the focus automatically. **/ void midori_browser_set_current_page (MidoriBrowser* browser, gint n) { GtkWidget* view; g_return_if_fail (MIDORI_IS_BROWSER (browser)); view = midori_browser_get_nth_tab (browser, n); g_return_if_fail (view != NULL); #ifdef HAVE_GRANITE granite_widgets_dynamic_notebook_set_current ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), midori_view_get_tab (MIDORI_VIEW (view))); #else gtk_notebook_set_current_page (GTK_NOTEBOOK (browser->notebook), n); #endif if (midori_view_is_blank (MIDORI_VIEW (view))) midori_browser_activate_action (browser, "Location"); else gtk_widget_grab_focus (view); g_object_freeze_notify (G_OBJECT (browser)); g_object_notify (G_OBJECT (browser), "uri"); g_object_notify (G_OBJECT (browser), "tab"); g_object_thaw_notify (G_OBJECT (browser)); } /** * midori_browser_get_current_page: * @browser: a #MidoriBrowser * * Determines the currently selected page. * * If there is no page present at all, %NULL is returned. * * Return value: the selected page, or -1 **/ gint midori_browser_get_current_page (MidoriBrowser* browser) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), -1); #ifdef HAVE_GRANITE GraniteWidgetsTab* tab = granite_widgets_dynamic_notebook_get_current ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook)); return tab ? granite_widgets_dynamic_notebook_get_tab_position ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), tab) : -1; #else return gtk_notebook_get_current_page (GTK_NOTEBOOK (browser->notebook)); #endif } /** * midori_browser_get_nth_tab: * @browser: a #MidoriBrowser * @page: the index of a tab * * Retrieves the tab at the position @page. * * If there is no page present at all, %NULL is returned. * * Return value: the selected page, or -1 * * Since: 0.1.9 **/ GtkWidget* midori_browser_get_nth_tab (MidoriBrowser* browser, gint page) { #ifdef HAVE_GRANITE GraniteWidgetsTab* tab; g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); tab = granite_widgets_dynamic_notebook_get_tab_by_index ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook), page); return tab != NULL ? granite_widgets_tab_get_page (tab) : NULL; #else g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); return gtk_notebook_get_nth_page (GTK_NOTEBOOK (browser->notebook), page); #endif } /** * midori_browser_set_tab: * @browser: a #MidoriBrowser * @view: a #GtkWidget * * Switches to the page containing @view. * * The widget will also grab the focus automatically. * * Since: 0.2.6 **/ void midori_browser_set_current_tab (MidoriBrowser* browser, GtkWidget* view) { gint n; g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_return_if_fail (GTK_IS_WIDGET (view)); n = midori_browser_page_num (browser, view); midori_browser_set_current_page (browser, n); } /** * midori_browser_get_tab: * @browser: a #MidoriBrowser * * Retrieves the currently selected tab. * * If there is no tab present at all, %NULL is returned. * * See also midori_browser_get_current_page(). * * Return value: the selected tab, or %NULL * * Since: 0.2.6 **/ GtkWidget* midori_browser_get_current_tab (MidoriBrowser* browser) { #ifdef HAVE_GRANITE GraniteWidgetsTab* tab; #else gint n; #endif g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); #ifdef HAVE_GRANITE tab = granite_widgets_dynamic_notebook_get_current ( GRANITE_WIDGETS_DYNAMIC_NOTEBOOK (browser->notebook)); return tab ? granite_widgets_tab_get_page (tab) : NULL; #else n = midori_browser_get_current_page (browser); return (n >= 0) ? midori_browser_get_nth_tab (browser, n) : NULL; #endif } /** * midori_browser_get_tabs: * @browser: a #MidoriBrowser * * Retrieves the tabs as a list. * * Return value: a newly allocated #GList of #MidoriView * * Since: 0.2.5 **/ GList* midori_browser_get_tabs (MidoriBrowser* browser) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); return gtk_container_get_children (GTK_CONTAINER (browser->notebook)); } /** * midori_browser_get_proxy_items: * @browser: a #MidoriBrowser * * Retrieves a proxy array representing the respective proxy items * of the present views that can be used for session management. * * The array is updated automatically. * * Note: Calling this function doesn't add a reference and the browser * may release its reference at some point. * * Return value: the proxy #KatzeArray * * Since: 0.2.5 **/ KatzeArray* midori_browser_get_proxy_items (MidoriBrowser* browser) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); return browser->proxy_array; } /** * midori_browser_get_proxy_array: * @browser: a #MidoriBrowser * * Retrieves a proxy array representing the respective proxy items. * * Return value: the proxy #KatzeArray * * Deprecated: 0.2.5: Use midori_browser_get_proxy_item instead. **/ KatzeArray* midori_browser_get_proxy_array (MidoriBrowser* browser) { g_return_val_if_fail (MIDORI_IS_BROWSER (browser), NULL); return browser->proxy_array; } /** * midori_browser_get_for_widget: * @widget: a #GtkWidget * * Determines the browser appropriate for the specified widget. * * Return value: a #MidoriBrowser * * Since 0.1.7 **/ MidoriBrowser* midori_browser_get_for_widget (GtkWidget* widget) { gpointer browser; g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL); browser = gtk_widget_get_toplevel (GTK_WIDGET (widget)); if (!MIDORI_IS_BROWSER (browser)) { if (!GTK_IS_WINDOW (browser)) return NULL; browser = gtk_window_get_transient_for (GTK_WINDOW (browser)); if (!MIDORI_IS_BROWSER (browser)) return NULL; } return MIDORI_BROWSER (browser); } /** * midori_browser_quit: * @browser: a #MidoriBrowser * * Quits the browser, including any other browser windows. * * This function relys on the application implementing * the MidoriBrowser::quit signal. If the browser was added * to the MidoriApp, this is handled automatically. **/ void midori_browser_quit (MidoriBrowser* browser) { g_return_if_fail (MIDORI_IS_BROWSER (browser)); g_signal_emit (browser, signals[QUIT], 0); }