blob: ee38fc76698c50f7292537821ec05c851ae5e7df [file] [log] [blame]
// Copyright 2015 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 <stdint.h>
#include <string>
#include <utility>
#include "base/command_line.h"
#include "base/stl_util.h"
#include "base/strings/string_piece.h"
#include "base/strings/stringprintf.h"
#include "base/test/scoped_feature_list.h"
#include "build/build_config.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_window.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/common/chrome_features.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/test/base/in_process_browser_test.h"
#include "chrome/test/base/test_switches.h"
#include "chrome/test/base/ui_test_utils.h"
#include "components/zoom/zoom_controller.h"
#include "content/public/browser/readback_types.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/render_view_host.h"
#include "content/public/browser/render_widget_host.h"
#include "content/public/common/content_switches.h"
#include "content/public/test/browser_test_utils.h"
#include "content/public/test/ppapi_test_utils.h"
#include "net/test/embedded_test_server/embedded_test_server.h"
#include "net/test/embedded_test_server/http_request.h"
#include "net/test/embedded_test_server/http_response.h"
#include "third_party/WebKit/public/web/WebInputEvent.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "ui/base/window_open_disposition.h"
#include "ui/display/display.h"
#include "ui/display/display_switches.h"
#include "ui/display/screen.h"
#include "ui/gfx/codec/png_codec.h"
#include "ui/gfx/geometry/point.h"
namespace {
// Use fixed browser dimensions for pixel tests.
const int kBrowserWidth = 600;
const int kBrowserHeight = 700;
// Compare only a portion of the snapshots, as different platforms will
// capture different sized snapshots (due to differences in browser chrome).
const int kComparisonWidth = 500;
const int kComparisonHeight = 600;
// Different platforms have slightly different pixel output, due to different
// graphics implementations. Slightly different pixels (in BGR space) are still
// counted as a matching pixel by this simple manhattan distance threshold.
const int kPixelManhattanDistanceTolerance = 25;
std::string RunTestScript(base::StringPiece test_script,
content::WebContents* contents,
const std::string& element_id) {
std::string script = base::StringPrintf(
"var plugin = window.document.getElementById('%s');"
"if (plugin === undefined ||"
" (plugin.nodeName !== 'OBJECT' && plugin.nodeName !== 'EMBED')) {"
" window.domAutomationController.send('error');"
"} else {"
" %s"
"}",
element_id.c_str(), test_script.data());
std::string result;
EXPECT_TRUE(
content::ExecuteScriptAndExtractString(contents, script, &result));
return result;
}
// This also tests that we have JavaScript access to the underlying plugin.
bool PluginLoaded(content::WebContents* contents,
const std::string& element_id) {
std::string result = RunTestScript(
"if (plugin.postMessage === undefined) {"
" window.domAutomationController.send('poster_only');"
"} else {"
" window.domAutomationController.send('plugin_loaded');"
"}",
contents, element_id);
EXPECT_NE("error", result);
return result == "plugin_loaded";
}
// Blocks until the placeholder is ready.
void WaitForPlaceholderReady(content::WebContents* contents,
const std::string& element_id) {
std::string result = RunTestScript(
"function handleEvent(event) {"
" if (event.data === 'placeholderReady') {"
" window.domAutomationController.send('ready');"
" plugin.removeEventListener('message', handleEvent);"
" }"
"}"
"plugin.addEventListener('message', handleEvent);"
"if (plugin.hasAttribute('placeholderReady')) {"
" window.domAutomationController.send('ready');"
" plugin.removeEventListener('message', handleEvent);"
"}",
contents, element_id);
ASSERT_EQ("ready", result);
}
// Also waits for the placeholder UI overlay to finish loading.
void VerifyPluginIsThrottled(content::WebContents* contents,
const std::string& element_id) {
std::string result = RunTestScript(
"function handleEvent(event) {"
" if (event.data.isPeripheral && event.data.isThrottled && "
" event.data.isHiddenForPlaceholder) {"
" window.domAutomationController.send('throttled');"
" plugin.removeEventListener('message', handleEvent);"
" }"
"}"
"plugin.addEventListener('message', handleEvent);"
"if (plugin.postMessage !== undefined) {"
" plugin.postMessage('getPowerSaverStatus');"
"}",
contents, element_id);
EXPECT_EQ("throttled", result);
// Page should continue to have JavaScript access to all throttled plugins.
EXPECT_TRUE(PluginLoaded(contents, element_id));
WaitForPlaceholderReady(contents, element_id);
}
void VerifyPluginMarkedEssential(content::WebContents* contents,
const std::string& element_id) {
std::string result = RunTestScript(
"function handleEvent(event) {"
" if (event.data.isPeripheral === false) {"
" window.domAutomationController.send('essential');"
" plugin.removeEventListener('message', handleEvent);"
" }"
"}"
"plugin.addEventListener('message', handleEvent);"
"if (plugin.postMessage !== undefined) {"
" plugin.postMessage('getPowerSaverStatus');"
"}",
contents, element_id);
EXPECT_EQ("essential", result);
EXPECT_TRUE(PluginLoaded(contents, element_id));
}
std::unique_ptr<net::test_server::HttpResponse> RespondWithHTML(
const std::string& html,
const net::test_server::HttpRequest& request) {
std::unique_ptr<net::test_server::BasicHttpResponse> response(
new net::test_server::BasicHttpResponse());
response->set_content_type("text/html");
response->set_content(html);
return std::move(response);
}
void VerifyVisualStateUpdated(const base::Closure& done_cb,
bool visual_state_updated) {
ASSERT_TRUE(visual_state_updated);
done_cb.Run();
}
bool SnapshotMatches(const base::FilePath& reference, const SkBitmap& bitmap) {
if (bitmap.width() < kComparisonWidth ||
bitmap.height() < kComparisonHeight) {
return false;
}
std::string reference_data;
if (!base::ReadFileToString(reference, &reference_data))
return false;
int w = 0;
int h = 0;
std::vector<unsigned char> decoded;
if (!gfx::PNGCodec::Decode(reinterpret_cast<unsigned char*>(
base::string_as_array(&reference_data)),
reference_data.size(), gfx::PNGCodec::FORMAT_BGRA,
&decoded, &w, &h)) {
return false;
}
if (w < kComparisonWidth || h < kComparisonHeight)
return false;
int32_t* ref_pixels = reinterpret_cast<int32_t*>(decoded.data());
SkAutoLockPixels lock_image(bitmap);
int32_t* pixels = static_cast<int32_t*>(bitmap.getPixels());
bool success = true;
for (int y = 0; y < kComparisonHeight; ++y) {
for (int x = 0; x < kComparisonWidth; ++x) {
int32_t pixel = pixels[y * bitmap.rowBytes() / sizeof(int32_t) + x];
int pixel_b = pixel & 0xFF;
int pixel_g = (pixel >> 8) & 0xFF;
int pixel_r = (pixel >> 16) & 0xFF;
int32_t ref_pixel = ref_pixels[y * w + x];
int ref_pixel_b = ref_pixel & 0xFF;
int ref_pixel_g = (ref_pixel >> 8) & 0xFF;
int ref_pixel_r = (ref_pixel >> 16) & 0xFF;
int manhattan_distance = abs(pixel_b - ref_pixel_b) +
abs(pixel_g - ref_pixel_g) +
abs(pixel_r - ref_pixel_r);
if (manhattan_distance > kPixelManhattanDistanceTolerance) {
ADD_FAILURE() << "Pixel test failed on (" << x << ", " << y << "). " <<
"Pixel manhattan distance: " << manhattan_distance << ".";
success = false;
}
}
}
return success;
}
// |snapshot_matches| is set to true if the snapshot matches the reference and
// the test passes. Otherwise, set to false.
void CompareSnapshotToReference(const base::FilePath& reference,
bool* snapshot_matches,
const base::Closure& done_cb,
const SkBitmap& bitmap,
content::ReadbackResponse response) {
DCHECK(snapshot_matches);
ASSERT_EQ(content::READBACK_SUCCESS, response);
*snapshot_matches = SnapshotMatches(reference, bitmap);
// When rebaselining the pixel test, the test may fail. However, the
// reference file will still be overwritten.
if (base::CommandLine::ForCurrentProcess()->HasSwitch(
switches::kRebaselinePixelTests)) {
SkBitmap clipped_bitmap;
bitmap.extractSubset(&clipped_bitmap,
SkIRect::MakeWH(kComparisonWidth, kComparisonHeight));
std::vector<unsigned char> png_data;
ASSERT_TRUE(
gfx::PNGCodec::EncodeBGRASkBitmap(clipped_bitmap, false, &png_data));
ASSERT_EQ(static_cast<int>(png_data.size()),
base::WriteFile(reference,
reinterpret_cast<const char*>(png_data.data()),
png_data.size()));
}
done_cb.Run();
}
} // namespace
class PluginPowerSaverBrowserTest : public InProcessBrowserTest {
public:
void SetUp() override {
if (PixelTestsEnabled())
EnablePixelOutput();
InProcessBrowserTest::SetUp();
}
void SetUpOnMainThread() override {
InProcessBrowserTest::SetUpOnMainThread();
ASSERT_TRUE(embedded_test_server()->Start());
embedded_test_server()->ServeFilesFromDirectory(
ui_test_utils::GetTestFilePath(
base::FilePath(FILE_PATH_LITERAL("plugin_power_saver")),
base::FilePath()));
}
void SetUpCommandLine(base::CommandLine* command_line) override {
command_line->AppendSwitch(switches::kEnablePluginPlaceholderTesting);
ASSERT_TRUE(ppapi::RegisterFlashTestPlugin(command_line));
// Allows us to use the same reference image on HiDPI/Retina displays.
command_line->AppendSwitchASCII(switches::kForceDeviceScaleFactor, "1");
// The pixel tests run more reliably in software mode.
if (PixelTestsEnabled())
command_line->AppendSwitch(switches::kDisableGpu);
}
void SetUpInProcessBrowserTestFixture() override {
// Although this is redundant with the Field Trial testing configuration,
// the official builders don't use those, so we also enable it here.
feature_list.InitAndEnableFeature(features::kBlockSmallContent);
// Disable HTML5 By Default feature to test Plugin Power Saver specifically.
feature_list.InitAndDisableFeature(features::kPreferHtmlOverPlugins);
}
protected:
void LoadHTML(const std::string& html) {
if (PixelTestsEnabled()) {
gfx::Rect bounds(gfx::Rect(0, 0, kBrowserWidth, kBrowserHeight));
gfx::Rect screen_bounds =
display::Screen::GetScreen()->GetPrimaryDisplay().bounds();
ASSERT_GT(screen_bounds.width(), kBrowserWidth);
ASSERT_GT(screen_bounds.height(), kBrowserHeight);
browser()->window()->SetBounds(bounds);
}
ASSERT_TRUE(embedded_test_server()->Started());
embedded_test_server()->RegisterRequestHandler(
base::Bind(&RespondWithHTML, html));
ui_test_utils::NavigateToURL(browser(), embedded_test_server()->base_url());
EXPECT_TRUE(content::WaitForRenderFrameReady(
GetActiveWebContents()->GetMainFrame()));
}
// Returns the background WebContents.
content::WebContents* LoadHTMLInBackgroundTab(const std::string& html) {
embedded_test_server()->RegisterRequestHandler(
base::Bind(&RespondWithHTML, html));
ui_test_utils::NavigateToURLWithDisposition(
browser(), embedded_test_server()->base_url(),
WindowOpenDisposition::NEW_BACKGROUND_TAB,
ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION);
int index = browser()->tab_strip_model()->GetIndexOfLastWebContentsOpenedBy(
GetActiveWebContents(), 0 /* start_index */);
content::WebContents* contents =
browser()->tab_strip_model()->GetWebContentsAt(index);
EXPECT_TRUE(content::WaitForRenderFrameReady(contents->GetMainFrame()));
return contents;
}
void ActivateTab(content::WebContents* contents) {
browser()->tab_strip_model()->ActivateTabAt(
browser()->tab_strip_model()->GetIndexOfWebContents(contents),
true /* user_gesture */);
}
content::WebContents* GetActiveWebContents() {
return browser()->tab_strip_model()->GetActiveWebContents();
}
// This sends a simulated click at |point| and waits for test plugin to send
// a status message indicating that it is essential. The test plugin sends a
// status message during:
// - Plugin creation, to handle a plugin freshly created from a poster.
// - Peripheral status change.
// - In response to the explicit 'getPowerSaverStatus' request, in case the
// test has missed the above two events.
void SimulateClickAndAwaitMarkedEssential(const std::string& element_id,
const gfx::Point& point) {
WaitForPlaceholderReady(GetActiveWebContents(), element_id);
content::SimulateMouseClickAt(GetActiveWebContents(), 0 /* modifiers */,
blink::WebMouseEvent::Button::Left, point);
VerifyPluginMarkedEssential(GetActiveWebContents(), element_id);
}
// |element_id| must be an element on the foreground tab.
void VerifyPluginIsPlaceholderOnly(const std::string& element_id) {
EXPECT_FALSE(PluginLoaded(GetActiveWebContents(), element_id));
WaitForPlaceholderReady(GetActiveWebContents(), element_id);
}
bool VerifySnapshot(const base::FilePath::StringType& expected_filename) {
if (!PixelTestsEnabled())
return true;
base::FilePath reference = ui_test_utils::GetTestFilePath(
base::FilePath(FILE_PATH_LITERAL("plugin_power_saver")),
base::FilePath(expected_filename));
GetActiveWebContents()->GetMainFrame()->InsertVisualStateCallback(
base::Bind(&VerifyVisualStateUpdated,
base::MessageLoop::QuitWhenIdleClosure()));
content::RunMessageLoop();
content::RenderWidgetHost* rwh =
GetActiveWebContents()->GetRenderViewHost()->GetWidget();
if (!rwh->CanCopyFromBackingStore()) {
ADD_FAILURE() << "Could not copy from backing store.";
return false;
}
bool snapshot_matches = false;
rwh->CopyFromBackingStore(
gfx::Rect(), gfx::Size(),
base::Bind(&CompareSnapshotToReference, reference, &snapshot_matches,
base::MessageLoop::QuitWhenIdleClosure()),
kN32_SkColorType);
content::RunMessageLoop();
return snapshot_matches;
}
// TODO(tommycli): Remove this once all flakiness resolved.
bool PixelTestsEnabled() {
#if defined(OS_WIN) || defined(ADDRESS_SANITIZER)
// Flaky on Windows and Asan bots. See crbug.com/549285.
return false;
#elif defined(OS_CHROMEOS)
// Because ChromeOS cannot use software rendering and the pixel tests
// continue to flake with hardware acceleration, disable these on ChromeOS.
return false;
#else
return true;
#endif
}
private:
base::test::ScopedFeatureList feature_list;
};
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, EssentialPlugins) {
LoadHTML(
"<object id='small_same_origin' data='fake.swf' "
" type='application/x-shockwave-flash' width='400' height='100'>"
"</object>"
"<object id='small_same_origin_poster' data='fake.swf' "
" type='application/x-shockwave-flash' width='400' height='100' "
" poster='click_me.png'>"
"</object>"
"<object id='large_cross_origin' data='http://b.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='500'>"
"</object>"
"<object id='medium_16_9_cross_origin' data='http://c.com/fake.swf' "
" type='application/x-shockwave-flash' width='480' height='270'>"
"</object>");
VerifyPluginMarkedEssential(GetActiveWebContents(), "small_same_origin");
VerifyPluginMarkedEssential(GetActiveWebContents(),
"small_same_origin_poster");
VerifyPluginMarkedEssential(GetActiveWebContents(), "large_cross_origin");
VerifyPluginMarkedEssential(GetActiveWebContents(),
"medium_16_9_cross_origin");
}
// Flaky on WebKit Mac dbg bots: crbug.com/599484.
#if defined(OS_MACOSX)
#define MAYBE_SmallCrossOrigin DISABLED_SmallCrossOrigin
#else
#define MAYBE_SmallCrossOrigin SmallCrossOrigin
#endif
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, MAYBE_SmallCrossOrigin) {
LoadHTML(
"<object id='plugin' data='http://otherorigin.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='100'>"
"</object>"
"<br>"
"<object id='plugin_poster' data='http://otherorigin.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='100' "
" poster='click_me.png'>"
"</object>");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin");
VerifyPluginIsPlaceholderOnly("plugin_poster");
EXPECT_TRUE(
VerifySnapshot(FILE_PATH_LITERAL("small_cross_origin_expected.png")));
SimulateClickAndAwaitMarkedEssential("plugin", gfx::Point(50, 50));
SimulateClickAndAwaitMarkedEssential("plugin_poster", gfx::Point(50, 150));
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, SmallerThanPlayIcon) {
LoadHTML(
"<object id='plugin_16' type='application/x-shockwave-flash' "
" width='16' height='16'></object>"
"<object id='plugin_32' type='application/x-shockwave-flash' "
" width='32' height='32'></object>"
"<object id='plugin_16_64' type='application/x-shockwave-flash' "
" width='16' height='64'></object>"
"<object id='plugin_64_16' type='application/x-shockwave-flash' "
" width='64' height='16'></object>");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin_16");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin_32");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin_16_64");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin_64_16");
EXPECT_TRUE(
VerifySnapshot(FILE_PATH_LITERAL("smaller_than_play_icon_expected.png")));
}
// Flaky on WebKit Mac dbg bots: crbug.com/599484.
#if defined(OS_MACOSX)
#define MAYBE_PosterTests DISABLED_PosterTests
#else
#define MAYBE_PosterTests PosterTests
#endif
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, MAYBE_PosterTests) {
// This test simultaneously verifies the varied supported poster syntaxes,
// as well as verifies that the poster is rendered correctly with various
// mismatched aspect ratios and sizes, following the same rules as VIDEO.
LoadHTML(
"<object id='plugin_src' type='application/x-shockwave-flash' "
" width='100' height='100' poster='click_me.png'></object>"
"<object id='plugin_srcset' type='application/x-shockwave-flash' "
" width='100' height='100' "
" poster='click_me.png 1x, click_me.png 2x'></object>"
"<br>"
"<object id='plugin_poster_param' type='application/x-shockwave-flash' "
" width='100' height='100'>"
" <param name='poster' value='click_me.png 1x, click_me.png 2x'>"
"</object>"
"<embed id='plugin_embed_src' type='application/x-shockwave-flash' "
" width='100' height='100' poster='click_me.png'></embed>"
"<embed id='plugin_embed_srcset' type='application/x-shockwave-flash' "
" width='100' height='100'"
" poster='click_me.png 1x, click_me.png 2x'></embed>"
"<br>"
"<object id='poster_missing' type='application/x-shockwave-flash' "
" width='100' height='100' poster='missing.png'></object>"
"<object id='poster_too_small' type='application/x-shockwave-flash' "
" width='100' height='50' poster='click_me.png'></object>"
"<object id='poster_too_big' type='application/x-shockwave-flash' "
" width='100' height='150' poster='click_me.png'></object>"
"<br>"
"<object id='poster_16' type='application/x-shockwave-flash' "
" width='16' height='16' poster='click_me.png'></object>"
"<object id='poster_32' type='application/x-shockwave-flash' "
" width='32' height='32' poster='click_me.png'></object>"
"<object id='poster_16_64' type='application/x-shockwave-flash' "
" width='16' height='64' poster='click_me.png'></object>"
"<object id='poster_64_16' type='application/x-shockwave-flash' "
" width='64' height='16' poster='click_me.png'></object>"
"<br>"
"<div id='container' "
" style='width: 400px; height: 100px; overflow: hidden;'>"
" <object id='poster_obscured' data='http://otherorigin.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='500' "
" poster='click_me.png'>"
" </object>"
"</div>");
VerifyPluginIsPlaceholderOnly("plugin_src");
VerifyPluginIsPlaceholderOnly("plugin_srcset");
VerifyPluginIsPlaceholderOnly("plugin_poster_param");
VerifyPluginIsPlaceholderOnly("plugin_embed_src");
VerifyPluginIsPlaceholderOnly("plugin_embed_srcset");
VerifyPluginIsPlaceholderOnly("poster_missing");
VerifyPluginIsPlaceholderOnly("poster_too_small");
VerifyPluginIsPlaceholderOnly("poster_too_big");
VerifyPluginIsPlaceholderOnly("poster_16");
VerifyPluginIsPlaceholderOnly("poster_32");
VerifyPluginIsPlaceholderOnly("poster_16_64");
VerifyPluginIsPlaceholderOnly("poster_64_16");
VerifyPluginIsPlaceholderOnly("poster_obscured");
EXPECT_TRUE(VerifySnapshot(FILE_PATH_LITERAL("poster_tests_expected.png")));
// Test that posters can be unthrottled via click.
SimulateClickAndAwaitMarkedEssential("plugin_src", gfx::Point(50, 50));
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, LargePostersNotThrottled) {
// This test verifies that small posters are throttled, large posters are not,
// and that large posters can whitelist origins for other plugins.
LoadHTML(
"<object id='poster_small' data='http://a.com/fake.swf' "
" type='application/x-shockwave-flash' width='50' height='50' "
" poster='click_me.png'></object>"
"<object id='poster_whitelisted_origin' data='http://b.com/fake.swf' "
" type='application/x-shockwave-flash' width='50' height='50' "
" poster='click_me.png'></object>"
"<object id='plugin_whitelisted_origin' data='http://b.com/fake.swf' "
" type='application/x-shockwave-flash' width='50' height='50'>"
"</object>"
"<br>"
"<object id='poster_large' data='http://b.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='300' "
" poster='click_me.png'></object>");
VerifyPluginIsPlaceholderOnly("poster_small");
VerifyPluginMarkedEssential(GetActiveWebContents(),
"poster_whitelisted_origin");
VerifyPluginMarkedEssential(GetActiveWebContents(),
"plugin_whitelisted_origin");
VerifyPluginMarkedEssential(GetActiveWebContents(), "poster_large");
}
// Flaky on ASAN bots: crbug.com/560765.
#if defined(ADDRESS_SANITIZER)
#define MAYBE_OriginWhitelisting DISABLED_OriginWhitelisting
#else
#define MAYBE_OriginWhitelisting OriginWhitelisting
#endif
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, MAYBE_OriginWhitelisting) {
LoadHTML(
"<object id='plugin_small' data='http://a.com/fake1.swf' "
" type='application/x-shockwave-flash' width='100' height='100'>"
"</object>"
"<object id='plugin_small_poster' data='http://a.com/fake1.swf' "
" type='application/x-shockwave-flash' width='100' height='100' "
" poster='click_me.png'></object>"
"<object id='plugin_large' data='http://a.com/fake2.swf' "
" type='application/x-shockwave-flash' width='400' height='500'>"
"</object>");
VerifyPluginMarkedEssential(GetActiveWebContents(), "plugin_small");
VerifyPluginMarkedEssential(GetActiveWebContents(), "plugin_small_poster");
VerifyPluginMarkedEssential(GetActiveWebContents(), "plugin_large");
}
// Flaky on almost all platforms: crbug.com/648827.
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest,
DISABLED_LargeCrossOriginObscured) {
LoadHTML(
"<div id='container' "
" style='width: 100px; height: 400px; overflow: hidden;'>"
" <object id='plugin' data='http://otherorigin.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='500' "
" style='float: right;'>"
" </object>"
"</div>");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin");
EXPECT_TRUE(VerifySnapshot(
FILE_PATH_LITERAL("large_cross_origin_obscured_expected.png")));
// Test that's unthrottled if it is unobscured.
std::string script =
"var container = window.document.getElementById('container');"
"container.setAttribute('style', 'width: 400px; height: 400px;');";
ASSERT_TRUE(content::ExecuteScript(GetActiveWebContents(), script));
VerifyPluginMarkedEssential(GetActiveWebContents(), "plugin");
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, ExpandingSmallPlugin) {
LoadHTML(
"<object id='plugin' data='http://otherorigin.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='80'>"
"</object>");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin");
std::string script = "window.document.getElementById('plugin').height = 400;";
ASSERT_TRUE(content::ExecuteScript(GetActiveWebContents(), script));
VerifyPluginMarkedEssential(GetActiveWebContents(), "plugin");
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, BackgroundTabPlugins) {
content::WebContents* background_contents = LoadHTMLInBackgroundTab(
"<object id='same_origin' data='fake.swf' "
" type='application/x-shockwave-flash'></object>"
"<object id='small_cross_origin' data='http://otherorigin.com/fake1.swf' "
" type='application/x-shockwave-flash' width='400' height='80'>"
"</object>");
EXPECT_FALSE(PluginLoaded(background_contents, "same_origin"));
EXPECT_FALSE(PluginLoaded(background_contents, "small_cross_origin"));
ActivateTab(background_contents);
VerifyPluginMarkedEssential(background_contents, "same_origin");
VerifyPluginIsThrottled(background_contents, "small_cross_origin");
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, ZoomIndependent) {
zoom::ZoomController::FromWebContents(GetActiveWebContents())
->SetZoomLevel(4.0);
LoadHTML(
"<object id='plugin' data='http://otherorigin.com/fake.swf' "
" type='application/x-shockwave-flash' width='400' height='200'>"
"</object>");
VerifyPluginIsThrottled(GetActiveWebContents(), "plugin");
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, BlockTinyPlugins) {
LoadHTML(
"<object id='tiny_same_origin' data='fake.swf' "
" type='application/x-shockwave-flash' width='3' height='3'>"
"</object>"
"<object id='tiny_cross_origin_1' data='http://a.com/fake.swf' "
" type='application/x-shockwave-flash' width='3' height='3'>"
"</object>"
"<object id='tiny_cross_origin_2' data='http://a.com/fake.swf' "
" type='application/x-shockwave-flash' width='1' height='1'>"
"</object>");
VerifyPluginMarkedEssential(GetActiveWebContents(), "tiny_same_origin");
VerifyPluginIsPlaceholderOnly("tiny_cross_origin_1");
VerifyPluginIsPlaceholderOnly("tiny_cross_origin_2");
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, BackgroundTabTinyPlugins) {
content::WebContents* background_contents = LoadHTMLInBackgroundTab(
"<object id='tiny' data='http://a.com/fake.swf' "
" type='application/x-shockwave-flash' width='3' height='3'>"
"</object>");
EXPECT_FALSE(PluginLoaded(background_contents, "tiny"));
ActivateTab(background_contents);
VerifyPluginIsPlaceholderOnly("tiny");
}
IN_PROC_BROWSER_TEST_F(PluginPowerSaverBrowserTest, ExpandingTinyPlugins) {
LoadHTML(
"<object id='expand_to_peripheral' data='http://a.com/fake.swf' "
" type='application/x-shockwave-flash' width='4' height='4'></object>"
"<object id='expand_to_essential' data='http://b.com/fake.swf' "
" type='application/x-shockwave-flash' width='4' height='4'>"
"</object>");
VerifyPluginIsPlaceholderOnly("expand_to_peripheral");
VerifyPluginIsPlaceholderOnly("expand_to_essential");
std::string script =
"window.document.getElementById('expand_to_peripheral').height = 200;"
"window.document.getElementById('expand_to_peripheral').width = 200;"
"window.document.getElementById('expand_to_essential').height = 400;"
"window.document.getElementById('expand_to_essential').width = 400;";
ASSERT_TRUE(content::ExecuteScript(GetActiveWebContents(), script));
VerifyPluginIsThrottled(GetActiveWebContents(), "expand_to_peripheral");
VerifyPluginMarkedEssential(GetActiveWebContents(), "expand_to_essential");
}
// Separate test case that allows tiny plugins. This requires a separate test
// case, because we need to initialize the renderer with a different feature
// setting.
class PluginPowerSaverAllowTinyBrowserTest
: public PluginPowerSaverBrowserTest {
public:
void SetUpInProcessBrowserTestFixture() override {
PluginPowerSaverBrowserTest::SetUpInProcessBrowserTestFixture();
feature_list.InitAndDisableFeature(features::kBlockSmallContent);
}
private:
base::test::ScopedFeatureList feature_list;
};
IN_PROC_BROWSER_TEST_F(PluginPowerSaverAllowTinyBrowserTest,
EssentialTinyPlugins) {
LoadHTML(
"<object id='tiny_cross_origin_1' data='http://a.com/fake.swf' "
" type='application/x-shockwave-flash' width='3' height='3'>"
"</object>"
"<object id='tiny_cross_origin_2' data='http://a.com/fake.swf' "
" type='application/x-shockwave-flash' width='1' height='1'>"
"</object>");
VerifyPluginMarkedEssential(GetActiveWebContents(), "tiny_cross_origin_1");
VerifyPluginMarkedEssential(GetActiveWebContents(), "tiny_cross_origin_2");
}