blob: b699ee3d0241171564d71bdd7c820347038ca60e [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/shell/webkit_test_runner.h"
#include <cmath>
#include "base/md5.h"
#include "base/memory/scoped_ptr.h"
#include "base/message_loop.h"
#include "base/stringprintf.h"
#include "base/sys_string_conversions.h"
#include "base/time.h"
#include "base/utf_string_conversions.h"
#include "content/public/renderer/render_view.h"
#include "content/public/test/layouttest_support.h"
#include "content/shell/shell_messages.h"
#include "content/shell/shell_render_process_observer.h"
#include "net/base/net_util.h"
#include "skia/ext/platform_canvas.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebCString.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebKitPlatformSupport.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebRect.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebSize.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebContextMenuData.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebDevToolsAgent.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebElement.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebKit.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h"
#include "third_party/WebKit/Tools/DumpRenderTree/chromium/TestRunner/public/WebTask.h"
#include "third_party/WebKit/Tools/DumpRenderTree/chromium/TestRunner/public/WebTestProxy.h"
#include "webkit/glue/webkit_glue.h"
#include "webkit/glue/webpreferences.h"
using WebKit::WebContextMenuData;
using WebKit::WebDevToolsAgent;
using WebKit::WebElement;
using WebKit::WebFrame;
using WebKit::WebGamepads;
using WebKit::WebRect;
using WebKit::WebSize;
using WebKit::WebString;
using WebKit::WebVector;
using WebKit::WebView;
using WebTestRunner::WebTask;
namespace content {
namespace {
void InvokeTaskHelper(void* context) {
WebTask* task = reinterpret_cast<WebTask*>(context);
task->run();
delete task;
}
std::string DumpDocumentText(WebFrame* frame) {
// We use the document element's text instead of the body text here because
// not all documents have a body, such as XML documents.
WebElement documentElement = frame->document().documentElement();
if (documentElement.isNull())
return std::string();
return documentElement.innerText().utf8();
}
std::string DumpDocumentPrintedText(WebFrame* frame) {
return frame->renderTreeAsText(WebFrame::RenderAsTextPrinting).utf8();
}
std::string DumpFramesAsText(WebFrame* frame, bool printing, bool recursive) {
std::string result;
// Cannot do printed format for anything other than HTML.
if (printing && !frame->document().isHTMLDocument())
return std::string();
// Add header for all but the main frame. Skip emtpy frames.
if (frame->parent() && !frame->document().documentElement().isNull()) {
result.append("\n--------\nFrame: '");
result.append(frame->uniqueName().utf8().data());
result.append("'\n--------\n");
}
result.append(
printing ? DumpDocumentPrintedText(frame) : DumpDocumentText(frame));
result.append("\n");
if (recursive) {
for (WebFrame* child = frame->firstChild(); child;
child = child->nextSibling()) {
result.append(DumpFramesAsText(child, printing, recursive));
}
}
return result;
}
std::string DumpFrameScrollPosition(WebFrame* frame, bool recursive) {
std::string result;
WebSize offset = frame->scrollOffset();
if (offset.width > 0 || offset.height > 0) {
if (frame->parent()) {
result.append(
base::StringPrintf("frame '%s' ", frame->uniqueName().utf8().data()));
}
result.append(
base::StringPrintf("scrolled to %d,%d\n", offset.width, offset.height));
}
if (recursive) {
for (WebFrame* child = frame->firstChild(); child;
child = child->nextSibling()) {
result.append(DumpFrameScrollPosition(child, recursive));
}
}
return result;
}
#if !defined(OS_MACOSX)
void MakeBitmapOpaque(SkBitmap* bitmap) {
SkAutoLockPixels lock(*bitmap);
DCHECK(bitmap->config() == SkBitmap::kARGB_8888_Config);
for (int y = 0; y < bitmap->height(); ++y) {
uint32_t* row = bitmap->getAddr32(0, y);
for (int x = 0; x < bitmap->width(); ++x)
row[x] |= 0xFF000000; // Set alpha bits to 1.
}
}
#endif
void CopyCanvasToBitmap(SkCanvas* canvas, SkBitmap* snapshot) {
SkDevice* device = skia::GetTopDevice(*canvas);
const SkBitmap& bitmap = device->accessBitmap(false);
bitmap.copyTo(snapshot, SkBitmap::kARGB_8888_Config);
#if !defined(OS_MACOSX)
// Only the expected PNGs for Mac have a valid alpha channel.
MakeBitmapOpaque(snapshot);
#endif
}
} // namespace
WebKitTestRunner::WebKitTestRunner(RenderView* render_view)
: RenderViewObserver(render_view) {
}
WebKitTestRunner::~WebKitTestRunner() {
}
// WebTestDelegate -----------------------------------------------------------
void WebKitTestRunner::clearContextMenuData() {
last_context_menu_data_.reset();
}
WebContextMenuData* WebKitTestRunner::lastContextMenuData() const {
return last_context_menu_data_.get();
}
void WebKitTestRunner::clearEditCommand() {
render_view()->ClearEditCommands();
}
void WebKitTestRunner::setEditCommand(const std::string& name,
const std::string& value) {
render_view()->SetEditCommandForNextKeyEvent(name, value);
}
void WebKitTestRunner::fillSpellingSuggestionList(
const WebString& word, WebVector<WebString>* suggestions) {
if (word == WebString::fromUTF8("wellcome")) {
WebVector<WebString> result(suggestions->size() + 1);
for (size_t i = 0; i < suggestions->size(); ++i)
result[i] = (*suggestions)[i];
result[suggestions->size()] = WebString::fromUTF8("welcome");
suggestions->swap(result);
}
}
void WebKitTestRunner::setGamepadData(const WebGamepads& gamepads) {
SetMockGamepads(gamepads);
}
void WebKitTestRunner::printMessage(const std::string& message) {
Send(new ShellViewHostMsg_PrintMessage(routing_id(), message));
}
void WebKitTestRunner::postTask(WebTask* task) {
WebKit::webKitPlatformSupport()->callOnMainThread(InvokeTaskHelper, task);
}
void WebKitTestRunner::postDelayedTask(WebTask* task, long long ms) {
MessageLoop::current()->PostDelayedTask(
FROM_HERE,
base::Bind(&WebTask::run, base::Owned(task)),
base::TimeDelta::FromMilliseconds(ms));
}
WebString WebKitTestRunner::registerIsolatedFileSystem(
const WebKit::WebVector<WebKit::WebString>& absolute_filenames) {
Send(new ShellViewHostMsg_NotImplemented(
routing_id(), "WebTestDelegate", "registerIsolatedFileSystem"));
return WebString();
}
long long WebKitTestRunner::getCurrentTimeInMillisecond() {
return base::TimeTicks::Now().ToInternalValue() /
base::Time::kMicrosecondsPerMillisecond;
}
WebString WebKitTestRunner::getAbsoluteWebStringFromUTF8Path(
const std::string& utf8_path) {
#if defined(OS_WIN)
FilePath path(UTF8ToWide(utf8_path));
#else
FilePath path(base::SysWideToNativeMB(base::SysUTF8ToWide(utf8_path)));
#endif
if (!path.IsAbsolute()) {
GURL base_url =
net::FilePathToFileURL(current_working_directory_.Append(
FILE_PATH_LITERAL("foo")));
net::FileURLToFilePath(base_url.Resolve(utf8_path), &path);
}
#if defined(OS_WIN)
return WebString(path.value());
#else
return WideToUTF16(base::SysNativeMBToWide(path.value()));
#endif
}
// RenderViewObserver --------------------------------------------------------
void WebKitTestRunner::DidClearWindowObject(WebFrame* frame) {
ShellRenderProcessObserver::GetInstance()->BindTestRunnersToWindow(frame);
}
void WebKitTestRunner::DidFinishLoad(WebFrame* frame) {
if (!frame->parent())
Send(new ShellViewHostMsg_DidFinishLoad(routing_id()));
}
void WebKitTestRunner::DidRequestShowContextMenu(
WebFrame* frame,
const WebContextMenuData& data) {
last_context_menu_data_.reset(new WebContextMenuData(data));
}
bool WebKitTestRunner::OnMessageReceived(const IPC::Message& message) {
bool handled = true;
IPC_BEGIN_MESSAGE_MAP(WebKitTestRunner, message)
IPC_MESSAGE_HANDLER(ShellViewMsg_CaptureTextDump, OnCaptureTextDump)
IPC_MESSAGE_HANDLER(ShellViewMsg_CaptureImageDump, OnCaptureImageDump)
IPC_MESSAGE_HANDLER(ShellViewMsg_SetCurrentWorkingDirectory,
OnSetCurrentWorkingDirectory)
IPC_MESSAGE_UNHANDLED(handled = false)
IPC_END_MESSAGE_MAP()
return handled;
}
// Public methods - -----------------------------------------------------------
void WebKitTestRunner::Display() {
const WebSize& size = render_view()->GetWebView()->size();
WebRect rect(0, 0, size.width, size.height);
proxy_->setPaintRect(rect);
PaintInvalidatedRegion();
DisplayRepaintMask();
}
void WebKitTestRunner::SetXSSAuditorEnabled(bool enabled) {
prefs_.xss_auditor_enabled = enabled;
webkit_glue::WebPreferences prefs = render_view()->GetWebkitPreferences();
prefs_.Apply(&prefs);
render_view()->SetWebkitPreferences(prefs);
Send(new ShellViewHostMsg_OverridePreferences(routing_id(), prefs_));
}
void WebKitTestRunner::NotifyDone() {
Send(new ShellViewHostMsg_NotifyDone(routing_id()));
}
void WebKitTestRunner::DumpAsText() {
Send(new ShellViewHostMsg_DumpAsText(routing_id()));
}
void WebKitTestRunner::DumpChildFramesAsText() {
Send(new ShellViewHostMsg_DumpChildFramesAsText(routing_id()));
}
void WebKitTestRunner::SetPrinting() {
Send(new ShellViewHostMsg_SetPrinting(routing_id()));
}
void WebKitTestRunner::SetShouldStayOnPageAfterHandlingBeforeUnload(
bool should_stay_on_page) {
Send(new ShellViewHostMsg_SetShouldStayOnPageAfterHandlingBeforeUnload(
routing_id(), should_stay_on_page));
}
void WebKitTestRunner::WaitUntilDone() {
Send(new ShellViewHostMsg_WaitUntilDone(routing_id()));
}
void WebKitTestRunner::CanOpenWindows() {
Send(new ShellViewHostMsg_CanOpenWindows(routing_id()));
}
void WebKitTestRunner::ShowWebInspector() {
Send(new ShellViewHostMsg_ShowWebInspector(routing_id()));
}
void WebKitTestRunner::CloseWebInspector() {
Send(new ShellViewHostMsg_CloseWebInspector(routing_id()));
}
void WebKitTestRunner::EvaluateInWebInspector(int32_t call_id,
const std::string& script) {
WebDevToolsAgent* agent = render_view()->GetWebView()->devToolsAgent();
if (agent)
agent->evaluateInWebInspector(call_id, WebString::fromUTF8(script));
}
void WebKitTestRunner::NotImplemented(const std::string& object,
const std::string& method) {
Send(new ShellViewHostMsg_NotImplemented(routing_id(), object, method));
}
void WebKitTestRunner::Reset() {
prefs_ = ShellWebPreferences();
webkit_glue::WebPreferences prefs = render_view()->GetWebkitPreferences();
prefs_.Apply(&prefs);
render_view()->SetWebkitPreferences(prefs);
}
// Private methods -----------------------------------------------------------
void WebKitTestRunner::OnCaptureTextDump(bool as_text,
bool printing,
bool recursive) {
WebFrame* frame = render_view()->GetWebView()->mainFrame();
std::string dump;
if (as_text) {
dump = DumpFramesAsText(frame, printing, recursive);
} else {
WebFrame::RenderAsTextControls render_text_behavior =
WebFrame::RenderAsTextNormal;
if (printing)
render_text_behavior |= WebFrame::RenderAsTextPrinting;
dump = frame->renderTreeAsText(render_text_behavior).utf8();
dump.append(DumpFrameScrollPosition(frame, recursive));
}
Send(new ShellViewHostMsg_TextDump(routing_id(), dump));
}
void WebKitTestRunner::OnCaptureImageDump(
const std::string& expected_pixel_hash) {
SkBitmap snapshot;
PaintInvalidatedRegion();
CopyCanvasToBitmap(GetCanvas(), &snapshot);
SkAutoLockPixels snapshot_lock(snapshot);
base::MD5Digest digest;
#if defined(OS_ANDROID)
// On Android, pixel layout is RGBA, however, other Chrome platforms use BGRA.
const uint8_t* raw_pixels =
reinterpret_cast<const uint8_t*>(snapshot.getPixels());
size_t snapshot_size = snapshot.getSize();
scoped_array<uint8_t> reordered_pixels(new uint8_t[snapshot_size]);
for (size_t i = 0; i < snapshot_size; i += 4) {
reordered_pixels[i] = raw_pixels[i + 2];
reordered_pixels[i + 1] = raw_pixels[i + 1];
reordered_pixels[i + 2] = raw_pixels[i];
reordered_pixels[i + 3] = raw_pixels[i + 3];
}
base::MD5Sum(reordered_pixels.get(), snapshot_size, &digest);
#else
base::MD5Sum(snapshot.getPixels(), snapshot.getSize(), &digest);
#endif
std::string actual_pixel_hash = base::MD5DigestToBase16(digest);
if (actual_pixel_hash == expected_pixel_hash) {
SkBitmap empty_image;
Send(new ShellViewHostMsg_ImageDump(
routing_id(), actual_pixel_hash, empty_image));
return;
}
Send(new ShellViewHostMsg_ImageDump(
routing_id(), actual_pixel_hash, snapshot));
}
void WebKitTestRunner::OnSetCurrentWorkingDirectory(
const FilePath& current_working_directory) {
current_working_directory_ = current_working_directory;
}
SkCanvas* WebKitTestRunner::GetCanvas() {
WebView* view = render_view()->GetWebView();
const WebSize& size = view->size();
float device_scale_factor = view->deviceScaleFactor();
int width = std::ceil(device_scale_factor * size.width);
int height = std::ceil(device_scale_factor * size.height);
if (canvas_ &&
canvas_->getDeviceSize().width() == width &&
canvas_->getDeviceSize().height() == height) {
return canvas_.get();
}
canvas_.reset(skia::CreatePlatformCanvas(
size.width, size.height, true, 0, skia::RETURN_NULL_ON_FAILURE));
return canvas_.get();
}
void WebKitTestRunner::PaintRect(const WebRect& rect) {
WebView* view = render_view()->GetWebView();
float device_scale_factor = view->deviceScaleFactor();
int scaled_x = device_scale_factor * rect.x;
int scaled_y = device_scale_factor * rect.y;
int scaled_width = std::ceil(device_scale_factor * rect.width);
int scaled_height = std::ceil(device_scale_factor * rect.height);
// TODO(jochen): Verify that the scaling is correct once the HiDPI tests
// actually work.
WebRect device_rect(scaled_x, scaled_y, scaled_width, scaled_height);
view->paint(webkit_glue::ToWebCanvas(GetCanvas()), device_rect);
}
void WebKitTestRunner::PaintInvalidatedRegion() {
WebView* view = render_view()->GetWebView();
view->animate(0.0);
view->layout();
const WebSize& widget_size = view->size();
WebRect client_rect(0, 0, widget_size.width, widget_size.height);
// Paint the canvas if necessary. Allow painting to generate extra rects
// for the first two calls. This is necessary because some WebCore rendering
// objects update their layout only when painted.
for (int i = 0; i < 3; ++i) {
// Make sure that paint_rect is always inside the RenderView's visible
// area.
WebRect paint_rect = proxy_->paintRect();
int left = std::max(paint_rect.x, client_rect.x);
int top = std::max(paint_rect.y, client_rect.y);
int right = std::min(paint_rect.x + paint_rect.width,
client_rect.x + client_rect.width);
int bottom = std::min(paint_rect.y + paint_rect.height,
client_rect.y + client_rect.height);
WebRect rect;
if (left < right && top < bottom)
rect = WebRect(left, top, right - left, bottom - top);
proxy_->setPaintRect(WebRect());
if (rect.isEmpty())
continue;
PaintRect(rect);
}
CHECK(proxy_->paintRect().isEmpty());
}
void WebKitTestRunner::DisplayRepaintMask() {
GetCanvas()->drawARGB(167, 0, 0, 0);
}
} // namespace content