/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8; -*- */
/*
* Copyright © 2004 Tommi Komulainen
* Copyright © 2004, 2005 Christian Persch
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
#include "config.h"
#include "ephy-find-toolbar.h"
#include "ephy-debug.h"
#include "ephy-embed-utils.h"
#include <gdk/gdkkeysyms.h>
#include <glib/gi18n.h>
#include <gtk/gtk.h>
#include <string.h>
#ifdef HAVE_WEBKIT2
#include <webkit2/webkit2.h>
#else
#include <webkit/webkit.h>
#endif
#define EPHY_FIND_TOOLBAR_GET_PRIVATE(object) (G_TYPE_INSTANCE_GET_PRIVATE ((object),EPHY_TYPE_FIND_TOOLBAR, EphyFindToolbarPrivate))
struct _EphyFindToolbarPrivate
{
EphyWindow *window;
WebKitWebView *web_view;
#ifdef HAVE_WEBKIT2
WebKitFindController *controller;
#endif
GtkWidget *entry;
GtkWidget *label;
GtkToolItem *next;
GtkToolItem *prev;
GtkToolItem *sep;
GtkToolItem *status_item;
GtkWidget *status_label;
GtkWidget *case_sensitive;
gulong set_focus_handler;
guint source_id;
guint find_again_source_id;
guint find_source_id;
char *find_string;
guint preedit_changed : 1;
guint prevent_activate : 1;
guint activated : 1;
guint links_only : 1;
guint typing_ahead : 1;
};
enum
{
PROP_0,
PROP_WINDOW
};
enum
{
NEXT,
PREVIOUS,
CLOSE,
LAST_SIGNAL
};
typedef enum
{
EPHY_FIND_RESULT_FOUND = 0,
EPHY_FIND_RESULT_NOTFOUND = 1,
EPHY_FIND_RESULT_FOUNDWRAPPED = 2
} EphyFindResult;
typedef enum
{
EPHY_FIND_DIRECTION_NEXT,
EPHY_FIND_DIRECTION_PREV
} EphyFindDirection;
static guint signals[LAST_SIGNAL];
/* private functions */
static void
scroll_lines (WebKitWebView *web_view,
int num_lines)
{
#ifdef HAVE_WEBKIT2
/* TODO: Scroll API? */
#else
GtkScrolledWindow *scrolled_window;
GtkAdjustment *vadj;
gdouble value;
scrolled_window = GTK_SCROLLED_WINDOW (gtk_widget_get_parent (GTK_WIDGET (web_view)));
vadj = gtk_scrolled_window_get_vadjustment (scrolled_window);
value = gtk_adjustment_get_value (vadj) + (num_lines * gtk_adjustment_get_step_increment (vadj));
gtk_adjustment_set_value (vadj, value);
#endif
}
static void
scroll_pages (WebKitWebView *web_view,
int num_pages)
{
#ifdef HAVE_WEBKIT2
/* TODO: Scroll API */
#else
GtkScrolledWindow *scrolled_window;
GtkAdjustment *vadj;
gdouble value;
scrolled_window = GTK_SCROLLED_WINDOW (gtk_widget_get_parent (GTK_WIDGET (web_view)));
vadj = gtk_scrolled_window_get_vadjustment (scrolled_window);
value = gtk_adjustment_get_value (vadj) + (num_pages * gtk_adjustment_get_page_increment (vadj));
gtk_adjustment_set_value (vadj, value);
#endif
}
static gboolean
set_status_notfound_cb (EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv;
PangoFontDescription *pango_desc = NULL;
priv = toolbar->priv;
pango_desc = pango_font_description_new ();
gtk_widget_override_font (priv->status_label, pango_desc);
pango_font_description_free (pango_desc);
priv->source_id = 0;
return FALSE;
}
static void
set_status (EphyFindToolbar *toolbar,
EphyFindResult result)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
char *text = NULL;
PangoFontDescription *pango_desc = NULL;
switch (result)
{
case EPHY_FIND_RESULT_FOUND:
text = NULL;
break;
case EPHY_FIND_RESULT_NOTFOUND:
{
text = _("Not found");
pango_desc = pango_font_description_new ();
pango_font_description_set_weight (pango_desc, PANGO_WEIGHT_BOLD);
gtk_widget_override_font (priv->status_label, pango_desc);
pango_font_description_free (pango_desc);
gtk_widget_error_bell (GTK_WIDGET (priv->window));
priv->source_id = g_timeout_add (500, (GSourceFunc) set_status_notfound_cb, toolbar);
}
break;
case EPHY_FIND_RESULT_FOUNDWRAPPED:
text = _("Wrapped");
break;
}
gtk_label_set_text (GTK_LABEL (priv->status_label),
text != NULL ? text : "");
g_object_set (priv->sep, "visible", text != NULL, NULL);
g_object_set (priv->status_item, "visible", text != NULL, NULL);
}
static void
clear_status (EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
gtk_widget_hide (GTK_WIDGET (priv->sep));
gtk_widget_hide (GTK_WIDGET (priv->status_item));
gtk_label_set_text (GTK_LABEL (priv->status_label), "");
gtk_label_set_text (GTK_LABEL (priv->label),
priv->links_only ? _("Find links:") : _("Find:"));
}
/* Code adapted from gtktreeview.c:gtk_tree_view_key_press() and
* gtk_tree_view_real_start_interactive_seach()
*/
static gboolean
tab_search_key_press_cb (EphyEmbed *embed,
GdkEventKey *event,
EphyFindToolbar *toolbar)
{
GtkWidget *widget = (GtkWidget *) toolbar;
g_return_val_if_fail (event != NULL, FALSE);
/* check for / and ' which open the find toolbar in text resp. link mode */
if (gtk_widget_get_visible (widget) == FALSE)
{
if (event->keyval == GDK_KEY_slash)
{
ephy_find_toolbar_open (toolbar, FALSE, TRUE);
return TRUE;
}
else if (event->keyval == GDK_KEY_apostrophe)
{
ephy_find_toolbar_open (toolbar, TRUE, TRUE);
return TRUE;
}
}
return FALSE;
}
static void
find_next_cb (EphyFindToolbar *toolbar)
{
g_signal_emit (toolbar, signals[NEXT], 0);
}
static void
find_prev_cb (EphyFindToolbar *toolbar)
{
g_signal_emit (toolbar, signals[PREVIOUS], 0);
}
#ifndef HAVE_WEBKIT2
static void
ephy_find_toolbar_mark_matches (EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
WebKitWebView *web_view = priv->web_view;
gboolean case_sensitive;
case_sensitive = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (priv->case_sensitive));
webkit_web_view_unmark_text_matches (web_view);
if (priv->find_string != NULL && priv->find_string[0] != '\0')
webkit_web_view_mark_text_matches (web_view,
priv->find_string,
case_sensitive,
0);
webkit_web_view_set_highlight_text_matches (web_view, TRUE);
}
#endif
#ifdef HAVE_WEBKIT2
static void
real_find (EphyFindToolbarPrivate *priv,
EphyFindDirection direction)
{
WebKitFindOptions options = WEBKIT_FIND_OPTIONS_NONE;
if (!g_strcmp0 (priv->find_string, ""))
return;
if (!gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (priv->case_sensitive)))
options |= WEBKIT_FIND_OPTIONS_CASE_INSENSITIVE;
if (direction == EPHY_FIND_DIRECTION_PREV)
options |= WEBKIT_FIND_OPTIONS_BACKWARDS;
webkit_find_controller_search (priv->controller, priv->find_string, options, G_MAXUINT);
}
#else
static EphyFindResult
real_find (EphyFindToolbarPrivate *priv,
EphyFindDirection direction)
{
WebKitWebView *web_view = priv->web_view;
gboolean case_sensitive = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (priv->case_sensitive));
gboolean forward = (direction == EPHY_FIND_DIRECTION_NEXT);
if (!priv->find_string || !g_strcmp0 (priv->find_string, ""))
return EPHY_FIND_RESULT_NOTFOUND;
if (!webkit_web_view_search_text
(web_view, priv->find_string, case_sensitive, forward, FALSE)) {
/* not found, try to wrap */
if (!webkit_web_view_search_text
(web_view, priv->find_string, case_sensitive, forward, TRUE)) {
/* there's no result */
return EPHY_FIND_RESULT_NOTFOUND;
} else {
/* found wrapped */
return EPHY_FIND_RESULT_FOUNDWRAPPED;
}
}
return EPHY_FIND_RESULT_FOUND;
}
#endif
static gboolean
do_search (EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
#ifndef HAVE_WEBKIT2
EphyFindResult result;
#endif
priv->find_source_id = 0;
#ifdef HAVE_WEBKIT2
real_find (priv, EPHY_FIND_DIRECTION_NEXT);
#else
ephy_find_toolbar_mark_matches (toolbar);
result = real_find (priv, EPHY_FIND_DIRECTION_NEXT);
set_status (toolbar, result);
#endif
return FALSE;
}
#ifdef HAVE_WEBKIT2
static void
found_text_cb (WebKitFindController *controller,
guint n_matches,
EphyFindToolbar *toolbar)
{
WebKitFindOptions options;
EphyFindResult result;
options = webkit_find_controller_get_options (controller);
/* FIXME: it's not possible to remove the wrap flag, so the status is now always wrapped. */
result = options & WEBKIT_FIND_OPTIONS_WRAP_AROUND ? EPHY_FIND_RESULT_FOUNDWRAPPED : EPHY_FIND_RESULT_FOUND;
set_status (toolbar, result);
}
static void
failed_to_find_text_cb (WebKitFindController *controller,
EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
WebKitFindOptions options;
options = webkit_find_controller_get_options (controller);
if (options & WEBKIT_FIND_OPTIONS_WRAP_AROUND) {
set_status (toolbar, EPHY_FIND_RESULT_NOTFOUND);
return;
}
options |= WEBKIT_FIND_OPTIONS_WRAP_AROUND;
webkit_find_controller_search (controller, priv->find_string, options, G_MAXUINT);
}
#endif
static void
entry_changed_cb (GtkEntry *entry,
EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
g_free (priv->find_string);
priv->find_string = g_strdup (gtk_entry_get_text (GTK_ENTRY (priv->entry)));
if (priv->find_source_id != 0) {
g_source_remove (priv->find_source_id);
priv->find_source_id = 0;
}
if (strlen (priv->find_string) == 0)
return;
priv->find_source_id = g_timeout_add (300, (GSourceFunc)do_search, toolbar);
}
static gboolean
ephy_find_toolbar_activate_link (EphyFindToolbar *toolbar,
GdkModifierType mask)
{
return FALSE;
}
static gboolean
entry_key_press_event_cb (GtkEntry *entry,
GdkEventKey *event,
EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
guint mask = gtk_accelerator_get_default_mod_mask ();
gboolean handled = FALSE;
if ((event->state & mask) == 0)
{
handled = TRUE;
switch (event->keyval)
{
case GDK_KEY_Up:
case GDK_KEY_KP_Up:
scroll_lines (priv->web_view, -1);
break;
case GDK_KEY_Down:
case GDK_KEY_KP_Down:
scroll_lines (priv->web_view, 1);
break;
case GDK_KEY_Page_Up:
case GDK_KEY_KP_Page_Up:
scroll_pages (priv->web_view, -1);
break;
case GDK_KEY_Page_Down:
case GDK_KEY_KP_Page_Down:
scroll_pages (priv->web_view, 1);
break;
case GDK_KEY_Escape:
/* Hide the toolbar when ESC is pressed */
ephy_find_toolbar_request_close (toolbar);
break;
default:
handled = FALSE;
break;
}
}
else if ((event->state & mask) == GDK_CONTROL_MASK &&
(event->keyval == GDK_KEY_Return ||
event->keyval == GDK_KEY_KP_Enter ||
event->keyval == GDK_KEY_ISO_Enter))
{
handled = ephy_find_toolbar_activate_link (toolbar, event->state);
}
else if ((event->state & mask) == GDK_SHIFT_MASK &&
(event->keyval == GDK_KEY_Return ||
event->keyval == GDK_KEY_KP_Enter ||
event->keyval == GDK_KEY_ISO_Enter))
{
handled = TRUE;
g_signal_emit (toolbar, signals[PREVIOUS], 0);
}
return handled;
}
static void
entry_activate_cb (GtkWidget *entry,
EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
if (priv->typing_ahead)
{
ephy_find_toolbar_activate_link (toolbar, 0);
}
else
{
g_signal_emit (toolbar, signals[NEXT], 0);
}
}
static void
case_sensitive_menu_toggled_cb (GtkWidget *check,
EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
gboolean case_sensitive;
case_sensitive = gtk_check_menu_item_get_active (GTK_CHECK_MENU_ITEM (check));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->case_sensitive),
case_sensitive);
}
static void
case_sensitive_toggled_cb (GtkWidget *check,
EphyFindToolbar *toolbar)
{
gboolean case_sensitive;
GtkWidget *proxy;
case_sensitive = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (check));
proxy = gtk_tool_item_get_proxy_menu_item (GTK_TOOL_ITEM (gtk_widget_get_parent (check)),
"menu-proxy");
if (proxy != NULL)
{
g_signal_handlers_block_by_func
(proxy, G_CALLBACK (case_sensitive_menu_toggled_cb), toolbar);
gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (proxy),
case_sensitive);
g_signal_handlers_unblock_by_func
(proxy, G_CALLBACK (case_sensitive_menu_toggled_cb), toolbar);
}
#ifdef HAVE_WEBKIT2
do_search (toolbar);
#else
ephy_find_toolbar_mark_matches (toolbar);
/*
* If we now use the stricter method (and are case sensitive),
* check that the current selection still matches. If not, find the
* next one.
* This currently requires going back and then forward, because
* there's no function in WebKit that would verify the current selection.
*/
if (case_sensitive)
{
EphyFindResult result;
result = real_find (toolbar->priv, EPHY_FIND_DIRECTION_PREV);
if (result != EPHY_FIND_RESULT_NOTFOUND)
result = real_find (toolbar->priv,
EPHY_FIND_DIRECTION_NEXT);
set_status (toolbar, result);
}
#endif
}
static gboolean
toolitem_create_menu_proxy_cb (GtkToolItem *toolitem,
EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
GtkWidget *checkbox_menu;
gboolean case_sensitive;
/* Create a menu item, and sync it */
checkbox_menu = gtk_check_menu_item_new_with_mnemonic (_("_Case sensitive"));
case_sensitive = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (priv->case_sensitive));
gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (checkbox_menu),
case_sensitive);
g_signal_connect (checkbox_menu, "toggled",
G_CALLBACK (case_sensitive_menu_toggled_cb), toolbar);
gtk_tool_item_set_proxy_menu_item (toolitem, "menu-proxy",
checkbox_menu);
return TRUE;
}
static void
set_focus_cb (EphyWindow *window,
GtkWidget *widget,
EphyFindToolbar *toolbar)
{
GtkWidget *wtoolbar = GTK_WIDGET (toolbar);
while (widget != NULL && widget != wtoolbar)
{
widget = gtk_widget_get_parent (widget);
}
/* if widget == toolbar, the new focus widget is in the toolbar */
if (widget != wtoolbar)
{
ephy_find_toolbar_request_close (toolbar);
}
}
static void
ephy_find_toolbar_set_window (EphyFindToolbar *toolbar,
EphyWindow *window)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
priv->window = window;
}
static void
ephy_find_toolbar_parent_set (GtkWidget *widget,
GtkWidget *previous_parent)
{
EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (widget);
EphyFindToolbarPrivate *priv = toolbar->priv;
GtkWidget *toplevel;
if (gtk_widget_get_parent (widget) != NULL &&
priv->set_focus_handler == 0)
{
toplevel = gtk_widget_get_toplevel (widget);
priv->set_focus_handler =
g_signal_connect (toplevel, "set-focus",
G_CALLBACK (set_focus_cb), toolbar);
}
}
static void
ephy_find_toolbar_grab_focus (GtkWidget *widget)
{
EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (widget);
EphyFindToolbarPrivate *priv = toolbar->priv;
gtk_widget_grab_focus (GTK_WIDGET (priv->entry));
}
static void
close_button_clicked_cb (GtkButton *button, EphyFindToolbar *toolbar)
{
ephy_find_toolbar_request_close (toolbar);
}
static void
ephy_find_toolbar_init (EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv;
GtkToolbar *gtoolbar;
GtkToolItem *item;
GtkWidget *alignment, *arrow, *box;
GtkWidget *checkbox, *close_button, *image;
priv = toolbar->priv = EPHY_FIND_TOOLBAR_GET_PRIVATE (toolbar);
gtoolbar = GTK_TOOLBAR (toolbar);
gtk_toolbar_set_style (gtoolbar, GTK_TOOLBAR_BOTH_HORIZ);
/* Find: |_____| */
alignment = gtk_alignment_new (0.0, 0.5, 1.0, 0.0);
gtk_alignment_set_padding (GTK_ALIGNMENT (alignment), 0, 0, 2, 2);
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 12);
gtk_container_add (GTK_CONTAINER (alignment), box);
close_button = gtk_button_new ();
image = gtk_image_new_from_icon_name ("window-close-symbolic", GTK_ICON_SIZE_BUTTON);
gtk_button_set_relief (GTK_BUTTON (close_button), GTK_RELIEF_NONE);
gtk_container_add (GTK_CONTAINER (close_button), image);
gtk_box_pack_start (GTK_BOX (box), close_button, FALSE, FALSE, 0);
priv->label = gtk_label_new (NULL);
gtk_box_pack_start (GTK_BOX (box), priv->label, FALSE, FALSE, 0);
priv->entry = gtk_entry_new ();
gtk_entry_set_width_chars (GTK_ENTRY (priv->entry), 32);
gtk_entry_set_max_length (GTK_ENTRY (priv->entry), 512);
gtk_box_pack_start (GTK_BOX (box), priv->entry, TRUE, TRUE, 0);
item = gtk_tool_item_new ();
gtk_container_add (GTK_CONTAINER (item), alignment);
/* gtk_tool_item_set_expand (item, TRUE); */
gtk_toolbar_insert (GTK_TOOLBAR (toolbar), item, -1);
gtk_widget_show_all (GTK_WIDGET (item));
/* Prev */
arrow = gtk_arrow_new (GTK_ARROW_LEFT, GTK_SHADOW_NONE);
priv->prev = gtk_tool_button_new (arrow, _("Find Previous"));
gtk_tool_item_set_is_important (priv->prev, TRUE);
gtk_tool_item_set_tooltip_text (priv->prev,
_("Find previous occurrence of the search string"));
gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->prev, -1);
gtk_widget_show_all (GTK_WIDGET (priv->prev));
/* Next */
arrow = gtk_arrow_new (GTK_ARROW_RIGHT, GTK_SHADOW_NONE);
priv->next = gtk_tool_button_new (arrow, _("Find Next"));
gtk_tool_item_set_is_important (priv->next, TRUE);
gtk_tool_item_set_tooltip_text (priv->next,
_("Find next occurrence of the search string"));
gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->next, -1);
gtk_widget_show_all (GTK_WIDGET (priv->next));
/* Case sensitivity */
checkbox = gtk_check_button_new_with_mnemonic (_("_Case sensitive"));
priv->case_sensitive = checkbox;
item = gtk_tool_item_new ();
gtk_container_add (GTK_CONTAINER (item), checkbox);
gtk_toolbar_insert (GTK_TOOLBAR (toolbar), item, -1);
gtk_widget_show_all (GTK_WIDGET (item));
/* Populate the overflow menu */
g_signal_connect (item, "create-menu-proxy",
G_CALLBACK (toolitem_create_menu_proxy_cb), toolbar);
priv->sep = gtk_separator_tool_item_new ();
gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->sep, -1);
priv->status_item = gtk_tool_item_new ();
gtk_tool_item_set_expand (priv->status_item, TRUE);
priv->status_label = gtk_label_new ("");
gtk_misc_set_alignment (GTK_MISC (priv->status_label), 0.0, 0.5);
gtk_label_set_ellipsize (GTK_LABEL (priv->status_label), PANGO_ELLIPSIZE_END);
gtk_container_add (GTK_CONTAINER (priv->status_item), priv->status_label);
gtk_widget_show (priv->status_label);
gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->status_item, -1);
/* connect signals */
g_signal_connect (priv->entry, "key-press-event",
G_CALLBACK (entry_key_press_event_cb), toolbar);
g_signal_connect_after (priv->entry, "changed",
G_CALLBACK (entry_changed_cb), toolbar);
g_signal_connect (priv->entry, "activate",
G_CALLBACK (entry_activate_cb), toolbar);
g_signal_connect_swapped (priv->next, "clicked",
G_CALLBACK (find_next_cb), toolbar);
g_signal_connect_swapped (priv->prev, "clicked",
G_CALLBACK (find_prev_cb), toolbar);
g_signal_connect (priv->case_sensitive, "toggled",
G_CALLBACK (case_sensitive_toggled_cb), toolbar);
g_signal_connect (close_button, "clicked",
G_CALLBACK (close_button_clicked_cb), toolbar);
}
G_DEFINE_TYPE (EphyFindToolbar, ephy_find_toolbar, GTK_TYPE_TOOLBAR)
static void
ephy_find_toolbar_dispose (GObject *object)
{
EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (object);
EphyFindToolbarPrivate *priv = toolbar->priv;
if (priv->source_id != 0)
{
g_source_remove (priv->source_id);
priv->source_id = 0;
}
if (priv->find_again_source_id != 0)
{
g_source_remove (priv->find_again_source_id);
priv->find_again_source_id = 0;
}
if (priv->find_source_id != 0)
{
g_source_remove (priv->find_source_id);
priv->find_source_id = 0;
}
G_OBJECT_CLASS (ephy_find_toolbar_parent_class)->dispose (object);
}
static void
ephy_find_toolbar_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
/* no readable properties */
g_assert_not_reached ();
}
static void
ephy_find_toolbar_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (object);
switch (prop_id)
{
case PROP_WINDOW:
ephy_find_toolbar_set_window (toolbar, (EphyWindow *) g_value_get_object (value));
break;
}
}
static void
ephy_find_toolbar_finalize (GObject *o)
{
EphyFindToolbarPrivate *priv = EPHY_FIND_TOOLBAR (o)->priv;
g_free (priv->find_string);
G_OBJECT_CLASS (ephy_find_toolbar_parent_class)->finalize (o);
}
static void
ephy_find_toolbar_class_init (EphyFindToolbarClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
object_class->dispose = ephy_find_toolbar_dispose;
object_class->finalize = ephy_find_toolbar_finalize;
object_class->get_property = ephy_find_toolbar_get_property;
object_class->set_property = ephy_find_toolbar_set_property;
widget_class->parent_set = ephy_find_toolbar_parent_set;
widget_class->grab_focus = ephy_find_toolbar_grab_focus;
klass->next = ephy_find_toolbar_find_next;
klass->previous = ephy_find_toolbar_find_previous;
signals[NEXT] =
g_signal_new ("next",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EphyFindToolbarClass, next),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
signals[PREVIOUS] =
g_signal_new ("previous",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST,
G_STRUCT_OFFSET (EphyFindToolbarClass, previous),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
signals[CLOSE] =
g_signal_new ("close",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
G_STRUCT_OFFSET (EphyFindToolbarClass, close),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
g_object_class_install_property
(object_class,
PROP_WINDOW,
g_param_spec_object ("window",
"Window",
"Parent window",
EPHY_TYPE_WINDOW,
(GParamFlags) (G_PARAM_WRITABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB | G_PARAM_CONSTRUCT_ONLY)));
g_type_class_add_private (klass, sizeof (EphyFindToolbarPrivate));
}
/* public functions */
EphyFindToolbar *
ephy_find_toolbar_new (EphyWindow *window)
{
return g_object_new (EPHY_TYPE_FIND_TOOLBAR,
"window", window,
NULL);
}
const char *
ephy_find_toolbar_get_text (EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
return gtk_entry_get_text (GTK_ENTRY (priv->entry));
}
void
ephy_find_toolbar_set_embed (EphyFindToolbar *toolbar,
EphyEmbed *embed)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
WebKitWebView *web_view = EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED(embed);
if (priv->web_view == web_view) return;
if (priv->web_view != NULL)
{
#ifdef HAVE_WEBKIT2
g_signal_handlers_disconnect_matched (priv->controller,
G_SIGNAL_MATCH_DATA,
0, 0, NULL, NULL, toolbar);
#endif
g_signal_handlers_disconnect_matched (EPHY_WEB_VIEW (web_view),
G_SIGNAL_MATCH_DATA,
0, 0, NULL, NULL, toolbar);
}
priv->web_view = web_view;
if (web_view != NULL)
{
#ifdef HAVE_WEBKIT2
priv->controller = webkit_web_view_get_find_controller (web_view);
g_signal_connect_object (priv->controller, "found-text",
G_CALLBACK (found_text_cb),
toolbar, 0);
g_signal_connect_object (priv->controller, "failed-to-find-text",
G_CALLBACK (failed_to_find_text_cb),
toolbar, 0);
#endif
clear_status (toolbar);
g_signal_connect_object (EPHY_WEB_VIEW (web_view), "search-key-press",
G_CALLBACK (tab_search_key_press_cb),
toolbar, 0);
}
}
#ifndef HAVE_WEBKIT2
typedef struct
{
EphyFindToolbar *toolbar;
gboolean direction;
gboolean highlight;
} FindAgainCBStruct;
static void
find_again_data_destroy_cb (FindAgainCBStruct *data)
{
g_slice_free (FindAgainCBStruct, data);
}
static gboolean
find_again_cb (FindAgainCBStruct *data)
{
EphyFindResult result;
EphyFindToolbarPrivate *priv = data->toolbar->priv;
result = real_find (priv, data->direction);
/* Highlight matches again if the toolbar was hidden when the user
* requested find-again. */
if (result != EPHY_FIND_RESULT_NOTFOUND && data->highlight)
ephy_find_toolbar_mark_matches (data->toolbar);
set_status (data->toolbar, result);
priv->find_again_source_id = 0;
return FALSE;
}
static void
find_again (EphyFindToolbar *toolbar, EphyFindDirection direction)
{
GtkWidget *widget = GTK_WIDGET (toolbar);
EphyFindToolbarPrivate *priv = toolbar->priv;
FindAgainCBStruct *data;
gboolean visible;
visible = gtk_widget_get_visible (widget);
if (!visible) {
gtk_widget_show (widget);
gtk_widget_grab_focus (widget);
}
/* We need to do this to give time to the embed to sync with the size
* change due to the toolbar being shown, otherwise the toolbar can
* obscure the result. See GNOME bug #415074.
*/
if (priv->find_again_source_id != 0) return;
data = g_slice_new0 (FindAgainCBStruct);
data->toolbar = toolbar;
data->direction = direction;
data->highlight = !visible;
priv->find_again_source_id = g_idle_add_full (G_PRIORITY_DEFAULT_IDLE,
(GSourceFunc) find_again_cb,
data,
(GDestroyNotify) find_again_data_destroy_cb);
}
#endif
void
ephy_find_toolbar_find_next (EphyFindToolbar *toolbar)
{
#ifdef HAVE_WEBKIT2
webkit_find_controller_search_next (toolbar->priv->controller);
#else
find_again (toolbar, EPHY_FIND_DIRECTION_NEXT);
#endif
}
void
ephy_find_toolbar_find_previous (EphyFindToolbar *toolbar)
{
#ifdef HAVE_WEBKIT2
webkit_find_controller_search_previous (toolbar->priv->controller);
#else
find_again (toolbar, EPHY_FIND_DIRECTION_PREV);
#endif
}
void
ephy_find_toolbar_open (EphyFindToolbar *toolbar,
gboolean links_only,
gboolean typing_ahead)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
g_return_if_fail (priv->web_view != NULL);
priv->typing_ahead = typing_ahead;
priv->links_only = links_only;
clear_status (toolbar);
gtk_editable_select_region (GTK_EDITABLE (priv->entry), 0, -1);
gtk_widget_show (GTK_WIDGET (toolbar));
gtk_widget_grab_focus (GTK_WIDGET (toolbar));
}
void
ephy_find_toolbar_close (EphyFindToolbar *toolbar)
{
EphyFindToolbarPrivate *priv = toolbar->priv;
gtk_widget_hide (GTK_WIDGET (toolbar));
if (priv->web_view == NULL) return;
#ifdef HAVE_WEBKIT2
webkit_find_controller_search_finish (toolbar->priv->controller);
#else
webkit_web_view_set_highlight_text_matches (priv->web_view, FALSE);
#endif
}
void
ephy_find_toolbar_request_close (EphyFindToolbar *toolbar)
{
if (gtk_widget_get_visible (GTK_WIDGET (toolbar)))
{
g_signal_emit (toolbar, signals[CLOSE], 0);
}
}