| // 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 |