/* * Copyright 2009 Mark Benjamin * * This file is part of NetSurf, http://www.netsurf-browser.org/ * * NetSurf 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; version 2 of the License. * * NetSurf 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, see . */ /** \file * Free text search (front component) */ #include #include #include #include "gtk/gtk_search.h" #include "gtk/gtk_scaffolding.h" #include "gtk/gtk_window.h" #include "utils/config.h" #include "content/content.h" #include "content/hlcache.h" #include "desktop/browser.h" #include "desktop/gui.h" #include "desktop/search.h" #include "desktop/searchweb.h" #include "desktop/selection.h" #include "render/box.h" #include "render/html.h" #include "utils/config.h" #include "utils/log.h" #include "utils/messages.h" #include "utils/utils.h" static void nsgtk_search_init(struct gtk_scaffolding *g); static void nsgtk_search_set_status(bool found, void *p); static void nsgtk_search_set_hourglass(bool active, void *p); static void nsgtk_search_add_recent(const char *string, void *p); static struct search_callbacks nsgtk_search_callbacks = { nsgtk_search_set_forward_state, nsgtk_search_set_back_state, nsgtk_search_set_status, nsgtk_search_set_hourglass, nsgtk_search_add_recent }; /** connected to the search forward button */ gboolean nsgtk_search_forward_button_clicked(GtkWidget *widget, gpointer data) { struct gtk_scaffolding *g = (struct gtk_scaffolding *)data; struct browser_window *bw = gui_window_get_browser_window( nsgtk_scaffolding_top_level(g)); nsgtk_search_init(g); search_flags_t flags = SEARCH_FLAG_FORWARDS | (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON( nsgtk_scaffolding_search(g)->caseSens)) ? SEARCH_FLAG_CASE_SENSITIVE : 0) | (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON( nsgtk_scaffolding_search(g)->checkAll)) ? SEARCH_FLAG_SHOWALL : 0); if (search_verify_new(bw, &nsgtk_search_callbacks, (void *)bw)) search_step(bw->search_context, flags, gtk_entry_get_text( nsgtk_scaffolding_search(g)->entry)); return TRUE; } /** connected to the search back button */ gboolean nsgtk_search_back_button_clicked(GtkWidget *widget, gpointer data) { struct gtk_scaffolding *g = (struct gtk_scaffolding *)data; struct browser_window *bw = gui_window_get_browser_window( nsgtk_scaffolding_top_level(g)); nsgtk_search_init(g); search_flags_t flags = 0 |(gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON( nsgtk_scaffolding_search(g)->caseSens)) ? SEARCH_FLAG_CASE_SENSITIVE : 0) | (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON( nsgtk_scaffolding_search(g)->checkAll)) ? SEARCH_FLAG_SHOWALL : 0); if (search_verify_new(bw, &nsgtk_search_callbacks, (void *)bw)) search_step(bw->search_context, flags, gtk_entry_get_text( nsgtk_scaffolding_search(g)->entry)); return TRUE; } /** preparatory code when the search bar is made visible initially */ void nsgtk_search_init(struct gtk_scaffolding *g) { hlcache_handle *c; assert(gui_window_get_browser_window(nsgtk_scaffolding_top_level(g)) != NULL); c = gui_window_get_browser_window(nsgtk_scaffolding_top_level(g))-> current_content; if ((!c) || (content_get_type(c) != CONTENT_HTML && content_get_type(c) != CONTENT_TEXTPLAIN)) return; } /** connected to the search close button */ gboolean nsgtk_search_close_button_clicked(GtkWidget *widget, gpointer data) { struct gtk_scaffolding *g = (struct gtk_scaffolding *)data; nsgtk_scaffolding_toggle_search_bar_visibility(g); return TRUE; } /** connected to the search entry [typing] */ gboolean nsgtk_search_entry_changed(GtkWidget *widget, gpointer data) { nsgtk_scaffolding *g = (nsgtk_scaffolding *)data; struct browser_window *bw = gui_window_get_browser_window( nsgtk_scaffolding_top_level(g)); if ((bw != NULL) && (bw->search_context != NULL)) search_destroy_context(bw->search_context); nsgtk_search_set_forward_state(true, (void *)bw); nsgtk_search_set_back_state(true, (void *)bw); return TRUE; } /** connected to the search entry [return key] */ gboolean nsgtk_search_entry_activate(GtkWidget *widget, gpointer data) { nsgtk_scaffolding *g = (nsgtk_scaffolding *)data; struct browser_window *bw = gui_window_get_browser_window( nsgtk_scaffolding_top_level(g)); nsgtk_search_init(g); search_flags_t flags = SEARCH_FLAG_FORWARDS | (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON( nsgtk_scaffolding_search(g)->caseSens)) ? SEARCH_FLAG_CASE_SENSITIVE : 0) | (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON( nsgtk_scaffolding_search(g)->checkAll)) ? SEARCH_FLAG_SHOWALL : 0); if (search_verify_new(bw, &nsgtk_search_callbacks, (void *)bw)) search_step(bw->search_context, flags, gtk_entry_get_text( nsgtk_scaffolding_search(g)->entry)); return FALSE; } /** allows escape key to close search bar too */ gboolean nsgtk_search_entry_key(GtkWidget *widget, GdkEventKey *event, gpointer data) { if (event->keyval == GDK_Escape) { struct gtk_scaffolding *g = (struct gtk_scaffolding *)data; nsgtk_scaffolding_toggle_search_bar_visibility(g); } return FALSE; } /** connected to the websearch entry [return key] */ gboolean nsgtk_websearch_activate(GtkWidget *widget, gpointer data) { struct gtk_scaffolding *g = (struct gtk_scaffolding *)data; temp_open_background = 0; search_web_new_window(gui_window_get_browser_window( nsgtk_scaffolding_top_level(g)), (char *)gtk_entry_get_text(GTK_ENTRY( nsgtk_scaffolding_websearch(g)))); temp_open_background = -1; return TRUE; } /** * allows a click in the websearch entry field to clear the name of the * provider */ gboolean nsgtk_websearch_clear(GtkWidget *widget, GdkEventFocus *f, gpointer data) { struct gtk_scaffolding *g = (struct gtk_scaffolding *)data; gtk_editable_select_region(GTK_EDITABLE( nsgtk_scaffolding_websearch(g)), 0, -1); gtk_widget_grab_focus(GTK_WIDGET(nsgtk_scaffolding_websearch(g))); return TRUE; } /** * Change the displayed search status. * \param found search pattern matched in text * \param p the pointer sent to search_verify_new() / search_create_context() */ void nsgtk_search_set_status(bool found, void *p) { } /** * display hourglass while searching * \param active start/stop indicator * \param p the pointer sent to search_verify_new() / search_create_context() */ void nsgtk_search_set_hourglass(bool active, void *p) { } /** * add search string to recent searches list * front is at liberty how to implement the bare notification * should normally store a strdup() of the string; * core gives no guarantee of the integrity of the const char * * \param string search pattern * \param p the pointer sent to search_verify_new() / search_create_context() */ void nsgtk_search_add_recent(const char *string, void *p) { } /** * activate search forwards button in gui * \param active activate/inactivate * \param p the pointer sent to search_verify_new() / search_create_context() */ void nsgtk_search_set_forward_state(bool active, void *p) { struct browser_window *bw = (struct browser_window *)p; if ((bw != NULL) && (bw->window != NULL)) { struct gtk_scaffolding *g = nsgtk_get_scaffold(bw->window); gtk_widget_set_sensitive(GTK_WIDGET(nsgtk_scaffolding_search( g)->buttons[1]), active); } } /** * activate search back button in gui * \param active activate/inactivate * \param p the pointer sent to search_verify_new() / search_create_context() */ void nsgtk_search_set_back_state(bool active, void *p) { struct browser_window *bw = (struct browser_window *)p; if ((bw != NULL) && (bw->window != NULL)) { struct gtk_scaffolding *g = nsgtk_get_scaffold(bw->window); gtk_widget_set_sensitive(GTK_WIDGET(nsgtk_scaffolding_search( g)->buttons[0]), active); } }