diff options
Diffstat (limited to 'embed/xulrunner/tests/testgeckoembed.cpp')
-rw-r--r-- | embed/xulrunner/tests/testgeckoembed.cpp | 1145 |
1 files changed, 1145 insertions, 0 deletions
diff --git a/embed/xulrunner/tests/testgeckoembed.cpp b/embed/xulrunner/tests/testgeckoembed.cpp new file mode 100644 index 000000000..1b9a888b6 --- /dev/null +++ b/embed/xulrunner/tests/testgeckoembed.cpp @@ -0,0 +1,1145 @@ +/* + * Copyright © Christopher Blizzard + * + * This program 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, 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + * + * --------------------------------------------------------------------------- + * Derived from Mozilla.org code, which had the following attributions: + * + * The Original Code is mozilla.org code. + * + * The Initial Developer of the Original Code is + * Christopher Blizzard. Portions created by Christopher Blizzard are Copyright © Christopher Blizzard. All Rights Reserved. + * Portions created by the Initial Developer are Copyright © 2001 + * the Initial Developer. All Rights Reserved. + * + * Contributor(s): + * Christopher Blizzard <blizzard@mozilla.org> + * --------------------------------------------------------------------------- + * + * $Id$ + */ + +#include <mozilla-config.h> +#include <config.h> + +#include "gecko-init.h" +#include "gecko-embed.h" +#include "gecko-embed-single.h" +#include "gecko-dom-event.h" +#include "gecko-dom-event-internal.h" +#include "gecko-embed-types.h" + +#include <gtk/gtk.h> +#include <stdio.h> +#include <string.h> +#include <stdlib.h> + +// mozilla specific headers +#include "nsIDOMKeyEvent.h" +#include "nsIDOMMouseEvent.h" +#include "nsIDOMUIEvent.h" + +#include <nsStringAPI.h> +#include <nsCOMPtr.h> + +#define XPCOM_GLUE +#include <nsXPCOMGlue.h> + +typedef struct _TestGeckoBrowser { + GtkWidget *topLevelWindow; + GtkWidget *topLevelVBox; + GtkWidget *menuBar; + GtkWidget *fileMenuItem; + GtkWidget *fileMenu; + GtkWidget *fileOpenNewBrowser; + GtkWidget *fileStream; + GtkWidget *fileClose; + GtkWidget *fileQuit; + GtkWidget *toolbarHBox; + GtkWidget *toolbar; + GtkWidget *backButton; + GtkWidget *stopButton; + GtkWidget *forwardButton; + GtkWidget *reloadButton; + GtkWidget *urlEntry; + GtkWidget *mozEmbed; + GtkWidget *progressAreaHBox; + GtkWidget *progressBar; + GtkWidget *statusAlign; + GtkWidget *statusBar; + const char *statusMessage; + int loadPercent; + int bytesLoaded; + int maxBytesLoaded; + char *tempMessage; + gboolean menuBarOn; + gboolean toolBarOn; + gboolean locationBarOn; + gboolean statusBarOn; + +} TestGeckoBrowser; + +// the list of browser windows currently open +GList *browser_list = g_list_alloc(); + +static TestGeckoBrowser *new_gecko_browser (guint32 chromeMask); +static void set_browser_visibility (TestGeckoBrowser *browser, + gboolean visibility); + +static int num_browsers = 0; + +// callbacks from the UI +static void back_clicked_cb (GtkButton *button, + TestGeckoBrowser *browser); +static void stop_clicked_cb (GtkButton *button, + TestGeckoBrowser *browser); +static void forward_clicked_cb (GtkButton *button, + TestGeckoBrowser *browser); +static void reload_clicked_cb (GtkButton *button, + TestGeckoBrowser *browser); +static void url_activate_cb (GtkEditable *widget, + TestGeckoBrowser *browser); +static void menu_open_new_cb (GtkMenuItem *menuitem, + TestGeckoBrowser *browser); +static void menu_stream_cb (GtkMenuItem *menuitem, + TestGeckoBrowser *browser); +static void menu_close_cb (GtkMenuItem *menuitem, + TestGeckoBrowser *browser); +static void menu_quit_cb (GtkMenuItem *menuitem, + TestGeckoBrowser *browser); +static gboolean delete_cb (GtkWidget *widget, GdkEventAny *event, + TestGeckoBrowser *browser); +static void destroy_cb (GtkWidget *widget, + TestGeckoBrowser *browser); + +// callbacks from the widget +static void location_changed_cb (GeckoEmbed *embed, TestGeckoBrowser *browser); +static void title_changed_cb (GeckoEmbed *embed, TestGeckoBrowser *browser); +static void load_started_cb (GeckoEmbed *embed, TestGeckoBrowser *browser); +static void load_finished_cb (GeckoEmbed *embed, TestGeckoBrowser *browser); +static void net_state_change_cb (GeckoEmbed *embed, gint flags, + guint status, TestGeckoBrowser *browser); +static void net_state_change_all_cb (GeckoEmbed *embed, const char *uri, + gint flags, guint status, + TestGeckoBrowser *browser); +static void progress_change_cb (GeckoEmbed *embed, gint cur, gint max, + TestGeckoBrowser *browser); +static void progress_change_all_cb (GeckoEmbed *embed, const char *uri, + gint cur, gint max, + TestGeckoBrowser *browser); +static void link_message_cb (GeckoEmbed *embed, TestGeckoBrowser *browser); +static void js_status_cb (GeckoEmbed *embed, TestGeckoBrowser *browser); +static void new_window_cb (GeckoEmbed *embed, + GeckoEmbed **retval, guint chromemask, + TestGeckoBrowser *browser); +static void visibility_cb (GeckoEmbed *embed, + gboolean visibility, + TestGeckoBrowser *browser); +static void destroy_brsr_cb (GeckoEmbed *embed, TestGeckoBrowser *browser); +static gint open_uri_cb (GeckoEmbed *embed, const char *uri, + TestGeckoBrowser *browser); +static void size_to_cb (GeckoEmbed *embed, gint width, + gint height, TestGeckoBrowser *browser); +static gboolean dom_key_down_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_key_press_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_key_up_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_mouse_down_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_mouse_up_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_mouse_click_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_mouse_dbl_click_cb (GeckoEmbed *embed, + GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_mouse_over_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_mouse_out_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_activate_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_focus_in_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_focus_out_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); +static gboolean dom_context_menu_cb (GeckoEmbed *, GeckoDOMEvent *event, + TestGeckoBrowser *browser); + +// callbacks from the singleton object +static void new_window_orphan_cb (GeckoEmbedSingle *embed, + GeckoEmbed **retval, guint chromemask, + gpointer data); + +// some utility functions +static void update_status_bar_text (TestGeckoBrowser *browser); +static void update_temp_message (TestGeckoBrowser *browser, + const char *message); +static void update_nav_buttons (TestGeckoBrowser *browser); + +int +main(int argc, char **argv) +{ + //g_setenv ("MOZILLA_FIVE_HOME", GECKO_HOME, FALSE); + + gtk_init(&argc, &argv); + + static const GREVersionRange greVersion = { + "1.9a", PR_TRUE, + "2", PR_TRUE + }; + + char xpcomPath[PATH_MAX]; + + nsresult rv = GRE_GetGREPathWithProperties(&greVersion, 1, nsnull, 0, + xpcomPath, sizeof(xpcomPath)); + if (NS_FAILED(rv)) { + fprintf(stderr, "Couldn't find a compatible GRE.\n"); + return 1; + } + + rv = XPCOMGlueStartup(xpcomPath); + if (NS_FAILED(rv)) { + fprintf(stderr, "Couldn't start XPCOM."); + return 1; + } + + char *lastSlash = strrchr(xpcomPath, '/'); + if (lastSlash) + *lastSlash = '\0'; + + + char *home_path; + char *full_path; + home_path = getenv("HOME"); + if (!home_path) { + fprintf(stderr, "Failed to get HOME\n"); + exit(1); + } + + + full_path = g_strdup_printf("%s/%s", home_path, ".TestGtkEmbed"); + + if (!gecko_init_with_profile (xpcomPath, full_path, "TestGtkEmbed")) { + fprintf(stderr, "Failed gecko_init_with_profile\n"); + exit(1); + } + + // get the singleton object and hook up to its new window callback + // so we can create orphaned windows. + + GeckoEmbedSingle *single; + + single = gecko_embed_single_get(); + if (!single) { + fprintf(stderr, "Failed to get singleton embed object!\n"); + exit(1); + } + + g_signal_connect (single, "new-window-orphan", + G_CALLBACK (new_window_orphan_cb), NULL); + + + TestGeckoBrowser *browser = new_gecko_browser(GECKO_EMBED_FLAG_DEFAULTCHROME); + + // set our minimum size + gtk_widget_set_usize(browser->mozEmbed, 400, 400); + + set_browser_visibility(browser, TRUE); + + if (argc > 1) + gecko_embed_load_url(GECKO_EMBED(browser->mozEmbed), argv[1]); + + gtk_main(); + + gecko_shutdown (); +} + +static TestGeckoBrowser * +new_gecko_browser(guint32 chromeMask) +{ + guint32 actualChromeMask = chromeMask; + TestGeckoBrowser *browser = 0; + + num_browsers++; + + browser = g_new0(TestGeckoBrowser, 1); + + browser_list = g_list_prepend(browser_list, browser); + + browser->menuBarOn = FALSE; + browser->toolBarOn = FALSE; + browser->locationBarOn = FALSE; + browser->statusBarOn = FALSE; + + g_print("new_gecko_browser\n"); + +// if (chromeMask == GECKO_EMBED_FLAG_DEFAULTCHROME) +// actualChromeMask = GECKO_EMBED_FLAG_ALLCHROME; + + /* FIXMEchpe find out WHY the chrome masks we get from gecko as so weird */ + actualChromeMask = 0xffffffff; + + if (actualChromeMask & GECKO_EMBED_FLAG_MENUBARON) + { + browser->menuBarOn = TRUE; + g_print("\tmenu bar\n"); + } + if (actualChromeMask & GECKO_EMBED_FLAG_TOOLBARON) + { + browser->toolBarOn = TRUE; + g_print("\ttool bar\n"); + } + if (actualChromeMask & GECKO_EMBED_FLAG_LOCATIONBARON) + { + browser->locationBarOn = TRUE; + g_print("\tlocation bar\n"); + } + if (actualChromeMask & GECKO_EMBED_FLAG_STATUSBARON) + { + browser->statusBarOn = TRUE; + g_print("\tstatus bar\n"); + } + + // create our new toplevel window + browser->topLevelWindow = gtk_window_new(GTK_WINDOW_TOPLEVEL); + // new vbox + browser->topLevelVBox = gtk_vbox_new(FALSE, 0); + // add it to the toplevel window + gtk_container_add(GTK_CONTAINER(browser->topLevelWindow), + browser->topLevelVBox); + // create our menu bar + browser->menuBar = gtk_menu_bar_new(); + // create the file menu + browser->fileMenuItem = gtk_menu_item_new_with_label("File"); + browser->fileMenu = gtk_menu_new(); + gtk_menu_item_set_submenu (GTK_MENU_ITEM(browser->fileMenuItem), + browser->fileMenu); + + browser->fileOpenNewBrowser = + gtk_menu_item_new_with_label("Open New Browser"); + gtk_menu_append(GTK_MENU(browser->fileMenu), + browser->fileOpenNewBrowser); + + browser->fileStream = + gtk_menu_item_new_with_label("Test Stream"); + gtk_menu_append(GTK_MENU(browser->fileMenu), + browser->fileStream); + + browser->fileClose = + gtk_menu_item_new_with_label("Close"); + gtk_menu_append(GTK_MENU(browser->fileMenu), + browser->fileClose); + + browser->fileQuit = + gtk_menu_item_new_with_label("Quit"); + gtk_menu_append(GTK_MENU(browser->fileMenu), + browser->fileQuit); + + // append it + gtk_menu_bar_append(GTK_MENU_BAR(browser->menuBar), browser->fileMenuItem); + + // add it to the vbox + gtk_box_pack_start(GTK_BOX(browser->topLevelVBox), + browser->menuBar, + FALSE, // expand + FALSE, // fill + 0); // padding + // create the hbox that will contain the toolbar and the url text entry bar + browser->toolbarHBox = gtk_hbox_new(FALSE, 0); + // add that hbox to the vbox + gtk_box_pack_start(GTK_BOX(browser->topLevelVBox), + browser->toolbarHBox, + FALSE, // expand + FALSE, // fill + 0); // padding + // new horiz toolbar with buttons + icons + browser->toolbar = gtk_toolbar_new(); + gtk_toolbar_set_orientation(GTK_TOOLBAR(browser->toolbar), + GTK_ORIENTATION_HORIZONTAL); + gtk_toolbar_set_style(GTK_TOOLBAR(browser->toolbar), + GTK_TOOLBAR_BOTH); + + // add it to the hbox + gtk_box_pack_start(GTK_BOX(browser->toolbarHBox), browser->toolbar, + FALSE, // expand + FALSE, // fill + 0); // padding + // new back button + browser->backButton = + gtk_toolbar_append_item(GTK_TOOLBAR(browser->toolbar), + "Back", + "Go Back", + "Go Back", + 0, // XXX replace with icon + GTK_SIGNAL_FUNC(back_clicked_cb), + browser); + // new stop button + browser->stopButton = + gtk_toolbar_append_item(GTK_TOOLBAR(browser->toolbar), + "Stop", + "Stop", + "Stop", + 0, // XXX replace with icon + GTK_SIGNAL_FUNC(stop_clicked_cb), + browser); + // new forward button + browser->forwardButton = + gtk_toolbar_append_item(GTK_TOOLBAR(browser->toolbar), + "Forward", + "Forward", + "Forward", + 0, // XXX replace with icon + GTK_SIGNAL_FUNC(forward_clicked_cb), + browser); + // new reload button + browser->reloadButton = + gtk_toolbar_append_item(GTK_TOOLBAR(browser->toolbar), + "Reload", + "Reload", + "Reload", + 0, // XXX replace with icon + GTK_SIGNAL_FUNC(reload_clicked_cb), + browser); + // create the url text entry + browser->urlEntry = gtk_entry_new(); + // add it to the hbox + gtk_box_pack_start(GTK_BOX(browser->toolbarHBox), browser->urlEntry, + TRUE, // expand + TRUE, // fill + 0); // padding + // create our new gtk moz embed widget + browser->mozEmbed = gecko_embed_new(); + // add it to the toplevel vbox + gtk_box_pack_start(GTK_BOX(browser->topLevelVBox), browser->mozEmbed, + TRUE, // expand + TRUE, // fill + 0); // padding + // create the new hbox for the progress area + browser->progressAreaHBox = gtk_hbox_new(FALSE, 0); + // add it to the vbox + gtk_box_pack_start(GTK_BOX(browser->topLevelVBox), browser->progressAreaHBox, + FALSE, // expand + FALSE, // fill + 0); // padding + // create our new progress bar + browser->progressBar = gtk_progress_bar_new(); + // add it to the hbox + gtk_box_pack_start(GTK_BOX(browser->progressAreaHBox), browser->progressBar, + FALSE, // expand + FALSE, // fill + 0); // padding + + // create our status area and the alignment object that will keep it + // from expanding + browser->statusAlign = gtk_alignment_new(0, 0, 1, 1); + gtk_widget_set_usize(browser->statusAlign, 1, -1); + // create the status bar + browser->statusBar = gtk_statusbar_new(); + gtk_container_add(GTK_CONTAINER(browser->statusAlign), browser->statusBar); + // add it to the hbox + gtk_box_pack_start(GTK_BOX(browser->progressAreaHBox), browser->statusAlign, + TRUE, // expand + TRUE, // fill + 0); // padding + // by default none of the buttons are marked as sensitive. + gtk_widget_set_sensitive(browser->backButton, FALSE); + gtk_widget_set_sensitive(browser->stopButton, FALSE); + gtk_widget_set_sensitive(browser->forwardButton, FALSE); + gtk_widget_set_sensitive(browser->reloadButton, FALSE); + + // catch the destruction of the toplevel window + gtk_signal_connect(GTK_OBJECT(browser->topLevelWindow), "delete_event", + GTK_SIGNAL_FUNC(delete_cb), browser); + + // hook up the activate signal to the right callback + gtk_signal_connect(GTK_OBJECT(browser->urlEntry), "activate", + GTK_SIGNAL_FUNC(url_activate_cb), browser); + + // hook up to the open new browser activation + gtk_signal_connect(GTK_OBJECT(browser->fileOpenNewBrowser), "activate", + GTK_SIGNAL_FUNC(menu_open_new_cb), browser); + // hook up to the stream test + gtk_signal_connect(GTK_OBJECT(browser->fileStream), "activate", + GTK_SIGNAL_FUNC(menu_stream_cb), browser); + // close this window + gtk_signal_connect(GTK_OBJECT(browser->fileClose), "activate", + GTK_SIGNAL_FUNC(menu_close_cb), browser); + // quit the application + gtk_signal_connect(GTK_OBJECT(browser->fileQuit), "activate", + GTK_SIGNAL_FUNC(menu_quit_cb), browser); + + // hook up the location change to update the urlEntry + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "location", + GTK_SIGNAL_FUNC(location_changed_cb), browser); + // hook up the title change to update the window title + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "title", + GTK_SIGNAL_FUNC(title_changed_cb), browser); + // hook up the start and stop signals + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "net_start", + GTK_SIGNAL_FUNC(load_started_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "net_stop", + GTK_SIGNAL_FUNC(load_finished_cb), browser); + // hook up to the change in network status + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "net_state", + GTK_SIGNAL_FUNC(net_state_change_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "net_state_all", + GTK_SIGNAL_FUNC(net_state_change_all_cb), browser); + // hookup to changes in progress + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "progress", + GTK_SIGNAL_FUNC(progress_change_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "progress_all", + GTK_SIGNAL_FUNC(progress_change_all_cb), browser); + // hookup to changes in over-link message + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "link_message", + GTK_SIGNAL_FUNC(link_message_cb), browser); + // hookup to changes in js status message + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "js_status", + GTK_SIGNAL_FUNC(js_status_cb), browser); + // hookup to see whenever a new window is requested + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "new_window", + GTK_SIGNAL_FUNC(new_window_cb), browser); + // hookup to any requested visibility changes + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "visibility", + GTK_SIGNAL_FUNC(visibility_cb), browser); + // hookup to the signal that says that the browser requested to be + // destroyed + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "destroy_browser", + GTK_SIGNAL_FUNC(destroy_brsr_cb), browser); + // hookup to the signal that is called when someone clicks on a link + // to load a new uri + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "open_uri", + GTK_SIGNAL_FUNC(open_uri_cb), browser); + // this signal is emitted when there's a request to change the + // containing browser window to a certain height, like with width + // and height args for a window.open in javascript + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "size_to", + GTK_SIGNAL_FUNC(size_to_cb), browser); + // key event signals + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_key_down", + GTK_SIGNAL_FUNC(dom_key_down_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_key_press", + GTK_SIGNAL_FUNC(dom_key_press_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_key_up", + GTK_SIGNAL_FUNC(dom_key_up_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_mouse_down", + GTK_SIGNAL_FUNC(dom_mouse_down_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_mouse_up", + GTK_SIGNAL_FUNC(dom_mouse_up_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_mouse_click", + GTK_SIGNAL_FUNC(dom_mouse_click_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_mouse_dbl_click", + GTK_SIGNAL_FUNC(dom_mouse_dbl_click_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_mouse_over", + GTK_SIGNAL_FUNC(dom_mouse_over_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_mouse_out", + GTK_SIGNAL_FUNC(dom_mouse_out_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_activate", + GTK_SIGNAL_FUNC(dom_activate_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_focus_in", + GTK_SIGNAL_FUNC(dom_focus_in_cb), browser); + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "dom_focus_out", + GTK_SIGNAL_FUNC(dom_focus_out_cb), browser); + g_signal_connect(browser->mozEmbed, "dom-context-menu", + GTK_SIGNAL_FUNC(dom_context_menu_cb), browser); + // hookup to when the window is destroyed + gtk_signal_connect(GTK_OBJECT(browser->mozEmbed), "destroy", + GTK_SIGNAL_FUNC(destroy_cb), browser); + + // set the chrome type so it's stored in the object + gecko_embed_set_chrome_mask(GECKO_EMBED(browser->mozEmbed), + actualChromeMask); + + gtk_widget_show_all (browser->topLevelVBox); + + return browser; +} + +void +set_browser_visibility (TestGeckoBrowser *browser, gboolean visibility) +{ + if (!visibility) + { + gtk_widget_hide(browser->topLevelWindow); + return; + } + + if (browser->menuBarOn) + gtk_widget_show_all(browser->menuBar); + else + gtk_widget_hide_all(browser->menuBar); + + // since they are on the same line here... + if (browser->toolBarOn || browser->locationBarOn) + gtk_widget_show_all(browser->toolbarHBox); + else + gtk_widget_hide_all(browser->toolbarHBox); + + if (browser->statusBarOn) + gtk_widget_show_all(browser->progressAreaHBox); + else + gtk_widget_hide_all(browser->progressAreaHBox); + + gtk_widget_show(browser->mozEmbed); + gtk_widget_show(browser->topLevelVBox); + gtk_widget_show(browser->topLevelWindow); +} + +void +back_clicked_cb (GtkButton *button, TestGeckoBrowser *browser) +{ + gecko_embed_go_back(GECKO_EMBED(browser->mozEmbed)); +} + +void +stop_clicked_cb (GtkButton *button, TestGeckoBrowser *browser) +{ + g_print("stop_clicked_cb\n"); + gecko_embed_stop_load(GECKO_EMBED(browser->mozEmbed)); +} + +void +forward_clicked_cb (GtkButton *button, TestGeckoBrowser *browser) +{ + g_print("forward_clicked_cb\n"); + gecko_embed_go_forward(GECKO_EMBED(browser->mozEmbed)); +} + +void +reload_clicked_cb (GtkButton *button, TestGeckoBrowser *browser) +{ + g_print("reload_clicked_cb\n"); + GdkModifierType state = (GdkModifierType)0; + gint x, y; + gdk_window_get_pointer(NULL, &x, &y, &state); + + gecko_embed_reload(GECKO_EMBED(browser->mozEmbed), + (state & GDK_SHIFT_MASK) ? + GECKO_EMBED_FLAG_RELOADBYPASSCACHE : + GECKO_EMBED_FLAG_RELOADNORMAL); +} + +void +stream_clicked_cb (GtkButton *button, TestGeckoBrowser *browser) +{ + const char *data; + const char *data2; + data = "<html>Hi"; + data2 = " there</html>\n"; + g_print("stream_clicked_cb\n"); + gecko_embed_open_stream(GECKO_EMBED(browser->mozEmbed), + "file://", "text/html"); + gecko_embed_append_data(GECKO_EMBED(browser->mozEmbed), + data, strlen(data)); + gecko_embed_append_data(GECKO_EMBED(browser->mozEmbed), + data2, strlen(data2)); + gecko_embed_close_stream(GECKO_EMBED(browser->mozEmbed)); +} + +void +url_activate_cb (GtkEditable *widget, TestGeckoBrowser *browser) +{ + gchar *text = gtk_editable_get_chars(widget, 0, -1); + g_print("loading url %s\n", text); + gecko_embed_load_url(GECKO_EMBED(browser->mozEmbed), text); + g_free(text); +} + +void +menu_open_new_cb (GtkMenuItem *menuitem, TestGeckoBrowser *browser) +{ + g_print("opening new browser.\n"); + TestGeckoBrowser *newBrowser = + new_gecko_browser(GECKO_EMBED_FLAG_DEFAULTCHROME); + gtk_widget_set_usize(newBrowser->mozEmbed, 400, 400); + set_browser_visibility(newBrowser, TRUE); +} + +void +menu_stream_cb (GtkMenuItem *menuitem, TestGeckoBrowser *browser) +{ + g_print("menu_stream_cb\n"); + const char *data; + const char *data2; + data = "<html>Hi"; + data2 = " there</html>\n"; + g_print("stream_clicked_cb\n"); + gecko_embed_open_stream(GECKO_EMBED(browser->mozEmbed), + "file://", "text/html"); + gecko_embed_append_data(GECKO_EMBED(browser->mozEmbed), + data, strlen(data)); + gecko_embed_append_data(GECKO_EMBED(browser->mozEmbed), + data2, strlen(data2)); + gecko_embed_close_stream(GECKO_EMBED(browser->mozEmbed)); +} + +void +menu_close_cb (GtkMenuItem *menuitem, TestGeckoBrowser *browser) +{ + gtk_widget_destroy(browser->topLevelWindow); +} + +void +menu_quit_cb (GtkMenuItem *menuitem, TestGeckoBrowser *browser) +{ + TestGeckoBrowser *tmpBrowser; + GList *tmp_list = browser_list; + tmpBrowser = (TestGeckoBrowser *)tmp_list->data; + while (tmpBrowser) { + tmp_list = tmp_list->next; + gtk_widget_destroy(tmpBrowser->topLevelWindow); + tmpBrowser = (TestGeckoBrowser *)tmp_list->data; + } +} + +gboolean +delete_cb(GtkWidget *widget, GdkEventAny *event, TestGeckoBrowser *browser) +{ + g_print("delete_cb\n"); + gtk_widget_destroy(widget); + return TRUE; +} + +void +destroy_cb (GtkWidget *widget, TestGeckoBrowser *browser) +{ + GList *tmp_list; + g_print("destroy_cb\n"); + num_browsers--; + tmp_list = g_list_find(browser_list, browser); + browser_list = g_list_remove_link(browser_list, tmp_list); + if (browser->tempMessage) + g_free(browser->tempMessage); + if (num_browsers == 0) + gtk_main_quit(); +} + +void +location_changed_cb (GeckoEmbed *embed, TestGeckoBrowser *browser) +{ + char *newLocation; + int newPosition = 0; + g_print("location_changed_cb\n"); + newLocation = gecko_embed_get_location(embed); + if (newLocation) + { + gtk_editable_delete_text(GTK_EDITABLE(browser->urlEntry), 0, -1); + gtk_editable_insert_text(GTK_EDITABLE(browser->urlEntry), + newLocation, strlen(newLocation), &newPosition); + g_free(newLocation); + } + else + g_print("failed to get location!\n"); + // always make sure to clear the tempMessage. it might have been + // set from the link before a click and we wouldn't have gotten the + // callback to unset it. + update_temp_message(browser, 0); + // update the nav buttons on a location change + update_nav_buttons(browser); +} + +void +title_changed_cb (GeckoEmbed *embed, TestGeckoBrowser *browser) +{ + char *newTitle; + g_print("title_changed_cb\n"); + newTitle = gecko_embed_get_title(embed); + if (newTitle) + { + gtk_window_set_title(GTK_WINDOW(browser->topLevelWindow), newTitle); + g_free(newTitle); + } + +} + +void +load_started_cb (GeckoEmbed *embed, TestGeckoBrowser *browser) +{ + g_print("load_started_cb\n"); + gtk_widget_set_sensitive(browser->stopButton, TRUE); + gtk_widget_set_sensitive(browser->reloadButton, FALSE); + browser->loadPercent = 0; + browser->bytesLoaded = 0; + browser->maxBytesLoaded = 0; + update_status_bar_text(browser); +} + +void +load_finished_cb (GeckoEmbed *embed, TestGeckoBrowser *browser) +{ + g_print("load_finished_cb\n"); + gtk_widget_set_sensitive(browser->stopButton, FALSE); + gtk_widget_set_sensitive(browser->reloadButton, TRUE); + browser->loadPercent = 0; + browser->bytesLoaded = 0; + browser->maxBytesLoaded = 0; + update_status_bar_text(browser); + gtk_progress_set_percentage(GTK_PROGRESS(browser->progressBar), 0); +} + + +void +net_state_change_cb (GeckoEmbed *embed, gint flags, guint status, + TestGeckoBrowser *browser) +{ + g_print("net_state_change_cb %d\n", flags); + if (flags & GECKO_EMBED_FLAG_IS_REQUEST) { + if (flags & GECKO_EMBED_FLAG_REDIRECTING) + browser->statusMessage = "Redirecting to site..."; + else if (flags & GECKO_EMBED_FLAG_TRANSFERRING) + browser->statusMessage = "Transferring data from site..."; + else if (flags & GECKO_EMBED_FLAG_NEGOTIATING) + browser->statusMessage = "Waiting for authorization..."; + } + + if (status == GECKO_EMBED_STATUS_FAILED_DNS) + browser->statusMessage = "Site not found."; + else if (status == GECKO_EMBED_STATUS_FAILED_CONNECT) + browser->statusMessage = "Failed to connect to site."; + else if (status == GECKO_EMBED_STATUS_FAILED_TIMEOUT) + browser->statusMessage = "Failed due to connection timeout."; + else if (status == GECKO_EMBED_STATUS_FAILED_USERCANCELED) + browser->statusMessage = "User canceled connecting to site."; + + if (flags & GECKO_EMBED_FLAG_IS_DOCUMENT) { + if (flags & GECKO_EMBED_FLAG_START) + browser->statusMessage = "Loading site..."; + else if (flags & GECKO_EMBED_FLAG_STOP) + browser->statusMessage = "Done."; + } + + update_status_bar_text(browser); + +} + +void net_state_change_all_cb (GeckoEmbed *embed, const char *uri, + gint flags, guint status, + TestGeckoBrowser *browser) +{ + // g_print("net_state_change_all_cb %s %d %d\n", uri, flags, status); +} + +void progress_change_cb (GeckoEmbed *embed, gint cur, gint max, + TestGeckoBrowser *browser) +{ + g_print("progress_change_cb cur %d max %d\n", cur, max); + + // avoid those pesky divide by zero errors + if (max < 1) + { + gtk_progress_set_activity_mode(GTK_PROGRESS(browser->progressBar), FALSE); + browser->loadPercent = 0; + browser->bytesLoaded = cur; + browser->maxBytesLoaded = 0; + update_status_bar_text(browser); + } + else + { + browser->bytesLoaded = cur; + browser->maxBytesLoaded = max; + if (cur > max) + browser->loadPercent = 100; + else + browser->loadPercent = (cur * 100) / max; + update_status_bar_text(browser); + gtk_progress_set_percentage(GTK_PROGRESS(browser->progressBar), browser->loadPercent / 100.0); + } + +} + +void progress_change_all_cb (GeckoEmbed *embed, const char *uri, + gint cur, gint max, + TestGeckoBrowser *browser) +{ + //g_print("progress_change_all_cb %s cur %d max %d\n", uri, cur, max); +} + +void +link_message_cb (GeckoEmbed *embed, TestGeckoBrowser *browser) +{ + char *message; + g_print("link_message_cb\n"); + message = gecko_embed_get_link_message(embed); + if (!message || !*message) + update_temp_message(browser, 0); + else + update_temp_message(browser, message); + if (message) + g_free(message); +} + +void +js_status_cb (GeckoEmbed *embed, TestGeckoBrowser *browser) +{ + char *message; + g_print("js_status_cb\n"); + message = gecko_embed_get_js_status(embed); + if (!message || !*message) + update_temp_message(browser, 0); + else + update_temp_message(browser, message); + if (message) + g_free(message); +} + +void +new_window_cb (GeckoEmbed *embed, GeckoEmbed **newEmbed, guint chromemask, TestGeckoBrowser *browser) +{ + g_print("new_window_cb\n"); + g_print("embed is %p chromemask is %d\n", (void *)embed, chromemask); + TestGeckoBrowser *newBrowser = new_gecko_browser(chromemask); + gtk_widget_set_usize(newBrowser->mozEmbed, 400, 400); + *newEmbed = GECKO_EMBED(newBrowser->mozEmbed); + g_print("new browser is %p\n", (void *)*newEmbed); +} + +void +visibility_cb (GeckoEmbed *embed, gboolean visibility, TestGeckoBrowser *browser) +{ + g_print("visibility_cb %d\n", visibility); + set_browser_visibility(browser, visibility); +} + +void +destroy_brsr_cb (GeckoEmbed *embed, TestGeckoBrowser *browser) +{ + g_print("destroy_brsr_cb\n"); + gtk_widget_destroy(browser->topLevelWindow); +} + +gint +open_uri_cb (GeckoEmbed *embed, const char *uri, TestGeckoBrowser *browser) +{ + g_print("open_uri_cb %s\n", uri); + + // interrupt this test load + if (!strcmp(uri, "http://people.redhat.com/blizzard/monkeys.txt")) + return TRUE; + // don't interrupt anything + return FALSE; +} + +void +size_to_cb (GeckoEmbed *embed, gint width, gint height, + TestGeckoBrowser *browser) +{ + g_print("*** size_to_cb %d %d\n", width, height); + gtk_widget_set_usize(browser->mozEmbed, width, height); +} + +gboolean dom_key_down_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + nsIDOMEvent *domevent = gecko_dom_event_get_I (gecko_event); + nsCOMPtr<nsIDOMKeyEvent> event (do_QueryInterface (domevent)); + + PRUint32 keyCode = 0; + // g_print("dom_key_down_cb\n"); + event->GetKeyCode(&keyCode); + // g_print("key code is %d\n", keyCode); + return NS_OK; +} + +gboolean dom_key_press_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + nsIDOMEvent *domevent = gecko_dom_event_get_I (gecko_event); + nsCOMPtr<nsIDOMKeyEvent> event (do_QueryInterface (domevent)); + + PRUint32 keyCode = 0; + // g_print("dom_key_press_cb\n"); + event->GetCharCode(&keyCode); + // g_print("char code is %d\n", keyCode); + return NS_OK; +} + +gboolean dom_key_up_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + nsIDOMEvent *domevent = gecko_dom_event_get_I (gecko_event); + nsCOMPtr<nsIDOMKeyEvent> event (do_QueryInterface (domevent)); + + PRUint32 keyCode = 0; + // g_print("dom_key_up_cb\n"); + event->GetKeyCode(&keyCode); + // g_print("key code is %d\n", keyCode); + return NS_OK; +} + +gboolean dom_mouse_down_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + // g_print("dom_mouse_down_cb\n"); + return NS_OK; + } + +gboolean dom_mouse_up_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + // g_print("dom_mouse_up_cb\n"); + return NS_OK; +} + +gboolean dom_mouse_click_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + nsIDOMEvent *domevent = gecko_dom_event_get_I (gecko_event); + nsCOMPtr<nsIDOMMouseEvent> event (do_QueryInterface (domevent)); + + // g_print("dom_mouse_click_cb\n"); + PRUint16 button; + event->GetButton(&button); + printf("button was %d\n", button); + return NS_OK; +} + +gboolean dom_mouse_dbl_click_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + // g_print("dom_mouse_dbl_click_cb\n"); + return NS_OK; +} + +gboolean dom_mouse_over_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + //g_print("dom_mouse_over_cb\n"); + return NS_OK; +} + +gboolean dom_mouse_out_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + //g_print("dom_mouse_out_cb\n"); + return NS_OK; +} + +gboolean dom_activate_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + //g_print("dom_activate_cb\n"); + return NS_OK; +} + +gboolean dom_focus_in_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + //g_print("dom_focus_in_cb\n"); + return NS_OK; +} + +gboolean dom_focus_out_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + //g_print("dom_focus_out_cb\n"); + return NS_OK; +} + +gboolean dom_context_menu_cb (GeckoEmbed *embed, + GeckoDOMEvent *gecko_event, + TestGeckoBrowser *browser) +{ + g_print("dom_context_menu_cb\n"); + return NS_OK; +} + +void new_window_orphan_cb (GeckoEmbedSingle *embed, + GeckoEmbed **retval, guint chromemask, + gpointer data) +{ + g_print("new_window_orphan_cb\n"); + g_print("chromemask is %d\n", chromemask); + TestGeckoBrowser *newBrowser = new_gecko_browser(chromemask); + *retval = GECKO_EMBED(newBrowser->mozEmbed); + g_print("new browser is %p\n", (void *)*retval); +} + +// utility functions + +void +update_status_bar_text(TestGeckoBrowser *browser) +{ + gchar message[256]; + + gtk_statusbar_pop(GTK_STATUSBAR(browser->statusBar), 1); + if (browser->tempMessage) + gtk_statusbar_push(GTK_STATUSBAR(browser->statusBar), 1, browser->tempMessage); + else + { + if (browser->loadPercent) + { + g_snprintf(message, 255, "%s (%d%% complete, %d bytes of %d loaded)", browser->statusMessage, browser->loadPercent, browser->bytesLoaded, browser->maxBytesLoaded); + } + else if (browser->bytesLoaded) + { + g_snprintf(message, 255, "%s (%d bytes loaded)", browser->statusMessage, browser->bytesLoaded); + } + else if (browser->statusMessage == NULL) + { + g_snprintf(message, 255, " "); + } + else + { + g_snprintf(message, 255, "%s", browser->statusMessage); + } + gtk_statusbar_push(GTK_STATUSBAR(browser->statusBar), 1, message); + } +} + +void +update_temp_message(TestGeckoBrowser *browser, const char *message) +{ + if (browser->tempMessage) + g_free(browser->tempMessage); + if (message) + browser->tempMessage = g_strdup(message); + else + browser->tempMessage = 0; + // now that we've updated the temp message, redraw the status bar + update_status_bar_text(browser); +} + + +void +update_nav_buttons (TestGeckoBrowser *browser) +{ + gboolean can_go_back; + gboolean can_go_forward; + can_go_back = gecko_embed_can_go_back(GECKO_EMBED(browser->mozEmbed)); + can_go_forward = gecko_embed_can_go_forward(GECKO_EMBED(browser->mozEmbed)); + if (can_go_back) + gtk_widget_set_sensitive(browser->backButton, TRUE); + else + gtk_widget_set_sensitive(browser->backButton, FALSE); + if (can_go_forward) + gtk_widget_set_sensitive(browser->forwardButton, TRUE); + else + gtk_widget_set_sensitive(browser->forwardButton, FALSE); + } |