| /* |
| * Copyright (C) 2011 Igalia S.L. |
| * |
| * 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. |
| * |
| * This library 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 |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| */ |
| |
| #include "config.h" |
| #include "WebViewTest.h" |
| #include <wtf/HashSet.h> |
| #include <wtf/RunLoop.h> |
| #include <wtf/glib/GRefPtr.h> |
| #include <wtf/text/StringHash.h> |
| |
| static const char* kAlertDialogMessage = "WebKitGTK alert dialog message"; |
| static const char* kConfirmDialogMessage = "WebKitGTK confirm dialog message"; |
| static const char* kPromptDialogMessage = "WebKitGTK prompt dialog message"; |
| static const char* kPromptDialogReturnedText = "WebKitGTK prompt dialog returned text"; |
| static const char* kBeforeUnloadConfirmDialogMessage = "WebKitGTK beforeunload dialog message"; |
| |
| class UIClientTest: public WebViewTest { |
| public: |
| MAKE_GLIB_TEST_FIXTURE(UIClientTest); |
| |
| enum WebViewEvents { |
| Create, |
| ReadyToShow, |
| RunAsModal, |
| Close |
| }; |
| |
| class WindowProperties { |
| public: |
| WindowProperties() |
| : m_isNull(true) |
| , m_toolbarVisible(true) |
| , m_statusbarVisible(true) |
| , m_scrollbarsVisible(true) |
| , m_menubarVisible(true) |
| , m_locationbarVisible(true) |
| , m_resizable(true) |
| , m_fullscreen(false) |
| { |
| #if PLATFORM(GTK) |
| memset(&m_geometry, 0, sizeof(GdkRectangle)); |
| #endif |
| } |
| |
| WindowProperties(WebKitWindowProperties* windowProperties) |
| : m_isNull(false) |
| , m_toolbarVisible(webkit_window_properties_get_toolbar_visible(windowProperties)) |
| , m_statusbarVisible(webkit_window_properties_get_statusbar_visible(windowProperties)) |
| , m_scrollbarsVisible(webkit_window_properties_get_scrollbars_visible(windowProperties)) |
| , m_menubarVisible(webkit_window_properties_get_menubar_visible(windowProperties)) |
| , m_locationbarVisible(webkit_window_properties_get_locationbar_visible(windowProperties)) |
| , m_resizable(webkit_window_properties_get_resizable(windowProperties)) |
| , m_fullscreen(webkit_window_properties_get_fullscreen(windowProperties)) |
| { |
| #if PLATFORM(GTK) |
| webkit_window_properties_get_geometry(windowProperties, &m_geometry); |
| #endif |
| } |
| |
| WindowProperties(cairo_rectangle_int_t* geometry, bool toolbarVisible, bool statusbarVisible, bool scrollbarsVisible, bool menubarVisible, bool locationbarVisible, bool resizable, bool fullscreen) |
| : m_isNull(false) |
| , m_geometry(*geometry) |
| , m_toolbarVisible(toolbarVisible) |
| , m_statusbarVisible(statusbarVisible) |
| , m_scrollbarsVisible(scrollbarsVisible) |
| , m_menubarVisible(menubarVisible) |
| , m_locationbarVisible(locationbarVisible) |
| , m_resizable(resizable) |
| , m_fullscreen(fullscreen) |
| { |
| } |
| |
| bool isNull() const { return m_isNull; } |
| |
| void assertEqual(const WindowProperties& other) const |
| { |
| #if PLATFORM(GTK) |
| g_assert_cmpint(m_geometry.x, ==, other.m_geometry.x); |
| g_assert_cmpint(m_geometry.y, ==, other.m_geometry.y); |
| g_assert_cmpint(m_geometry.width, ==, other.m_geometry.width); |
| g_assert_cmpint(m_geometry.height, ==, other.m_geometry.height); |
| #endif |
| g_assert_cmpint(static_cast<int>(m_toolbarVisible), ==, static_cast<int>(other.m_toolbarVisible)); |
| g_assert_cmpint(static_cast<int>(m_statusbarVisible), ==, static_cast<int>(other.m_statusbarVisible)); |
| g_assert_cmpint(static_cast<int>(m_scrollbarsVisible), ==, static_cast<int>(other.m_scrollbarsVisible)); |
| g_assert_cmpint(static_cast<int>(m_menubarVisible), ==, static_cast<int>(other.m_menubarVisible)); |
| g_assert_cmpint(static_cast<int>(m_locationbarVisible), ==, static_cast<int>(other.m_locationbarVisible)); |
| g_assert_cmpint(static_cast<int>(m_resizable), ==, static_cast<int>(other.m_resizable)); |
| g_assert_cmpint(static_cast<int>(m_fullscreen), ==, static_cast<int>(other.m_fullscreen)); |
| } |
| |
| private: |
| bool m_isNull; |
| |
| cairo_rectangle_int_t m_geometry; |
| |
| bool m_toolbarVisible; |
| bool m_statusbarVisible; |
| bool m_scrollbarsVisible; |
| bool m_menubarVisible; |
| bool m_locationbarVisible; |
| |
| bool m_resizable; |
| bool m_fullscreen; |
| }; |
| |
| static void windowPropertiesNotifyCallback(GObject*, GParamSpec* paramSpec, UIClientTest* test) |
| { |
| test->m_windowPropertiesChanged.add(g_param_spec_get_name(paramSpec)); |
| } |
| |
| static WebKitWebView* viewCreateCallback(WebKitWebView* webView, WebKitNavigationAction* navigation, UIClientTest* test) |
| { |
| return test->viewCreate(webView, navigation); |
| } |
| |
| static void viewReadyToShowCallback(WebKitWebView* webView, UIClientTest* test) |
| { |
| test->viewReadyToShow(webView); |
| } |
| |
| static void viewCloseCallback(WebKitWebView* webView, UIClientTest* test) |
| { |
| test->viewClose(webView); |
| } |
| |
| void scriptAlert(WebKitScriptDialog* dialog) |
| { |
| switch (m_scriptDialogType) { |
| case WEBKIT_SCRIPT_DIALOG_ALERT: |
| g_assert_cmpstr(webkit_script_dialog_get_message(dialog), ==, kAlertDialogMessage); |
| break; |
| case WEBKIT_SCRIPT_DIALOG_CONFIRM: |
| g_assert_true(m_scriptDialogConfirmed); |
| g_assert_cmpstr(webkit_script_dialog_get_message(dialog), ==, "confirmed"); |
| |
| break; |
| case WEBKIT_SCRIPT_DIALOG_PROMPT: |
| g_assert_cmpstr(webkit_script_dialog_get_message(dialog), ==, kPromptDialogReturnedText); |
| break; |
| case WEBKIT_SCRIPT_DIALOG_BEFORE_UNLOAD_CONFIRM: |
| g_assert_not_reached(); |
| break; |
| } |
| |
| if (m_delayedScriptDialogs) { |
| webkit_script_dialog_ref(dialog); |
| RunLoop::main().dispatch([this, dialog] { |
| webkit_script_dialog_close(dialog); |
| webkit_script_dialog_unref(dialog); |
| g_main_loop_quit(m_mainLoop); |
| }); |
| } else |
| g_main_loop_quit(m_mainLoop); |
| } |
| |
| void scriptConfirm(WebKitScriptDialog* dialog) |
| { |
| g_assert_cmpstr(webkit_script_dialog_get_message(dialog), ==, kConfirmDialogMessage); |
| m_scriptDialogConfirmed = !m_scriptDialogConfirmed; |
| webkit_script_dialog_confirm_set_confirmed(dialog, m_scriptDialogConfirmed); |
| } |
| |
| void scriptPrompt(WebKitScriptDialog* dialog) |
| { |
| g_assert_cmpstr(webkit_script_dialog_get_message(dialog), ==, kPromptDialogMessage); |
| g_assert_cmpstr(webkit_script_dialog_prompt_get_default_text(dialog), ==, "default"); |
| webkit_script_dialog_prompt_set_text(dialog, kPromptDialogReturnedText); |
| } |
| |
| void scriptBeforeUnloadConfirm(WebKitScriptDialog* dialog) |
| { |
| g_assert_cmpstr(webkit_script_dialog_get_message(dialog), ==, kBeforeUnloadConfirmDialogMessage); |
| m_scriptDialogConfirmed = true; |
| webkit_script_dialog_confirm_set_confirmed(dialog, m_scriptDialogConfirmed); |
| } |
| |
| static gboolean scriptDialog(WebKitWebView*, WebKitScriptDialog* dialog, UIClientTest* test) |
| { |
| switch (webkit_script_dialog_get_dialog_type(dialog)) { |
| case WEBKIT_SCRIPT_DIALOG_ALERT: |
| test->scriptAlert(dialog); |
| break; |
| case WEBKIT_SCRIPT_DIALOG_CONFIRM: |
| test->scriptConfirm(dialog); |
| break; |
| case WEBKIT_SCRIPT_DIALOG_PROMPT: |
| test->scriptPrompt(dialog); |
| break; |
| case WEBKIT_SCRIPT_DIALOG_BEFORE_UNLOAD_CONFIRM: |
| test->scriptBeforeUnloadConfirm(dialog); |
| break; |
| } |
| |
| return TRUE; |
| } |
| |
| static void mouseTargetChanged(WebKitWebView*, WebKitHitTestResult* hitTestResult, guint modifiers, UIClientTest* test) |
| { |
| g_assert_true(WEBKIT_IS_HIT_TEST_RESULT(hitTestResult)); |
| test->assertObjectIsDeletedWhenTestFinishes(G_OBJECT(hitTestResult)); |
| |
| test->m_mouseTargetHitTestResult = hitTestResult; |
| test->m_mouseTargetModifiers = modifiers; |
| if (test->m_waitingForMouseTargetChange) |
| g_main_loop_quit(test->m_mainLoop); |
| } |
| |
| static gboolean permissionRequested(WebKitWebView*, WebKitPermissionRequest* request, UIClientTest* test) |
| { |
| g_assert_true(WEBKIT_IS_PERMISSION_REQUEST(request)); |
| test->assertObjectIsDeletedWhenTestFinishes(G_OBJECT(request)); |
| |
| if (test->m_verifyMediaTypes && WEBKIT_IS_USER_MEDIA_PERMISSION_REQUEST(request)) { |
| WebKitUserMediaPermissionRequest* userMediaRequest = WEBKIT_USER_MEDIA_PERMISSION_REQUEST(request); |
| g_assert_true(webkit_user_media_permission_is_for_audio_device(userMediaRequest) == test->m_expectedAudioMedia); |
| g_assert_true(webkit_user_media_permission_is_for_video_device(userMediaRequest) == test->m_expectedVideoMedia); |
| } |
| |
| if (test->m_allowPermissionRequests) |
| webkit_permission_request_allow(request); |
| else |
| webkit_permission_request_deny(request); |
| |
| return TRUE; |
| } |
| |
| static void permissionResultMessageReceivedCallback(WebKitUserContentManager* userContentManager, WebKitJavascriptResult* javascriptResult, UIClientTest* test) |
| { |
| test->m_permissionResult.reset(WebViewTest::javascriptResultToCString(javascriptResult)); |
| g_main_loop_quit(test->m_mainLoop); |
| } |
| |
| UIClientTest() |
| : m_scriptDialogType(WEBKIT_SCRIPT_DIALOG_ALERT) |
| , m_scriptDialogConfirmed(true) |
| , m_allowPermissionRequests(false) |
| , m_verifyMediaTypes(false) |
| , m_expectedAudioMedia(false) |
| , m_expectedVideoMedia(false) |
| , m_mouseTargetModifiers(0) |
| { |
| webkit_settings_set_javascript_can_open_windows_automatically(webkit_web_view_get_settings(m_webView), TRUE); |
| g_signal_connect(m_webView, "create", G_CALLBACK(viewCreateCallback), this); |
| g_signal_connect(m_webView, "script-dialog", G_CALLBACK(scriptDialog), this); |
| g_signal_connect(m_webView, "mouse-target-changed", G_CALLBACK(mouseTargetChanged), this); |
| g_signal_connect(m_webView, "permission-request", G_CALLBACK(permissionRequested), this); |
| webkit_user_content_manager_register_script_message_handler(m_userContentManager.get(), "permission"); |
| g_signal_connect(m_userContentManager.get(), "script-message-received::permission", G_CALLBACK(permissionResultMessageReceivedCallback), this); |
| } |
| |
| ~UIClientTest() |
| { |
| g_signal_handlers_disconnect_matched(m_webView, G_SIGNAL_MATCH_DATA, 0, 0, 0, 0, this); |
| g_signal_handlers_disconnect_matched(m_userContentManager.get(), G_SIGNAL_MATCH_DATA, 0, 0, 0, 0, this); |
| webkit_user_content_manager_unregister_script_message_handler(m_userContentManager.get(), "permission"); |
| } |
| |
| static void tryWebViewCloseCallback(UIClientTest* test) |
| { |
| g_main_loop_quit(test->m_mainLoop); |
| } |
| |
| void tryCloseAndWaitUntilClosed() |
| { |
| gulong handler = g_signal_connect_swapped(m_webView, "close", G_CALLBACK(tryWebViewCloseCallback), this); |
| // Use an idle because webkit_web_view_try_close can emit the close signal in the |
| // current run loop iteration. |
| g_idle_add([](gpointer data) -> gboolean { |
| webkit_web_view_try_close(WEBKIT_WEB_VIEW(data)); |
| return G_SOURCE_REMOVE; |
| }, m_webView); |
| g_main_loop_run(m_mainLoop); |
| g_signal_handler_disconnect(m_webView, handler); |
| } |
| |
| void waitUntilMainLoopFinishes() |
| { |
| g_main_loop_run(m_mainLoop); |
| } |
| |
| const char* waitUntilPermissionResultMessageReceived() |
| { |
| m_permissionResult = nullptr; |
| g_main_loop_run(m_mainLoop); |
| return m_permissionResult.get(); |
| } |
| |
| void setExpectedWindowProperties(const WindowProperties& windowProperties) |
| { |
| m_windowProperties = windowProperties; |
| } |
| |
| WebKitHitTestResult* moveMouseAndWaitUntilMouseTargetChanged(int x, int y, unsigned mouseModifiers = 0) |
| { |
| m_waitingForMouseTargetChange = true; |
| mouseMoveTo(x, y, mouseModifiers); |
| g_main_loop_run(m_mainLoop); |
| m_waitingForMouseTargetChange = false; |
| return m_mouseTargetHitTestResult.get(); |
| } |
| |
| void simulateUserInteraction() |
| { |
| runJavaScriptAndWaitUntilFinished("document.getElementById('testInput').focus()", nullptr); |
| // FIXME: Implement keyStroke in WPE. |
| #if PLATFORM(GTK) |
| keyStroke(GDK_KEY_a); |
| keyStroke(GDK_KEY_b); |
| while (gtk_events_pending()) |
| gtk_main_iteration(); |
| #endif |
| } |
| |
| virtual WebKitWebView* viewCreate(WebKitWebView* webView, WebKitNavigationAction* navigation) |
| { |
| g_assert_true(webView == m_webView); |
| g_assert_nonnull(navigation); |
| |
| auto* newWebView = Test::createWebView(webView); |
| #if PLATFORM(GTK) |
| g_object_ref_sink(newWebView); |
| #endif |
| |
| m_webViewEvents.append(Create); |
| |
| WebKitWindowProperties* windowProperties = webkit_web_view_get_window_properties(WEBKIT_WEB_VIEW(newWebView)); |
| g_assert_nonnull(windowProperties); |
| assertObjectIsDeletedWhenTestFinishes(G_OBJECT(windowProperties)); |
| m_windowPropertiesChanged.clear(); |
| |
| g_signal_connect(windowProperties, "notify", G_CALLBACK(windowPropertiesNotifyCallback), this); |
| g_signal_connect(newWebView, "ready-to-show", G_CALLBACK(viewReadyToShowCallback), this); |
| g_signal_connect(newWebView, "close", G_CALLBACK(viewCloseCallback), this); |
| |
| return WEBKIT_WEB_VIEW(newWebView); |
| } |
| |
| virtual void viewReadyToShow(WebKitWebView* webView) |
| { |
| g_assert_true(webView != m_webView); |
| |
| WebKitWindowProperties* windowProperties = webkit_web_view_get_window_properties(webView); |
| g_assert_nonnull(windowProperties); |
| if (!m_windowProperties.isNull()) |
| WindowProperties(windowProperties).assertEqual(m_windowProperties); |
| |
| m_webViewEvents.append(ReadyToShow); |
| } |
| |
| virtual void viewClose(WebKitWebView* webView) |
| { |
| g_assert_true(webView != m_webView); |
| |
| m_webViewEvents.append(Close); |
| g_object_unref(webView); |
| |
| g_main_loop_quit(m_mainLoop); |
| } |
| |
| Vector<WebViewEvents> m_webViewEvents; |
| WebKitScriptDialogType m_scriptDialogType; |
| bool m_scriptDialogConfirmed; |
| bool m_delayedScriptDialogs { false }; |
| bool m_allowPermissionRequests; |
| gboolean m_verifyMediaTypes; |
| gboolean m_expectedAudioMedia; |
| gboolean m_expectedVideoMedia; |
| WindowProperties m_windowProperties; |
| HashSet<WTF::String> m_windowPropertiesChanged; |
| GRefPtr<WebKitHitTestResult> m_mouseTargetHitTestResult; |
| unsigned m_mouseTargetModifiers; |
| GUniquePtr<char> m_permissionResult; |
| bool m_waitingForMouseTargetChange { false }; |
| }; |
| |
| static void testWebViewCreateReadyClose(UIClientTest* test, gconstpointer) |
| { |
| test->loadHtml("<html><body onLoad=\"window.open().close();\"></html>", 0); |
| test->waitUntilMainLoopFinishes(); |
| |
| Vector<UIClientTest::WebViewEvents>& events = test->m_webViewEvents; |
| g_assert_cmpint(events.size(), ==, 3); |
| g_assert_cmpint(events[0], ==, UIClientTest::Create); |
| g_assert_cmpint(events[1], ==, UIClientTest::ReadyToShow); |
| g_assert_cmpint(events[2], ==, UIClientTest::Close); |
| } |
| |
| #if PLATFORM(GTK) |
| class CreateNavigationDataTest: public UIClientTest { |
| public: |
| MAKE_GLIB_TEST_FIXTURE(CreateNavigationDataTest); |
| |
| CreateNavigationDataTest() |
| : m_navigation(nullptr) |
| { |
| } |
| |
| ~CreateNavigationDataTest() |
| { |
| clearNavigation(); |
| } |
| |
| void clearNavigation() |
| { |
| if (m_navigation) |
| webkit_navigation_action_free(m_navigation); |
| m_navigation = nullptr; |
| } |
| |
| WebKitWebView* viewCreate(WebKitWebView* webView, WebKitNavigationAction* navigation) |
| { |
| g_assert_nonnull(navigation); |
| g_assert_null(m_navigation); |
| m_navigation = webkit_navigation_action_copy(navigation); |
| g_main_loop_quit(m_mainLoop); |
| return nullptr; |
| } |
| |
| void loadHTML(const char* html) |
| { |
| clearNavigation(); |
| WebViewTest::loadHtml(html, nullptr); |
| } |
| |
| void clickAndWaitUntilMainLoopFinishes(int x, int y) |
| { |
| clearNavigation(); |
| clickMouseButton(x, y, 1); |
| g_main_loop_run(m_mainLoop); |
| } |
| |
| WebKitNavigationAction* m_navigation; |
| }; |
| |
| static void testWebViewCreateNavigationData(CreateNavigationDataTest* test, gconstpointer) |
| { |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(); |
| #endif |
| |
| test->loadHTML( |
| "<html><body>" |
| "<input style=\"position:absolute; left:0; top:0; margin:0; padding:0\" type=\"button\" value=\"click to show a popup\" onclick=\"window.open('data:foo');\"/>" |
| "<a style=\"position:absolute; left:20; top:20;\" href=\"data:bar\" target=\"_blank\">popup link</a>" |
| "</body></html>"); |
| test->waitUntilLoadFinished(); |
| |
| // Click on a button. |
| test->clickAndWaitUntilMainLoopFinishes(5, 5); |
| g_assert_cmpstr(webkit_uri_request_get_uri(webkit_navigation_action_get_request(test->m_navigation)), ==, "data:foo"); |
| g_assert_cmpuint(webkit_navigation_action_get_navigation_type(test->m_navigation), ==, WEBKIT_NAVIGATION_TYPE_OTHER); |
| // FIXME: This should be button 1. |
| g_assert_cmpuint(webkit_navigation_action_get_mouse_button(test->m_navigation), ==, 0); |
| g_assert_cmpuint(webkit_navigation_action_get_modifiers(test->m_navigation), ==, 0); |
| g_assert_true(webkit_navigation_action_is_user_gesture(test->m_navigation)); |
| |
| // Click on a link. |
| test->clickAndWaitUntilMainLoopFinishes(21, 21); |
| g_assert_cmpstr(webkit_uri_request_get_uri(webkit_navigation_action_get_request(test->m_navigation)), ==, "data:bar"); |
| g_assert_cmpuint(webkit_navigation_action_get_navigation_type(test->m_navigation), ==, WEBKIT_NAVIGATION_TYPE_LINK_CLICKED); |
| g_assert_cmpuint(webkit_navigation_action_get_mouse_button(test->m_navigation), ==, 1); |
| g_assert_cmpuint(webkit_navigation_action_get_modifiers(test->m_navigation), ==, 0); |
| g_assert_true(webkit_navigation_action_is_user_gesture(test->m_navigation)); |
| |
| // No user interaction. |
| test->loadHTML("<html><body onLoad=\"window.open();\"></html>"); |
| test->waitUntilMainLoopFinishes(); |
| |
| g_assert_cmpstr(webkit_uri_request_get_uri(webkit_navigation_action_get_request(test->m_navigation)), ==, ""); |
| g_assert_cmpuint(webkit_navigation_action_get_navigation_type(test->m_navigation), ==, WEBKIT_NAVIGATION_TYPE_OTHER); |
| g_assert_cmpuint(webkit_navigation_action_get_mouse_button(test->m_navigation), ==, 0); |
| g_assert_cmpuint(webkit_navigation_action_get_modifiers(test->m_navigation), ==, 0); |
| g_assert_false(webkit_navigation_action_is_user_gesture(test->m_navigation)); |
| } |
| #endif // PLATFORM(GTK) |
| |
| #if PLATFORM(GTK) |
| static gboolean checkMimeTypeForFilter(GtkFileFilter* filter, const gchar* mimeType) |
| { |
| GtkFileFilterInfo filterInfo; |
| filterInfo.contains = GTK_FILE_FILTER_MIME_TYPE; |
| filterInfo.mime_type = mimeType; |
| return gtk_file_filter_filter(filter, &filterInfo); |
| } |
| #endif |
| |
| class ModalDialogsTest: public UIClientTest { |
| public: |
| MAKE_GLIB_TEST_FIXTURE(ModalDialogsTest); |
| |
| static void dialogRunAsModalCallback(WebKitWebView* webView, ModalDialogsTest* test) |
| { |
| g_assert_true(webView != test->m_webView); |
| test->m_webViewEvents.append(RunAsModal); |
| } |
| |
| WebKitWebView* viewCreate(WebKitWebView* webView, WebKitNavigationAction* navigation) |
| { |
| g_assert_true(webView == m_webView); |
| |
| auto* newWebView = UIClientTest::viewCreate(webView, navigation); |
| g_signal_connect(newWebView, "run-as-modal", G_CALLBACK(dialogRunAsModalCallback), this); |
| return newWebView; |
| } |
| |
| void viewReadyToShow(WebKitWebView* webView) |
| { |
| g_assert_true(webView != m_webView); |
| m_webViewEvents.append(ReadyToShow); |
| } |
| }; |
| |
| static void testWebViewAllowModalDialogs(ModalDialogsTest* test, gconstpointer) |
| { |
| WebKitSettings* settings = webkit_web_view_get_settings(test->m_webView); |
| webkit_settings_set_allow_modal_dialogs(settings, TRUE); |
| |
| test->loadHtml("<html><body onload=\"window.showModalDialog('data:text/html,<html><body/><script>window.close();</script></html>')\"></body></html>", 0); |
| test->waitUntilMainLoopFinishes(); |
| |
| Vector<UIClientTest::WebViewEvents>& events = test->m_webViewEvents; |
| g_assert_cmpint(events.size(), ==, 4); |
| g_assert_cmpint(events[0], ==, UIClientTest::Create); |
| g_assert_cmpint(events[1], ==, UIClientTest::ReadyToShow); |
| g_assert_cmpint(events[2], ==, UIClientTest::RunAsModal); |
| g_assert_cmpint(events[3], ==, UIClientTest::Close); |
| } |
| |
| static void testWebViewDisallowModalDialogs(ModalDialogsTest* test, gconstpointer) |
| { |
| WebKitSettings* settings = webkit_web_view_get_settings(test->m_webView); |
| webkit_settings_set_allow_modal_dialogs(settings, FALSE); |
| |
| test->loadHtml("<html><body onload=\"window.showModalDialog('data:text/html,<html><body/><script>window.close();</script></html>')\"></body></html>", 0); |
| // We need to use a timeout here because the viewClose() function |
| // won't ever be called as the dialog won't be created. |
| test->wait(1); |
| |
| Vector<UIClientTest::WebViewEvents>& events = test->m_webViewEvents; |
| g_assert_cmpint(events.size(), ==, 0); |
| } |
| |
| static void testWebViewJavaScriptDialogs(UIClientTest* test, gconstpointer) |
| { |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(GTK_WINDOW_TOPLEVEL); |
| #endif |
| |
| static const char* htmlOnLoadFormat = "<html><body onLoad=\"%s\"></body></html>"; |
| static const char* jsAlertFormat = "alert('%s')"; |
| static const char* jsConfirmFormat = "do { confirmed = confirm('%s'); } while (!confirmed); alert('confirmed');"; |
| static const char* jsPromptFormat = "alert(prompt('%s', 'default'));"; |
| #if PLATFORM(GTK) |
| static const char* htmlOnBeforeUnloadFormat = |
| "<html><body onbeforeunload=\"return beforeUnloadHandler();\"><input id=\"testInput\" type=\"text\"></input><script>function beforeUnloadHandler() { return \"%s\"; }</script></body></html>"; |
| #endif |
| |
| for (unsigned i = 0; i <= 1; ++i) { |
| test->m_delayedScriptDialogs = !!i; |
| |
| test->m_scriptDialogType = WEBKIT_SCRIPT_DIALOG_ALERT; |
| GUniquePtr<char> alertDialogMessage(g_strdup_printf(jsAlertFormat, kAlertDialogMessage)); |
| GUniquePtr<char> alertHTML(g_strdup_printf(htmlOnLoadFormat, alertDialogMessage.get())); |
| test->loadHtml(alertHTML.get(), nullptr); |
| test->waitUntilMainLoopFinishes(); |
| webkit_web_view_stop_loading(test->m_webView); |
| test->waitUntilLoadFinished(); |
| |
| test->m_scriptDialogType = WEBKIT_SCRIPT_DIALOG_CONFIRM; |
| GUniquePtr<char> confirmDialogMessage(g_strdup_printf(jsConfirmFormat, kConfirmDialogMessage)); |
| GUniquePtr<char> confirmHTML(g_strdup_printf(htmlOnLoadFormat, confirmDialogMessage.get())); |
| test->loadHtml(confirmHTML.get(), nullptr); |
| test->waitUntilMainLoopFinishes(); |
| webkit_web_view_stop_loading(test->m_webView); |
| test->waitUntilLoadFinished(); |
| |
| test->m_scriptDialogType = WEBKIT_SCRIPT_DIALOG_PROMPT; |
| GUniquePtr<char> promptDialogMessage(g_strdup_printf(jsPromptFormat, kPromptDialogMessage)); |
| GUniquePtr<char> promptHTML(g_strdup_printf(htmlOnLoadFormat, promptDialogMessage.get())); |
| test->loadHtml(promptHTML.get(), nullptr); |
| test->waitUntilMainLoopFinishes(); |
| webkit_web_view_stop_loading(test->m_webView); |
| test->waitUntilLoadFinished(); |
| } |
| |
| test->m_delayedScriptDialogs = false; |
| |
| // FIXME: implement simulateUserInteraction in WPE. |
| #if PLATFORM(GTK) |
| test->m_scriptDialogType = WEBKIT_SCRIPT_DIALOG_BEFORE_UNLOAD_CONFIRM; |
| GUniquePtr<char> beforeUnloadDialogHTML(g_strdup_printf(htmlOnBeforeUnloadFormat, kBeforeUnloadConfirmDialogMessage)); |
| test->loadHtml(beforeUnloadDialogHTML.get(), nullptr); |
| test->waitUntilLoadFinished(); |
| |
| // Reload should trigger onbeforeunload. |
| #if 0 |
| test->simulateUserInteraction(); |
| // FIXME: reloading HTML data doesn't emit finished load event. |
| // See https://bugs.webkit.org/show_bug.cgi?id=139089. |
| test->m_scriptDialogConfirmed = false; |
| webkit_web_view_reload(test->m_webView); |
| test->waitUntilLoadFinished(); |
| g_assert_true(test->m_scriptDialogConfirmed); |
| #endif |
| |
| // Navigation should trigger onbeforeunload. |
| test->simulateUserInteraction(); |
| test->m_scriptDialogConfirmed = false; |
| test->loadHtml("<html></html>", nullptr); |
| test->waitUntilLoadFinished(); |
| g_assert_true(test->m_scriptDialogConfirmed); |
| |
| // Try close should trigger onbeforeunload. |
| test->m_scriptDialogConfirmed = false; |
| test->loadHtml(beforeUnloadDialogHTML.get(), nullptr); |
| test->waitUntilLoadFinished(); |
| test->simulateUserInteraction(); |
| test->tryCloseAndWaitUntilClosed(); |
| g_assert_true(test->m_scriptDialogConfirmed); |
| |
| // Try close on a page with no unload handlers should not trigger onbeforeunload, |
| // but should actually close the page. |
| test->m_scriptDialogConfirmed = false; |
| test->loadHtml("<html><body></body></html>", nullptr); |
| test->waitUntilLoadFinished(); |
| // We got a onbeforeunload of the previous page. |
| g_assert_true(test->m_scriptDialogConfirmed); |
| test->m_scriptDialogConfirmed = false; |
| test->tryCloseAndWaitUntilClosed(); |
| g_assert_false(test->m_scriptDialogConfirmed); |
| #endif // PLATFORM(GTK) |
| } |
| |
| static void testWebViewWindowProperties(UIClientTest* test, gconstpointer) |
| { |
| static const char* windowProrpertiesString = "left=100,top=150,width=400,height=400,location=no,menubar=no,status=no,toolbar=no,scrollbars=no"; |
| cairo_rectangle_int_t geometry = { 100, 150, 400, 400 }; |
| test->setExpectedWindowProperties(UIClientTest::WindowProperties(&geometry, false, false, false, false, false, true, false)); |
| |
| GUniquePtr<char> htmlString(g_strdup_printf("<html><body onLoad=\"window.open('', '', '%s').close();\"></body></html>", windowProrpertiesString)); |
| test->loadHtml(htmlString.get(), 0); |
| test->waitUntilMainLoopFinishes(); |
| |
| static const char* propertiesChanged[] = { |
| #if PLATFORM(GTK) |
| "geometry", |
| #endif |
| "locationbar-visible", "menubar-visible", "statusbar-visible", "toolbar-visible", "scrollbars-visible" |
| }; |
| for (size_t i = 0; i < G_N_ELEMENTS(propertiesChanged); ++i) |
| g_assert_true(test->m_windowPropertiesChanged.contains(propertiesChanged[i])); |
| |
| Vector<UIClientTest::WebViewEvents>& events = test->m_webViewEvents; |
| g_assert_cmpint(events.size(), ==, 3); |
| g_assert_cmpint(events[0], ==, UIClientTest::Create); |
| g_assert_cmpint(events[1], ==, UIClientTest::ReadyToShow); |
| g_assert_cmpint(events[2], ==, UIClientTest::Close); |
| } |
| |
| #if PLATFORM(GTK) |
| static void testWebViewMouseTarget(UIClientTest* test, gconstpointer) |
| { |
| test->showInWindowAndWaitUntilMapped(GTK_WINDOW_TOPLEVEL); |
| |
| const char* linksHoveredHTML = |
| "<html><head>" |
| " <script>" |
| " window.onload = function () {" |
| " window.getSelection().removeAllRanges();" |
| " var select_range = document.createRange();" |
| " select_range.selectNodeContents(document.getElementById('text_to_select'));" |
| " window.getSelection().addRange(select_range);" |
| " }" |
| " </script>" |
| "</head><body>" |
| " <a style='position:absolute; left:1; top:1' href='http://www.webkitgtk.org' title='WebKitGTK Title'>WebKitGTK Website</a>" |
| " <img style='position:absolute; left:1; top:10' src='0xdeadbeef' width=5 height=5></img>" |
| " <a style='position:absolute; left:1; top:20' href='http://www.webkitgtk.org/logo' title='WebKitGTK Logo'><img src='0xdeadbeef' width=5 height=5></img></a>" |
| " <input style='position:absolute; left:1; top:30' size='10'></input>" |
| " <video style='position:absolute; left:1; top:100' width='300' height='300' controls='controls' preload='none'><source src='movie.ogg' type='video/ogg' /></video>" |
| " <p style='position:absolute; left:1; top:120' id='text_to_select'>Lorem ipsum.</p>" |
| "</body></html>"; |
| |
| test->loadHtml(linksHoveredHTML, "file:///"); |
| test->waitUntilLoadFinished(); |
| |
| // Move over link. |
| WebKitHitTestResult* hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(1, 1); |
| g_assert_true(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_cmpstr(webkit_hit_test_result_get_link_uri(hitTestResult), ==, "http://www.webkitgtk.org/"); |
| g_assert_cmpstr(webkit_hit_test_result_get_link_title(hitTestResult), ==, "WebKitGTK Title"); |
| g_assert_cmpstr(webkit_hit_test_result_get_link_label(hitTestResult), ==, "WebKitGTK Website"); |
| g_assert_cmpuint(test->m_mouseTargetModifiers, ==, 0); |
| |
| // Move out of the link. |
| hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(0, 0); |
| g_assert_false(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_cmpuint(test->m_mouseTargetModifiers, ==, 0); |
| |
| // Move over image with GDK_CONTROL_MASK. |
| hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(1, 10, GDK_CONTROL_MASK); |
| g_assert_false(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_true(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_scrollbar(hitTestResult)); |
| g_assert_cmpstr(webkit_hit_test_result_get_image_uri(hitTestResult), ==, "file:///0xdeadbeef"); |
| g_assert_true(test->m_mouseTargetModifiers & GDK_CONTROL_MASK); |
| |
| // Move over image link. |
| hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(1, 20); |
| g_assert_true(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_true(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_scrollbar(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_cmpstr(webkit_hit_test_result_get_link_uri(hitTestResult), ==, "http://www.webkitgtk.org/logo"); |
| g_assert_cmpstr(webkit_hit_test_result_get_image_uri(hitTestResult), ==, "file:///0xdeadbeef"); |
| g_assert_cmpstr(webkit_hit_test_result_get_link_title(hitTestResult), ==, "WebKitGTK Logo"); |
| g_assert_false(webkit_hit_test_result_get_link_label(hitTestResult)); |
| g_assert_cmpuint(test->m_mouseTargetModifiers, ==, 0); |
| |
| // Move over media. |
| hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(1, 100); |
| g_assert_false(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_true(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_scrollbar(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_cmpstr(webkit_hit_test_result_get_media_uri(hitTestResult), ==, "file:///movie.ogg"); |
| g_assert_cmpuint(test->m_mouseTargetModifiers, ==, 0); |
| |
| // Mover over input. |
| hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(5, 35); |
| g_assert_false(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_scrollbar(hitTestResult)); |
| g_assert_true(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_cmpuint(test->m_mouseTargetModifiers, ==, 0); |
| |
| // Move over scrollbar. |
| hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(gtk_widget_get_allocated_width(GTK_WIDGET(test->m_webView)) - 4, 5); |
| g_assert_false(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_true(webkit_hit_test_result_context_is_scrollbar(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_cmpuint(test->m_mouseTargetModifiers, ==, 0); |
| |
| // Move over selection. |
| hitTestResult = test->moveMouseAndWaitUntilMouseTargetChanged(2, 145); |
| g_assert_false(webkit_hit_test_result_context_is_link(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_image(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_media(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_editable(hitTestResult)); |
| g_assert_false(webkit_hit_test_result_context_is_scrollbar(hitTestResult)); |
| g_assert_true(webkit_hit_test_result_context_is_selection(hitTestResult)); |
| g_assert_cmpuint(test->m_mouseTargetModifiers, ==, 0); |
| } |
| #endif // PLATFORM(GTK) |
| |
| static void testWebViewGeolocationPermissionRequests(UIClientTest* test, gconstpointer) |
| { |
| // Some versions of geoclue give a runtime warning because it tries |
| // to register the error quark twice. See https://bugs.webkit.org/show_bug.cgi?id=89858. |
| // Make warnings non-fatal for this test to make it pass. |
| Test::removeLogFatalFlag(G_LOG_LEVEL_WARNING); |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(); |
| #endif |
| static const char* geolocationRequestHTML = |
| "<html>" |
| " <script>" |
| " function runTest()" |
| " {" |
| " navigator.geolocation.getCurrentPosition(function(p) { window.webkit.messageHandlers.permission.postMessage('OK'); }," |
| " function(e) { window.webkit.messageHandlers.permission.postMessage(e.code.toString()); });" |
| " }" |
| " </script>" |
| " <body onload='runTest();'></body>" |
| "</html>"; |
| |
| // Geolocation is not allowed from insecure connections like HTTP, |
| // PERMISSION_DENIED ('1') is returned in that case without even |
| // asking the API layer. |
| test->m_allowPermissionRequests = false; |
| test->loadHtml(geolocationRequestHTML, "http://foo.com/bar"); |
| const gchar* result = test->waitUntilPermissionResultMessageReceived(); |
| g_assert_cmpstr(result, ==, "1"); |
| |
| // Test denying a permission request. PERMISSION_DENIED ('1') is |
| // returned in this case. |
| test->m_allowPermissionRequests = false; |
| test->loadHtml(geolocationRequestHTML, "https://foo.com/bar"); |
| result = test->waitUntilPermissionResultMessageReceived(); |
| g_assert_cmpstr(result, ==, "1"); |
| |
| // Test allowing a permission request. Result should be different |
| // to PERMISSION_DENIED ('1'). |
| test->m_allowPermissionRequests = true; |
| test->loadHtml(geolocationRequestHTML, "https://foo.com/bar"); |
| result = test->waitUntilPermissionResultMessageReceived(); |
| g_assert_cmpstr(result, !=, "1"); |
| Test::addLogFatalFlag(G_LOG_LEVEL_WARNING); |
| } |
| |
| #if ENABLE(MEDIA_STREAM) |
| static void testWebViewUserMediaEnumerateDevicesPermissionCheck(UIClientTest* test, gconstpointer) |
| { |
| WebKitSettings* settings = webkit_web_view_get_settings(test->m_webView); |
| gboolean enabled = webkit_settings_get_enable_media_stream(settings); |
| webkit_settings_set_enable_media_stream(settings, TRUE); |
| |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(); |
| #endif |
| static const char* userMediaRequestHTML = |
| "<html>" |
| " <script>" |
| " function runTest()" |
| " {" |
| " navigator.mediaDevices.enumerateDevices().then(" |
| " function(devices) { " |
| " devices.forEach(function(device) {" |
| " if (device.label) document.title = \"OK\";" |
| " else document.title = \"Permission denied\";" |
| " })" |
| " })" |
| " }" |
| " </script>" |
| " <body onload='runTest();'></body>" |
| "</html>"; |
| |
| test->m_verifyMediaTypes = TRUE; |
| |
| // Test denying a permission request. |
| test->m_allowPermissionRequests = false; |
| test->loadHtml(userMediaRequestHTML, nullptr); |
| test->waitUntilTitleChangedTo("Permission denied"); |
| |
| // Test allowing a permission request. |
| test->m_allowPermissionRequests = true; |
| test->loadHtml(userMediaRequestHTML, nullptr); |
| test->waitUntilTitleChangedTo("OK"); |
| |
| webkit_settings_set_enable_media_stream(settings, enabled); |
| } |
| |
| static void testWebViewUserMediaPermissionRequests(UIClientTest* test, gconstpointer) |
| { |
| WebKitSettings* settings = webkit_web_view_get_settings(test->m_webView); |
| gboolean enabled = webkit_settings_get_enable_media_stream(settings); |
| webkit_settings_set_enable_media_stream(settings, TRUE); |
| |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(); |
| #endif |
| static const char* userMediaRequestHTML = |
| "<html>" |
| " <script>" |
| " function runTest()" |
| " {" |
| " navigator.webkitGetUserMedia({audio: true, video: true}," |
| " function(s) { document.title = \"OK\" }," |
| " function(e) { document.title = e.name });" |
| " }" |
| " </script>" |
| " <body onload='runTest();'></body>" |
| "</html>"; |
| |
| test->m_verifyMediaTypes = TRUE; |
| test->m_expectedAudioMedia = TRUE; |
| test->m_expectedVideoMedia = TRUE; |
| |
| // Test denying a permission request. |
| test->m_allowPermissionRequests = false; |
| test->loadHtml(userMediaRequestHTML, nullptr); |
| test->waitUntilTitleChangedTo("PermissionDeniedError"); |
| |
| // Test allowing a permission request. |
| test->m_allowPermissionRequests = true; |
| test->loadHtml(userMediaRequestHTML, nullptr); |
| test->waitUntilTitleChangedTo("OK"); |
| |
| webkit_settings_set_enable_media_stream(settings, enabled); |
| } |
| |
| static void testWebViewAudioOnlyUserMediaPermissionRequests(UIClientTest* test, gconstpointer) |
| { |
| WebKitSettings* settings = webkit_web_view_get_settings(test->m_webView); |
| gboolean enabled = webkit_settings_get_enable_media_stream(settings); |
| webkit_settings_set_enable_media_stream(settings, TRUE); |
| |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(); |
| #endif |
| static const char* userMediaRequestHTML = |
| "<html>" |
| " <script>" |
| " function runTest()" |
| " {" |
| " navigator.webkitGetUserMedia({audio: true, video: false}," |
| " function(s) { document.title = \"OK\" }," |
| " function(e) { document.title = e.name });" |
| " }" |
| " </script>" |
| " <body onload='runTest();'></body>" |
| "</html>"; |
| |
| test->m_verifyMediaTypes = TRUE; |
| test->m_expectedAudioMedia = TRUE; |
| test->m_expectedVideoMedia = FALSE; |
| |
| // Test denying a permission request. |
| test->m_allowPermissionRequests = false; |
| test->loadHtml(userMediaRequestHTML, nullptr); |
| test->waitUntilTitleChangedTo("PermissionDeniedError"); |
| |
| webkit_settings_set_enable_media_stream(settings, enabled); |
| } |
| #endif // ENABLE(MEDIA_STREAM) |
| |
| #if ENABLE(POINTER_LOCK) |
| static void testWebViewPointerLockPermissionRequest(UIClientTest* test, gconstpointer) |
| { |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(GTK_WINDOW_TOPLEVEL); |
| #endif |
| |
| static const char* pointerLockRequestHTML = |
| "<html>" |
| " <script>" |
| " function runTest()" |
| " {" |
| " document.onpointerlockchange = function () { document.title = \"Locked\" };" |
| " document.onpointerlockerror = function () { document.title = \"Error\" };" |
| " document.getElementById('target').requestPointerLock();" |
| " }" |
| " </script>" |
| " <body>" |
| " <input style='position:absolute; left:0; top:0; margin:0; padding:0' type='button' value='click to lock pointer' onclick='runTest()'/>" |
| " <div id='target'></div>" |
| " </body>" |
| "</html>"; |
| |
| test->loadHtml(pointerLockRequestHTML, nullptr); |
| test->waitUntilLoadFinished(); |
| |
| // Test denying a permission request. |
| test->m_allowPermissionRequests = false; |
| test->clickMouseButton(5, 5, 1); |
| test->waitUntilTitleChangedTo("Error"); |
| |
| // Test allowing a permission request. |
| test->m_allowPermissionRequests = true; |
| test->clickMouseButton(5, 5, 1); |
| test->waitUntilTitleChangedTo("Locked"); |
| } |
| #endif |
| |
| #if PLATFORM(GTK) |
| class FileChooserTest: public UIClientTest { |
| public: |
| MAKE_GLIB_TEST_FIXTURE(FileChooserTest); |
| |
| FileChooserTest() |
| { |
| g_signal_connect(m_webView, "run-file-chooser", G_CALLBACK(runFileChooserCallback), this); |
| } |
| |
| static gboolean runFileChooserCallback(WebKitWebView*, WebKitFileChooserRequest* request, FileChooserTest* test) |
| { |
| test->runFileChooser(request); |
| return TRUE; |
| } |
| |
| void runFileChooser(WebKitFileChooserRequest* request) |
| { |
| assertObjectIsDeletedWhenTestFinishes(G_OBJECT(request)); |
| m_fileChooserRequest = request; |
| g_main_loop_quit(m_mainLoop); |
| } |
| |
| WebKitFileChooserRequest* clickMouseButtonAndWaitForFileChooserRequest(int x, int y) |
| { |
| clickMouseButton(x, y); |
| g_main_loop_run(m_mainLoop); |
| return m_fileChooserRequest.get(); |
| } |
| |
| private: |
| GRefPtr<WebKitFileChooserRequest> m_fileChooserRequest; |
| }; |
| |
| static void testWebViewFileChooserRequest(FileChooserTest* test, gconstpointer) |
| { |
| #if PLATFORM(GTK) |
| test->showInWindowAndWaitUntilMapped(); |
| #endif |
| static const char* fileChooserHTMLFormat = "<html><body><input style='position:absolute;left:0;top:0;margin:0;padding:0' type='file' %s/></body></html>"; |
| |
| // Multiple selections not allowed, no MIME filtering. |
| GUniquePtr<char> simpleFileUploadHTML(g_strdup_printf(fileChooserHTMLFormat, "")); |
| test->loadHtml(simpleFileUploadHTML.get(), 0); |
| test->waitUntilLoadFinished(); |
| WebKitFileChooserRequest* fileChooserRequest = test->clickMouseButtonAndWaitForFileChooserRequest(5, 5); |
| g_assert_false(webkit_file_chooser_request_get_select_multiple(fileChooserRequest)); |
| |
| const gchar* const* mimeTypes = webkit_file_chooser_request_get_mime_types(fileChooserRequest); |
| g_assert_null(mimeTypes); |
| #if PLATFORM(GTK) |
| GtkFileFilter* filter = webkit_file_chooser_request_get_mime_types_filter(fileChooserRequest); |
| g_assert_null(filter); |
| #endif |
| const gchar* const* selectedFiles = webkit_file_chooser_request_get_selected_files(fileChooserRequest); |
| g_assert_null(selectedFiles); |
| webkit_file_chooser_request_cancel(fileChooserRequest); |
| |
| // Multiple selections allowed, no MIME filtering, some pre-selected files. |
| GUniquePtr<char> multipleSelectionFileUploadHTML(g_strdup_printf(fileChooserHTMLFormat, "multiple")); |
| test->loadHtml(multipleSelectionFileUploadHTML.get(), 0); |
| test->waitUntilLoadFinished(); |
| fileChooserRequest = test->clickMouseButtonAndWaitForFileChooserRequest(5, 5); |
| g_assert_true(webkit_file_chooser_request_get_select_multiple(fileChooserRequest)); |
| |
| mimeTypes = webkit_file_chooser_request_get_mime_types(fileChooserRequest); |
| g_assert_null(mimeTypes); |
| #if PLATFORM(GTK) |
| filter = webkit_file_chooser_request_get_mime_types_filter(fileChooserRequest); |
| g_assert_null(filter); |
| #endif |
| selectedFiles = webkit_file_chooser_request_get_selected_files(fileChooserRequest); |
| g_assert_null(selectedFiles); |
| |
| // Select some files. |
| const gchar* filesToSelect[4] = { "/foo", "/foo/bar", "/foo/bar/baz", 0 }; |
| webkit_file_chooser_request_select_files(fileChooserRequest, filesToSelect); |
| |
| // Check the files that have been just selected. |
| selectedFiles = webkit_file_chooser_request_get_selected_files(fileChooserRequest); |
| g_assert_nonnull(selectedFiles); |
| g_assert_cmpstr(selectedFiles[0], ==, "/foo"); |
| g_assert_cmpstr(selectedFiles[1], ==, "/foo/bar"); |
| g_assert_cmpstr(selectedFiles[2], ==, "/foo/bar/baz"); |
| g_assert_null(selectedFiles[3]); |
| |
| // Perform another request to check if the list of files selected |
| // in the previous step appears now as part of the new request. |
| fileChooserRequest = test->clickMouseButtonAndWaitForFileChooserRequest(5, 5); |
| selectedFiles = webkit_file_chooser_request_get_selected_files(fileChooserRequest); |
| g_assert_nonnull(selectedFiles); |
| g_assert_cmpstr(selectedFiles[0], ==, "/foo"); |
| g_assert_cmpstr(selectedFiles[1], ==, "/foo/bar"); |
| g_assert_cmpstr(selectedFiles[2], ==, "/foo/bar/baz"); |
| g_assert_null(selectedFiles[3]); |
| webkit_file_chooser_request_cancel(fileChooserRequest); |
| |
| // Multiple selections not allowed, only accept images, audio and video files.. |
| GUniquePtr<char> mimeFilteredFileUploadHTML(g_strdup_printf(fileChooserHTMLFormat, "accept='audio/*,video/*,image/*'")); |
| test->loadHtml(mimeFilteredFileUploadHTML.get(), 0); |
| test->waitUntilLoadFinished(); |
| fileChooserRequest = test->clickMouseButtonAndWaitForFileChooserRequest(5, 5); |
| g_assert_false(webkit_file_chooser_request_get_select_multiple(fileChooserRequest)); |
| |
| mimeTypes = webkit_file_chooser_request_get_mime_types(fileChooserRequest); |
| g_assert_nonnull(mimeTypes); |
| g_assert_cmpstr(mimeTypes[0], ==, "audio/*"); |
| g_assert_cmpstr(mimeTypes[1], ==, "video/*"); |
| g_assert_cmpstr(mimeTypes[2], ==, "image/*"); |
| g_assert_null(mimeTypes[3]); |
| |
| #if PLATFORM(GTK) |
| filter = webkit_file_chooser_request_get_mime_types_filter(fileChooserRequest); |
| g_assert_true(GTK_IS_FILE_FILTER(filter)); |
| g_assert_true(checkMimeTypeForFilter(filter, "audio/*")); |
| g_assert_true(checkMimeTypeForFilter(filter, "video/*")); |
| g_assert_true(checkMimeTypeForFilter(filter, "image/*")); |
| #endif |
| |
| selectedFiles = webkit_file_chooser_request_get_selected_files(fileChooserRequest); |
| g_assert_null(selectedFiles); |
| webkit_file_chooser_request_cancel(fileChooserRequest); |
| } |
| #endif // PLATFORM(GTK) |
| |
| #if PLATFORM(GTK) |
| class ColorChooserTest: public WebViewTest { |
| public: |
| MAKE_GLIB_TEST_FIXTURE(ColorChooserTest); |
| |
| static gboolean runColorChooserCallback(WebKitWebView*, WebKitColorChooserRequest* request, ColorChooserTest* test) |
| { |
| test->runColorChooser(request); |
| return TRUE; |
| } |
| |
| static void requestFinishedCallback(WebKitColorChooserRequest* request, ColorChooserTest* test) |
| { |
| g_assert_true(test->m_request.get() == request); |
| test->m_request = nullptr; |
| if (g_main_loop_is_running(test->m_mainLoop)) |
| g_main_loop_quit(test->m_mainLoop); |
| } |
| |
| ColorChooserTest() |
| { |
| g_signal_connect(m_webView, "run-color-chooser", G_CALLBACK(runColorChooserCallback), this); |
| } |
| |
| void runColorChooser(WebKitColorChooserRequest* request) |
| { |
| g_assert_true(WEBKIT_IS_COLOR_CHOOSER_REQUEST(request)); |
| assertObjectIsDeletedWhenTestFinishes(G_OBJECT(request)); |
| m_request = request; |
| g_signal_connect(request, "finished", G_CALLBACK(requestFinishedCallback), this); |
| g_main_loop_quit(m_mainLoop); |
| } |
| |
| void finishRequest() |
| { |
| g_assert_nonnull(m_request.get()); |
| webkit_color_chooser_request_finish(m_request.get()); |
| g_assert_null(m_request); |
| } |
| |
| void cancelRequest() |
| { |
| g_assert_nonnull(m_request.get()); |
| webkit_color_chooser_request_cancel(m_request.get()); |
| g_assert_null(m_request); |
| } |
| |
| WebKitColorChooserRequest* clickMouseButtonAndWaitForColorChooserRequest(int x, int y) |
| { |
| clickMouseButton(x, y); |
| g_main_loop_run(m_mainLoop); |
| g_assert_nonnull(m_request.get()); |
| return m_request.get(); |
| } |
| |
| private: |
| GRefPtr<WebKitColorChooserRequest> m_request; |
| }; |
| |
| static void testWebViewColorChooserRequest(ColorChooserTest* test, gconstpointer) |
| { |
| static const char* colorChooserHTMLFormat = "<html><body><input style='position:absolute;left:1;top:1;margin:0;padding:0;width:45;height:25' type='color' %s/></body></html>"; |
| test->showInWindowAndWaitUntilMapped(); |
| |
| GUniquePtr<char> defaultColorHTML(g_strdup_printf(colorChooserHTMLFormat, "")); |
| test->loadHtml(defaultColorHTML.get(), nullptr); |
| test->waitUntilLoadFinished(); |
| WebKitColorChooserRequest* request = test->clickMouseButtonAndWaitForColorChooserRequest(5, 5); |
| |
| // Default color is black (#000000). |
| GdkRGBA rgba1; |
| GdkRGBA rgba2 = { 0., 0., 0., 1. }; |
| webkit_color_chooser_request_get_rgba(request, &rgba1); |
| g_assert_true(gdk_rgba_equal(&rgba1, &rgba2)); |
| |
| // Set a different color. |
| rgba2.green = 1; |
| webkit_color_chooser_request_set_rgba(request, &rgba2); |
| webkit_color_chooser_request_get_rgba(request, &rgba1); |
| g_assert_true(gdk_rgba_equal(&rgba1, &rgba2)); |
| |
| GdkRectangle rect; |
| webkit_color_chooser_request_get_element_rectangle(request, &rect); |
| g_assert_cmpint(rect.x, == , 1); |
| g_assert_cmpint(rect.y, == , 1); |
| g_assert_cmpint(rect.width, == , 45); |
| g_assert_cmpint(rect.height, == , 25); |
| |
| test->finishRequest(); |
| |
| // Use an initial color. |
| GUniquePtr<char> initialColorHTML(g_strdup_printf(colorChooserHTMLFormat, "value='#FF00FF'")); |
| test->loadHtml(initialColorHTML.get(), nullptr); |
| test->waitUntilLoadFinished(); |
| request = test->clickMouseButtonAndWaitForColorChooserRequest(5, 5); |
| |
| webkit_color_chooser_request_get_rgba(request, &rgba1); |
| GdkRGBA rgba3 = { 1., 0., 1., 1. }; |
| g_assert_true(gdk_rgba_equal(&rgba1, &rgba3)); |
| |
| test->cancelRequest(); |
| } |
| #endif // PLATFORM(GTK) |
| |
| void beforeAll() |
| { |
| UIClientTest::add("WebKitWebView", "create-ready-close", testWebViewCreateReadyClose); |
| // FIXME: Implement mouse clicks in WPE. |
| #if PLATFORM(GTK) |
| CreateNavigationDataTest::add("WebKitWebView", "create-navigation-data", testWebViewCreateNavigationData); |
| #endif |
| ModalDialogsTest::add("WebKitWebView", "allow-modal-dialogs", testWebViewAllowModalDialogs); |
| ModalDialogsTest::add("WebKitWebView", "disallow-modal-dialogs", testWebViewDisallowModalDialogs); |
| UIClientTest::add("WebKitWebView", "javascript-dialogs", testWebViewJavaScriptDialogs); |
| UIClientTest::add("WebKitWebView", "window-properties", testWebViewWindowProperties); |
| // FIXME: Implement mouse move in WPE. |
| #if PLATFORM(GTK) |
| UIClientTest::add("WebKitWebView", "mouse-target", testWebViewMouseTarget); |
| #endif |
| UIClientTest::add("WebKitWebView", "geolocation-permission-requests", testWebViewGeolocationPermissionRequests); |
| #if ENABLE(MEDIA_STREAM) |
| UIClientTest::add("WebKitWebView", "usermedia-enumeratedevices-permission-check", testWebViewUserMediaEnumerateDevicesPermissionCheck); |
| UIClientTest::add("WebKitWebView", "usermedia-permission-requests", testWebViewUserMediaPermissionRequests); |
| UIClientTest::add("WebKitWebView", "audio-usermedia-permission-request", testWebViewAudioOnlyUserMediaPermissionRequests); |
| #endif |
| // FIXME: Implement mouse click in WPE. |
| #if PLATFORM(GTK) |
| FileChooserTest::add("WebKitWebView", "file-chooser-request", testWebViewFileChooserRequest); |
| #endif |
| #if PLATFORM(GTK) |
| ColorChooserTest::add("WebKitWebView", "color-chooser-request", testWebViewColorChooserRequest); |
| #endif |
| #if ENABLE(POINTER_LOCK) |
| UIClientTest::add("WebKitWebView", "pointer-lock-permission-request", testWebViewPointerLockPermissionRequest); |
| #endif |
| } |
| |
| void afterAll() |
| { |
| } |