443 lines
14 KiB
C++
443 lines
14 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// Name: tests/controls/webtest.cpp
|
|
// Purpose: wxWebView unit test
|
|
// Author: Steven Lamerton
|
|
// Created: 2011-07-08
|
|
// Copyright: (c) 2011 Steven Lamerton
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "testprec.h"
|
|
|
|
#if wxUSE_WEBVIEW && (wxUSE_WEBVIEW_WEBKIT || wxUSE_WEBVIEW_WEBKIT2 || wxUSE_WEBVIEW_IE)
|
|
|
|
|
|
#ifndef WX_PRECOMP
|
|
#include "wx/app.h"
|
|
#endif // WX_PRECOMP
|
|
|
|
#include "testableframe.h"
|
|
#include "wx/webview.h"
|
|
#include "asserthelper.h"
|
|
#if wxUSE_WEBVIEW_IE
|
|
#include "wx/msw/webview_ie.h"
|
|
#endif
|
|
#if wxUSE_WEBVIEW_WEBKIT2
|
|
#include "wx/stopwatch.h"
|
|
#endif
|
|
|
|
//Convenience macro
|
|
#define ENSURE_LOADED CHECK( m_loaded->WaitEvent() )
|
|
|
|
class WebViewTestCase
|
|
{
|
|
public:
|
|
WebViewTestCase()
|
|
: m_browser(wxWebView::New()),
|
|
m_loaded(new EventCounter(m_browser, wxEVT_WEBVIEW_LOADED))
|
|
{
|
|
#ifdef __WXMSW__
|
|
if (wxWebView::IsBackendAvailable(wxWebViewBackendEdge))
|
|
{
|
|
// The blank page does not have an empty title with edge
|
|
m_blankTitle = "about:blank";
|
|
// Edge does not support about: url use a different URL instead
|
|
m_alternateHistoryURL = "about:blank";
|
|
}
|
|
else
|
|
#endif
|
|
m_alternateHistoryURL = "about:";
|
|
}
|
|
|
|
~WebViewTestCase()
|
|
{
|
|
delete m_loaded;
|
|
delete m_browser;
|
|
}
|
|
|
|
protected:
|
|
void LoadUrl(int times = 1)
|
|
{
|
|
//We alternate between urls as otherwise webkit merges them in the history
|
|
//we use about and about blank to avoid the need for a network connection
|
|
for(int i = 0; i < times; i++)
|
|
{
|
|
if(i % 2 == 1)
|
|
m_browser->LoadURL("about:blank");
|
|
else
|
|
m_browser->LoadURL(m_alternateHistoryURL);
|
|
ENSURE_LOADED;
|
|
}
|
|
}
|
|
|
|
void OnScriptResult(const wxWebViewEvent& evt)
|
|
{
|
|
m_asyncScriptResult = (evt.IsError()) ? 0 : 1;
|
|
m_asyncScriptString = evt.GetString();
|
|
}
|
|
|
|
void RunAsyncScript(const wxString& javascript)
|
|
{
|
|
m_browser->Bind(wxEVT_WEBVIEW_SCRIPT_RESULT, &WebViewTestCase::OnScriptResult, this);
|
|
m_asyncScriptResult = -1;
|
|
m_browser->RunScriptAsync(javascript);
|
|
while (m_asyncScriptResult == -1)
|
|
wxYield();
|
|
m_browser->Unbind(wxEVT_WEBVIEW_SCRIPT_RESULT, &WebViewTestCase::OnScriptResult, this);
|
|
}
|
|
|
|
wxWebView* const m_browser;
|
|
EventCounter* const m_loaded;
|
|
wxString m_blankTitle;
|
|
wxString m_alternateHistoryURL;
|
|
int m_asyncScriptResult;
|
|
wxString m_asyncScriptString;
|
|
};
|
|
|
|
TEST_CASE_METHOD(WebViewTestCase, "WebView", "[wxWebView]")
|
|
{
|
|
#if defined(__WXGTK__) && !defined(__WXGTK3__)
|
|
wxString value;
|
|
if ( !wxGetEnv("wxTEST_WEBVIEW_GTK2", &value) || value != "1" )
|
|
{
|
|
WARN("Skipping WebView tests known to fail with wxGTK 2, set "
|
|
"wxTEST_WEBVIEW_GTK2=1 to force running them.");
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
m_browser -> Create(wxTheApp->GetTopWindow(), wxID_ANY);
|
|
ENSURE_LOADED;
|
|
|
|
SECTION("Title")
|
|
{
|
|
CHECK(m_browser->GetCurrentTitle() == "");
|
|
|
|
//Test title after loading raw html
|
|
m_browser->SetPage("<html><title>Title</title><body>Text</body></html>", "");
|
|
ENSURE_LOADED;
|
|
CHECK(m_browser->GetCurrentTitle() == "Title");
|
|
|
|
//Test title after loading a url, we yield to let events process
|
|
LoadUrl();
|
|
CHECK(m_browser->GetCurrentTitle() == m_blankTitle);
|
|
}
|
|
|
|
SECTION("URL")
|
|
{
|
|
CHECK(m_browser->GetCurrentURL() == "about:blank");
|
|
|
|
//After first loading about:blank the next in the sequence is about:
|
|
LoadUrl();
|
|
CHECK(m_browser->GetCurrentURL() == m_alternateHistoryURL);
|
|
}
|
|
|
|
SECTION("History")
|
|
{
|
|
LoadUrl(3);
|
|
|
|
CHECK(m_browser->CanGoBack());
|
|
CHECK(!m_browser->CanGoForward());
|
|
|
|
m_browser->GoBack();
|
|
ENSURE_LOADED;
|
|
|
|
CHECK(m_browser->CanGoBack());
|
|
CHECK(m_browser->CanGoForward());
|
|
|
|
m_browser->GoBack();
|
|
ENSURE_LOADED;
|
|
m_browser->GoBack();
|
|
ENSURE_LOADED;
|
|
|
|
//We should now be at the start of the history
|
|
CHECK(!m_browser->CanGoBack());
|
|
CHECK(m_browser->CanGoForward());
|
|
}
|
|
|
|
#if !wxUSE_WEBVIEW_WEBKIT2 && !defined(__WXOSX__)
|
|
SECTION("HistoryEnable")
|
|
{
|
|
LoadUrl();
|
|
m_browser->EnableHistory(false);
|
|
|
|
CHECK(!m_browser->CanGoForward());
|
|
CHECK(!m_browser->CanGoBack());
|
|
|
|
LoadUrl();
|
|
|
|
CHECK(!m_browser->CanGoForward());
|
|
CHECK(!m_browser->CanGoBack());
|
|
}
|
|
#endif
|
|
|
|
#if !wxUSE_WEBVIEW_WEBKIT2 && !defined(__WXOSX__)
|
|
SECTION("HistoryClear")
|
|
{
|
|
LoadUrl(2);
|
|
|
|
//Now we are in the 'middle' of the history
|
|
m_browser->GoBack();
|
|
ENSURE_LOADED;
|
|
|
|
CHECK(m_browser->CanGoForward());
|
|
CHECK(m_browser->CanGoBack());
|
|
|
|
m_browser->ClearHistory();
|
|
|
|
CHECK(!m_browser->CanGoForward());
|
|
CHECK(!m_browser->CanGoBack());
|
|
}
|
|
#endif
|
|
|
|
SECTION("HistoryList")
|
|
{
|
|
LoadUrl(2);
|
|
m_browser->GoBack();
|
|
ENSURE_LOADED;
|
|
|
|
CHECK(m_browser->GetBackwardHistory().size() == 1);
|
|
CHECK(m_browser->GetForwardHistory().size() == 1);
|
|
|
|
m_browser->LoadHistoryItem(m_browser->GetForwardHistory()[0]);
|
|
ENSURE_LOADED;
|
|
|
|
CHECK(!m_browser->CanGoForward());
|
|
CHECK(m_browser->GetBackwardHistory().size() == 2);
|
|
}
|
|
|
|
#if !defined(__WXOSX__) && (!defined(wxUSE_WEBVIEW_EDGE) || !wxUSE_WEBVIEW_EDGE)
|
|
SECTION("Editable")
|
|
{
|
|
CHECK(!m_browser->IsEditable());
|
|
|
|
m_browser->SetEditable(true);
|
|
|
|
CHECK(m_browser->IsEditable());
|
|
|
|
m_browser->SetEditable(false);
|
|
|
|
CHECK(!m_browser->IsEditable());
|
|
}
|
|
#endif
|
|
|
|
SECTION("Selection")
|
|
{
|
|
m_browser->SetPage("<html><body>Some <strong>strong</strong> text</body></html>", "");
|
|
ENSURE_LOADED;
|
|
CHECK(!m_browser->HasSelection());
|
|
|
|
m_browser->SelectAll();
|
|
|
|
#if wxUSE_WEBVIEW_WEBKIT2
|
|
// With WebKit SelectAll() sends a request to perform the selection to
|
|
// another process via proxy and there doesn't seem to be any way to
|
|
// wait until this request is actually handled, so loop here for some a
|
|
// bit before giving up.
|
|
for ( wxStopWatch sw; !m_browser->HasSelection() && sw.Time() < 50; )
|
|
wxMilliSleep(1);
|
|
#endif // wxUSE_WEBVIEW_WEBKIT2
|
|
|
|
CHECK(m_browser->HasSelection());
|
|
CHECK(m_browser->GetSelectedText() == "Some strong text");
|
|
|
|
#if !defined(__WXOSX__) && (!defined(wxUSE_WEBVIEW_EDGE) || !wxUSE_WEBVIEW_EDGE)
|
|
// The web engine doesn't necessarily represent the HTML in the same way as
|
|
// we used above, e.g. IE uses upper case for all the tags while WebKit
|
|
// under OS X inserts plenty of its own <span> tags, so don't test for
|
|
// equality and just check that the source contains things we'd expect it
|
|
// to.
|
|
const wxString selSource = m_browser->GetSelectedSource();
|
|
WX_ASSERT_MESSAGE
|
|
(
|
|
("Unexpected selection source: \"%s\"", selSource),
|
|
selSource.Lower().Matches("*some*<strong*strong</strong>*text*")
|
|
);
|
|
#endif // !defined(__WXOSX__)
|
|
|
|
m_browser->ClearSelection();
|
|
CHECK(!m_browser->HasSelection());
|
|
}
|
|
|
|
SECTION("Zoom")
|
|
{
|
|
if(m_browser->CanSetZoomType(wxWEBVIEW_ZOOM_TYPE_LAYOUT))
|
|
{
|
|
m_browser->SetZoomType(wxWEBVIEW_ZOOM_TYPE_LAYOUT);
|
|
CHECK(m_browser->GetZoomType() == wxWEBVIEW_ZOOM_TYPE_LAYOUT);
|
|
|
|
m_browser->SetZoom(wxWEBVIEW_ZOOM_TINY);
|
|
CHECK(m_browser->GetZoom() == wxWEBVIEW_ZOOM_TINY);
|
|
}
|
|
|
|
//Reset the zoom level
|
|
m_browser->SetZoom(wxWEBVIEW_ZOOM_MEDIUM);
|
|
|
|
if(m_browser->CanSetZoomType(wxWEBVIEW_ZOOM_TYPE_TEXT))
|
|
{
|
|
m_browser->SetZoomType(wxWEBVIEW_ZOOM_TYPE_TEXT);
|
|
CHECK(m_browser->GetZoomType() == wxWEBVIEW_ZOOM_TYPE_TEXT);
|
|
|
|
m_browser->SetZoom(wxWEBVIEW_ZOOM_TINY);
|
|
CHECK(m_browser->GetZoom() == wxWEBVIEW_ZOOM_TINY);
|
|
}
|
|
}
|
|
|
|
SECTION("RunScript")
|
|
{
|
|
m_browser->
|
|
SetPage("<html><head><script></script></head><body></body></html>", "");
|
|
ENSURE_LOADED;
|
|
|
|
wxString result;
|
|
#if wxUSE_WEBVIEW_IE && !wxUSE_WEBVIEW_EDGE
|
|
// Define a specialized scope guard ensuring that we reset the emulation
|
|
// level to its default value even if any asserts below fail.
|
|
class ResetEmulationLevel
|
|
{
|
|
public:
|
|
ResetEmulationLevel()
|
|
{
|
|
// Allow this to fail because it doesn't work in GitHub Actions
|
|
// environment, but the tests below still pass there.
|
|
if ( !wxWebViewIE::MSWSetModernEmulationLevel() )
|
|
{
|
|
WARN("Setting IE modern emulation level failed.");
|
|
m_reset = false;
|
|
}
|
|
else
|
|
{
|
|
m_reset = true;
|
|
}
|
|
}
|
|
|
|
void DoReset()
|
|
{
|
|
if ( m_reset )
|
|
{
|
|
m_reset = false;
|
|
if ( !wxWebViewIE::MSWSetModernEmulationLevel(false) )
|
|
{
|
|
WARN("Resetting IE modern emulation level failed.");
|
|
}
|
|
}
|
|
}
|
|
|
|
~ResetEmulationLevel()
|
|
{
|
|
DoReset();
|
|
}
|
|
|
|
private:
|
|
bool m_reset;
|
|
} resetEmulationLevel;
|
|
|
|
CHECK(m_browser->RunScript("function f(){var person = new Object();person.name = 'Bar'; \
|
|
person.lastName = 'Foo';return person;}f();", &result));
|
|
CHECK(result == "{\"name\":\"Bar\",\"lastName\":\"Foo\"}");
|
|
|
|
CHECK(m_browser->RunScript("function f(){ return [\"foo\", \"bar\"]; }f();", &result));
|
|
CHECK(result == "[\"foo\",\"bar\"]");
|
|
|
|
CHECK(m_browser->RunScript("function f(){var d = new Date('10/08/2017 21:30:40'); \
|
|
var tzoffset = d.getTimezoneOffset() * 60000; return new Date(d.getTime() - tzoffset);}f();",
|
|
&result));
|
|
CHECK(result == "\"2017-10-08T21:30:40.000Z\"");
|
|
|
|
resetEmulationLevel.DoReset();
|
|
#endif // wxUSE_WEBVIEW_IE
|
|
|
|
CHECK(m_browser->RunScript("document.write(\"Hello World!\");"));
|
|
CHECK(m_browser->GetPageText() == "Hello World!");
|
|
|
|
CHECK(m_browser->RunScript("function f(a){return a;}f('Hello World!');", &result));
|
|
CHECK(result == _("Hello World!"));
|
|
|
|
CHECK(m_browser->RunScript("function f(a){return a;}f('a\\\'aa\\n\\rb\\tb\\\\ccc\\\"ddd\\b\\fx');", &result));
|
|
CHECK(result == _("a\'aa\n\rb\tb\\ccc\"ddd\b\fx"));
|
|
|
|
CHECK(m_browser->RunScript("function f(a){return a;}f(123);", &result));
|
|
CHECK(wxAtoi(result) == 123);
|
|
|
|
CHECK(m_browser->
|
|
RunScript("function f(a){return a;}f(2.34);", &result));
|
|
double value;
|
|
result.ToDouble(&value);
|
|
CHECK(value == 2.34);
|
|
|
|
CHECK(m_browser->RunScript("function f(a){return a;}f(false);", &result));
|
|
CHECK(result == "false");
|
|
|
|
CHECK(m_browser->RunScript("function f(){var person = new Object();person.lastName = 'Bar'; \
|
|
person.name = 'Foo';return person;}f();", &result));
|
|
CHECK(result == "{\"lastName\":\"Bar\",\"name\":\"Foo\"}");
|
|
|
|
CHECK(m_browser->RunScript("function f(){ return [\"foo\", \"bar\"]; }f();", &result));
|
|
CHECK(result == "[\"foo\",\"bar\"]");
|
|
|
|
CHECK(m_browser->RunScript("function f(){var person = new Object();}f();", &result));
|
|
CHECK(result == "undefined");
|
|
|
|
CHECK(m_browser->RunScript("function f(){return null;}f();", &result));
|
|
CHECK(result == "null");
|
|
|
|
result = "";
|
|
CHECK(!m_browser->RunScript("int main() { return 0; }", &result));
|
|
CHECK(!result);
|
|
|
|
CHECK(m_browser->RunScript("function a() { return eval(\"function b() { \
|
|
return eval(\\\"function c() { return eval(\\\\\\\"function d() { \
|
|
return \\\\\\\\\\\\\\\"test\\\\\\\\\\\\\\\"; } d();\\\\\\\"); } \
|
|
c();\\\"); } b();\"); } a();", &result));
|
|
CHECK(result == "test");
|
|
|
|
CHECK(m_browser->RunScript("function f(a){return a;}f(\"This is a backslash: \\\\\");",
|
|
&result));
|
|
CHECK(result == "This is a backslash: \\");
|
|
|
|
CHECK(m_browser->RunScript("function f(){var d = new Date('10/08/2016 21:30:40'); \
|
|
var tzoffset = d.getTimezoneOffset() * 60000; return new Date(d.getTime() - tzoffset);}f();",
|
|
&result));
|
|
CHECK(result == "\"2016-10-08T21:30:40.000Z\"");
|
|
|
|
// Check for C++-style comments which used to be broken.
|
|
CHECK(m_browser->RunScript("function f() {\n"
|
|
" // A C++ style comment\n"
|
|
" return 17;\n"
|
|
"}f();", &result));
|
|
CHECK(result == "17");
|
|
|
|
// Check for errors too.
|
|
CHECK(!m_browser->RunScript("syntax(error"));
|
|
CHECK(!m_browser->RunScript("syntax(error", &result));
|
|
CHECK(!m_browser->RunScript("x.y.z"));
|
|
}
|
|
|
|
SECTION("RunScriptAsync")
|
|
{
|
|
#ifdef __WXMSW__
|
|
// IE doesn't support async script execution
|
|
if (!wxWebView::IsBackendAvailable(wxWebViewBackendEdge))
|
|
return;
|
|
#endif
|
|
RunAsyncScript("function f(a){return a;}f('Hello World!');");
|
|
CHECK(m_asyncScriptResult == 1);
|
|
CHECK(m_asyncScriptString == "Hello World!");
|
|
|
|
RunAsyncScript("int main() { return 0; }");
|
|
CHECK(m_asyncScriptResult == 0);
|
|
}
|
|
|
|
SECTION("SetPage")
|
|
{
|
|
m_browser->SetPage("<html><body>text</body></html>", "");
|
|
ENSURE_LOADED;
|
|
CHECK(m_browser->GetPageText() == "text");
|
|
|
|
m_browser->SetPage("<html><body>other text</body></html>", "");
|
|
ENSURE_LOADED;
|
|
CHECK(m_browser->GetPageText() == "other text");
|
|
}
|
|
}
|
|
|
|
#endif //wxUSE_WEBVIEW && (wxUSE_WEBVIEW_WEBKIT || wxUSE_WEBVIEW_WEBKIT2 || wxUSE_WEBVIEW_IE)
|