8224 lines
291 KiB
C
8224 lines
291 KiB
C
/*
|
|
Copyright (C) 2007-2011 Christian Dywan <christian@twotoasts.de>
|
|
Copyright (C) 2008 Dale Whittaker <dayul@users.sf.net>
|
|
Copyright (C) 2009 Jérôme Geulfucci <jeromeg@xfce.org>
|
|
|
|
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 <glib/gi18n.h>
|
|
#include <glib/gstdio.h>
|
|
#include <gdk/gdkkeysyms.h>
|
|
#include <string.h>
|
|
|
|
#include <config.h>
|
|
|
|
#ifdef HAVE_GRANITE
|
|
#include <granite.h>
|
|
#endif
|
|
|
|
#if HAVE_ZEITGEIST
|
|
#include <zeitgeist.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_HILDON_2_2
|
|
#include <dbus/dbus.h>
|
|
#include <mce/mode-names.h>
|
|
#include <mce/dbus-names.h>
|
|
#define MCE_SIGNAL_MATCH "type='signal'," \
|
|
"sender='" MCE_SERVICE "'," \
|
|
"path='" MCE_SIGNAL_PATH "'," \
|
|
"interface='" MCE_SIGNAL_IF "'"
|
|
#include <gdk/gdkx.h>
|
|
#include <X11/Xatom.h>
|
|
#endif
|
|
|
|
#include <sqlite3.h>
|
|
|
|
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;
|
|
GKeyFile* speeddial;
|
|
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)
|
|
{
|
|
gboolean loading = midori_view_get_load_status (view) != MIDORI_LOAD_FINISHED;
|
|
gboolean can_reload = midori_view_can_reload (view);
|
|
GtkAction* action;
|
|
GSList* proxies;
|
|
|
|
_action_set_sensitive (browser, "Reload", can_reload);
|
|
_action_set_sensitive (browser, "Stop", can_reload && loading);
|
|
_action_set_sensitive (browser, "Back", midori_view_can_go_back (view));
|
|
_action_set_sensitive (browser, "Forward", midori_view_can_go_forward (view));
|
|
proxies = gtk_action_get_proxies (_action_by_name (browser, "Forward"));
|
|
for (; proxies != NULL; proxies = g_slist_next (proxies))
|
|
if (GTK_IS_TOOL_ITEM (proxies->data))
|
|
gtk_widget_set_visible (proxies->data, 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, "Print", midori_view_can_print (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_set_sensitive (browser, "Find",
|
|
midori_view_can_find (view));
|
|
_action_set_sensitive (browser, "FindNext",
|
|
midori_view_can_find (view));
|
|
_action_set_sensitive (browser, "FindPrevious",
|
|
midori_view_can_find (view));
|
|
midori_findbar_set_can_find (MIDORI_FINDBAR (browser->find),
|
|
midori_view_can_find (view));
|
|
|
|
action = _action_by_name (browser, "ReloadStop");
|
|
if (!loading)
|
|
{
|
|
g_object_set (action,
|
|
"stock-id", GTK_STOCK_REFRESH,
|
|
"tooltip", _("Reload the current page"),
|
|
"sensitive", can_reload, 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);
|
|
}
|
|
|
|
#if HAVE_HILDON
|
|
#if HILDON_CHECK_VERSION (2, 2, 0)
|
|
hildon_gtk_window_set_progress_indicator (GTK_WINDOW (browser), loading);
|
|
#endif
|
|
#else
|
|
gtk_widget_set_sensitive (browser->throbber, loading);
|
|
katze_throbber_set_animated (KATZE_THROBBER (browser->throbber), loading);
|
|
#endif
|
|
|
|
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)
|
|
{
|
|
MidoriLocationAction* action;
|
|
gdouble progress;
|
|
|
|
action = MIDORI_LOCATION_ACTION (_action_by_name (browser, "Location"));
|
|
progress = midori_view_get_progress (view);
|
|
/* When we are finished, we don't want to *see* progress anymore */
|
|
if (midori_view_get_load_status (view) == MIDORI_LOAD_FINISHED)
|
|
progress = 0.0;
|
|
/* When loading we want to see at minimum 10% progress */
|
|
else
|
|
progress = CLAMP (progress, 0.1, 1.0);
|
|
midori_location_action_set_progress (action, progress);
|
|
}
|
|
|
|
/**
|
|
* 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)
|
|
{
|
|
#if 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 (sokoke_is_app_or_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 ("<b>%s</b>", 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)
|
|
|
|
{
|
|
guint64 total_size = webkit_download_get_total_size (download);
|
|
GFile* file = g_file_new_for_uri (uri);
|
|
GFile* folder = g_file_get_parent (file);
|
|
GError* error = NULL;
|
|
GFileInfo* info = g_file_query_filesystem_info (folder,
|
|
G_FILE_ATTRIBUTE_FILESYSTEM_FREE, NULL, &error);
|
|
guint64 free_space = g_file_info_get_attribute_uint64 (info,
|
|
G_FILE_ATTRIBUTE_FILESYSTEM_FREE);
|
|
gchar* path = g_file_get_path (folder);
|
|
gboolean can_write = g_access (path, W_OK) == 0;
|
|
g_free (path);
|
|
g_object_unref (file);
|
|
g_object_unref (folder);
|
|
if (free_space < total_size || !can_write)
|
|
{
|
|
gchar* message;
|
|
gchar* detailed_message;
|
|
|
|
if (!can_write)
|
|
{
|
|
message = g_strdup_printf (
|
|
_("The file \"%s\" can't be saved in this folder."), &uri[7]);
|
|
detailed_message = g_strdup_printf (
|
|
_("You don't have permission to write in this location."));
|
|
}
|
|
else if (free_space < total_size)
|
|
{
|
|
gchar* total_size_string = g_format_size (total_size);
|
|
gchar* free_space_string = g_format_size (free_space);
|
|
message = g_strdup_printf (
|
|
_("There is not enough free space to download \"%s\"."),
|
|
&uri[7]);
|
|
detailed_message = g_strdup_printf (
|
|
_("The file needs %s but only %s are left."),
|
|
total_size_string, free_space_string);
|
|
g_free (total_size_string);
|
|
g_free (free_space_string);
|
|
}
|
|
else
|
|
g_assert_not_reached ();
|
|
|
|
sokoke_message_dialog (GTK_MESSAGE_ERROR, message, detailed_message, FALSE);
|
|
g_free (message);
|
|
g_free (detailed_message);
|
|
g_object_unref (download);
|
|
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 gchar*
|
|
midori_browser_get_folder_for_uri (MidoriBrowser* browser,
|
|
const gchar* uri)
|
|
{
|
|
if (uri)
|
|
{
|
|
/* Base the start folder on the current view's uri if it is local */
|
|
gchar* filename = g_filename_from_uri (uri, NULL, NULL);
|
|
if (filename)
|
|
{
|
|
gchar* dirname = g_path_get_dirname (filename);
|
|
g_free (filename);
|
|
if (dirname && g_file_test (dirname, G_FILE_TEST_IS_DIR))
|
|
return dirname;
|
|
}
|
|
}
|
|
return katze_object_get_string (browser->settings, "download-folder");
|
|
}
|
|
|
|
static gchar*
|
|
midori_browser_fixup_filename (gchar* filename)
|
|
{
|
|
#ifdef G_OS_WIN32
|
|
g_strdelimit (filename, "/\\<>:\"|?*", '_');
|
|
#else
|
|
g_strdelimit (filename, "/", '_');
|
|
#endif
|
|
return filename;
|
|
}
|
|
|
|
static gchar*
|
|
midori_browser_get_filename_suggestion_for_uri (MidoriView* view,
|
|
const gchar* uri)
|
|
{
|
|
/* Try to provide a good default filename, UTF-8 encoded */
|
|
gchar* filename = soup_uri_decode (uri);
|
|
gchar* last_slash = g_strrstr (filename, "/") + 1;
|
|
/* Take the rest of the URI if needed */
|
|
if (*last_slash == '\0')
|
|
{
|
|
const gchar* extension = midori_view_fallback_extension (view, NULL);
|
|
gchar* guessed;
|
|
last_slash = midori_browser_fixup_filename (filename);
|
|
guessed = g_strconcat (filename, extension, NULL);
|
|
g_free (filename);
|
|
return guessed;
|
|
}
|
|
last_slash = g_strdup (last_slash);
|
|
g_free (filename);
|
|
return midori_browser_fixup_filename (last_slash);
|
|
}
|
|
|
|
static gchar*
|
|
midori_browser_download_prepare_filename (gchar* filename);
|
|
|
|
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;
|
|
|
|
g_mkdir (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;
|
|
|
|
/* FIXME: mime type fallback should respect the resource's type */
|
|
gchar* sub_filename = midori_browser_get_filename_suggestion_for_uri (
|
|
view, webkit_web_resource_get_uri (resource));
|
|
gchar* sub_path = g_build_filename (folder, sub_filename, NULL);
|
|
sub_path = midori_browser_download_prepare_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 (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_browser_get_folder_for_uri (browser, uri);
|
|
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), dirname);
|
|
g_free (dirname);
|
|
}
|
|
|
|
if (uri == NULL)
|
|
uri = midori_view_get_display_uri (view);
|
|
if (!file_only && !g_str_equal (title, uri))
|
|
filename = midori_browser_fixup_filename (g_strdup (title));
|
|
else
|
|
filename = midori_browser_get_filename_suggestion_for_uri (view, uri);
|
|
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 gchar*
|
|
midori_browser_speed_dial_get_next_free_slot (MidoriView* view)
|
|
{
|
|
MidoriBrowser* browser = midori_browser_get_for_widget (GTK_WIDGET (view));
|
|
GKeyFile* key_file;
|
|
guint slot_count = 0, slot = 1, i;
|
|
gchar** groups;
|
|
|
|
g_object_get (browser, "speed-dial", &key_file, NULL);
|
|
|
|
groups = g_key_file_get_groups (key_file, NULL);
|
|
for (i = 0; groups[i]; i++)
|
|
{
|
|
if (g_key_file_has_key (key_file, groups[i], "uri", NULL))
|
|
slot_count++;
|
|
}
|
|
|
|
while (slot <= slot_count)
|
|
{
|
|
gchar* dial_id = g_strdup_printf ("Dial %d", slot);
|
|
if (!g_key_file_has_group (key_file, dial_id))
|
|
{
|
|
g_free (dial_id);
|
|
return g_strdup_printf ("s%d", slot);
|
|
}
|
|
g_free (dial_id);
|
|
slot++;
|
|
}
|
|
return g_strdup_printf ("s%d", slot_count + 1);
|
|
}
|
|
|
|
static void
|
|
midori_browser_add_speed_dial (MidoriBrowser* browser)
|
|
{
|
|
GdkPixbuf* img;
|
|
GtkWidget* view = midori_browser_get_current_tab (browser);
|
|
gchar* uri = g_strdup (midori_view_get_display_uri (MIDORI_VIEW (view)));
|
|
gchar* title = g_strdup (midori_view_get_display_title (MIDORI_VIEW (view)));
|
|
gchar* slot_id = midori_browser_speed_dial_get_next_free_slot (MIDORI_VIEW (view));
|
|
|
|
if (slot_id == NULL)
|
|
{
|
|
g_free (uri);
|
|
g_free (title);
|
|
return;
|
|
}
|
|
|
|
if ((img = midori_view_get_snapshot (MIDORI_VIEW (view), 240, 160)))
|
|
{
|
|
GKeyFile* key_file;
|
|
gchar* dial_id = g_strdup_printf ("Dial %s", slot_id + 1);
|
|
gchar* file_path = sokoke_build_thumbnail_path (uri);
|
|
gchar* thumb_dir = g_build_path (G_DIR_SEPARATOR_S, g_get_user_cache_dir (),
|
|
PACKAGE_NAME, "thumbnails", NULL);
|
|
g_object_get (browser, "speed-dial", &key_file, NULL);
|
|
|
|
g_key_file_set_string (key_file, dial_id, "uri", uri);
|
|
g_key_file_set_string (key_file, dial_id, "title", title);
|
|
|
|
if (!g_file_test (thumb_dir, G_FILE_TEST_EXISTS))
|
|
katze_mkdir_with_parents (thumb_dir, 0700);
|
|
|
|
gdk_pixbuf_save (img, file_path, "png", NULL, "compression", "7", NULL);
|
|
|
|
midori_view_save_speed_dial_config (MIDORI_VIEW (view), key_file);
|
|
|
|
g_object_unref (img);
|
|
g_free (file_path);
|
|
g_free (thumb_dir);
|
|
g_free (dial_id);
|
|
}
|
|
g_free (uri);
|
|
g_free (title);
|
|
g_free (slot_id);
|
|
}
|
|
|
|
|
|
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 gchar*
|
|
midori_browser_download_prepare_filename (gchar* filename)
|
|
{
|
|
if (g_file_test (filename, G_FILE_TEST_EXISTS))
|
|
{
|
|
int i = 1;
|
|
const gchar* dot_pos;
|
|
const gchar* last_separator;
|
|
gchar* serial;
|
|
GString* tmp_filename;
|
|
gssize position;
|
|
|
|
last_separator = strrchr (filename, G_DIR_SEPARATOR);
|
|
dot_pos = strrchr ((last_separator) ? last_separator : filename, '.');
|
|
position = dot_pos ? (dot_pos - filename) : (gssize) strlen (filename);
|
|
tmp_filename = g_string_new (NULL);
|
|
|
|
do
|
|
{
|
|
serial = g_strdup_printf ("-%d", i++);
|
|
g_string_assign (tmp_filename, filename);
|
|
g_string_insert (tmp_filename, position, serial);
|
|
g_free (serial);
|
|
} while (g_file_test (tmp_filename->str, G_FILE_TEST_EXISTS));
|
|
|
|
g_free (filename);
|
|
filename = g_string_free (tmp_filename, FALSE);
|
|
}
|
|
return filename;
|
|
}
|
|
|
|
static gchar*
|
|
midori_browser_download_prepare_destination_uri (WebKitDownload* download,
|
|
const gchar* folder)
|
|
{
|
|
gchar* suggested_filename;
|
|
GFile* file_source;
|
|
gchar* file_basename;
|
|
gchar* download_dir = NULL;
|
|
gchar* destination_uri;
|
|
gchar* destination_filename;
|
|
|
|
suggested_filename = sokoke_get_download_filename (download);
|
|
file_source = g_file_new_for_uri (suggested_filename);
|
|
g_free (suggested_filename);
|
|
file_basename = g_file_get_basename (file_source);
|
|
if (folder == NULL)
|
|
download_dir = midori_view_get_tmp_dir ();
|
|
else
|
|
download_dir = (gchar*)folder;
|
|
destination_filename = g_build_filename (download_dir, file_basename, NULL);
|
|
destination_filename = midori_browser_download_prepare_filename (destination_filename);
|
|
destination_uri = g_filename_to_uri (destination_filename, NULL, NULL);
|
|
|
|
g_free (file_basename);
|
|
if (folder == NULL)
|
|
g_free (download_dir);
|
|
g_free (destination_filename);
|
|
g_object_unref (file_source);
|
|
|
|
return destination_uri;
|
|
}
|
|
|
|
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 = GPOINTER_TO_INT (
|
|
g_object_get_data (G_OBJECT (download), "midori-download-type"));
|
|
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_browser_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_browser_get_folder_for_uri (browser,
|
|
webkit_download_get_uri (download));
|
|
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 = sokoke_get_download_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_browser_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
|
|
&& !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
|
|
*/
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_SPEED_DIAL,
|
|
g_param_spec_pointer (
|
|
"speed-dial",
|
|
"Speeddial",
|
|
"Pointer to key-value object with speed dial items",
|
|
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 && sokoke_is_app_or_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",
|
|
"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_app_find_res_filename ("faq.css");
|
|
stylesheet = NULL;
|
|
if (!g_file_get_contents (filename, &stylesheet, NULL, NULL))
|
|
{
|
|
katze_assign (filename, sokoke_find_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 (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 <christian@twotoasts.de>", NULL };
|
|
static const gchar* credits_documenters[] =
|
|
{ "Christian Dywan <christian@twotoasts.de>", NULL };
|
|
static const gchar* credits_artists[] =
|
|
{ "Nancy Runge <nancy@twotoasts.de>", 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 = sokoke_find_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-2011 Christian Dywan",
|
|
#else
|
|
gtk_show_about_dialog (GTK_WINDOW (browser),
|
|
"wrap-license", TRUE,
|
|
"copyright", "Copyright © 2007-2011 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.twotoasts.de",
|
|
"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 (sokoke_is_app_or_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"), "<Ctrl>n",
|
|
N_("Open a new window"), G_CALLBACK (_action_window_new_activate) },
|
|
{ "TabNew", STOCK_TAB_NEW,
|
|
NULL, "<Ctrl>t",
|
|
N_("Open a new tab"), G_CALLBACK (_action_tab_new_activate) },
|
|
{ "PrivateBrowsing", NULL,
|
|
N_("New P_rivate Browsing Window"), "<Ctrl><Shift>n",
|
|
NULL, G_CALLBACK (_action_private_browsing_activate), },
|
|
{ "Open", GTK_STOCK_OPEN,
|
|
NULL, "<Ctrl>o",
|
|
N_("Open a file"), G_CALLBACK (_action_open_activate) },
|
|
{ "SaveAs", GTK_STOCK_SAVE_AS,
|
|
N_("_Save Page As…"), "<Ctrl>s",
|
|
N_("Save to a file"), G_CALLBACK (_action_save_as_activate) },
|
|
{ "AddSpeedDial", NULL,
|
|
N_("Add to Speed _dial"), "<Ctrl>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"), "<Ctrl>w",
|
|
N_("Close the current tab"), G_CALLBACK (_action_tab_close_activate) },
|
|
{ "WindowClose", NULL,
|
|
N_("C_lose Window"), "<Ctrl><Shift>w",
|
|
NULL, G_CALLBACK (_action_window_close_activate) },
|
|
#ifdef HAVE_GRANITE
|
|
{ "Print", "document-export",
|
|
N_("_Share"), "<Ctrl>p",
|
|
NULL, G_CALLBACK (_action_print_activate) },
|
|
#else
|
|
{ "Print", GTK_STOCK_PRINT,
|
|
NULL, "<Ctrl>p",
|
|
N_("Print the current page"), G_CALLBACK (_action_print_activate) },
|
|
#endif
|
|
{ "Quit", GTK_STOCK_QUIT,
|
|
N_("Close a_ll Windows"), "<Ctrl><Shift>q",
|
|
NULL, G_CALLBACK (_action_quit_activate) },
|
|
|
|
{ "Edit", NULL, N_("_Edit"), NULL, NULL, G_CALLBACK (_action_edit_activate) },
|
|
{ "Undo", GTK_STOCK_UNDO,
|
|
NULL, "<Ctrl>z",
|
|
NULL, G_CALLBACK (_action_undo_activate) },
|
|
{ "Redo", GTK_STOCK_REDO,
|
|
NULL, "<Ctrl><Shift>z",
|
|
NULL, G_CALLBACK (_action_redo_activate) },
|
|
{ "Cut", GTK_STOCK_CUT,
|
|
NULL, "<Ctrl>x",
|
|
NULL, G_CALLBACK (_action_cut_activate) },
|
|
{ "Copy", GTK_STOCK_COPY,
|
|
NULL, "<Ctrl>c",
|
|
NULL, G_CALLBACK (_action_copy_activate) },
|
|
{ "Paste", GTK_STOCK_PASTE,
|
|
NULL, "<Ctrl>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, "<Ctrl>a",
|
|
NULL, G_CALLBACK (_action_select_all_activate) },
|
|
{ "Find", GTK_STOCK_FIND,
|
|
N_("_Find…"), "<Ctrl>f",
|
|
N_("Find a word or phrase in the page"), G_CALLBACK (_action_find_activate) },
|
|
{ "FindNext", GTK_STOCK_GO_FORWARD,
|
|
N_("Find _Next"), "<Ctrl>g",
|
|
NULL, G_CALLBACK (_action_find_next_activate) },
|
|
{ "FindPrevious", GTK_STOCK_GO_BACK,
|
|
N_("Find _Previous"), "<Ctrl><Shift>g",
|
|
NULL, G_CALLBACK (_action_find_previous_activate) },
|
|
{ "Preferences", GTK_STOCK_PREFERENCES,
|
|
NULL, "<Ctrl><Alt>p",
|
|
N_("Configure the application preferences"), G_CALLBACK (_action_preferences_activate) },
|
|
|
|
{ "View", NULL, N_("_View") },
|
|
{ "Toolbars", NULL, N_("_Toolbars") },
|
|
{ "Reload", GTK_STOCK_REFRESH,
|
|
NULL, "<Ctrl>r",
|
|
N_("Reload the current page"), G_CALLBACK (_action_reload_stop_activate) },
|
|
{ "ReloadUncached", GTK_STOCK_REFRESH,
|
|
N_("Reload page without caching"), "<Ctrl><Shift>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, "<Ctrl>r",
|
|
N_("Reload the current page"), G_CALLBACK (_action_reload_stop_activate) },
|
|
{ "ZoomIn", GTK_STOCK_ZOOM_IN,
|
|
NULL, "<Ctrl>plus",
|
|
N_("Increase the zoom level"), G_CALLBACK (_action_zoom_activate) },
|
|
{ "ZoomOut", GTK_STOCK_ZOOM_OUT,
|
|
NULL, "<Ctrl>minus",
|
|
N_("Decrease the zoom level"), G_CALLBACK (_action_zoom_activate) },
|
|
{ "ZoomNormal", GTK_STOCK_ZOOM_100,
|
|
NULL, "<Ctrl>0",
|
|
NULL, G_CALLBACK (_action_zoom_activate) },
|
|
{ "Encoding", NULL, N_("_Encoding") },
|
|
{ "SourceView", NULL,
|
|
N_("View So_urce"), "<Ctrl><Alt>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"), "<Ctrl><Alt>R",
|
|
NULL, G_CALLBACK (_action_readable_activate) },
|
|
|
|
{ "Go", NULL, N_("_Go") },
|
|
{ "Back", GTK_STOCK_GO_BACK,
|
|
NULL, "<Alt>Left",
|
|
N_("Go back to the previous page"), G_CALLBACK (_action_navigation_activate) },
|
|
{ "Forward", GTK_STOCK_GO_FORWARD,
|
|
NULL, "<Alt>Right",
|
|
N_("Go forward to the next page"), G_CALLBACK (_action_navigation_activate) },
|
|
{ "Previous", GTK_STOCK_MEDIA_PREVIOUS,
|
|
NULL, "<Alt><Shift>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, "<Alt><Shift>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) },
|
|
{ "Homepage", GTK_STOCK_HOME,
|
|
N_("_Homepage"), "<Alt>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"), "<Ctrl><Shift>t",
|
|
NULL, G_CALLBACK (_action_undo_tab_close_activate) },
|
|
|
|
{ "BookmarkAdd", STOCK_BOOKMARK_ADD,
|
|
NULL, "<Ctrl>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"), "<Ctrl><Alt>s",
|
|
NULL, G_CALLBACK (_action_manage_search_engines_activate) },
|
|
{ "ClearPrivateData", NULL,
|
|
N_("_Clear Private Data"), "<Ctrl><Shift>Delete",
|
|
NULL, G_CALLBACK (_action_clear_private_data_activate) },
|
|
{ "InspectPage", NULL,
|
|
N_("_Inspect Page"), "<Ctrl><Shift>i",
|
|
NULL, G_CALLBACK (_action_inspect_page_activate) },
|
|
|
|
{ "TabPrevious", GTK_STOCK_GO_BACK,
|
|
N_("_Previous Tab"), "<Ctrl>Page_Up",
|
|
NULL, G_CALLBACK (_action_tab_previous_activate) },
|
|
{ "TabNext", GTK_STOCK_GO_FORWARD,
|
|
N_("_Next Tab"), "<Ctrl>Page_Down",
|
|
NULL, G_CALLBACK (_action_tab_next_activate) },
|
|
{ "TabMoveBackward", NULL, N_("Move Tab _Backward"), "<Ctrl><Shift>Page_Up",
|
|
NULL, G_CALLBACK (_action_tab_move_backward_activate) },
|
|
{ "TabMoveForward", NULL, N_("_Move Tab Forward"), "<Ctrl><Shift>Page_Down",
|
|
NULL, G_CALLBACK (_action_tab_move_forward_activate) },
|
|
{ "TabCurrent", NULL,
|
|
N_("Focus _Current Tab"), "<Ctrl><Alt>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"), "<Ctrl>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 =
|
|
"<ui>"
|
|
"<menubar>"
|
|
"<menu action='File'>"
|
|
"<menuitem action='WindowNew'/>"
|
|
"<menuitem action='TabNew'/>"
|
|
"<menuitem action='PrivateBrowsing'/>"
|
|
"<separator/>"
|
|
"<menuitem action='Open'/>"
|
|
"<separator/>"
|
|
"<menuitem action='SaveAs'/>"
|
|
"<menuitem action='AddSpeedDial'/>"
|
|
"<menuitem action='AddDesktopShortcut'/>"
|
|
"<separator/>"
|
|
"<menuitem action='TabClose'/>"
|
|
"<menuitem action='WindowClose'/>"
|
|
"<separator/>"
|
|
"<menuitem action='Print'/>"
|
|
"<menuitem action='BookmarksImport'/>"
|
|
"<menuitem action='BookmarksExport'/>"
|
|
"<separator/>"
|
|
"<menuitem action='Quit'/>"
|
|
"</menu>"
|
|
"<menu action='Edit'>"
|
|
"<menuitem action='Undo'/>"
|
|
"<menuitem action='Redo'/>"
|
|
"<separator/>"
|
|
"<menuitem action='Cut'/>"
|
|
"<menuitem action='Copy'/>"
|
|
"<menuitem action='Paste'/>"
|
|
"<menuitem action='Delete'/>"
|
|
"<separator/>"
|
|
"<menuitem action='SelectAll'/>"
|
|
"<separator/>"
|
|
"<menuitem action='Find'/>"
|
|
"<menuitem action='FindNext'/>"
|
|
#ifndef G_OS_WIN32
|
|
"<separator/>"
|
|
"<menuitem action='Preferences'/>"
|
|
#endif
|
|
"</menu>"
|
|
"<menu action='View'>"
|
|
"<menu action='Toolbars'>"
|
|
"<menuitem action='Menubar'/>"
|
|
"<menuitem action='Navigationbar'/>"
|
|
"<menuitem action='Bookmarkbar'/>"
|
|
"<menuitem action='Statusbar'/>"
|
|
"</menu>"
|
|
"<menuitem action='Panel'/>"
|
|
"<separator/>"
|
|
"<menuitem action='Stop'/>"
|
|
"<menuitem action='Reload'/>"
|
|
"<separator/>"
|
|
"<menuitem action='ZoomIn'/>"
|
|
"<menuitem action='ZoomOut'/>"
|
|
"<menuitem action='ZoomNormal'/>"
|
|
"<separator/>"
|
|
"<menu action='Encoding'>"
|
|
"<menuitem action='EncodingAutomatic'/>"
|
|
"<menuitem action='EncodingChinese'/>"
|
|
"<menuitem action='EncodingChineseSimplified'/>"
|
|
"<menuitem action='EncodingJapanese'/>"
|
|
"<menuitem action='EncodingKorean'/>"
|
|
"<menuitem action='EncodingRussian'/>"
|
|
"<menuitem action='EncodingUnicode'/>"
|
|
"<menuitem action='EncodingWestern'/>"
|
|
"<menuitem action='EncodingCustom'/>"
|
|
"</menu>"
|
|
"<menuitem action='SourceView'/>"
|
|
"<menuitem action='Fullscreen'/>"
|
|
"<menuitem action='Readable'/>"
|
|
"</menu>"
|
|
"<menu action='Go'>"
|
|
"<menuitem action='Back'/>"
|
|
"<menuitem action='Forward'/>"
|
|
"<menuitem action='Previous'/>"
|
|
"<menuitem action='Next'/>"
|
|
"<menuitem action='Homepage'/>"
|
|
"<menuitem action='Location'/>"
|
|
"<menuitem action='Search'/>"
|
|
"<menuitem action='Trash'/>"
|
|
"</menu>"
|
|
"<menuitem action='Bookmarks'/>"
|
|
"<menuitem action='Tools'/>"
|
|
"<menuitem action='Window'/>"
|
|
"<menu action='Help'>"
|
|
"<menuitem action='HelpFAQ'/>"
|
|
"<menuitem action='HelpBugs'/>"
|
|
"<separator/>"
|
|
"<menuitem action='About'/>"
|
|
"</menu>"
|
|
/* For accelerators to work all actions need to be used
|
|
*somewhere* in the UI definition */
|
|
/* These also show up in Unity's HUD */
|
|
"<menu action='Dummy'>"
|
|
"<menuitem action='TabMoveBackward'/>"
|
|
"<menuitem action='TabMoveForward'/>"
|
|
"<menuitem action='ScrollLeft'/>"
|
|
"<menuitem action='ScrollDown'/>"
|
|
"<menuitem action='ScrollUp'/>"
|
|
"<menuitem action='ScrollRight'/>"
|
|
"<menuitem action='FindPrevious'/>"
|
|
"<menuitem action='BookmarkAdd'/>"
|
|
"<menuitem action='BookmarkFolderAdd'/>"
|
|
"<menuitem action='ManageSearchEngines'/>"
|
|
"<menuitem action='ClearPrivateData'/>"
|
|
"<menuitem action='TabPrevious'/>"
|
|
"<menuitem action='TabNext'/>"
|
|
"<menuitem action='TabCurrent'/>"
|
|
"<menuitem action='NextView'/>"
|
|
"<menuitem action='TabMinimize'/>"
|
|
"<menuitem action='TabDuplicate'/>"
|
|
"<menuitem action='TabCloseOther'/>"
|
|
"<menuitem action='LastSession'/>"
|
|
"<menuitem action='UndoTabClose'/>"
|
|
"<menuitem action='TrashEmpty'/>"
|
|
"<menuitem action='Preferences'/>"
|
|
"<menuitem action='InspectPage'/>"
|
|
"<menuitem action='ReloadUncached'/>"
|
|
"<menuitem action='CaretBrowsing'/>"
|
|
"</menu>"
|
|
"</menubar>"
|
|
"<toolbar name='toolbar_navigation'>"
|
|
"</toolbar>"
|
|
"</ui>";
|
|
|
|
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->speeddial = 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 ("<Manual>/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, "<Ctrl>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, "<Ctrl>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->speeddial = 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 (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));
|
|
}
|
|
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->speeddial = g_value_get_pointer (value);
|
|
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->speeddial);
|
|
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);
|
|
}
|