diff --git a/DEPS b/DEPS index 2920dc4..a9c65e1a 100644 --- a/DEPS +++ b/DEPS
@@ -311,11 +311,11 @@ # Three lines of non-changing comments so that # the commit queue can handle CLs rolling V8 # and whatever else without interference from each other. - 'v8_revision': 'f1bc03fd6b4c201abd9f0fd9d51fb989150f97b9', + 'v8_revision': 'c90f939a79eab36ff08b04ae6a619c718b363b22', # Three lines of non-changing comments so that # the commit queue can handle CLs rolling ANGLE # and whatever else without interference from each other. - 'angle_revision': '3ca4cde711b0ee03861d1294213c5b63dbee5ef9', + 'angle_revision': '19fb11b67c6f30f440112ca5d9725d1bed9b232e', # Three lines of non-changing comments so that # the commit queue can handle CLs rolling SwiftShader # and whatever else without interference from each other. @@ -334,7 +334,7 @@ # Three lines of non-changing comments so that # the commit queue can handle CLs rolling Fuchsia sdk # and whatever else without interference from each other. - 'fuchsia_version': 'version:10.20221013.0.1', + 'fuchsia_version': 'version:9.20221006.5.1', # Three lines of non-changing comments so that # the commit queue can handle CLs rolling google-toolbox-for-mac # and whatever else without interference from each other. @@ -378,7 +378,7 @@ # Three lines of non-changing comments so that # the commit queue can handle CLs rolling catapult # and whatever else without interference from each other. - 'catapult_revision': 'df8440aca35dcbdaf56ecc3a5ded229fedf80cc7', + 'catapult_revision': 'faaae893d5d2c4050d9ba741a2cbd70b4528c7fe', # Three lines of non-changing comments so that # the commit queue can handle CLs rolling libFuzzer # and whatever else without interference from each other. @@ -422,7 +422,7 @@ # Three lines of non-changing comments so that # the commit queue can handle CLs rolling feed # and whatever else without interference from each other. - 'dawn_revision': 'aecf1a2ab5106c4a08f8e2ae79717483f3abf8ee', + 'dawn_revision': '18bcdebc0256ad3d14bd1290163c98c84baddfa3', # Three lines of non-changing comments so that # the commit queue can handle CLs rolling feed # and whatever else without interference from each other. @@ -818,7 +818,7 @@ 'src/clank': { 'url': 'https://chrome-internal.googlesource.com/clank/internal/apps.git' + '@' + - 'f2f33b925ca49fa7e2c44d27e6a2f5b741855c0a', + 'aa0b389580b88beb9ef353b3a7bb11cfa3f8bac6', 'condition': 'checkout_android and checkout_src_internal and not checkout_clank_via_src_internal', }, @@ -923,7 +923,7 @@ 'packages': [ { 'package': 'chromium/rts/model/mac-amd64', - 'version': 'xH8MfShB-S7HYkM3gLOUa916ukoEtDJa-8X1bOwfevsC', + 'version': 'Yl7cUc03IbwQqGpCBGiiefZ83rN4O-7cUEWedTJ0qLcC', }, ], 'dep_type': 'cipd', @@ -934,7 +934,7 @@ 'packages': [ { 'package': 'chromium/rts/model/windows-amd64', - 'version': 'SWCvrm3LQO_Y0XbcVVs0q2CJOVKn0ImNLJ0WPQDKx5YC', + 'version': 'h3TMxlBpShy8lGRaIG18Il0sqnAYBWm1qQ3jArztOiQC', }, ], 'dep_type': 'cipd', @@ -1251,7 +1251,7 @@ Var('chromium_git') + '/devtools/devtools-frontend' + '@' + Var('devtools_frontend_revision'), 'src/third_party/devtools-frontend-internal': { - 'url': 'https://chrome-internal.googlesource.com/devtools/devtools-internal.git' + '@' + 'b2aab03500a9810f3d267f1bd72ae155937d45a0', + 'url': 'https://chrome-internal.googlesource.com/devtools/devtools-internal.git' + '@' + 'd26f0c7dfd01f6ddc0a5fab146c54fa7cc31e8c0', 'condition': 'checkout_src_internal', }, @@ -1406,7 +1406,7 @@ Var('chromium_git') + '/chromium/deps/hunspell_dictionaries.git' + '@' + '41cdffd71c9948f63c7ad36e1fb0ff519aa7a37e', 'src/third_party/icu': - Var('chromium_git') + '/chromium/deps/icu.git' + '@' + '20f8ac695af59b6c830def7d4e95bfeb13dd7be5', + Var('chromium_git') + '/chromium/deps/icu.git' + '@' + 'da07448619763d1cde255b361324242646f5b268', 'src/third_party/icu4j': { 'packages': [ @@ -1921,7 +1921,7 @@ Var('chromium_git') + '/v8/v8.git' + '@' + Var('v8_revision'), 'src-internal': { - 'url': 'https://chrome-internal.googlesource.com/chrome/src-internal.git@f4fd7a75dd91cc4c56f00d09ee098d86775c6e10', + 'url': 'https://chrome-internal.googlesource.com/chrome/src-internal.git@4bd291c858980c53e3127bcb8a7076502f87d6a1', 'condition': 'checkout_src_internal', },
diff --git a/ash/BUILD.gn b/ash/BUILD.gn index 0adc954..7a9efbd4 100644 --- a/ash/BUILD.gn +++ b/ash/BUILD.gn
@@ -2772,7 +2772,6 @@ "metrics/feature_discovery_duration_reporter_impl_unittest.cc", "metrics/histogram_macros_unittest.cc", "metrics/login_metrics_recorder_unittest.cc", - "metrics/login_unlock_throughput_recorder_unittest.cc", "metrics/pointer_metrics_recorder_unittest.cc", "metrics/stylus_metrics_recorder_unittest.cc", "metrics/task_switch_metrics_recorder_unittest.cc", @@ -3215,7 +3214,6 @@ "//chromeos/crosapi/cpp", "//chromeos/dbus/power", "//chromeos/dbus/power:power_manager_proto", - "//chromeos/login/login_state:login_state", "//chromeos/services/network_config/public/cpp", "//chromeos/services/network_config/public/mojom", "//chromeos/strings:strings_grit",
diff --git a/ash/app_list/BUILD.gn b/ash/app_list/BUILD.gn index 1719f47..6da7baa 100644 --- a/ash/app_list/BUILD.gn +++ b/ash/app_list/BUILD.gn
@@ -155,10 +155,6 @@ "views/search_result_page_dialog_controller.h", "views/search_result_page_view.cc", "views/search_result_page_view.h", - "views/search_result_tile_item_list_view.cc", - "views/search_result_tile_item_list_view.h", - "views/search_result_tile_item_view.cc", - "views/search_result_tile_item_view.h", "views/search_result_view.cc", "views/search_result_view.h", "views/suggested_content_info_view.cc", @@ -311,7 +307,6 @@ "views/search_box_view_unittest.cc", "views/search_result_list_view_unittest.cc", "views/search_result_page_view_unittest.cc", - "views/search_result_tile_item_list_view_unittest.cc", "views/search_result_view_unittest.cc", ]
diff --git a/ash/app_list/app_list_color_provider_impl.cc b/ash/app_list/app_list_color_provider_impl.cc index baf96e4c..049eae0 100644 --- a/ash/app_list/app_list_color_provider_impl.cc +++ b/ash/app_list/app_list_color_provider_impl.cc
@@ -245,13 +245,6 @@ kColorAshControlBackgroundColorInactive); } -ui::ColorId AppListColorProviderImpl::GetSeparatorColorId() const { - if (ShouldUseDarkLightColors()) { - return ui::kColorAshAppListSeparatorLight; - } - return ui::kColorAshAppListSeparator; // default_color -} - SkColor AppListColorProviderImpl::GetFocusRingColor( const views::Widget* app_list_widget) const { DCHECK(app_list_widget);
diff --git a/ash/app_list/app_list_color_provider_impl.h b/ash/app_list/app_list_color_provider_impl.h index 0a4a6384..639274d 100644 --- a/ash/app_list/app_list_color_provider_impl.h +++ b/ash/app_list/app_list_color_provider_impl.h
@@ -55,7 +55,6 @@ const views::Widget* app_list_widget) const override; SkColor GetGridBackgroundCardInactiveColor( const views::Widget* app_list_widget) const override; - ui::ColorId GetSeparatorColorId() const override; SkColor GetFocusRingColor( const views::Widget* app_list_widget) const override; SkColor GetInkDropBaseColor(
diff --git a/ash/app_list/app_list_metrics_unittest.cc b/ash/app_list/app_list_metrics_unittest.cc index 6dc6946d..7d978703 100644 --- a/ash/app_list/app_list_metrics_unittest.cc +++ b/ash/app_list/app_list_metrics_unittest.cc
@@ -26,7 +26,6 @@ #include "ash/app_list/views/recent_apps_view.h" #include "ash/app_list/views/search_result_container_view.h" #include "ash/app_list/views/search_result_page_view.h" -#include "ash/app_list/views/search_result_tile_item_list_view.h" #include "ash/constants/ash_features.h" #include "ash/public/cpp/app_list/app_list_types.h" #include "ash/public/cpp/shelf_item_delegate.h" @@ -116,45 +115,6 @@ LeftClickOn(view_model->view_at(kBrowserAppIndexOnShelf)); } - void PopulateAndLaunchSearchBoxTileItem() { - // Populate 4 tile items. - for (size_t i = 0; i < 4; i++) { - auto search_result = std::make_unique<SearchResult>(); - search_result->set_display_type(SearchResultDisplayType::kTile); - - // Give each item a name so that the accessibility paint checks pass. - // (Focusable items should have accessible names.) - search_result->SetAccessibleName( - base::UTF8ToUTF16(base::StringPrintf("item %zu", i))); - - search_model_->results()->Add(std::move(search_result)); - } - GetAppListTestHelper()->WaitUntilIdle(); - - // Mark the privacy notices as dismissed so that the tile items will be the - // first search container. - ContentsView* contents_view = Shell::Get() - ->app_list_controller() - ->fullscreen_presenter() - ->GetView() - ->app_list_main_view() - ->contents_view(); - Shell::Get()->app_list_controller()->MarkSuggestedContentInfoDismissed(); - contents_view->search_result_page_view() - ->GetPrivacyContainerViewForTest() - ->Update(); - - SearchResultContainerView* search_result_container_view = - contents_view->search_result_page_view() - ->GetSearchResultTileItemListViewForTest(); - - // Request focus on the first tile item view. - search_result_container_view->GetFirstResultView()->RequestFocus(); - - // Press return to simulate an app launch from the tile item. - PressAndReleaseKey(ui::KeyboardCode::VKEY_RETURN); - } - void PopulateAndLaunchAppInGrid(int num = 4) { // Populate apps in the root app grid. AppListModel* model = AppListModelProvider::Get()->model(); @@ -253,33 +213,6 @@ 1 /* Number of times launched from shelf */); } -// Test that the histogram records an app launch from the search box while the -// homercher search state is showing. -TEST_P(AppListMetricsTabletTest, HomecherSearchLaunchFromSearchBox) { - // ProductivityLauncher does not tile search results. - if (features::IsProductivityLauncherEnabled()) - return; - - base::HistogramTester histogram_tester; - - // Enable tablet mode. - GetAppListTestHelper()->WaitUntilIdle(); - Shell::Get()->tablet_mode_controller()->SetEnabledForTest(true); - - // Press a letter key, the AppListView should transition to kFullscreenSearch. - PressAndReleaseKey(ui::KeyboardCode::VKEY_H); - GetAppListTestHelper()->WaitUntilIdle(); - GetAppListTestHelper()->CheckState(AppListViewState::kFullscreenSearch); - - // Populate search box with tile items and launch a tile item. - PopulateAndLaunchSearchBoxTileItem(); - - histogram_tester.ExpectBucketCount( - "Apps.AppListAppLaunchedV2.HomecherSearch", - AppListLaunchedFrom::kLaunchedFromSearchBox, - 1 /* Number of times launched from search box */); -} - // Tests with feature ProductivityLauncher enabled. class AppListMetricsProductivityLauncherTest : public AppListMetricsTest { public:
diff --git a/ash/app_list/app_list_presenter_impl.cc b/ash/app_list/app_list_presenter_impl.cc index c801b68..1d68b14d 100644 --- a/ash/app_list/app_list_presenter_impl.cc +++ b/ash/app_list/app_list_presenter_impl.cc
@@ -29,6 +29,7 @@ #include "base/bind.h" #include "base/callback_helpers.h" #include "base/containers/contains.h" +#include "base/functional/callback_forward.h" #include "base/metrics/histogram_macros.h" #include "base/metrics/user_metrics.h" #include "chromeos/ash/services/assistant/public/cpp/assistant_enums.h" @@ -37,6 +38,8 @@ #include "ui/aura/window.h" #include "ui/compositor/animation_throughput_reporter.h" #include "ui/compositor/layer.h" +#include "ui/compositor/layer_animation_element.h" +#include "ui/compositor/layer_animation_observer.h" #include "ui/compositor/scoped_layer_animation_settings.h" #include "ui/display/screen.h" #include "ui/display/types/display_constants.h" @@ -53,6 +56,14 @@ using assistant::AssistantExitPoint; +// The target scale to which (or from which) the fullscreen launcher will +// animate between tablet <-> clamshell mode transition. +constexpr float kFullscreenLauncherFadeAnimationScale = 0.92f; + +// The fade in/out animation duration for tablet <-> clamshell mode transition. +constexpr base::TimeDelta kFullscreenLauncherTransitionDuration = + base::Milliseconds(350); + inline ui::Layer* GetLayer(views::Widget* widget) { return widget->GetNativeView()->layer(); } @@ -105,6 +116,42 @@ } } +// Invokes `complete_callback_` at the end of animation. +class FullscreenLauncherAnimationObserver + : public ui::ImplicitAnimationObserver { + public: + explicit FullscreenLauncherAnimationObserver( + base::OnceClosure complete_callback) + : complete_callback_(std::move(complete_callback)) {} + FullscreenLauncherAnimationObserver( + const FullscreenLauncherAnimationObserver& other) = delete; + FullscreenLauncherAnimationObserver& operator=( + const FullscreenLauncherAnimationObserver& other) = delete; + ~FullscreenLauncherAnimationObserver() override { + StopObservingImplicitAnimations(); + } + + // ui::ImplicitAnimationObserver: + void OnImplicitAnimationsCompleted() override { + std::move(complete_callback_).Run(); + delete this; + } + + private: + base::OnceClosure complete_callback_; +}; + +void UpdateTabletModeTransitionAnimationSettings( + FullscreenLauncherAnimationObserver* animation_observer, + ui::ScopedLayerAnimationSettings* settings) { + settings->SetTransitionDuration(kFullscreenLauncherTransitionDuration); + settings->SetTweenType(gfx::Tween::FAST_OUT_SLOW_IN); + settings->SetPreemptionStrategy( + ui::LayerAnimator::IMMEDIATELY_ANIMATE_TO_NEW_TARGET); + if (animation_observer) + settings->AddObserver(animation_observer); +} + // Implicit animation observer that runs a scoped closure runner, and deletes // itself when the observed implicit animations complete. class CallbackRunnerLayerAnimationObserver @@ -227,8 +274,35 @@ last_open_source_ = show_source; last_open_time_ = base::Time::Now(); + auto* layer = view_->GetWidget()->GetNativeWindow()->layer(); + + float initial_opacity; + if (app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) { + // `0.01f` prevents a DCHECK error (widgets cannot be shown when visible and + // fully transparent at the same time). + initial_opacity = layer->opacity() == 0.0f ? 0.01f : layer->opacity(); + } else { + initial_opacity = 1.0f; + } + layer->SetOpacity(initial_opacity); + view_->Show(preferred_state, IsSideShelf(shelf)); + if (app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) { + // Set initial transform only if layer is not currently animating one (for + // example during hide animation). + if (!layer->GetAnimator()->IsAnimatingProperty( + ui::LayerAnimationElement::TRANSFORM)) { + layer->SetTransform( + gfx::GetScaleTransform(gfx::Rect(layer->size()).CenterPoint(), + kFullscreenLauncherFadeAnimationScale)); + } + UpdateScaleAndOpacityForHomeLauncher( + 1.0f, 1.0f, absl::nullopt, + base::BindRepeating(&UpdateTabletModeTransitionAnimationSettings, + nullptr)); + } + SnapAppListBoundsToDisplayEdge(); event_filter_ = @@ -304,8 +378,20 @@ last_open_source_.reset(); last_open_time_.reset(); - if (!view_->GetWidget()->GetNativeWindow()->is_destroying()) + if (!view_->GetWidget()->GetNativeWindow()->is_destroying()) { + if (app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) { + base::OnceClosure animation_complete_callback = base::BindOnce( + &AppListPresenterImpl::OnTabletToClamshellTransitionAnimationDone, + weak_ptr_factory_.GetWeakPtr()); + auto* animation_observer = new FullscreenLauncherAnimationObserver( + std::move(animation_complete_callback)); + UpdateScaleAndOpacityForHomeLauncher( + kFullscreenLauncherFadeAnimationScale, 0.0f, absl::nullopt, + base::BindRepeating(&UpdateTabletModeTransitionAnimationSettings, + animation_observer)); + } view_->SetState(AppListViewState::kClosed); + } view_->SetDragAndDropHostOfCurrentAppList(nullptr); @@ -400,17 +486,12 @@ float opacity, absl::optional<TabletModeAnimationTransition> transition, UpdateHomeLauncherAnimationSettingsCallback callback) { - if (!view_) + // Exiting from overview in clamshell mode should not affect the hidden + // fullscreen launcher. + if (!view_ || view_->app_list_state() == AppListViewState::kClosed) return; ui::Layer* layer = view_->GetWidget()->GetNativeWindow()->layer(); - if (!Shell::Get()->IsInTabletMode()) { - // In clamshell mode, set the opacity of the AppList immediately to - // instantly hide it. Opacity of the AppList is reset when it is shown - // again. - layer->SetOpacity(opacity); - return; - } if (layer->GetAnimator()->is_animating()) { layer->GetAnimator()->StopAnimating(); @@ -695,4 +776,13 @@ SetViewIgnoredForAccessibility(view_->search_box_view(), false); } +void AppListPresenterImpl::OnTabletToClamshellTransitionAnimationDone() { + if (!view_) + return; + + auto* window = view_->GetWidget()->GetNativeWindow(); + if (window->layer()->GetTargetOpacity() == 0.0f) + window->Hide(); +} + } // namespace ash
diff --git a/ash/app_list/app_list_presenter_impl.h b/ash/app_list/app_list_presenter_impl.h index 2955f37c..ed06dbdb 100644 --- a/ash/app_list/app_list_presenter_impl.h +++ b/ash/app_list/app_list_presenter_impl.h
@@ -205,6 +205,10 @@ // Called when the reorder animation completes. void OnAppListReorderAnimationDone(); + // Called when the tablet to clamshell transition animation completes. + // Hides the `AppListView`'s window if target opacity is zero. + void OnTabletToClamshellTransitionAnimationDone(); + // Owns |this|. AppListControllerImpl* const controller_;
diff --git a/ash/app_list/app_list_presenter_unittest.cc b/ash/app_list/app_list_presenter_unittest.cc index dc2ec57..5de9776e 100644 --- a/ash/app_list/app_list_presenter_unittest.cc +++ b/ash/app_list/app_list_presenter_unittest.cc
@@ -37,13 +37,13 @@ #include "ash/app_list/views/search_result_list_view.h" #include "ash/app_list/views/search_result_page_anchored_dialog.h" #include "ash/app_list/views/search_result_page_view.h" -#include "ash/app_list/views/search_result_tile_item_list_view.h" #include "ash/assistant/ui/assistant_ui_constants.h" #include "ash/constants/ash_features.h" #include "ash/keyboard/keyboard_controller_impl.h" #include "ash/keyboard/ui/keyboard_ui_controller.h" #include "ash/keyboard/ui/test/keyboard_test_util.h" #include "ash/public/cpp/app_list/app_list_color_provider.h" +#include "ash/public/cpp/app_list/app_list_features.h" #include "ash/public/cpp/app_list/app_list_types.h" #include "ash/public/cpp/keyboard/keyboard_switches.h" #include "ash/public/cpp/shelf_config.h" @@ -89,6 +89,9 @@ #include "ui/display/test/display_manager_test_api.h" #include "ui/events/event_constants.h" #include "ui/events/test/event_generator.h" +#include "ui/gfx/geometry/rect.h" +#include "ui/gfx/geometry/transform.h" +#include "ui/gfx/geometry/transform_util.h" #include "ui/touch_selection/touch_selection_menu_runner.h" #include "ui/views/controls/button/label_button.h" #include "ui/views/controls/textfield/textfield.h" @@ -3353,12 +3356,6 @@ EXPECT_TRUE(search_box_view->is_search_box_active()); GetAppListTestHelper()->CheckState(AppListViewState::kFullscreenSearch); - SearchResultContainerView* const tile_item_container = - search_result_page()->GetSearchResultTileItemListViewForTest(); - ASSERT_EQ(1, tile_item_container->num_results()); - EXPECT_EQ("test_tile", - tile_item_container->GetResultViewAt(0)->result()->id()); - SearchResultContainerView* item_list_container = search_result_page()->GetSearchResultListViewForTest(); ASSERT_EQ(1, item_list_container->num_results()); @@ -3400,10 +3397,6 @@ EXPECT_TRUE(search_box_view->is_search_box_active()); GetAppListTestHelper()->CheckState(AppListViewState::kFullscreenSearch); - ASSERT_EQ(1, tile_item_container->num_results()); - EXPECT_EQ("test_tile_override", - tile_item_container->GetResultViewAt(0)->result()->id()); - ASSERT_EQ(1, item_list_container->num_results()); EXPECT_EQ("test_override", item_list_container->GetResultViewAt(0)->result()->id()); @@ -4413,4 +4406,88 @@ EXPECT_LT(textfield->GetCursorPosition(), initial_cursor_position); } +// Temporary class to test `kAnimateScaleOnTabletModeTransition` related +// animations. +class AppListPresenterWithScaleAnimationOnTabletModeTransitionTest + : public AppListPresenterTest { + public: + AppListPresenterWithScaleAnimationOnTabletModeTransitionTest() { + scoped_feature_list_.InitAndEnableFeature( + app_list_features::kAnimateScaleOnTabletModeTransition); + } + + protected: + void EnsureAppListViewIsCached() { + ASSERT_FALSE(GetAppListTestHelper()->GetAppListView()); + + EnableTabletMode(true); + GetAppListTestHelper()->ShowAndRunLoop(GetPrimaryDisplayId()); + GetAppListTestHelper()->CheckState(AppListViewState::kFullscreenAllApps); + + EnableTabletMode(false); + GetAppListTestHelper()->WaitUntilIdle(); + GetAppListTestHelper()->CheckState(AppListViewState::kClosed); + + // Entering and exiting from tablet mode should keep `AppListView` cached. + ASSERT_TRUE(GetAppListTestHelper()->GetAppListView()); + } + + base::test::ScopedFeatureList scoped_feature_list_; +}; + +TEST_F(AppListPresenterWithScaleAnimationOnTabletModeTransitionTest, + UpdatesScaleAndOpacity) { + // Enter tablet mode. + EnableTabletMode(true); + GetAppListTestHelper()->ShowAndRunLoop(GetPrimaryDisplayId()); + GetAppListTestHelper()->CheckState(AppListViewState::kFullscreenAllApps); + + auto* const window = + GetAppListTestHelper()->GetAppListView()->GetWidget()->GetNativeWindow(); + auto* const layer = window->layer(); + const auto center_point = gfx::Rect(layer->size()).CenterPoint(); + const auto no_transform = gfx::GetScaleTransform(center_point, 1.0f); + const auto scaled_down_transform = + gfx::GetScaleTransform(center_point, 0.92f); + + // The layer is fully visible and without applied transform. + EXPECT_EQ(layer->opacity(), 1.0f); + EXPECT_EQ(layer->transform(), no_transform); + EXPECT_TRUE(window->IsVisible()); + + // Exit tablet mode. + EnableTabletMode(false); + GetAppListTestHelper()->WaitUntilIdle(); + GetAppListTestHelper()->CheckState(AppListViewState::kClosed); + + // The layer is fully transparent, scaled down and the window is hidden. + EXPECT_EQ(layer->opacity(), 0.00f); + EXPECT_EQ(layer->transform(), scaled_down_transform); + EXPECT_FALSE(window->IsVisible()); +} + +TEST_F(AppListPresenterWithScaleAnimationOnTabletModeTransitionTest, + ExitingFromOverviewInClamshellModeShouldNotAffectFullscreenLauncher) { + EnsureAppListViewIsCached(); + + auto* const layer = GetAppListTestHelper() + ->GetAppListView() + ->GetWidget() + ->GetNativeWindow() + ->layer(); + const auto expected_opacity = layer->opacity(); + const auto expected_transform = layer->transform(); + + OverviewController* overview_controller = Shell::Get()->overview_controller(); + + EnterOverview(); + EXPECT_TRUE(overview_controller->InOverviewSession()); + ExitOverview(); + EXPECT_FALSE(overview_controller->InOverviewSession()); + + // Layer's opacity and transform should stay the same. + EXPECT_EQ(layer->opacity(), expected_opacity); + EXPECT_EQ(layer->transform(), expected_transform); +} + } // namespace ash
diff --git a/ash/app_list/views/app_list_view.cc b/ash/app_list/views/app_list_view.cc index 05da13f..fdbfb7e1 100644 --- a/ash/app_list/views/app_list_view.cc +++ b/ash/app_list/views/app_list_view.cc
@@ -21,6 +21,7 @@ #include "ash/app_list/views/paged_apps_grid_view.h" #include "ash/app_list/views/search_box_view.h" #include "ash/keyboard/ui/keyboard_ui_controller.h" +#include "ash/public/cpp/app_list/app_list_features.h" #include "ash/public/cpp/app_list/app_list_types.h" #include "ash/public/cpp/metrics_util.h" #include "ash/shell.h" @@ -499,9 +500,6 @@ void AppListView::Show(AppListViewState preferred_state, bool is_side_shelf) { if (!time_shown_.has_value()) time_shown_ = base::Time::Now(); - // The opacity of the AppListView may have been manipulated by overview mode, - // so reset it before it is shown. - GetWidget()->GetLayer()->SetOpacity(1.0f); is_side_shelf_ = is_side_shelf; AddAccelerator(ui::Accelerator(ui::VKEY_ESCAPE, ui::EF_NONE)); @@ -730,10 +728,20 @@ AppListState::kStateApps, !is_side_shelf_); } - if (target_state == AppListViewState::kClosed && is_side_shelf_) { - // Reset the search box to be shown again. This is done after the animation - // is complete normally, but there is no animation when |is_side_shelf_|. - search_box_view_->ClearSearchAndDeactivateSearchBox(); + if (target_state == AppListViewState::kClosed) { + if (app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) { + if (app_list_state_ == AppListViewState::kFullscreenSearch) + search_box_view_->ClearSearchAndDeactivateSearchBox(); + + auto* contents_view = app_list_main_view_->contents_view(); + if (contents_view->IsShowingEmbeddedAssistantUI()) + contents_view->ShowEmbeddedAssistantUI(false); + } else if (is_side_shelf_) { + // Reset the search box to be shown again. This is done after the + // animation is complete normally, but there is no animation when + // |is_side_shelf_|. + search_box_view_->ClearSearchAndDeactivateSearchBox(); + } } } @@ -1086,7 +1094,9 @@ base::TimeDelta animation_duration = GetStateTransitionAnimationDuration(target_state); - ApplyBoundsAnimation(target_state, animation_duration); + if (!app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) + ApplyBoundsAnimation(target_state, animation_duration); + app_list_main_view_->contents_view()->OnAppListViewTargetStateChanged( target_state); app_list_main_view_->contents_view()->AnimateToViewState(target_state, @@ -1413,6 +1423,8 @@ case AppListViewState::kFullscreenSearch: return fullscreen_height; case AppListViewState::kClosed: + if (app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) + return fullscreen_height; // Align the widget y with shelf y to avoid flicker in show animation. In // side shelf mode, the widget y is the top of work area because the // widget does not animate.
diff --git a/ash/app_list/views/app_list_view_unittest.cc b/ash/app_list/views/app_list_view_unittest.cc index f0aa533..b76d98d 100644 --- a/ash/app_list/views/app_list_view_unittest.cc +++ b/ash/app_list/views/app_list_view_unittest.cc
@@ -33,8 +33,6 @@ #include "ash/app_list/views/search_result_container_view.h" #include "ash/app_list/views/search_result_list_view.h" #include "ash/app_list/views/search_result_page_view.h" -#include "ash/app_list/views/search_result_tile_item_list_view.h" -#include "ash/app_list/views/search_result_tile_item_view.h" #include "ash/app_list/views/search_result_view.h" #include "ash/constants/ash_features.h" #include "ash/keyboard/ui/keyboard_ui_controller.h" @@ -633,12 +631,6 @@ view_->Show(AppListViewState::kFullscreenAllApps, /*is_side_shelf=*/false); } - SearchResultTileItemListView* GetSearchResultTileItemListView() { - return contents_view() - ->search_result_page_view() - ->GetSearchResultTileItemListViewForTest(); - } - SearchResultListView* GetSearchResultListView() { return contents_view() ->search_result_page_view() @@ -658,28 +650,18 @@ } // Add search results for test on focus movement. - void SetUpSearchResults(int tile_results_num, int list_results_num) { - std::vector<std::pair<SearchResult::DisplayType, int>> result_types; - result_types.emplace_back(SearchResultDisplayType::kTile, tile_results_num); - result_types.emplace_back(SearchResultDisplayType::kList, list_results_num); - + void SetUpSearchResults(int list_results_num) { SearchModel::SearchResults* results = GetSearchModel()->results(); results->DeleteAll(); - double display_score = result_types.size(); - for (const auto& data : result_types) { - // Set the display score of the results in each group in decreasing order - // (so the earlier groups have higher display score, and therefore appear - // first). - display_score -= 0.5; - for (int i = 0; i < data.second; ++i) { - std::unique_ptr<TestSearchResult> result = - std::make_unique<TestSearchResult>(); - result->set_display_type(data.first); - result->set_display_score(display_score); - result->SetTitle(u"Test"); - result->set_best_match(true); - results->Add(std::move(result)); - } + + for (int i = 0; i < list_results_num; ++i) { + std::unique_ptr<TestSearchResult> result = + std::make_unique<TestSearchResult>(); + result->set_display_type(SearchResultDisplayType::kList); + result->set_display_score(1); + result->SetTitle(u"Test"); + result->set_best_match(true); + results->Add(std::move(result)); } // Adding results will schedule Update(). @@ -1104,9 +1086,8 @@ search_box_view()->search_box()->InsertText( u"test", ui::TextInputClient::InsertTextCursorBehavior::kMoveCursorAfterText); - constexpr int kTileResults = 3; constexpr int kListResults = 2; - SetUpSearchResults(kTileResults, kListResults); + SetUpSearchResults(kListResults); // Move focus to the first search result. SimulateKeyPress(ui::VKEY_TAB, false); @@ -1153,7 +1134,7 @@ u"test", ui::TextInputClient::InsertTextCursorBehavior::kMoveCursorAfterText); const int kListResults = 2; - SetUpSearchResults(0, kListResults); + SetUpSearchResults(kListResults); SearchResultListView* list_view = GetSearchResultListView(); EXPECT_EQ(search_box_view()->search_box(), focused_view()); @@ -1161,28 +1142,15 @@ contents_view()->search_result_page_view()->first_result_view()); EXPECT_TRUE(list_view->GetResultViewAt(0)->selected()); - // Populate both fake list results and tile results. - const int kTileResults = 3; - SetUpSearchResults(kTileResults, kListResults); - const std::vector<SearchResultTileItemView*>& tile_views = - GetSearchResultTileItemListView()->tile_views_for_test(); - EXPECT_EQ(search_box_view()->search_box(), focused_view()); - EXPECT_EQ(tile_views[0], - contents_view()->search_result_page_view()->first_result_view()); - EXPECT_TRUE(tile_views[0]->selected()); - ResultSelectionController* selection_controller = contents_view()->search_result_page_view()->result_selection_controller(); - // Ensures the |ResultSelectionController| selects the correct result - EXPECT_EQ(selection_controller->selected_result(), tile_views[0]); - - // Ensure current highlighted result loses highlight on transition - SimulateKeyPress(ui::VKEY_TAB, false); - EXPECT_FALSE(tile_views[0]->selected()); + // Ensures the |ResultSelectionController| selects the correct result. + EXPECT_EQ(selection_controller->selected_result(), + list_view->GetResultViewAt(0)); // Clear up all search results. - SetUpSearchResults(0, 0); + SetUpSearchResults(0); EXPECT_EQ(search_box_view()->search_box(), focused_view()); EXPECT_EQ(nullptr, contents_view()->search_result_page_view()->first_result_view()); @@ -1206,23 +1174,16 @@ u"test", ui::TextInputClient::InsertTextCursorBehavior::kMoveCursorAfterText); const int kListResults = 2; - SetUpSearchResults(0, kListResults); + SetUpSearchResults(kListResults); SimulateKeyPress(ui::VKEY_RETURN, false); EXPECT_EQ(1, GetOpenFirstSearchResultCount()); EXPECT_EQ(1, GetTotalOpenSearchResultCount()); - // Populate both fake list results and tile results. Then hit Enter key. - const int kTileResults = 3; - SetUpSearchResults(kTileResults, kListResults); - SimulateKeyPress(ui::VKEY_RETURN, false); - EXPECT_EQ(2, GetOpenFirstSearchResultCount()); - EXPECT_EQ(2, GetTotalOpenSearchResultCount()); - // Clear up all search results. Then hit Enter key. - SetUpSearchResults(0, 0); + SetUpSearchResults(0); SimulateKeyPress(ui::VKEY_RETURN, false); - EXPECT_EQ(2, GetOpenFirstSearchResultCount()); - EXPECT_EQ(2, GetTotalOpenSearchResultCount()); + EXPECT_EQ(1, GetOpenFirstSearchResultCount()); + EXPECT_EQ(1, GetTotalOpenSearchResultCount()); } // Tests that search box becomes focused when it is activated.
diff --git a/ash/app_list/views/contents_view.cc b/ash/app_list/views/contents_view.cc index aadacba..c79e83e 100644 --- a/ash/app_list/views/contents_view.cc +++ b/ash/app_list/views/contents_view.cc
@@ -18,7 +18,6 @@ #include "ash/app_list/views/search_box_view.h" #include "ash/app_list/views/search_result_list_view.h" #include "ash/app_list/views/search_result_page_view.h" -#include "ash/app_list/views/search_result_tile_item_list_view.h" #include "ash/constants/ash_features.h" #include "ash/keyboard/ui/keyboard_ui_controller.h" #include "ash/public/cpp/app_list/app_list_config.h" @@ -611,8 +610,13 @@ ConvertRectToWidgetWithoutTransform(search_box_bounds)); search_box->GetWidget()->SetBounds(search_rect); - const float search_box_opacity = - target_view_state() != AppListViewState::kClosed ? 1.0f : 0.0f; + float search_box_opacity; + if (app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) { + search_box_opacity = 1.0f; + } else { + search_box_opacity = + target_view_state() != AppListViewState::kClosed ? 1.0f : 0.0f; + } search_box->layer()->SetOpacity(search_box_opacity); for (AppListPage* page : app_list_pages_) { @@ -631,6 +635,12 @@ ? pagination_model_.transition().target_page : pagination_model_.selected_page()); + if (app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) { + last_target_view_state_ = target_view_state; + target_page_for_last_view_state_update_ = target_page; + return; + } + // Animates layer's opacity. // |duration| - The default transition duration. The actual transition gets // halved when animating to hidden state. @@ -786,7 +796,9 @@ AppListViewState view_state) const { switch (view_state) { case AppListViewState::kClosed: - return 0; + if (!app_list_features::IsAnimateScaleOnTabletModeTransitionEnabled()) + return 0; + [[fallthrough]]; case AppListViewState::kFullscreenAllApps: case AppListViewState::kFullscreenSearch: return apps_container_view_
diff --git a/ash/app_list/views/search_box_view_unittest.cc b/ash/app_list/views/search_box_view_unittest.cc index 6dd1064..f3b34834 100644 --- a/ash/app_list/views/search_box_view_unittest.cc +++ b/ash/app_list/views/search_box_view_unittest.cc
@@ -23,7 +23,6 @@ #include "ash/app_list/views/search_box_view_delegate.h" #include "ash/app_list/views/search_result_list_view.h" #include "ash/app_list/views/search_result_page_view.h" -#include "ash/app_list/views/search_result_tile_item_list_view.h" #include "ash/constants/ash_features.h" #include "ash/public/cpp/app_list/app_list_color_provider.h" #include "ash/public/cpp/app_list/app_list_features.h" @@ -604,19 +603,6 @@ selection = GetResultSelectionController()->selected_result(); EXPECT_EQ(u"testing", selection->result()->title()); - // Tile containers are deprecated for productivity launcher. - if (!IsProductivityLauncherEnabled()) { - // Add a new result in a tile container - verify the selected result remains - // the same. - CreateSearchResultAt(0, ash::SearchResultDisplayType::kTile, 0.9, - u"test tile", std::u16string()); - // Finish results update. - base::RunLoop().RunUntilIdle(); - - selection = GetResultSelectionController()->selected_result(); - EXPECT_EQ(u"testing", selection->result()->title()); - } - // Add a new result at the end, and verify the selection stays the same. CreateSearchResult(ash::SearchResultDisplayType::kList, 0.2, u"testing almost", std::u16string()); @@ -627,35 +613,28 @@ // Remove the result before the selected one, and verify the selection remains // the same. - results()->RemoveAt(IsProductivityLauncherEnabled() ? 0 : 1); + results()->RemoveAt(0); base::RunLoop().RunUntilIdle(); selection = GetResultSelectionController()->selected_result(); EXPECT_EQ(u"testing", selection->result()->title()); // Result should be reset if the selected result is removed. - results()->RemoveAt(IsProductivityLauncherEnabled() ? 0 : 1); + results()->RemoveAt(0); base::RunLoop().RunUntilIdle(); // Tile results are not created when testing productivity launcher. selection = GetResultSelectionController()->selected_result(); - if (IsProductivityLauncherEnabled()) { + EXPECT_EQ(u"testing almost", selection->result()->title()); // New result can override the default selection. CreateSearchResultAt(0, ash::SearchResultDisplayType::kList, 1.0, u"test", std::u16string()); - } else { - EXPECT_EQ(u"test tile", selection->result()->title()); + base::RunLoop().RunUntilIdle(); - // New result can override the default selection. - CreateSearchResultAt(0, ash::SearchResultDisplayType::kTile, 1.0, u"test", - std::u16string()); - } - base::RunLoop().RunUntilIdle(); - - selection = GetResultSelectionController()->selected_result(); - EXPECT_EQ(u"test", selection->result()->title()); + selection = GetResultSelectionController()->selected_result(); + EXPECT_EQ(u"test", selection->result()->title()); } // Tests that the default selection is reset after resetting and reactivating @@ -942,93 +921,39 @@ // titles. TEST_P(SearchBoxViewAutocompleteTest, SearchBoxAutocompletesTopListResultTitle) { - // Search result tile tests are not relevant for productivity launcher. - if (IsProductivityLauncherEnabled()) - return; - SimulateQuery(u"he"); - // Add two SearchResults, one tile and one list result. Initialize their title - // field to a non-empty string. - CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0, u"hello list", + // Add two SearchResults. The higher ranked result should be selected by + // default and it's title should be autocompleted into the search box. + CreateSearchResult(ash::SearchResultDisplayType::kList, 2.0, u"hello list", std::u16string()); - CreateSearchResult(ash::SearchResultDisplayType::kTile, 0.5, u"hello tile", + CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0, u"hello list2", std::u16string()); base::RunLoop().RunUntilIdle(); ProcessAutocomplete(); - EXPECT_EQ(view()->search_box()->GetText(), u"hello tile"); - EXPECT_EQ(view()->search_box()->GetSelectedText(), u"llo tile"); -} - -// Tests that autocomplete suggestions are consistent with top SearchResult tile -// titles. -TEST_P(SearchBoxViewAutocompleteTest, - SearchBoxAutocompletesTopTileResultTitle) { - // Search result tile tests are not relevant for productivity launcher. - if (IsProductivityLauncherEnabled()) - return; - - SimulateQuery(u"he"); - - // Add two SearchResults, one tile and one list result. Initialize their title - // field to a non-empty string. - CreateSearchResult(ash::SearchResultDisplayType::kTile, 1.0, u"hello tile", - std::u16string()); - CreateSearchResult(ash::SearchResultDisplayType::kList, 0.5, u"hello list", - std::u16string()); + EXPECT_EQ(view()->search_box()->GetText(), u"hello list"); + EXPECT_EQ(view()->search_box()->GetSelectedText(), u"llo list"); base::RunLoop().RunUntilIdle(); - - ProcessAutocomplete(); - EXPECT_EQ(view()->search_box()->GetText(), u"hello tile"); - EXPECT_EQ(view()->search_box()->GetSelectedText(), u"llo tile"); } // Tests that autocomplete suggestions are consistent with top SearchResult list // details. TEST_P(SearchBoxViewAutocompleteTest, SearchBoxAutocompletesTopListResultDetails) { - // Search result tile tests are not relevant for productivity launcher. - if (IsProductivityLauncherEnabled()) - return; - SimulateQuery(u"he"); - // Add two SearchResults, one tile and one list result. The tile should - // display first, despite having a lower score. Initialize their details field - // to a non-empty string. + // Add two SearchResults. The higher ranked result should be selected by + // default and it's title should be autocompleted into the search box. + CreateSearchResult(ash::SearchResultDisplayType::kList, 2.0, std::u16string(), + u"hello list"); CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0, std::u16string(), - u"hello list"); - CreateSearchResult(ash::SearchResultDisplayType::kTile, 0.5, std::u16string(), - u"hello tile"); + u"hello list2"); base::RunLoop().RunUntilIdle(); ProcessAutocomplete(); - EXPECT_EQ(view()->search_box()->GetText(), u"hello tile"); - EXPECT_EQ(view()->search_box()->GetSelectedText(), u"llo tile"); -} - -// Tests that autocomplete suggestions are consistent with top SearchResult tile -// details. -TEST_P(SearchBoxViewAutocompleteTest, - SearchBoxAutocompletesTopTileResultDetails) { - // Search result tile tests are not relevant for productivity launcher. - if (IsProductivityLauncherEnabled()) - return; - - SimulateQuery(u"he"); - - // Add two SearchResults, one tile and one list result. Initialize their - // details field to a non-empty string. - CreateSearchResult(ash::SearchResultDisplayType::kTile, 1.0, std::u16string(), - u"hello tile"); - CreateSearchResult(ash::SearchResultDisplayType::kList, 0.5, std::u16string(), - u"hello list"); - base::RunLoop().RunUntilIdle(); - - ProcessAutocomplete(); - EXPECT_EQ(view()->search_box()->GetText(), u"hello tile"); - EXPECT_EQ(view()->search_box()->GetSelectedText(), u"llo tile"); + EXPECT_EQ(view()->search_box()->GetText(), u"hello list"); + EXPECT_EQ(view()->search_box()->GetSelectedText(), u"llo list"); } // Tests that SearchBoxView's textfield text does not autocomplete if the top
diff --git a/ash/app_list/views/search_result_page_view.cc b/ash/app_list/views/search_result_page_view.cc index 51838ed5..92c29f2 100644 --- a/ash/app_list/views/search_result_page_view.cc +++ b/ash/app_list/views/search_result_page_view.cc
@@ -18,7 +18,6 @@ #include "ash/app_list/views/search_result_base_view.h" #include "ash/app_list/views/search_result_list_view.h" #include "ash/app_list/views/search_result_page_anchored_dialog.h" -#include "ash/app_list/views/search_result_tile_item_list_view.h" #include "ash/constants/ash_features.h" #include "ash/public/cpp/app_list/app_list_color_provider.h" #include "ash/public/cpp/app_list/app_list_config.h" @@ -61,19 +60,9 @@ constexpr int kMinHeight = 440; constexpr int kWidth = 640; -// The horizontal padding of the separator. -constexpr int kSeparatorPadding = 12; -constexpr int kSeparatorThickness = 1; - // The height of the active search box in this page. constexpr int kActiveSearchBoxHeight = 56; -// The spacing between search box bottom and separator line. -// Add 1 pixel spacing so that the search bbox bottom will not paint over -// the separator line drawn by SearchResultPageBackground in some scale factors -// due to the round up. -constexpr int kSearchBoxBottomSpacing = 1; - // Minimum spacing between shelf and bottom of search box. constexpr int kSearchResultPageMinimumBottomMargin = 24; @@ -133,64 +122,8 @@ } }; -class SearchResultPageBackground : public views::Background { - public: - SearchResultPageBackground() = default; - SearchResultPageBackground(const SearchResultPageBackground&) = delete; - SearchResultPageBackground& operator=(const SearchResultPageBackground&) = - delete; - ~SearchResultPageBackground() override = default; - - private: - // views::Background overrides: - void Paint(gfx::Canvas* canvas, views::View* view) const override { - canvas->DrawColor(get_color()); - gfx::Rect bounds = view->GetContentsBounds(); - if (bounds.height() <= kActiveSearchBoxHeight) - return; - // Draw a separator between SearchBoxView and SearchResultPageView. - bounds.set_y(kActiveSearchBoxHeight + kSearchBoxBottomSpacing); - bounds.set_height(kSeparatorThickness); - canvas->FillRect(bounds, - view->GetColorProvider()->GetColor( - AppListColorProvider::Get()->GetSeparatorColorId())); - } -}; - } // namespace -class SearchResultPageView::HorizontalSeparator : public views::View { - public: - explicit HorizontalSeparator(int preferred_width) - : preferred_width_(preferred_width) { - SetBorder(views::CreateEmptyBorder( - gfx::Insets::TLBR(0, kSeparatorPadding, 0, kSeparatorPadding))); - } - - HorizontalSeparator(const HorizontalSeparator&) = delete; - HorizontalSeparator& operator=(const HorizontalSeparator&) = delete; - - ~HorizontalSeparator() override = default; - - // views::View overrides: - const char* GetClassName() const override { return "HorizontalSeparator"; } - - gfx::Size CalculatePreferredSize() const override { - return gfx::Size(preferred_width_, kSeparatorThickness); - } - - void OnPaint(gfx::Canvas* canvas) override { - gfx::Rect rect = GetContentsBounds(); - canvas->FillRect(rect, - GetColorProvider()->GetColor( - AppListColorProvider::Get()->GetSeparatorColorId())); - View::OnPaint(canvas); - } - - private: - const int preferred_width_; -}; - SearchResultPageView::SearchResultPageView() : contents_view_(new views::View) { SetPaintToLayer(); layer()->SetFillsBoundsOpaquely(false); @@ -204,7 +137,7 @@ // Hides this view behind the search box by using the same color and // background border corner radius. All child views' background should be // set transparent so that the rounded corner is not overwritten. - SetBackground(std::make_unique<SearchResultPageBackground>()); + SetBackground(views::CreateSolidBackground(SK_ColorTRANSPARENT)); if (features::IsProductivityLauncherEnabled()) { layer()->SetBackgroundBlur(ColorProvider::kBackgroundBlurSigma); layer()->SetBackdropFilterQuality(ColorProvider::kBackgroundBlurQuality); @@ -216,16 +149,14 @@ // App list bubble search page has its own scroller and result selection // controller so we do not need to construct new ones here. if (features::IsProductivityLauncherEnabled()) { - contents_view_->SetBorder(views::CreateEmptyBorder(gfx::Insets::TLBR( - kActiveSearchBoxHeight + kSearchBoxBottomSpacing + kSeparatorThickness, - 0, 0, 0))); + contents_view_->SetBorder(views::CreateEmptyBorder( + gfx::Insets::TLBR(kActiveSearchBoxHeight, 0, 0, 0))); AddChildView(contents_view_); } else { auto scroller = std::make_unique<views::ScrollView>(); // Leaves a placeholder area for the search box and the separator below it. - scroller->SetBorder(views::CreateEmptyBorder(gfx::Insets::TLBR( - kActiveSearchBoxHeight + kSearchBoxBottomSpacing + kSeparatorThickness, - 0, 0, 0))); + scroller->SetBorder(views::CreateEmptyBorder( + gfx::Insets::TLBR(kActiveSearchBoxHeight, 0, 0, 0))); scroller->SetDrawOverflowIndicator(false); scroller->SetContents(base::WrapUnique(contents_view_)); // Setting clip height is necessary to make ScrollView take into account its @@ -274,11 +205,6 @@ std::make_unique<SearchResultPageDialogController>(this); privacy_container_view_ = AddSearchResultContainerView( std::make_unique<PrivacyContainerView>(view_delegate)); - search_result_tile_item_list_view_ = AddSearchResultContainerView( - std::make_unique<SearchResultTileItemListView>( - search_box_view->search_box(), view_delegate)); - result_lists_separator_ = contents_view_->AddChildView( - std::make_unique<HorizontalSeparator>(bounds().width())); // productivity_launcher_index is not set as the feature is not enabled. search_result_list_view_ = AddSearchResultContainerView(std::make_unique<SearchResultListView>( @@ -327,8 +253,7 @@ int adjusted_height = std::min( std::max(kMinHeight, productivity_launcher_search_view_->TabletModePreferredHeight() + - kActiveSearchBoxHeight + kSearchBoxBottomSpacing + - kSeparatorThickness + + kActiveSearchBoxHeight + kExpandedSearchBoxCornerRadiusForProductivityLauncher), AppListPage::contents_view()->height()); return gfx::Size(kWidth, adjusted_height); @@ -424,10 +349,6 @@ should_show_page_view || should_show_container_view; } - result_lists_separator_->SetVisible( - search_result_tile_item_list_view_->num_results() && - search_result_list_view_->num_results() && - ShouldShowSearchResultView()); AppListPage::contents_view() ->GetSearchBoxView() ->OnResultContainerVisibilityChanged(should_show_page_view); @@ -766,11 +687,6 @@ return privacy_container_view_; } -SearchResultTileItemListView* -SearchResultPageView::GetSearchResultTileItemListViewForTest() { - return search_result_tile_item_list_view_; -} - SearchResultListView* SearchResultPageView::GetSearchResultListViewForTest() { return search_result_list_view_; }
diff --git a/ash/app_list/views/search_result_page_view.h b/ash/app_list/views/search_result_page_view.h index 3b69eb8b..55ae04f5 100644 --- a/ash/app_list/views/search_result_page_view.h +++ b/ash/app_list/views/search_result_page_view.h
@@ -27,7 +27,6 @@ class SearchBoxView; class SearchResultBaseView; class SearchResultListView; -class SearchResultTileItemListView; class SearchResultPageAnchoredDialog; class SystemShadow; @@ -126,7 +125,6 @@ void UpdateResultContainersVisibility(); PrivacyContainerView* GetPrivacyContainerViewForTest(); - SearchResultTileItemListView* GetSearchResultTileItemListViewForTest(); SearchResultListView* GetSearchResultListViewForTest(); private: @@ -134,9 +132,6 @@ // launcher. enum class SearchResultsState { kClosed, kActive, kExpanded }; - // Separator between SearchResultContainerView. - class HorizontalSeparator; - // Passed to |result_selection_controller_| as a callback that gets called // when the currently selected result changes. // Ensures that |scroller_| visible rect contains the newly selected result. @@ -209,15 +204,10 @@ // Search result containers shown within search results page (and added to // `result_container_views_`). PrivacyContainerView* privacy_container_view_ = nullptr; - SearchResultTileItemListView* search_result_tile_item_list_view_ = nullptr; SearchResultListView* search_result_list_view_ = nullptr; // Search result container used for productivity launcher. ProductivityLauncherSearchView* productivity_launcher_search_view_ = nullptr; - // Separator view shown between search result tile item list and search - // results list. - HorizontalSeparator* result_lists_separator_ = nullptr; - // View containing SearchCardView instances. Owned by view hierarchy. views::View* const contents_view_;
diff --git a/ash/app_list/views/search_result_page_view_unittest.cc b/ash/app_list/views/search_result_page_view_unittest.cc index a6cce84..5ec227a 100644 --- a/ash/app_list/views/search_result_page_view_unittest.cc +++ b/ash/app_list/views/search_result_page_view_unittest.cc
@@ -15,7 +15,6 @@ #include "ash/app_list/views/app_list_view.h" #include "ash/app_list/views/contents_view.h" #include "ash/app_list/views/search_result_list_view.h" -#include "ash/app_list/views/search_result_tile_item_list_view.h" #include "ash/app_list/views/search_result_view.h" #include "ash/constants/ash_features.h" #include "ash/public/cpp/app_list/app_list_features.h" @@ -56,8 +55,6 @@ ContentsView* contents_view = app_list_view_->app_list_main_view()->contents_view(); view_ = contents_view->search_result_page_view(); - tile_list_view_ = contents_view->search_result_page_view() - ->GetSearchResultTileItemListViewForTest(); list_view_ = contents_view->search_result_page_view() ->GetSearchResultListViewForTest(); } @@ -69,9 +66,6 @@ protected: SearchResultPageView* view() const { return view_; } - SearchResultTileItemListView* tile_list_view() const { - return tile_list_view_; - } SearchResultListView* list_view() const { return list_view_; } SearchModel::SearchResults* GetResults() const { @@ -82,8 +76,6 @@ TestAppListColorProvider color_provider_; // Needed by AppListView. AppListView* app_list_view_ = nullptr; // Owned by native widget. SearchResultPageView* view_ = nullptr; // Owned by views hierarchy. - SearchResultTileItemListView* tile_list_view_ = - nullptr; // Owned by views hierarchy. SearchResultListView* list_view_ = nullptr; // Owned by views hierarchy. std::unique_ptr<AppListTestViewDelegate> delegate_; base::test::ScopedFeatureList scoped_feature_list_; @@ -114,13 +106,11 @@ // Adding results will schedule Update(). RunPendingMessages(); - ASSERT_GE(view()->result_container_views().size(), 3u); + ASSERT_GE(view()->result_container_views().size(), 2u); // Privacy container is hidden. EXPECT_FALSE(view()->result_container_views()[0]->GetVisible()); - EXPECT_TRUE(tile_list_view()->GetVisible()); - EXPECT_EQ(tile_list_view(), view()->result_container_views()[1]); EXPECT_TRUE(list_view()->GetVisible()); - EXPECT_EQ(list_view(), view()->result_container_views()[2]); + EXPECT_EQ(list_view(), view()->result_container_views()[1]); // Change the relevance of the tile result to be lower than list results. The // tile container should still be displayed first. @@ -131,9 +121,7 @@ // Privacy container is hidden. EXPECT_FALSE(view()->result_container_views()[0]->GetVisible()); - EXPECT_EQ(tile_list_view(), view()->result_container_views()[1]); - EXPECT_TRUE(tile_list_view()->GetVisible()); - EXPECT_EQ(list_view(), view()->result_container_views()[2]); + EXPECT_EQ(list_view(), view()->result_container_views()[1]); EXPECT_TRUE(list_view()->GetVisible()); } @@ -150,7 +138,6 @@ // Adding results will schedule Update(). RunPendingMessages(); - EXPECT_TRUE(tile_list_view()->GetVisible()); EXPECT_FALSE(list_view()->GetVisible()); } @@ -168,7 +155,6 @@ RunPendingMessages(); EXPECT_TRUE(list_view()->GetVisible()); - EXPECT_FALSE(tile_list_view()->GetVisible()); } } // namespace test
diff --git a/ash/app_list/views/search_result_tile_item_list_view.cc b/ash/app_list/views/search_result_tile_item_list_view.cc deleted file mode 100644 index 6f5bd45..0000000 --- a/ash/app_list/views/search_result_tile_item_list_view.cc +++ /dev/null
@@ -1,368 +0,0 @@ -// Copyright 2014 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#include "ash/app_list/views/search_result_tile_item_list_view.h" - -#include <stddef.h> - -#include <algorithm> -#include <memory> -#include <set> -#include <string> - -#include "ash/app_list/app_list_model_provider.h" -#include "ash/app_list/app_list_util.h" -#include "ash/app_list/app_list_view_delegate.h" -#include "ash/app_list/model/search/search_result.h" -#include "ash/app_list/views/search_result_page_view.h" -#include "ash/public/cpp/app_list/app_list_color_provider.h" -#include "ash/public/cpp/app_list/app_list_config.h" -#include "ash/public/cpp/app_list/app_list_features.h" -#include "ash/public/cpp/app_list/app_list_notifier.h" -#include "ash/public/cpp/app_list/app_list_types.h" -#include "ash/public/cpp/app_list/internal_app_id_constants.h" -#include "base/bind.h" -#include "base/callback.h" -#include "base/i18n/rtl.h" -#include "base/metrics/histogram_macros.h" -#include "base/metrics/user_metrics.h" -#include "base/stl_util.h" -#include "ui/gfx/color_palette.h" -#include "ui/gfx/geometry/insets.h" -#include "ui/views/background.h" -#include "ui/views/border.h" -#include "ui/views/controls/separator.h" -#include "ui/views/controls/textfield/textfield.h" -#include "ui/views/focus/focus_manager.h" -#include "ui/views/layout/box_layout.h" -#include "ui/views/widget/widget.h" - -namespace ash { - -namespace { - -// Layout constants used when fullscreen app list feature is enabled. -constexpr int kItemListVerticalSpacing = 16; -constexpr int kItemListHorizontalSpacing = 12; -constexpr int kBetweenItemSpacing = 8; -constexpr int kSeparatorLeftRightPadding = 4; -constexpr int kSeparatorHeight = 46; -constexpr int kSeparatorTopPadding = 10; - -// The Delay before recording play store app results impression, i.e., if the -// play store results are displayed less than the duration, we assume user -// won't have chance to see them clearly and click on them, and wont' log -// the impression. -constexpr int kPlayStoreImpressionDelayInMs = 1000; - -// Returns true if the search result is an installable app. -bool IsResultAnInstallableApp(SearchResult* result) { - SearchResult::ResultType result_type = result->result_type(); - return result_type == AppListSearchResultType::kPlayStoreApp || - result_type == AppListSearchResultType::kPlayStoreReinstallApp || - result_type == AppListSearchResultType::kInstantApp; -} - -bool IsPlayStoreApp(SearchResult* result) { - return result->result_type() == AppListSearchResultType::kPlayStoreApp; -} - -} // namespace - -SearchResultTileItemListView::SearchResultTileItemListView( - views::Textfield* search_box, - AppListViewDelegate* view_delegate) - : SearchResultContainerView(view_delegate), - search_box_(search_box), - is_app_reinstall_recommendation_enabled_( - app_list_features::IsAppReinstallZeroStateEnabled()), - max_search_result_tiles_( - SharedAppListConfig::instance().max_search_result_tiles()) { - layout_ = SetLayoutManager(std::make_unique<views::BoxLayout>( - views::BoxLayout::Orientation::kHorizontal, - gfx::Insets::VH(kItemListVerticalSpacing, kItemListHorizontalSpacing), - kBetweenItemSpacing)); - for (size_t i = 0; i < max_search_result_tiles_; ++i) { - views::Separator* separator = - AddChildView(std::make_unique<views::Separator>()); - separator->SetVisible(false); - separator->SetBorder(views::CreateEmptyBorder(gfx::Insets::TLBR( - kSeparatorTopPadding, kSeparatorLeftRightPadding, - SharedAppListConfig::instance().search_tile_height() - kSeparatorHeight, - kSeparatorLeftRightPadding))); - separator->SetColorId(AppListColorProvider::Get()->GetSeparatorColorId()); - separator_views_.push_back(separator); - layout_->SetFlexForView(separator, 0); - - SearchResultTileItemView* tile_item = - AddChildView(std::make_unique<SearchResultTileItemView>(view_delegate)); - tile_item->set_index_in_container(i); - tile_views_.push_back(tile_item); - AddObservedResultView(tile_item); - } - - // Tile items are shown horizontally. - set_horizontally_traversable(true); -} - -SearchResultTileItemListView::~SearchResultTileItemListView() = default; - -SearchResultTileItemView* SearchResultTileItemListView::GetResultViewAt( - size_t index) { - DCHECK(index >= 0 && index < tile_views_.size()); - return tile_views_[index]; -} - -int SearchResultTileItemListView::DoUpdate() { - if (!GetWidget() || !GetWidget()->IsVisible() || !GetWidget()->IsActive()) { - for (size_t i = 0; i < max_search_result_tiles_; ++i) { - SearchResultBaseView* result_view = GetResultViewAt(i); - result_view->SetResult(nullptr); - result_view->SetVisible(false); - } - return 0; - } - - std::vector<SearchResult*> display_results = GetDisplayResults(); - - std::set<std::string> result_id_removed, result_id_added; - bool is_result_an_installable_app = false; - bool is_previous_result_installable_app = false; - int installed_app_index = -1; - int playstore_app_index = -1; - int app_group_index = -1; - bool found_playstore_results = false; - - for (size_t i = 0; i < max_search_result_tiles_; ++i) { - // If the current result at i exists, wants to be notified and is a - // different id, notify it that it is being hidden. - SearchResult* current_result = tile_views_[i]->result(); - if (current_result != nullptr) { - result_id_removed.insert(current_result->id()); - } - - if (i >= display_results.size()) { - separator_views_[i]->SetVisible(false); - - GetResultViewAt(i)->SetResult(nullptr); - continue; - } - - SearchResult* item = display_results[i]; - if (IsPlayStoreApp(item)) { - ++playstore_app_index; - app_group_index = playstore_app_index; - found_playstore_results = true; - } else if (item->result_type() == - AppListSearchResultType::kPlayStoreReinstallApp) { - app_group_index = playstore_app_index; - } else { - ++installed_app_index; - app_group_index = installed_app_index; - } - - GetResultViewAt(i)->SetResult(item); - GetResultViewAt(i)->set_group_index_in_container_view(app_group_index); - result_id_added.insert(item->id()); - is_result_an_installable_app = IsResultAnInstallableApp(item); - - if (i > 0 && - (is_result_an_installable_app != is_previous_result_installable_app)) { - // Add a separator between installed apps and installable apps. - // This assumes the search results are already separated in groups for - // installed and installable apps. - separator_views_[i]->SetVisible(true); - } else { - separator_views_[i]->SetVisible(false); - } - - is_previous_result_installable_app = is_result_an_installable_app; - } - - auto* notifier = view_delegate()->GetNotifier(); - if (notifier) { - std::vector<AppListNotifier::Result> notifier_results; - for (const auto* result : display_results) - notifier_results.emplace_back(result->id(), result->metrics_type()); - notifier->NotifyResultsUpdated(SearchResultDisplayType::kTile, - notifier_results); - } - - // Track play store results and start the timer for recording their impression - // UMA metrics. - std::u16string user_typed_query = GetUserTypedQuery(); - if (found_playstore_results && user_typed_query != recent_playstore_query_) { - recent_playstore_query_ = user_typed_query; - playstore_impression_timer_.Stop(); - playstore_impression_timer_.Start( - FROM_HERE, base::Milliseconds(kPlayStoreImpressionDelayInMs), this, - &SearchResultTileItemListView::OnPlayStoreImpressionTimer); - // Set the starting time in result view for play store results. - base::TimeTicks result_display_start = base::TimeTicks::Now(); - for (size_t i = 0; i < max_search_result_tiles_; ++i) { - SearchResult* result = GetResultViewAt(i)->result(); - if (result && IsPlayStoreApp(result)) { - GetResultViewAt(i)->set_result_display_start_time(result_display_start); - } - } - } else if (!found_playstore_results) { - playstore_impression_timer_.Stop(); - } - - return display_results.size(); -} - -std::vector<SearchResult*> SearchResultTileItemListView::GetDisplayResults() { - std::u16string raw_query = search_box_->GetText(); - std::u16string query; - base::TrimWhitespace(raw_query, base::TRIM_ALL, &query); - - // We ask for |max_search_result_tiles_| policy tile results first, - // then add them to their preferred position in the tile list if found. - // Note: Policy tile provides a mechanism to display the result tile at the - // preferred position recommended by display_index() property of the search - // result. This is what policy referred to. It has nothing to do with - // Enterprise policy. - auto policy_tiles_filter = - base::BindRepeating([](const SearchResult& r) -> bool { - return r.display_index() != SearchResultDisplayIndex::kUndefined && - r.display_type() == SearchResultDisplayType::kTile && - r.is_recommendation(); - }); - std::vector<SearchResult*> policy_tiles_results = - is_app_reinstall_recommendation_enabled_ && query.empty() - ? SearchModel::FilterSearchResultsByFunction( - results(), policy_tiles_filter, max_search_result_tiles_) - : std::vector<SearchResult*>(); - - SearchResult::DisplayType display_type = SearchResultDisplayType::kTile; - size_t display_num = max_search_result_tiles_ - policy_tiles_results.size(); - - // Do not display the repeat reinstall results or continue reading app in the - // search result list. - auto non_policy_tiles_filter = base::BindRepeating( - [](const SearchResult::DisplayType& display_type, - const SearchResult& r) -> bool { - return r.display_type() == display_type && - r.result_type() != - AppListSearchResultType::kPlayStoreReinstallApp && - r.id() != kInternalAppIdContinueReading; - }, - display_type); - std::vector<SearchResult*> display_results = - SearchModel::FilterSearchResultsByFunction( - results(), non_policy_tiles_filter, display_num); - - // Policy tile results will be appended to the final tiles list - // based on their specified index. If the requested index is out of - // range of the current list, the result will be appended to the back. - std::sort(policy_tiles_results.begin(), policy_tiles_results.end(), - [](const SearchResult* r1, const SearchResult* r2) -> bool { - return r1->display_index() < r2->display_index(); - }); - - const SearchResultDisplayIndex display_results_last_index = - static_cast<SearchResultDisplayIndex>(display_results.size() - 1); - for (auto* result : policy_tiles_results) { - const SearchResultDisplayIndex result_index = result->display_index(); - if (result_index > display_results_last_index) { - display_results.emplace_back(result); - } else { - // TODO(newcomer): Remove this check once we determine the root cause for - // https://crbug.com/992344. - CHECK_GE(result_index, SearchResultDisplayIndex::kFirstIndex); - display_results.emplace(display_results.begin() + result_index, result); - } - } - - return display_results; -} - -std::u16string SearchResultTileItemListView::GetUserTypedQuery() { - std::u16string search_box_text = search_box_->GetText(); - gfx::Range range = search_box_->GetSelectedRange(); - std::u16string raw_query = range.is_empty() - ? search_box_text - : search_box_text.substr(0, range.start()); - std::u16string query; - base::TrimWhitespace(raw_query, base::TRIM_ALL, &query); - return query; -} - -void SearchResultTileItemListView::OnPlayStoreImpressionTimer() { - size_t playstore_app_num = 0; - for (const auto* tile_view : tile_views_) { - SearchResult* result = tile_view->result(); - if (result == nullptr) - continue; - if (IsPlayStoreApp(result)) - ++playstore_app_num; - } - - // Log the UMA metrics of play store impression. - base::RecordAction( - base::UserMetricsAction("AppList_ShowPlayStoreQueryResults")); - - DCHECK_LE(playstore_app_num, max_search_result_tiles_); - UMA_HISTOGRAM_EXACT_LINEAR("Apps.AppListPlayStoreSearchAppsDisplayed", - playstore_app_num, max_search_result_tiles_); -} - -void SearchResultTileItemListView::CleanUpOnViewHide() { - playstore_impression_timer_.Stop(); - recent_playstore_query_.clear(); -} - -const char* SearchResultTileItemListView::GetClassName() const { - return "SearchResultTileItemListView"; -} - -void SearchResultTileItemListView::Layout() { - const bool flex = GetContentsBounds().width() < GetPreferredSize().width(); - layout_->SetDefaultFlex(flex ? 1 : 0); - layout_->set_between_child_spacing(flex ? 1 : kBetweenItemSpacing); - - views::View::Layout(); -} - -void SearchResultTileItemListView::OnShownChanged() { - SearchResultContainerView::OnShownChanged(); - for (const auto* tile_view : tile_views_) { - SearchResult* result = tile_view->result(); - if (result == nullptr) { - continue; - } - } -} - -void SearchResultTileItemListView::OnThemeChanged() { - SearchResultContainerView::OnThemeChanged(); - - const auto parent_background_color = - AppListColorProvider::Get()->GetSearchBoxCardBackgroundColor(GetWidget()); - - for (auto* tile_view : tile_views_) - tile_view->SetParentBackgroundColor(parent_background_color); -} - -void SearchResultTileItemListView::VisibilityChanged(View* starting_from, - bool is_visible) { - SearchResultContainerView::VisibilityChanged(starting_from, is_visible); - // We only do this work when is_visible is false, since this is how we - // receive the event. We filter and only run when shown. - if (is_visible && shown()) { - return; - } - - CleanUpOnViewHide(); - - for (const auto* tile_view : tile_views_) { - SearchResult* result = tile_view->result(); - if (result == nullptr) { - continue; - } - } -} - -} // namespace ash
diff --git a/ash/app_list/views/search_result_tile_item_list_view.h b/ash/app_list/views/search_result_tile_item_list_view.h deleted file mode 100644 index 886c14c..0000000 --- a/ash/app_list/views/search_result_tile_item_list_view.h +++ /dev/null
@@ -1,87 +0,0 @@ -// Copyright 2014 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#ifndef ASH_APP_LIST_VIEWS_SEARCH_RESULT_TILE_ITEM_LIST_VIEW_H_ -#define ASH_APP_LIST_VIEWS_SEARCH_RESULT_TILE_ITEM_LIST_VIEW_H_ - -#include <vector> - -#include "ash/app_list/views/search_result_container_view.h" -#include "ash/app_list/views/search_result_tile_item_view.h" -#include "base/timer/timer.h" - -namespace views { -class BoxLayout; -class Separator; -class Textfield; -} // namespace views - -namespace ash { - -class AppListViewDelegate; - -// Displays a list of SearchResultTileItemView. -class ASH_EXPORT SearchResultTileItemListView - : public SearchResultContainerView { - public: - SearchResultTileItemListView(views::Textfield* search_box, - AppListViewDelegate* view_delegate); - SearchResultTileItemListView(const SearchResultTileItemListView&) = delete; - SearchResultTileItemListView& operator=(const SearchResultTileItemListView&) = - delete; - ~SearchResultTileItemListView() override; - - // Overridden from SearchResultContainerView: - SearchResultTileItemView* GetResultViewAt(size_t index) override; - - // Overridden from views::View: - const char* GetClassName() const override; - void Layout() override; - - const std::vector<SearchResultTileItemView*>& tile_views_for_test() const { - return tile_views_; - } - - // Overridden from SearchResultContainerView: - void OnShownChanged() override; - void OnThemeChanged() override; - - protected: - // View overrides: - void VisibilityChanged(View* starting_from, bool is_visible) override; - - private: - // Overridden from SearchResultContainerView: - int DoUpdate() override; - - std::vector<SearchResult*> GetDisplayResults(); - - std::u16string GetUserTypedQuery(); - - void OnPlayStoreImpressionTimer(); - - // Cleans up when the view is hid due to closing the suggestion widow - // or closing the launcher. - void CleanUpOnViewHide(); - - std::vector<SearchResultTileItemView*> tile_views_; - - std::vector<views::Separator*> separator_views_; - - // Owned by the views hierarchy. - views::Textfield* search_box_ = nullptr; - views::BoxLayout* layout_ = nullptr; - - std::u16string recent_playstore_query_; - - base::OneShotTimer playstore_impression_timer_; - - const bool is_app_reinstall_recommendation_enabled_; - - const size_t max_search_result_tiles_; -}; - -} // namespace ash - -#endif // ASH_APP_LIST_VIEWS_SEARCH_RESULT_TILE_ITEM_LIST_VIEW_H_
diff --git a/ash/app_list/views/search_result_tile_item_list_view_unittest.cc b/ash/app_list/views/search_result_tile_item_list_view_unittest.cc deleted file mode 100644 index 036c87b..0000000 --- a/ash/app_list/views/search_result_tile_item_list_view_unittest.cc +++ /dev/null
@@ -1,331 +0,0 @@ -// Copyright 2017 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#include "ash/app_list/views/search_result_tile_item_list_view.h" - -#include <algorithm> -#include <memory> -#include <utility> - -#include "ash/app_list/app_list_test_view_delegate.h" -#include "ash/app_list/model/app_list_model.h" -#include "ash/app_list/model/search/test_search_result.h" -#include "ash/app_list/views/search_result_tile_item_view.h" -#include "ash/app_list/views/search_result_view.h" -#include "ash/public/cpp/app_list/app_list_features.h" -#include "ash/public/cpp/test/test_app_list_color_provider.h" -#include "ash/strings/grit/ash_strings.h" -#include "base/strings/string_number_conversions.h" -#include "base/strings/utf_string_conversions.h" -#include "base/test/scoped_feature_list.h" -#include "testing/gtest/include/gtest/gtest.h" -#include "ui/accessibility/ax_node_data.h" -#include "ui/base/l10n/l10n_util.h" -#include "ui/views/controls/textfield/textfield.h" -#include "ui/views/test/widget_test.h" - -namespace ash { - -namespace { -constexpr size_t kInstalledApps = 4; -constexpr size_t kPlayStoreApps = 2; -constexpr size_t kRecommendedApps = 1; - -// used to test when multiple chips with specified display -// indexes have been added -constexpr size_t kRecommendedAppsWithDisplayIndex = 3; -} // namespace - -class SearchResultTileItemListViewTest - : public views::test::WidgetTest, - public ::testing::WithParamInterface<std::pair<bool, bool>> { - public: - SearchResultTileItemListViewTest() = default; - - SearchResultTileItemListViewTest(const SearchResultTileItemListViewTest&) = - delete; - SearchResultTileItemListViewTest& operator=( - const SearchResultTileItemListViewTest&) = delete; - - ~SearchResultTileItemListViewTest() override = default; - - // Overridden from testing::Test: - void SetUp() override { - views::test::WidgetTest::SetUp(); - widget_ = CreateTopLevelPlatformWidget(); - } - - void TearDown() override { - view_.reset(); - widget_->CloseNow(); - views::test::WidgetTest::TearDown(); - } - - protected: - void CreateSearchResultTileItemListView() { - scoped_feature_list_.InitWithFeatureState( - app_list_features::kEnableAppReinstallZeroState, - IsReinstallAppRecommendationEnabled()); - - // Sets up the views. - textfield_ = std::make_unique<views::Textfield>(); - view_ = std::make_unique<SearchResultTileItemListView>(textfield_.get(), - &view_delegate_); - widget_->SetBounds(gfx::Rect(0, 0, 300, 200)); - widget_->GetContentsView()->AddChildView(view_.get()); - widget_->Show(); - view_->SetResults(GetResults()); - } - - bool IsReinstallAppRecommendationEnabled() const { return GetParam().first; } - - SearchResultTileItemListView* view() { return view_.get(); } - - SearchModel::SearchResults* GetResults() { - return AppListModelProvider::Get()->search_model()->results(); - } - - void SetUpSearchResults() { - SearchModel::SearchResults* results = GetResults(); - - // Populate results for installed applications. - for (size_t i = 0; i < kInstalledApps; ++i) { - std::unique_ptr<TestSearchResult> result = - std::make_unique<TestSearchResult>(); - result->set_result_id("InstalledApp " + base::NumberToString(i)); - result->set_display_type(SearchResultDisplayType::kTile); - result->set_result_type(AppListSearchResultType::kInstalledApp); - result->SetTitle(u"InstalledApp " + base::NumberToString16(i)); - results->Add(std::move(result)); - } - - // Populate results for Play Store search applications. - for (size_t i = 0; i < kPlayStoreApps; ++i) { - std::unique_ptr<TestSearchResult> result = - std::make_unique<TestSearchResult>(); - result->set_result_id("PlayStoreApp " + base::NumberToString(i)); - result->set_display_type(SearchResultDisplayType::kTile); - result->set_result_type(AppListSearchResultType::kPlayStoreApp); - result->SetTitle(u"PlayStoreApp " + base::NumberToString16(i)); - result->SetRating(1 + i); - result->SetFormattedPrice(u"Price " + base::NumberToString16(i)); - results->Add(std::move(result)); - } - - if (IsReinstallAppRecommendationEnabled()) { - for (size_t i = 0; i < kRecommendedApps; ++i) { - std::unique_ptr<TestSearchResult> result = - std::make_unique<TestSearchResult>(); - result->set_result_id("RecommendedApp " + base::NumberToString(i)); - result->set_display_type(SearchResultDisplayType::kTile); - result->set_is_recommendation(true); - result->set_result_type( - AppListSearchResultType::kPlayStoreReinstallApp); - result->set_display_index(SearchResultDisplayIndex::kSixthIndex); - result->SetTitle(u"RecommendedApp " + base::NumberToString16(i)); - result->SetRating(1 + i); - results->Add(std::move(result)); - } - } - - // Adding results calls SearchResultContainerView::ScheduleUpdate(). - // It will post a delayed task to update the results and relayout. - RunPendingMessages(); - } - - void SetUpSearchResultsWithMultipleDisplayIndexesRequested() { - SearchModel::SearchResults* results = GetResults(); - - // Populate results for installed applications. - for (size_t i = 0; i < kInstalledApps; ++i) { - std::unique_ptr<TestSearchResult> result = - std::make_unique<TestSearchResult>(); - result->set_result_id("InstalledApp " + base::NumberToString(i)); - result->set_display_type(SearchResultDisplayType::kTile); - result->set_result_type(AppListSearchResultType::kInstalledApp); - result->SetTitle(u"InstalledApp " + base::NumberToString16(i)); - results->Add(std::move(result)); - } - - // Populate results for Play Store search applications. - for (size_t i = 0; i < kPlayStoreApps; ++i) { - std::unique_ptr<TestSearchResult> result = - std::make_unique<TestSearchResult>(); - result->set_result_id("PlayStoreApp " + base::NumberToString(i)); - result->set_display_type(SearchResultDisplayType::kTile); - result->set_result_type(AppListSearchResultType::kPlayStoreApp); - result->SetTitle(u"PlayStoreApp " + base::NumberToString16(i)); - result->SetRating(1 + i); - result->SetFormattedPrice(u"Price " + base::NumberToString16(i)); - results->Add(std::move(result)); - } - - const SearchResultDisplayIndex - display_indexes[kRecommendedAppsWithDisplayIndex] = { - SearchResultDisplayIndex::kFourthIndex, - SearchResultDisplayIndex::kFifthIndex, - SearchResultDisplayIndex::kSixthIndex, - }; - - if (IsReinstallAppRecommendationEnabled()) { - for (size_t i = 0; i < kRecommendedAppsWithDisplayIndex; ++i) { - std::unique_ptr<TestSearchResult> result = - std::make_unique<TestSearchResult>(); - result->set_result_id("RecommendedApp " + base::NumberToString(i)); - result->set_display_type(SearchResultDisplayType::kTile); - result->set_is_recommendation(true); - result->set_result_type( - AppListSearchResultType::kPlayStoreReinstallApp); - result->set_display_index(display_indexes[i]); - result->SetTitle(u"RecommendedApp " + base::NumberToString16(i)); - result->SetRating(1 + i); - results->AddAt(display_indexes[i], std::move(result)); - } - } - - // Adding results calls SearchResultContainerView::ScheduleUpdate(). - // It will post a delayed task to update the results and relayout. - RunPendingMessages(); - } - - size_t GetOpenResultCount(int ranking) { - return view_delegate_.open_search_result_counts()[ranking]; - } - - void ResetOpenResultCount() { - view_delegate_.open_search_result_counts().clear(); - } - - size_t GetResultCount() const { return view_->num_results(); } - - private: - TestAppListColorProvider color_provider_; // Needed by AppListView. - test::AppListTestViewDelegate view_delegate_; - std::unique_ptr<SearchResultTileItemListView> view_; - views::Widget* widget_; - std::unique_ptr<views::Textfield> textfield_; - base::test::ScopedFeatureList scoped_feature_list_; -}; - -TEST_P(SearchResultTileItemListViewTest, Basic) { - CreateSearchResultTileItemListView(); - SetUpSearchResults(); - - const size_t results = GetResultCount(); - size_t expected_results = kInstalledApps; - expected_results += kPlayStoreApps; - - if (IsReinstallAppRecommendationEnabled()) { - expected_results += kRecommendedApps; - } - constexpr size_t kMaxNumSearchResultTiles = 6; - expected_results = std::min(kMaxNumSearchResultTiles, expected_results); - - ASSERT_EQ(expected_results, results); - - // When the Play Store app search or app reinstallation results are - // present, for each result, we added a separator for result type grouping. - const size_t child_step = 2; - const size_t expected_num_children = kMaxNumSearchResultTiles * child_step; - EXPECT_EQ(expected_num_children, view()->children().size()); - - /// Test accessibility descriptions of tile views. - const size_t first_child = child_step - 1; - for (size_t i = 0; i < kInstalledApps; ++i) { - ui::AXNodeData node_data; - view()->children()[first_child + i * child_step]->GetAccessibleNodeData( - &node_data); - EXPECT_EQ(ax::mojom::Role::kListBoxOption, node_data.role); - EXPECT_EQ(l10n_util::GetStringFUTF8( - IDS_APP_ACCESSIBILITY_INSTALLED_APP_ANNOUNCEMENT, - base::UTF8ToUTF16("InstalledApp " + base::NumberToString(i))), - node_data.GetStringAttribute(ax::mojom::StringAttribute::kName)); - } - - const size_t expected_install_apps = - expected_results - - (IsReinstallAppRecommendationEnabled() ? kRecommendedApps : 0) - - kInstalledApps; - for (size_t i = 0; i < expected_install_apps; ++i) { - ui::AXNodeData node_data; - view() - ->children()[first_child + (i + kInstalledApps) * child_step] - ->GetAccessibleNodeData(&node_data); - EXPECT_EQ(ax::mojom::Role::kListBoxOption, node_data.role); - EXPECT_EQ( - l10n_util::GetStringFUTF8( - IDS_APP_ACCESSIBILITY_ARC_APP_ANNOUNCEMENT, - base::UTF8ToUTF16("PlayStoreApp " + base::NumberToString(i))) + - ", Star rating " + base::NumberToString(i + 1) + ".0", - node_data.GetStringAttribute(ax::mojom::StringAttribute::kName)); - } - - // Recommendations. - const size_t start_index = kInstalledApps + expected_install_apps; - for (size_t i = 0; i < expected_results - start_index; ++i) { - ui::AXNodeData node_data; - view() - ->children()[first_child + (i + start_index) * child_step] - ->GetAccessibleNodeData(&node_data); - EXPECT_EQ(ax::mojom::Role::kListBoxOption, node_data.role); - EXPECT_EQ( - l10n_util::GetStringFUTF8( - IDS_APP_ACCESSIBILITY_APP_RECOMMENDATION_ARC, - base::UTF8ToUTF16("RecommendedApp " + base::NumberToString(i))) + - ", Star rating " + base::NumberToString(i + 1) + ".0", - node_data.GetStringAttribute(ax::mojom::StringAttribute::kName)); - } - - ResetOpenResultCount(); - for (size_t i = 0; i < results; ++i) { - ui::KeyEvent event(ui::ET_KEY_PRESSED, ui::VKEY_RETURN, ui::EF_NONE); - for (size_t j = 0; j <= i; ++j) - view()->tile_views_for_test()[i]->OnKeyEvent(&event); - // When app reinstalls is enabled, we actually instantiate 7 results, - // but only show 6. So we have to look, for exactly 1 result, a "skip" - // ahead for the reinstall result. - if (IsReinstallAppRecommendationEnabled() && i == (results - 1)) { - EXPECT_EQ(i + 1, GetOpenResultCount(i + 1)); - } else { - EXPECT_EQ(i + 1, GetOpenResultCount(i)); - } - } -} - -// Tests that when multiple apps with specified indexes are added to the app -// results list, they are found at the indexes they requested. -TEST_P(SearchResultTileItemListViewTest, TestRecommendations) { - if (!IsReinstallAppRecommendationEnabled()) - return; - - CreateSearchResultTileItemListView(); - SetUpSearchResultsWithMultipleDisplayIndexesRequested(); - - const size_t child_step = 2; - - size_t first_index = kInstalledApps + kRecommendedAppsWithDisplayIndex; - - size_t stepper = 3; - for (size_t i = 0; i < stepper; ++i) { - ui::AXNodeData node_data; - view()->children()[first_index + i * child_step]->GetAccessibleNodeData( - &node_data); - EXPECT_EQ(ax::mojom::Role::kListBoxOption, node_data.role); - EXPECT_EQ( - l10n_util::GetStringFUTF8( - IDS_APP_ACCESSIBILITY_APP_RECOMMENDATION_ARC, - base::UTF8ToUTF16("RecommendedApp " + base::NumberToString(i))) + - ", Star rating " + base::NumberToString(i + 1) + ".0", - node_data.GetStringAttribute(ax::mojom::StringAttribute::kName)); - } -} - -INSTANTIATE_TEST_SUITE_P(, - SearchResultTileItemListViewTest, - testing::ValuesIn({std::make_pair(false, false), - std::make_pair(false, true), - std::make_pair(true, false), - std::make_pair(true, true)})); - -} // namespace ash
diff --git a/ash/app_list/views/search_result_tile_item_view.cc b/ash/app_list/views/search_result_tile_item_view.cc deleted file mode 100644 index 603ddbe9..0000000 --- a/ash/app_list/views/search_result_tile_item_view.cc +++ /dev/null
@@ -1,579 +0,0 @@ -// Copyright 2014 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#include "ash/app_list/views/search_result_tile_item_view.h" - -#include <utility> - -#include "ash/app_list/app_list_metrics.h" -#include "ash/app_list/app_list_util.h" -#include "ash/app_list/app_list_view_delegate.h" -#include "ash/app_list/model/app_list_model.h" -#include "ash/app_list/model/search/search_model.h" -#include "ash/app_list/model/search/search_result.h" -#include "ash/app_list/views/app_list_item_view.h" -#include "ash/public/cpp/app_list/app_list_color_provider.h" -#include "ash/public/cpp/app_list/app_list_config.h" -#include "ash/public/cpp/app_list/app_list_features.h" -#include "ash/public/cpp/app_list/app_list_types.h" -#include "ash/public/cpp/app_list/vector_icons/vector_icons.h" -#include "ash/public/cpp/ash_typography.h" -#include "ash/public/cpp/pagination/pagination_model.h" -#include "ash/strings/grit/ash_strings.h" -#include "ash/style/ash_color_id.h" -#include "base/bind.h" -#include "base/i18n/number_formatting.h" -#include "base/metrics/histogram_macros.h" -#include "base/metrics/user_metrics.h" -#include "base/strings/utf_string_conversions.h" -#include "ui/accessibility/ax_node_data.h" -#include "ui/base/l10n/l10n_util.h" -#include "ui/base/models/image_model.h" -#include "ui/base/themed_vector_icon.h" -#include "ui/chromeos/styles/cros_styles.h" -#include "ui/gfx/canvas.h" -#include "ui/gfx/color_palette.h" -#include "ui/gfx/geometry/rect_conversions.h" -#include "ui/gfx/image/image_skia_operations.h" -#include "ui/gfx/paint_vector_icon.h" -#include "ui/views/accessibility/view_accessibility.h" -#include "ui/views/background.h" -#include "ui/views/controls/image_view.h" -#include "ui/views/controls/label.h" -#include "ui/views/controls/menu/menu_runner.h" -#include "ui/views/focus/focus_manager.h" -#include "ui/views/style/typography.h" - -namespace ash { - -namespace { - -// The width of the focus ring. -constexpr int kSelectionRingWidth = 2; - -constexpr int kSearchTileWidth = 80; -constexpr int kSearchTileTopPadding = 4; -constexpr int kSearchTitleSpacing = 7; -constexpr int kSearchPriceSize = 37; -constexpr int kSearchRatingSize = 26; -constexpr int kSearchRatingStarSize = 12; -constexpr int kSearchRatingStarHorizontalSpacing = 1; -constexpr int kSearchRatingStarVerticalSpacing = 2; -// Text line height in the search result tile. -constexpr int kTileTextLineHeight = 16; -constexpr int kBadgeIconShadowWidth = 1; - -// Delta applied to the font size of SearchResultTile title. -constexpr int kSearchResultTileTitleTextSizeDelta = 1; - -constexpr int kIconSelectedSize = 58; -constexpr int kIconSelectedCornerRadius = 4; - -// Offset for centering star rating when there is no price. -constexpr int kSearchRatingCenteringOffset = - ((kSearchTileWidth - - (kSearchRatingSize + kSearchRatingStarHorizontalSpacing + - kSearchRatingStarSize)) / - 2); - -} // namespace - -SearchResultTileItemView::SearchResultTileItemView( - AppListViewDelegate* view_delegate) - : view_delegate_(view_delegate), - is_app_reinstall_recommendation_enabled_( - app_list_features::IsAppReinstallZeroStateEnabled()) { - SetCallback(base::BindRepeating(&SearchResultTileItemView::OnButtonPressed, - base::Unretained(this))); - SetFocusBehavior(FocusBehavior::ALWAYS); - - // When |result_| is null, the tile is invisible. Calling SetSearchResult with - // a non-null item makes the tile visible. - SetVisible(false); - - GetViewAccessibility().OverrideIsLeaf(true); - - // Prevent the icon view from interfering with our mouse events. - icon_ = AddChildView(std::make_unique<views::ImageView>()); - icon_->SetCanProcessEventsWithinSubtree(false); - icon_->SetVerticalAlignment(views::ImageView::Alignment::kLeading); - - badge_ = AddChildView(std::make_unique<views::ImageView>()); - badge_->SetCanProcessEventsWithinSubtree(false); - badge_->SetVerticalAlignment(views::ImageView::Alignment::kLeading); - badge_->SetVisible(false); - - title_ = AddChildView(std::make_unique<views::Label>()); - title_->SetEnabledColorId(cros_tokens::kTextColorSecondary); - title_->SetAutoColorReadabilityEnabled(false); - title_->SetLineHeight(kTileTextLineHeight); - title_->SetHorizontalAlignment(gfx::ALIGN_CENTER); - title_->SetHandlesTooltips(false); - title_->SetAllowCharacterBreak(true); - - rating_ = AddChildView(std::make_unique<views::Label>()); - rating_->SetLineHeight(kTileTextLineHeight); - rating_->SetHorizontalAlignment(gfx::ALIGN_RIGHT); - rating_->SetVisible(false); - - rating_star_ = AddChildView(std::make_unique<views::ImageView>()); - rating_star_->SetCanProcessEventsWithinSubtree(false); - rating_star_->SetVerticalAlignment(views::ImageView::Alignment::kLeading); - rating_star_->SetVisible(false); - - price_ = AddChildView(std::make_unique<views::Label>()); - price_->SetEnabledColorId(cros_tokens::kTextColorSecondary); - price_->SetLineHeight(kTileTextLineHeight); - price_->SetHorizontalAlignment(gfx::ALIGN_LEFT); - price_->SetVisible(false); - - set_context_menu_controller(this); -} - -SearchResultTileItemView::~SearchResultTileItemView() = default; - -void SearchResultTileItemView::OnResultChanged() { - // Handle the case where this may be called from a nested run loop while its - // context menu is showing. This cancels the menu (it's for the old item). - context_menu_.reset(); - - SetVisible(!!result()); - - if (!result()) - return; - - SetTitle(result()->title()); - SetTitleTags(result()->title_tags()); - SetRating(result()->rating()); - SetPrice(result()->formatted_price()); - - const gfx::FontList& font = - SharedAppListConfig::instance().search_result_recommendation_title_font(); - if (rating_) { - if (!IsSuggestedAppTile()) { - // App search results use different fonts than AppList apps. - rating_->SetFontList(ui::ResourceBundle::GetSharedInstance().GetFontList( - SharedAppListConfig::instance().search_result_title_font_style())); - } else { - rating_->SetFontList(font); - } - } - if (price_) { - if (!IsSuggestedAppTile()) { - // App search results use different fonts than AppList apps. - price_->SetFontList(ui::ResourceBundle::GetSharedInstance().GetFontList( - SharedAppListConfig::instance().search_result_title_font_style())); - } else { - price_->SetFontList(font); - } - } - if (!IsSuggestedAppTile()) { - // App search results use different fonts than AppList apps. - title_->SetFontList( - ui::ResourceBundle::GetSharedInstance() - .GetFontList(SharedAppListConfig::instance() - .search_result_title_font_style()) - .DeriveWithSizeDelta(kSearchResultTileTitleTextSizeDelta)); - } else { - title_->SetFontList(font); - } - title_->SetEnabledColorId(cros_tokens::kTextColorPrimary); - title_->SetMaxLines(2); - title_->SetMultiLine( - (result()->display_type() == SearchResultDisplayType::kTile || - IsSuggestedAppTile()) && - (result()->result_type() == AppListSearchResultType::kInstalledApp || - result()->result_type() == AppListSearchResultType::kArcAppShortcut)); - - // If the new icon is null, it's being decoded asynchronously. Not updating it - // now to prevent flickering from showing an empty icon while decoding. - if (!result()->icon().icon.isNull()) - OnMetadataChanged(); - - UpdateAccessibleName(); -} - -void SearchResultTileItemView::SetParentBackgroundColor(SkColor color) { - parent_background_color_ = color; - UpdateBackgroundColor(); -} - -void SearchResultTileItemView::GetAccessibleNodeData( - ui::AXNodeData* node_data) { - views::Button::GetAccessibleNodeData(node_data); - - // The tile is a list item in the search result page's result list. - node_data->role = ax::mojom::Role::kListBoxOption; - node_data->SetDefaultActionVerb(ax::mojom::DefaultActionVerb::kClick); - - // Specify |ax::mojom::StringAttribute::kDescription| with an empty string, so - // that long truncated names are not read twice. Details of this issue: - The - // Play Store app's name is shown in a label |title_|. - If the name is too - // long, it'll get truncated and the full name will - // go to the label's tooltip. - // - SearchResultTileItemView uses that label's tooltip as its tooltip. - // - If a view doesn't have |ax::mojom::StringAttribute::kDescription| defined - // in the - // |AXNodeData|, |AXViewObjWrapper::Serialize| will use the tooltip text - // as its description. - // - We're customizing this view's accessible name, so it get focused - // ChromeVox will read its accessible name and then its description. - node_data->AddStringAttribute(ax::mojom::StringAttribute::kDescription, ""); -} - -bool SearchResultTileItemView::OnKeyPressed(const ui::KeyEvent& event) { - // Return early if |result()| was deleted due to the search result list - // changing. see crbug.com/801142 - if (!result()) - return true; - - if (event.key_code() == ui::VKEY_RETURN) { - ActivateResult(event.flags(), false /* by_button_press */); - return true; - } - return false; -} - -void SearchResultTileItemView::StateChanged(ButtonState old_state) { - SearchResultBaseView::StateChanged(old_state); - UpdateBackgroundColor(); -} - -void SearchResultTileItemView::PaintButtonContents(gfx::Canvas* canvas) { - if (!result() || !selected()) - return; - - cc::PaintFlags flags; - flags.setAntiAlias(true); - flags.setStyle(cc::PaintFlags::kStroke_Style); - flags.setStrokeWidth(kSelectionRingWidth); - flags.setColor(AppListColorProvider::Get()->GetFocusRingColor(GetWidget())); - - gfx::RectF selection_ring = GetSelectionRingBounds(); - selection_ring.Inset(gfx::InsetsF::VH(kSelectionRingWidth / 2.0, 0)); - canvas->DrawRoundRect(selection_ring, kIconSelectedCornerRadius, flags); -} - -void SearchResultTileItemView::OnThemeChanged() { - SearchResultBaseView::OnThemeChanged(); - rating_star_->SetImage(gfx::CreateVectorIcon( - kBadgeRatingIcon, kSearchRatingStarSize, - AppListColorProvider::Get()->GetSearchBoxSecondaryTextColor( - gfx::kGoogleGrey700, GetWidget()))); -} - -gfx::RectF SearchResultTileItemView::GetSelectionRingBounds() const { - gfx::RectF bounds(GetContentsBounds()); - const float horizontal_padding = (bounds.width() - kIconSelectedSize) / 2.0; - bounds.Inset(gfx::InsetsF::VH(0, horizontal_padding)); - bounds.set_height(kIconSelectedSize); - return bounds; -} - -void SearchResultTileItemView::OnMetadataChanged() { - SetIcon(result()->icon().icon); - SetTitle(result()->title()); - SetTitleTags(result()->title_tags()); - SetBadgeIcon(result()->badge_icon(), result()->use_badge_icon_background()); - SetRating(result()->rating()); - SetPrice(result()->formatted_price()); - Layout(); -} - -void SearchResultTileItemView::ShowContextMenuForViewImpl( - views::View* source, - const gfx::Point& point, - ui::MenuSourceType source_type) { - // |result()| could be null when result list is changing. - if (!result()) - return; - - view_delegate_->GetSearchResultContextMenuModel( - result()->id(), - base::BindOnce(&SearchResultTileItemView::OnGetContextMenuModel, - weak_ptr_factory_.GetWeakPtr(), source, point, - source_type)); -} - -void SearchResultTileItemView::OnGetContextMenuModel( - views::View* source, - const gfx::Point& point, - ui::MenuSourceType source_type, - std::unique_ptr<ui::SimpleMenuModel> menu_model) { - if (!menu_model || (context_menu_ && context_menu_->IsShowingMenu())) - return; - - // Anchor the menu to the same rect that is used for selection ring. - gfx::Rect anchor_rect = gfx::ToEnclosingRect(GetSelectionRingBounds()); - views::View::ConvertRectToScreen(this, &anchor_rect); - - AppLaunchedMetricParams metric_params( - AppListLaunchedFrom::kLaunchedFromSearchBox, - AppListLaunchType::kAppSearchResult); - view_delegate_->GetAppLaunchedMetricParams(&metric_params); - - context_menu_ = std::make_unique<AppListMenuModelAdapter>( - result()->id(), std::move(menu_model), GetWidget(), source_type, - metric_params, GetAppType(), - base::BindOnce(&SearchResultTileItemView::OnMenuClosed, - weak_ptr_factory_.GetWeakPtr()), - view_delegate_->IsInTabletMode()); - context_menu_->Run(anchor_rect, views::MenuAnchorPosition::kBubbleRight, - views::MenuRunner::HAS_MNEMONICS | - views::MenuRunner::USE_ASH_SYS_UI_LAYOUT | - views::MenuRunner::CONTEXT_MENU | - views::MenuRunner::FIXED_ANCHOR); - if (!selected()) { - selected_for_context_menu_ = true; - SetSelected(true, absl::nullopt); - } -} - -void SearchResultTileItemView::OnMenuClosed() { - // Release menu since its menu model delegate (AppContextMenu) could be - // released as a result of menu command execution. - context_menu_.reset(); - if (selected_for_context_menu_) { - selected_for_context_menu_ = false; - SetSelected(false, absl::nullopt); - } -} - -void SearchResultTileItemView::OnButtonPressed(const ui::Event& event) { - ActivateResult(event.flags(), true /* by_button_press */); -} - -void SearchResultTileItemView::ActivateResult(int event_flags, - bool by_button_press) { - const bool launch_as_default = is_default_result() && !by_button_press; - if (result()->result_type() == AppListSearchResultType::kPlayStoreApp) { - const base::TimeDelta activation_delay = - base::TimeTicks::Now() - result_display_start_time(); - UMA_HISTOGRAM_MEDIUM_TIMES("Arc.PlayStoreSearch.ResultClickLatency", - activation_delay); - UMA_HISTOGRAM_EXACT_LINEAR( - "Apps.AppListPlayStoreAppLaunchedIndex", - group_index_in_container_view(), - SharedAppListConfig::instance().max_search_result_tiles()); - if (launch_as_default) { - UMA_HISTOGRAM_MEDIUM_TIMES( - "Arc.PlayStoreSearch.DefaultResultClickLatency", activation_delay); - } - } - - LogAppLaunchForSuggestedApp(); - - RecordSearchResultOpenSource(result(), view_delegate_->GetAppListViewState(), - view_delegate_->IsInTabletMode()); - view_delegate_->OpenSearchResult(result()->id(), event_flags, - AppListLaunchedFrom::kLaunchedFromSearchBox, - AppListLaunchType::kAppSearchResult, - index_in_container(), launch_as_default); -} - -void SearchResultTileItemView::SetIcon(const gfx::ImageSkia& icon) { - const int icon_size = - SharedAppListConfig::instance().search_tile_icon_dimension(); - gfx::ImageSkia resized(gfx::ImageSkiaOperations::CreateResizedImage( - icon, skia::ImageOperations::RESIZE_BEST, - gfx::Size(icon_size, icon_size))); - icon_->SetImage(resized); -} - -void SearchResultTileItemView::SetBadgeIcon(const ui::ImageModel& badge_icon, - bool use_badge_icon_background) { - if (badge_icon.IsEmpty()) { - badge_->SetVisible(false); - return; - } - - gfx::ImageSkia badge_icon_skia = badge_icon.Rasterize(GetColorProvider()); - - if (use_badge_icon_background) { - badge_icon_skia = CreateIconWithCircleBackground(badge_icon_skia); - } - - gfx::ImageSkia resized_badge_icon( - gfx::ImageSkiaOperations::CreateResizedImage( - badge_icon_skia, skia::ImageOperations::RESIZE_BEST, - SharedAppListConfig::instance().search_tile_badge_icon_size())); - - gfx::ShadowValues shadow_values; - shadow_values.push_back( - gfx::ShadowValue(gfx::Vector2d(0, kBadgeIconShadowWidth), 0, - SkColorSetARGB(0x33, 0, 0, 0))); - shadow_values.push_back( - gfx::ShadowValue(gfx::Vector2d(0, kBadgeIconShadowWidth), 2, - SkColorSetARGB(0x33, 0, 0, 0))); - badge_->SetImage(gfx::ImageSkiaOperations::CreateImageWithDropShadow( - resized_badge_icon, shadow_values)); - badge_->SetVisible(true); -} - -void SearchResultTileItemView::SetTitle(const std::u16string& title) { - title_->SetText(title); -} - -void SearchResultTileItemView::SetTitleTags(const SearchResultTags& tags) { - for (const auto& tag : tags) { - if (tag.styles & SearchResult::Tag::MATCH) { - title_->SetTextStyleRange(views::style::STYLE_EMPHASIZED, tag.range); - } - } -} - -void SearchResultTileItemView::SetRating(float rating) { - if (!rating_) - return; - - if (rating < 0) { - rating_->SetVisible(false); - rating_star_->SetVisible(false); - return; - } - - rating_->SetText(base::FormatDouble(rating, 1)); - rating_->SetVisible(true); - rating_star_->SetVisible(true); -} - -void SearchResultTileItemView::SetPrice(const std::u16string& price) { - if (!price_) - return; - - if (price.empty()) { - price_->SetVisible(false); - return; - } - - price_->SetText(price); - price_->SetVisible(true); -} - -AppListMenuModelAdapter::AppListViewAppType -SearchResultTileItemView::GetAppType() const { - if (IsSuggestedAppTile()) - return AppListMenuModelAdapter::FULLSCREEN_SUGGESTED; - - if (view_delegate_->GetAppListViewState() == - AppListViewState::kFullscreenSearch) { - return AppListMenuModelAdapter::FULLSCREEN_SEARCH_RESULT; - } - - NOTREACHED(); - return AppListMenuModelAdapter::APP_LIST_APP_TYPE_LAST; -} - -bool SearchResultTileItemView::IsSuggestedAppTile() const { - return result() && result()->is_recommendation(); -} - -void SearchResultTileItemView::LogAppLaunchForSuggestedApp() const { - // Only log the app launch if the class is being used as a suggested app. - if (!IsSuggestedAppTile()) - return; - - // We only need to record opening the installed app in zero state, no need to - // record the opening of a fast re-installed app, since the latter is already - // recorded in ArcAppReinstallAppResult::Open. - if (result()->result_type() != - AppListSearchResultType::kPlayStoreReinstallApp) { - base::RecordAction( - base::UserMetricsAction("AppList_ZeroStateOpenInstalledApp")); - } -} - -void SearchResultTileItemView::UpdateBackgroundColor() { - // Tell the label what color it will be drawn onto. It will use whether the - // background color is opaque or transparent to decide whether to use subpixel - // rendering. Does not actually set the label's background color. - title_->SetBackgroundColor(parent_background_color_); - SchedulePaint(); -} - -void SearchResultTileItemView::Layout() { - gfx::Rect rect(GetContentsBounds()); - if (rect.IsEmpty() || !result()) - return; - - gfx::Rect icon_rect(rect); - icon_rect.ClampToCenteredSize(icon_->GetImage().size()); - icon_rect.set_y(kSearchTileTopPadding); - icon_->SetBoundsRect(icon_rect); - - const int badge_icon_dimension = - SharedAppListConfig::instance().search_tile_badge_icon_dimension() + - 2 * kBadgeIconShadowWidth; - const int badge_icon_offset = - SharedAppListConfig::instance().search_tile_badge_icon_offset(); - const gfx::Rect badge_rect( - icon_rect.right() - badge_icon_dimension + badge_icon_offset, - icon_rect.bottom() - badge_icon_dimension + badge_icon_offset, - badge_icon_dimension, badge_icon_dimension); - badge_->SetBoundsRect(badge_rect); - - rect.set_y(icon_rect.bottom() + kSearchTitleSpacing); - rect.set_height(title_->GetPreferredSize().height()); - title_->SetBoundsRect(rect); - - // If there is no price set, we center the rating. - const bool center_rating = - rating_ && rating_star_ && price_ && price_->GetText().empty(); - const int rating_horizontal_offset = - center_rating ? kSearchRatingCenteringOffset : 0; - - if (rating_) { - gfx::Rect rating_rect(rect); - rating_rect.Inset(gfx::Insets::TLBR(title_->GetPreferredSize().height(), - rating_horizontal_offset, 0, 0)); - rating_rect.set_size(rating_->GetPreferredSize()); - rating_rect.set_width(kSearchRatingSize); - rating_->SetBoundsRect(rating_rect); - } - - if (rating_star_) { - gfx::Rect rating_star_rect(rect); - rating_star_rect.Inset(gfx::Insets::TLBR( - title_->GetPreferredSize().height() + kSearchRatingStarVerticalSpacing, - rating_horizontal_offset + kSearchRatingSize + - kSearchRatingStarHorizontalSpacing, - 0, 0)); - rating_star_rect.set_size(rating_star_->GetPreferredSize()); - rating_star_->SetBoundsRect(rating_star_rect); - } - - if (price_) { - gfx::Rect price_rect(rect); - price_rect.Inset(gfx::Insets::TLBR(title_->GetPreferredSize().height(), - rect.width() - kSearchPriceSize, 0, 0)); - price_rect.set_size(price_->GetPreferredSize()); - price_->SetBoundsRect(price_rect); - } -} - -const char* SearchResultTileItemView::GetClassName() const { - return "SearchResultTileItemView"; -} - -gfx::Size SearchResultTileItemView::CalculatePreferredSize() const { - if (!result()) - return gfx::Size(); - - return gfx::Size(kSearchTileWidth, - SharedAppListConfig::instance().search_tile_height()); -} - -std::u16string SearchResultTileItemView::GetTooltipText( - const gfx::Point& p) const { - // Use the label to generate a tooltip, so that it will consider its text - // truncation in making the tooltip. We do not want the label itself to have a - // tooltip, so we only temporarily enable it to get the tooltip text from the - // label, then disable it again. - title_->SetHandlesTooltips(true); - std::u16string tooltip = title_->GetTooltipText(p); - title_->SetHandlesTooltips(false); - return tooltip; -} - -} // namespace ash
diff --git a/ash/app_list/views/search_result_tile_item_view.h b/ash/app_list/views/search_result_tile_item_view.h deleted file mode 100644 index 3930629..0000000 --- a/ash/app_list/views/search_result_tile_item_view.h +++ /dev/null
@@ -1,146 +0,0 @@ -// Copyright 2014 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#ifndef ASH_APP_LIST_VIEWS_SEARCH_RESULT_TILE_ITEM_VIEW_H_ -#define ASH_APP_LIST_VIEWS_SEARCH_RESULT_TILE_ITEM_VIEW_H_ - -#include <memory> -#include <vector> - -#include "ash/app_list/views/app_list_menu_model_adapter.h" -#include "ash/app_list/views/search_result_base_view.h" -#include "ash/ash_export.h" -#include "ash/public/cpp/app_list/app_list_types.h" -#include "ui/views/context_menu_controller.h" - -namespace ui { -class ImageModel; -} // namespace ui - -namespace views { -class ImageView; -class Label; -} // namespace views - -namespace ash { - -class AppListViewDelegate; -class SearchResult; - -// A tile view that displays a search result. It hosts view for search result -// that has SearchResult::DisplayType DISPLAY_TILE or DISPLAY_RECOMMENDATION. -class ASH_EXPORT SearchResultTileItemView - : public SearchResultBaseView, - public views::ContextMenuController { - public: - explicit SearchResultTileItemView(AppListViewDelegate* view_delegate); - - SearchResultTileItemView(const SearchResultTileItemView&) = delete; - SearchResultTileItemView& operator=(const SearchResultTileItemView&) = delete; - - ~SearchResultTileItemView() override; - - void OnResultChanged() override; - - // Informs the SearchResultTileItemView of its parent's background color. The - // controls within the SearchResultTileItemView will adapt to suit the given - // color. - void SetParentBackgroundColor(SkColor color); - - void set_group_index_in_container_view(int index) { - group_index_in_container_view_ = index; - } - int group_index_in_container_view() const { - return group_index_in_container_view_; - } - - // Overridden from views::Button: - void GetAccessibleNodeData(ui::AXNodeData* node_data) override; - bool OnKeyPressed(const ui::KeyEvent& event) override; - void StateChanged(ButtonState old_state) override; - void PaintButtonContents(gfx::Canvas* canvas) override; - void OnThemeChanged() override; - - // Overridden from SearchResultObserver: - void OnMetadataChanged() override; - - // views::ContextMenuController overrides: - void ShowContextMenuForViewImpl(views::View* source, - const gfx::Point& point, - ui::MenuSourceType source_type) override; - - private: - // Launch the result and log to various histograms. - // |by_button_press|: True if |result_| is activated by button pressing; - // otherwise |result| is activated by ENTER key pressing. - void ActivateResult(int event_flags, bool by_button_press); - - // Bound by ShowContextMenuForViewImpl(). - void OnGetContextMenuModel(views::View* source, - const gfx::Point& point, - ui::MenuSourceType source_type, - std::unique_ptr<ui::SimpleMenuModel> menu_model); - - // The callback used when a menu closes. - void OnMenuClosed(); - - void OnButtonPressed(const ui::Event& event); - - void SetIcon(const gfx::ImageSkia& icon); - void SetBadgeIcon(const ui::ImageModel& badge_icon, - bool use_badge_icon_background); - void SetTitle(const std::u16string& title); - void SetTitleTags(const SearchResultTags& tags); - void SetRating(float rating); - void SetPrice(const std::u16string& price); - - AppListMenuModelAdapter::AppListViewAppType GetAppType() const; - - // Whether the tile view is a suggested app. - bool IsSuggestedAppTile() const; - - // Records an app being launched. - void LogAppLaunchForSuggestedApp() const; - - void UpdateBackgroundColor(); - - // Gets the bounds for the selection ring (shown when the result is selected). - gfx::RectF GetSelectionRingBounds() const; - - // Overridden from views::View: - void Layout() override; - const char* GetClassName() const override; - gfx::Size CalculatePreferredSize() const override; - std::u16string GetTooltipText(const gfx::Point& p) const override; - - AppListViewDelegate* const view_delegate_; // Owned by AppListView. - - views::ImageView* icon_ = nullptr; // Owned by views hierarchy. - views::ImageView* badge_ = nullptr; // Owned by views hierarchy. - views::Label* title_ = nullptr; // Owned by views hierarchy. - views::Label* rating_ = nullptr; // Owned by views hierarchy. - views::Label* price_ = nullptr; // Owned by views hierarchy. - views::ImageView* rating_star_ = nullptr; // Owned by views hierarchy. - - SkColor parent_background_color_ = SK_ColorTRANSPARENT; - - // The index of the app in its display group in its container view. Currently, - // there are three separately displayed groups for apps in launcher's - // suggestion window: Installed apps, play store apps, play store reinstalled - // app. - int group_index_in_container_view_; - const bool is_app_reinstall_recommendation_enabled_; - - // Whether the result view moved into selected state only because a context - // menu was shown. - bool selected_for_context_menu_ = false; - - std::unique_ptr<AppListMenuModelAdapter> context_menu_; - - base::WeakPtrFactory<SearchResultTileItemView> weak_ptr_factory_{this}; -}; - -} // namespace ash - -#endif // ASH_APP_LIST_VIEWS_SEARCH_RESULT_TILE_ITEM_VIEW_H_
diff --git a/ash/ash_strings.grd b/ash/ash_strings.grd index c53c7bb5..2a2ddde 100644 --- a/ash/ash_strings.grd +++ b/ash/ash_strings.grd
@@ -2,6 +2,15 @@ <!-- This file contains the strings for ash. + +Style notes: +* Most strings are "Sentence case" with the first word capitalized and other + words not capitalized +* Specific features may be capitalized (e.g. "Bluetooth", "Nearby Share") even + if they appear in the middle of a string +* Most strings do not end in a period (e.g. "Show settings") +* Multi-phrase strings that contain a period in the middle also have a period + at the end (e.g. "Show Bluetooth settings. Bluetooth is on.") --> <grit base_dir="." latest_public_release="0" current_release="1" @@ -427,10 +436,10 @@ <ph name="SECONDS">$1<ex>40</ex></ph> sec </message> <message name="IDS_ASH_STATUS_TRAY_NEARBY_SHARE_SETTINGS_TOOLTIP" desc="Tooltip text for the status tray button that shows settings for the Nearby Share feature."> - Show Nearby Share settings. + Show Nearby Share settings </message> <message name="IDS_ASH_STATUS_TRAY_NEARBY_SHARE_TOGGLE_TOOLTIP" desc="Tooltip text used for the Nearby Share status tray button, which toggles Nearby Share high visibility mode on/off."> - Toggle Nearby Share high visibility. + Toggle Nearby Share high visibility </message> <message name="IDS_ASH_STATUS_TRAY_NIGHT_LIGHT_BUTTON_LABEL" desc="The shorter label used for the button in the status tray to toggle the Night Light feature (which controls the color temperature of the screen) on or off. [CHAR_LIMIT=14]"> Night Light @@ -472,7 +481,7 @@ Cast </message> <message name="IDS_ASH_STATUS_TRAY_CAST_TOOLTIP" desc="The tooltip text used for Cast button in system tray bubble."> - Show cast devices. + Show cast devices </message> <message name="IDS_ASH_STATUS_TRAY_CAST_NOTIFICATION_TITLE" desc="The notification title to tell the user we are casting to a cast device."> Casting to <ph name="RECEIVER_NAME">$1<ex>Living Room</ex></ph> @@ -487,7 +496,7 @@ Dark theme </message> <message name="IDS_ASH_STATUS_TRAY_DARK_THEME_TOGGLE_TOOLTIP" desc="The tooltip text used for the button in the status tray to toggle the Dark theme feature on or off."> - Toggle Dark theme. <ph name="STATE_TEXT">$1<ex>Dark theme is on.</ex></ph> + Toggle Dark theme. <ph name="STATE_TEXT">$1<ex>Dark theme is on</ex></ph>. </message> <message name="IDS_ASH_STATUS_TRAY_DARK_THEME_ENABLED_STATE_TOOLTIP" desc="The tooltip text indicating the Dark theme feature is on."> Dark theme is on @@ -578,10 +587,10 @@ Bluetooth </message> <message name="IDS_ASH_STATUS_TRAY_BLUETOOTH_TOGGLE_TOOLTIP" desc="The tooltip text in the button that toggles Blueooth enabled/disabled."> - Toggle Bluetooth. <ph name="STATE_TEXT">$1<ex>Connected to 2 devices</ex></ph> + Toggle Bluetooth. <ph name="STATE_TEXT">$1<ex>Connected to 2 devices</ex></ph>. </message> <message name="IDS_ASH_STATUS_TRAY_BLUETOOTH_SETTINGS_TOOLTIP" desc="The tooltip text in the button that shows Blueooth settings."> - Show Bluetooth settings. <ph name="STATE_TEXT">$1<ex>Connected to 2 devices</ex></ph> + Show Bluetooth settings. <ph name="STATE_TEXT">$1<ex>Connected to 2 devices</ex></ph>. </message> <message name="IDS_ASH_STATUS_TRAY_BLUETOOTH_ENABLED_TOOLTIP" desc="The tooltip text to notify that bluetooth is enabled."> Bluetooth is on @@ -949,7 +958,7 @@ Show keyboard settings </message> <message name="IDS_ASH_STATUS_TRAY_IME_TOOLTIP_WITH_NAME" desc="The tooltip text used for IME button in system tray bubble."> - Show keyboard settings. <ph name="KEYBOARD_NAME">$1<ex>US keyboard</ex></ph> is selected + Show keyboard settings. <ph name="KEYBOARD_NAME">$1<ex>US keyboard</ex></ph> is selected. </message> <message name="IDS_ASH_STATUS_TRAY_IME" desc="The label used as the header in the IME popup. [CHAR_LIMIT=18]"> Input methods @@ -2423,10 +2432,10 @@ </message> <message name="IDS_ASH_STATUS_TRAY_NETWORK_TOGGLE_TOOLTIP" desc="The tooltip text for the button that toggles network enabled/disabled state."> - Toggle network connection. <ph name="STATE_TEXT">$1<ex>Connected to public wifi</ex></ph> + Toggle network connection. <ph name="STATE_TEXT">$1<ex>Connected to public wifi</ex></ph>. </message> <message name="IDS_ASH_STATUS_TRAY_NETWORK_SETTINGS_TOOLTIP" desc="The tooltip text for the button that shows network list."> - Show network list. <ph name="STATE_TEXT">$1<ex>Connected to public wifi</ex></ph> + Show network list. <ph name="STATE_TEXT">$1<ex>Connected to public wifi</ex></ph>. </message> <message name="IDS_ASH_STATUS_TRAY_NETWORK_PROGRESS_ACCESSIBLE_NAME" desc="The accessible name for the progress bar shown in the network settings."> Scanning
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_BLUETOOTH_SETTINGS_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_BLUETOOTH_SETTINGS_TOOLTIP.png.sha1 new file mode 100644 index 0000000..85a08e3 --- /dev/null +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_BLUETOOTH_SETTINGS_TOOLTIP.png.sha1
@@ -0,0 +1 @@ +14506683f4860247eb998fefef35e68d8d5ba681 \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_BLUETOOTH_TOGGLE_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_BLUETOOTH_TOGGLE_TOOLTIP.png.sha1 new file mode 100644 index 0000000..0b4171ee --- /dev/null +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_BLUETOOTH_TOGGLE_TOOLTIP.png.sha1
@@ -0,0 +1 @@ +5e34a89e2a19ab9c3309ecb3db36a2fcc7caafb1 \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_CAST_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_CAST_TOOLTIP.png.sha1 new file mode 100644 index 0000000..fd8132f --- /dev/null +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_CAST_TOOLTIP.png.sha1
@@ -0,0 +1 @@ +0122e9b5756ac230cb7b723e042098776c119cf6 \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_DARK_THEME_TOGGLE_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_DARK_THEME_TOGGLE_TOOLTIP.png.sha1 index 3b00f1d..a5d652c 100644 --- a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_DARK_THEME_TOGGLE_TOOLTIP.png.sha1 +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_DARK_THEME_TOGGLE_TOOLTIP.png.sha1
@@ -1 +1 @@ -a3cbce9b0bd7654915d9ea33f1b1b0a151fc3388 \ No newline at end of file +24e630801e46d8a1e2adb054186bef9cf2619281 \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_IME_TOOLTIP_WITH_NAME.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_IME_TOOLTIP_WITH_NAME.png.sha1 new file mode 100644 index 0000000..9c55b4ea --- /dev/null +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_IME_TOOLTIP_WITH_NAME.png.sha1
@@ -0,0 +1 @@ +fe626ac766446f602d894e35d882a5558d81cda3 \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_SETTINGS_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_SETTINGS_TOOLTIP.png.sha1 index e264bcc..c71f6cc 100644 --- a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_SETTINGS_TOOLTIP.png.sha1 +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_SETTINGS_TOOLTIP.png.sha1
@@ -1 +1 @@ -14fa83bd6816e26df5e9ac72cbceb09818990031 \ No newline at end of file +c0e2a814eab217357d2fe53af41a110f70276f2a \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_TOGGLE_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_TOGGLE_TOOLTIP.png.sha1 index 3371c2cd..0253090 100644 --- a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_TOGGLE_TOOLTIP.png.sha1 +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NEARBY_SHARE_TOGGLE_TOOLTIP.png.sha1
@@ -1 +1 @@ -14577c7eda36f1ee897e9d6974c02c969d478199 \ No newline at end of file +7992b9ffd8afbe69bbe52a87329d14b41537c2ac \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NETWORK_SETTINGS_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NETWORK_SETTINGS_TOOLTIP.png.sha1 new file mode 100644 index 0000000..0d52da5 --- /dev/null +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NETWORK_SETTINGS_TOOLTIP.png.sha1
@@ -0,0 +1 @@ +b18e7c706cfa38cd962fe35e810a1fc7ed7df4b6 \ No newline at end of file
diff --git a/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NETWORK_TOGGLE_TOOLTIP.png.sha1 b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NETWORK_TOGGLE_TOOLTIP.png.sha1 new file mode 100644 index 0000000..c90d6e4 --- /dev/null +++ b/ash/ash_strings_grd/IDS_ASH_STATUS_TRAY_NETWORK_TOGGLE_TOOLTIP.png.sha1
@@ -0,0 +1 @@ +f1346ee4f1f4af37d07953ffff206e5d72118225 \ No newline at end of file
diff --git a/ash/metrics/login_unlock_throughput_recorder.cc b/ash/metrics/login_unlock_throughput_recorder.cc index 28520c2..76b25081 100644 --- a/ash/metrics/login_unlock_throughput_recorder.cc +++ b/ash/metrics/login_unlock_throughput_recorder.cc
@@ -5,67 +5,20 @@ #include "ash/metrics/login_unlock_throughput_recorder.h" #include "ash/public/cpp/metrics_util.h" -#include "ash/public/cpp/shelf_model.h" #include "ash/session/session_controller_impl.h" -#include "ash/shelf/shelf_view.h" #include "ash/shell.h" #include "ash/wm/tablet_mode/tablet_mode_controller.h" #include "base/bind.h" #include "base/metrics/histogram_functions.h" -#include "base/metrics/histogram_macros.h" -#include "base/strings/utf_string_conversions.h" #include "chromeos/login/login_state/login_state.h" #include "chromeos/metrics/login_event_recorder.h" -#include "components/app_constants/constants.h" -#include "components/app_restore/window_properties.h" #include "ui/aura/window.h" #include "ui/aura/window_tree_host.h" -#include "ui/compositor/compositor.h" -#include "ui/compositor/layer.h" #include "ui/compositor/total_animation_throughput_reporter.h" -#include "ui/views/animation/bounds_animator.h" -#include "ui/views/animation/bounds_animator_observer.h" namespace ash { namespace { -// A class used to wait for animations. -class AnimationObserver : public views::BoundsAnimatorObserver { - public: - AnimationObserver(ShelfView* shelf_view, base::OnceClosure& on_animation_end) - : shelf_view_(shelf_view), - on_animation_end_(std::move(on_animation_end)) {} - - AnimationObserver(const AnimationObserver&) = delete; - AnimationObserver& operator=(const AnimationObserver&) = delete; - - ~AnimationObserver() override = default; - - // ShelfViewObserver overrides: - void OnBoundsAnimatorProgressed(views::BoundsAnimator* animator) override {} - void OnBoundsAnimatorDone(views::BoundsAnimator* animator) override { - shelf_view_->RemoveAnimationObserver(this); - RunCallbackAndDestroy(); - } - - void StartObserving() { - if (shelf_view_->IsAnimating()) { - shelf_view_->AddAnimationObserver(this); - return; - } - RunCallbackAndDestroy(); - } - - private: - void RunCallbackAndDestroy() { - std::move(on_animation_end_).Run(); - delete this; - } - - base::raw_ptr<ShelfView> shelf_view_; - base::OnceClosure on_animation_end_; -}; - std::string GetDeviceModeSuffix() { return Shell::Get()->tablet_mode_controller()->InTabletMode() ? "TabletMode" @@ -104,7 +57,6 @@ LOG(WARNING) << "Zero frames expected in login animation throughput data"; return; } - chromeos::LoginEventRecorder::Get()->AddLoginTimeMarker( "LoginAnimationEnd", /*send_to_uma=*/false, @@ -124,14 +76,6 @@ "Ash.UnlockAnimation.Jank.", "Ash.UnlockAnimation.Duration."); } -void OnRestoredWindowPresentationFeedbackReceived( - int restore_window_id, - const gfx::PresentationFeedback& feedback) { - LoginUnlockThroughputRecorder* throughput_recorder = - Shell::Get()->login_unlock_throughput_recorder(); - throughput_recorder->OnRestoredWindowPresented(restore_window_id); -} - } // namespace LoginUnlockThroughputRecorder::LoginUnlockThroughputRecorder() { @@ -154,160 +98,24 @@ new ui::TotalAnimationThroughputReporter( primary_root->GetHost()->compositor(), base::BindOnce(&ReportUnlock, base::TimeTicks::Now()), - /*should_delete=*/true); + /*self_destruct=*/true); } } void LoginUnlockThroughputRecorder::LoggedInStateChanged() { auto* login_state = chromeos::LoginState::Get(); auto logged_in_user = login_state->GetLoggedInUserType(); - - if (user_logged_in_) - return; - - if (!login_state->IsUserLoggedIn()) - return; - - if (logged_in_user != chromeos::LoginState::LOGGED_IN_USER_OWNER && - logged_in_user != chromeos::LoginState::LOGGED_IN_USER_REGULAR) { - return; + if (login_state->IsUserLoggedIn() && + (logged_in_user == chromeos::LoginState::LOGGED_IN_USER_OWNER || + logged_in_user == chromeos::LoginState::LOGGED_IN_USER_REGULAR)) { + ui_recorder_.OnUserLoggedIn(); + auto* primary_root = Shell::GetPrimaryRootWindow(); + new ui::TotalAnimationThroughputReporter( + primary_root->GetHost()->compositor(), + base::BindOnce(&LoginUnlockThroughputRecorder::OnLoginAnimationFinish, + weak_ptr_factory_.GetWeakPtr(), base::TimeTicks::Now()), + /*self_destruct=*/true); } - - user_logged_in_ = true; - ui_recorder_.OnUserLoggedIn(); - auto* primary_root = Shell::GetPrimaryRootWindow(); - primary_user_logged_in_ = base::TimeTicks::Now(); - auto* rec = new ui::TotalAnimationThroughputReporter( - primary_root->GetHost()->compositor(), - base::BindOnce(&LoginUnlockThroughputRecorder::OnLoginAnimationFinish, - weak_ptr_factory_.GetWeakPtr(), base::TimeTicks::Now()), - /*should_delete=*/true); - login_animation_throughput_reporter_ = rec->GetWeakPtr(); - DCHECK(!scoped_throughput_reporter_blocker_); - // Login animation metrics should not be reported until all shelf icons - // were loaded. - scoped_throughput_reporter_blocker_ = - login_animation_throughput_reporter_->NewScopedBlocker(); -} - -void LoginUnlockThroughputRecorder::AddScheduledRestoreWindow( - int restore_window_id, - const std::string& app_id, - RestoreWindowType window_type) { - switch (window_type) { - case LoginUnlockThroughputRecorder::kBrowser: - DCHECK(restore_window_id); - if (app_id.empty() || app_id == app_constants::kLacrosAppId) - windows_to_restore_.insert(restore_window_id); - - break; - default: - NOTREACHED(); - } -} - -void LoginUnlockThroughputRecorder::OnRestoredWindowCreated( - int restore_window_id) { - auto it = windows_to_restore_.find(restore_window_id); - if (it == windows_to_restore_.end()) - return; - windows_to_restore_.erase(it); - if (windows_to_restore_.empty() && !primary_user_logged_in_.is_null()) { - const base::TimeDelta duration_ms = - base::TimeTicks::Now() - primary_user_logged_in_; - UMA_HISTOGRAM_CUSTOM_TIMES( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated", duration_ms, - base::Milliseconds(1), base::Seconds(100), 100); - } - restore_windows_not_shown_.insert(restore_window_id); -} - -void LoginUnlockThroughputRecorder::OnBeforeRestoredWindowShown( - int restore_window_id, - ui::Compositor* compositor) { - auto it = restore_windows_not_shown_.find(restore_window_id); - if (it == restore_windows_not_shown_.end()) - return; - - restore_windows_not_shown_.erase(it); - if (windows_to_restore_.empty() && restore_windows_not_shown_.empty() && - !primary_user_logged_in_.is_null()) { - const base::TimeDelta duration_ms = - base::TimeTicks::Now() - primary_user_logged_in_; - UMA_HISTOGRAM_CUSTOM_TIMES("Ash.LoginSessionRestore.AllBrowserWindowsShown", - duration_ms, base::Milliseconds(1), - base::Seconds(100), 100); - } - - if (!compositor) - return; - - restore_windows_presentation_time_requested_.insert(restore_window_id); - compositor->RequestPresentationTimeForNextFrame(base::BindOnce( - &OnRestoredWindowPresentationFeedbackReceived, restore_window_id)); -} - -void LoginUnlockThroughputRecorder::OnRestoredWindowPresented( - int restore_window_id) { - auto it = - restore_windows_presentation_time_requested_.find(restore_window_id); - if (it == restore_windows_presentation_time_requested_.end()) - return; - - restore_windows_presentation_time_requested_.erase(it); - if (windows_to_restore_.empty() && restore_windows_not_shown_.empty() && - restore_windows_presentation_time_requested_.empty() && - !primary_user_logged_in_.is_null()) { - const base::TimeDelta duration_ms = - base::TimeTicks::Now() - primary_user_logged_in_; - UMA_HISTOGRAM_CUSTOM_TIMES( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented", duration_ms, - base::Milliseconds(1), base::Seconds(100), 100); - } - restore_windows_presented_.insert(restore_window_id); -} - -void LoginUnlockThroughputRecorder::InitShelfIconList(const ShelfModel* model) { - shelf_initialized_ = true; - - // Copy shelf icons to the expected list. - for (int index = 0; index < model->item_count(); ++index) { - const ShelfID& id = model->items()[index].id; - const ShelfItem& item = model->items()[index]; - if (item.image.isNull()) - expected_shelf_icons_.insert(id); - } - - if (expected_shelf_icons_.empty()) - OnAllExpectedShelfIconsLoaded(); -} - -void LoginUnlockThroughputRecorder::UpdateShelfIconList( - const ShelfModel* model) { - if (!shelf_initialized_) - return; - - // Remove IDs that have icons loaded or were already deleted. - base::flat_set<ShelfID> expected_ids_without_icons; - - for (int index = 0; index < model->item_count(); ++index) { - const ShelfItem& item = model->items()[index]; - const ShelfID& id = item.id; - if (!expected_shelf_icons_.contains(id)) - continue; - - if (item.image.isNull()) - expected_ids_without_icons.insert(id); - } - expected_shelf_icons_ = expected_ids_without_icons; - - if (expected_shelf_icons_.empty()) - OnAllExpectedShelfIconsLoaded(); -} - -void LoginUnlockThroughputRecorder:: - ResetScopedThroughputReporterBlockerForTesting() { - scoped_throughput_reporter_blocker_.reset(); } void LoginUnlockThroughputRecorder::OnLoginAnimationFinish( @@ -317,43 +125,4 @@ ReportLogin(start, data); } -void LoginUnlockThroughputRecorder::SetShelfViewIfNotSet( - ShelfView* shelf_view) { - if (!shelf_view_) - shelf_view_ = shelf_view; -} - -void LoginUnlockThroughputRecorder::ScheduleWaitForShelfAnimationEnd() { - DCHECK(shelf_view_); - if (!shelf_view_) - return; - - base::OnceCallback on_animation_end = base::BindOnce( - [](base::TimeTicks primary_user_logged_in) { - const base::TimeDelta duration_ms = - base::TimeTicks::Now() - primary_user_logged_in; - UMA_HISTOGRAM_CUSTOM_TIMES( - "Ash.LoginSessionRestore.ShelfLoginAnimationEnd", duration_ms, - base::Milliseconds(1), base::Seconds(100), 100); - }, - primary_user_logged_in_); - - (new AnimationObserver(shelf_view_, on_animation_end))->StartObserving(); -} - -void LoginUnlockThroughputRecorder::OnAllExpectedShelfIconsLoaded() { - if (shelf_icons_loaded_) - return; - - scoped_throughput_reporter_blocker_.reset(); - - shelf_icons_loaded_ = true; - const base::TimeDelta duration_ms = - base::TimeTicks::Now() - primary_user_logged_in_; - UMA_HISTOGRAM_CUSTOM_TIMES("Ash.LoginSessionRestore.AllShelfIconsLoaded", - duration_ms, base::Milliseconds(1), - base::Seconds(100), 100); - ScheduleWaitForShelfAnimationEnd(); -} - } // namespace ash
diff --git a/ash/metrics/login_unlock_throughput_recorder.h b/ash/metrics/login_unlock_throughput_recorder.h index 8c1dd7e9..55f5bbc 100644 --- a/ash/metrics/login_unlock_throughput_recorder.h +++ b/ash/metrics/login_unlock_throughput_recorder.h
@@ -5,38 +5,20 @@ #ifndef ASH_METRICS_LOGIN_UNLOCK_THROUGHPUT_RECORDER_H_ #define ASH_METRICS_LOGIN_UNLOCK_THROUGHPUT_RECORDER_H_ -#include <string> - #include "ash/ash_export.h" #include "ash/metrics/ui_throughput_recorder.h" #include "ash/public/cpp/session/session_observer.h" -#include "ash/public/cpp/shelf_types.h" -#include "base/containers/flat_set.h" -#include "base/memory/raw_ptr.h" #include "base/memory/weak_ptr.h" #include "base/time/time.h" #include "cc/metrics/frame_sequence_metrics.h" #include "chromeos/login/login_state/login_state.h" -#include "ui/compositor/total_animation_throughput_reporter.h" - -namespace ui { -class Compositor; -} namespace ash { -class ShelfModel; -class ShelfView; - class ASH_EXPORT LoginUnlockThroughputRecorder : public SessionObserver, public chromeos::LoginState::Observer { public: - enum RestoreWindowType { - kBrowser, - kArc, - }; - LoginUnlockThroughputRecorder(); LoginUnlockThroughputRecorder(const LoginUnlockThroughputRecorder&) = delete; LoginUnlockThroughputRecorder& operator=( @@ -49,85 +31,12 @@ // chromeos::LoginState::Observer: void LoggedInStateChanged() override; - // Adds "restore_window_id" to the list of potentially restored windows. - // See - // https://source.chromium.org/chromium/chromium/src/+/main:ui/views/widget/widget.h;l=404-415. - void AddScheduledRestoreWindow(int restore_window_id, - const std::string& app_id, - RestoreWindowType window_type); - - // This is called when restored window was created. - void OnRestoredWindowCreated(int restore_window_id); - - // This is called before window is shown to request presentation feedback. - void OnBeforeRestoredWindowShown(int restore_window_id, - ui::Compositor* compositor); - - // This is called when restored window was presented. - void OnRestoredWindowPresented(int restore_window_id); - - // This is called when the list of shelf icons is initialized. - void InitShelfIconList(const ShelfModel* model); - - // This is called when the list of shelf icons is updated. - void UpdateShelfIconList(const ShelfModel* model); - - // Remembers ShelfView pointer to watch for shelf animation finish. - void SetShelfViewIfNotSet(ShelfView* shelf_view); - - void ResetScopedThroughputReporterBlockerForTesting(); - - const ui::TotalAnimationThroughputReporter* - login_animation_throughput_reporter() const { - return login_animation_throughput_reporter_.get(); - } - private: void OnLoginAnimationFinish( base::TimeTicks start, const cc::FrameSequenceMetrics::CustomReportData& data); - void ScheduleWaitForShelfAnimationEnd(); - - void OnAllExpectedShelfIconsLoaded(); - UiThroughputRecorder ui_recorder_; - - // Set of window IDs ("restore_window_id") that could be restored but - // for which windows have not been created yet. - base::flat_set<int> windows_to_restore_; - - // Set of window IDs ("restore_window_id") that were created as a part of the - // session restore but not yet shown. - base::flat_set<int> restore_windows_not_shown_; - - // Set of window IDs ("restore_window_id") that were shown and presentation - // time was requested. - base::flat_set<int> restore_windows_presentation_time_requested_; - - // Set of window IDs ("restore_window_id") for which presentation time - // was received. - base::flat_set<int> restore_windows_presented_; - - base::TimeTicks primary_user_logged_in_; - - base::raw_ptr<ShelfView> shelf_view_ = nullptr; - - bool shelf_initialized_ = false; - - bool shelf_icons_loaded_ = false; - - bool user_logged_in_ = false; - - base::WeakPtr<ui::TotalAnimationThroughputReporter> - login_animation_throughput_reporter_; - - std::unique_ptr< - ui::TotalAnimationThroughputReporter::ScopedThroughputReporterBlocker> - scoped_throughput_reporter_blocker_; - - base::flat_set<ShelfID> expected_shelf_icons_; - base::WeakPtrFactory<LoginUnlockThroughputRecorder> weak_ptr_factory_{this}; };
diff --git a/ash/metrics/login_unlock_throughput_recorder_unittest.cc b/ash/metrics/login_unlock_throughput_recorder_unittest.cc deleted file mode 100644 index 9051400..0000000 --- a/ash/metrics/login_unlock_throughput_recorder_unittest.cc +++ /dev/null
@@ -1,542 +0,0 @@ -// Copyright 2022 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -#include "ash/metrics/login_unlock_throughput_recorder.h" - -#include <memory> -#include <string> - -#include "ash/login/ui/login_test_base.h" -#include "ash/metrics/user_metrics_recorder.h" -#include "ash/public/cpp/shelf_item_delegate.h" -#include "ash/public/cpp/shelf_model.h" -#include "ash/session/test_session_controller_client.h" -#include "ash/shell.h" -#include "ash/wm/tablet_mode/tablet_mode_controller.h" -#include "base/run_loop.h" -#include "base/test/metrics/histogram_tester.h" -#include "chromeos/login/login_state/login_state.h" -#include "components/app_constants/constants.h" -#include "ui/aura/window_tree_host.h" -#include "ui/compositor/test/draw_waiter_for_test.h" -#include "ui/gfx/image/image_unittest_util.h" - -namespace ash { -namespace { - -constexpr char kAshLoginAnimationDurationTabletMode[] = - "Ash.LoginAnimation.Duration.TabletMode"; -constexpr char kAshLoginAnimationDurationClamshellMode[] = - "Ash.LoginAnimation.Duration.ClamshellMode"; -constexpr char kAshUnlockAnimationDurationTabletMode[] = - "Ash.UnlockAnimation.Duration.TabletMode"; -constexpr char kAshUnlockAnimationDurationClamshellMode[] = - "Ash.UnlockAnimation.Duration.ClamshellMode"; - -// A test shelf item delegate that simulates an activated window when a shelf -// item is selected. -class TestShelfItemDelegate : public ShelfItemDelegate { - public: - explicit TestShelfItemDelegate(const ShelfID& shelf_id) - : ShelfItemDelegate(shelf_id) {} - - // ShelfItemDelegate: - void ItemSelected(std::unique_ptr<ui::Event> event, - int64_t display_id, - ash::ShelfLaunchSource source, - ItemSelectedCallback callback, - const ItemFilterPredicate& filter_predicate) override { - std::move(callback).Run(SHELF_ACTION_WINDOW_ACTIVATED, {}); - } - void ExecuteCommand(bool from_context_menu, - int64_t command_id, - int32_t event_flags, - int64_t display_id) override {} - void Close() override {} -}; - -class TestShelfModel : public ShelfModel { - public: - TestShelfModel() = default; - TestShelfModel(const TestShelfModel&) = delete; - TestShelfModel& operator=(const TestShelfModel&) = delete; - - ~TestShelfModel() = default; - - void InitializeIconList(const std::vector<int>& ids) { - while (!items().empty()) - RemoveItemAt(0); - - for (int n : ids) { - ShelfItem item; - item.id = ShelfID(base::StringPrintf("item%d", n)); - item.type = TYPE_PINNED_APP; - Add(item, std::make_unique<TestShelfItemDelegate>(item.id)); - } - } - - void SetIconsLoadedFor(const std::vector<int>& ids) { - for (int n : ids) { - const ShelfID id(base::StringPrintf("item%d", n)); - int index = ItemIndexByID(id); - // Expect item exists. - ASSERT_GE(index, 0); - - ShelfItem item = items()[index]; - item.image = gfx::test::CreateImageSkia(/*width=*/10, /*height=*/10); - - Set(index, item); - } - } -}; - -class TestObserver final : public ui::CompositorAnimationObserver { - public: - explicit TestObserver(ui::Compositor* compositor) : compositor_(compositor) { - compositor_->AddAnimationObserver(this); - } - TestObserver(const TestObserver&) = delete; - TestObserver& operator=(const TestObserver&) = delete; - ~TestObserver() override = default; - - // ui::CompositorAnimationObserver: - void OnAnimationStep(base::TimeTicks timestamp) override { - ++count_; - if (count_ < 3) - compositor_->ScheduleFullRedraw(); - else - compositor_->RemoveAnimationObserver(this); - } - - void OnCompositingShuttingDown(ui::Compositor* compositor) override {} - - private: - int count_ = 0; - const base::raw_ptr<ui::Compositor> compositor_; -}; - -class BeginMainFrameWaiter : public ui::CompositorObserver { - public: - explicit BeginMainFrameWaiter(ui::Compositor* compositor) - : compositor_(compositor) { - compositor->AddObserver(this); - } - - ~BeginMainFrameWaiter() override { compositor_->RemoveObserver(this); } - - // ui::CompositorObserver - void OnDidBeginMainFrame(ui::Compositor* compositor) override { - DCHECK_EQ(compositor_, compositor); - done_ = true; - if (run_loop_) - run_loop_->Quit(); - } - - void Wait() { - if (done_) - return; - - run_loop_ = std::make_unique<base::RunLoop>( - base::RunLoop::Type::kNestableTasksAllowed); - run_loop_->Run(); - run_loop_.reset(); - } - - private: - const base::raw_ptr<ui::Compositor> compositor_; - bool done_ = false; - std::unique_ptr<base::RunLoop> run_loop_; -}; - -class FirstNonAnimatedFrameStartedWaiter : public ui::CompositorObserver { - public: - explicit FirstNonAnimatedFrameStartedWaiter(ui::Compositor* compositor) - : compositor_(compositor) { - compositor->AddObserver(this); - } - - ~FirstNonAnimatedFrameStartedWaiter() override { - compositor_->RemoveObserver(this); - } - - // ui::CompositorObserver - void OnFirstNonAnimatedFrameStarted(ui::Compositor* compositor) override { - DCHECK_EQ(compositor_, compositor); - done_ = true; - if (run_loop_) - run_loop_->Quit(); - } - - void Wait() { - if (done_) - return; - - run_loop_ = std::make_unique<base::RunLoop>( - base::RunLoop::Type::kNestableTasksAllowed); - run_loop_->Run(); - run_loop_.reset(); - } - - private: - ui::Compositor* compositor_; - bool done_ = false; - std::unique_ptr<base::RunLoop> run_loop_; -}; - -void GiveItSomeTime(base::TimeDelta delta) { - // Due to the |frames_to_terminate_tracker|=3 constant in - // FrameSequenceTracker::ReportSubmitFrame we need to continue generating - // frames to receive feedback. - base::RepeatingTimer begin_main_frame_scheduler( - FROM_HERE, base::Milliseconds(16), base::BindRepeating([]() { - auto* compositor = - Shell::GetPrimaryRootWindow()->GetHost()->compositor(); - compositor->ScheduleFullRedraw(); - })); - begin_main_frame_scheduler.Reset(); - - base::RunLoop run_loop(base::RunLoop::Type::kNestableTasksAllowed); - auto* compositor = Shell::GetPrimaryRootWindow()->GetHost()->compositor(); - compositor->ScheduleFullRedraw(); - base::ThreadTaskRunnerHandle::Get()->PostDelayedTask( - FROM_HERE, run_loop.QuitClosure(), delta); - run_loop.Run(); -} - -class MetricsWaiter { - public: - MetricsWaiter(base::HistogramTester* histogram_tester, - std::string metrics_name) - : histogram_tester_(histogram_tester), metrics_name_(metrics_name) {} - - MetricsWaiter(const MetricsWaiter&) = delete; - MetricsWaiter& operator=(const MetricsWaiter&) = delete; - - ~MetricsWaiter() = default; - - void Wait() { - while (histogram_tester_->GetTotalSum(metrics_name_) == 0) { - GiveItSomeTime(base::Milliseconds(16)); - } - } - - private: - base::raw_ptr<base::HistogramTester> histogram_tester_; - const std::string metrics_name_; -}; - -} // namespace - -// Test fixture for the LoginUnlockThroughputRecorder class. -class LoginUnlockThroughputRecorderTestBase - : public LoginTestBase, - public testing::WithParamInterface<bool> { - public: - LoginUnlockThroughputRecorderTestBase() = default; - - LoginUnlockThroughputRecorderTestBase( - const LoginUnlockThroughputRecorderTestBase&) = delete; - LoginUnlockThroughputRecorderTestBase& operator=( - const LoginUnlockThroughputRecorderTestBase&) = delete; - - ~LoginUnlockThroughputRecorderTestBase() override = default; - - // LoginTestBase: - void SetUp() override { - LoginTestBase::SetUp(); - histogram_tester_ = std::make_unique<base::HistogramTester>(); - } - - void LoginOwner() { - CreateUserSessions(1); - LoginState::Get()->SetLoggedInState( - LoginState::LOGGED_IN_ACTIVE, - chromeos::LoginState::LOGGED_IN_USER_OWNER); - } - - void LockScreenAndAnimate() { - GetSessionControllerClient()->LockScreen(); - RunSimpleAnimation(); - } - - void UnlockScreenAndAnimate() { - GetSessionControllerClient()->UnlockScreen(); - RunSimpleAnimation(); - } - - void AddScheduledRestoreBrowserWindows(const std::vector<int>& ids, - bool is_lacros) { - for (int n : ids) { - throughput_recorder()->AddScheduledRestoreWindow( - n, is_lacros ? app_constants::kLacrosAppId : "", - LoginUnlockThroughputRecorder::kBrowser); - } - } - - void AddScheduledRestoreNonBrowserWindows(const std::vector<int>& ids) { - for (int n : ids) { - throughput_recorder()->AddScheduledRestoreWindow( - n, base::StringPrintf("some_app%d", n), - LoginUnlockThroughputRecorder::kBrowser); - } - } - - void RestoredWindowsCreated(const std::vector<int>& ids) { - for (int n : ids) { - throughput_recorder()->OnRestoredWindowCreated(n); - } - } - - void RestoredWindowsShown(const std::vector<int>& ids) { - ui::Compositor* compositor = - Shell::GetPrimaryRootWindow()->GetHost()->compositor(); - for (int n : ids) { - throughput_recorder()->OnBeforeRestoredWindowShown(n, compositor); - } - } - - void RestoredWindowsPresented(const std::vector<int>& ids) { - for (int n : ids) { - throughput_recorder()->OnRestoredWindowPresented(n); - } - } - - protected: - void RunSimpleAnimation() { - ui::Compositor* compositor = - Shell::GetPrimaryRootWindow()->GetHost()->compositor(); - TestObserver observer(compositor); - BeginMainFrameWaiter(compositor).Wait(); - FirstNonAnimatedFrameStartedWaiter(compositor).Wait(); - ui::DrawWaiterForTest::WaitForCompositingEnded(compositor); - } - - void EnableTabletMode(bool enable) { - Shell::Get()->tablet_mode_controller()->SetEnabledForTest(enable); - } - - LoginUnlockThroughputRecorder* throughput_recorder() { - return Shell::Get()->login_unlock_throughput_recorder(); - } - - bool IsThroughputRecorderBlocked() { - return throughput_recorder() - ->login_animation_throughput_reporter() - ->IsBlocked(); - } - - // Used to verify recorded data. - std::unique_ptr<base::HistogramTester> histogram_tester_; -}; - -using LoginUnlockThroughputRecorderLoginAnimationTest = - LoginUnlockThroughputRecorderTestBase; - -// Boolean parameter controls tablet mode. -INSTANTIATE_TEST_SUITE_P(All, - LoginUnlockThroughputRecorderLoginAnimationTest, - testing::Bool()); - -// Verifies that login animation metrics are reported correctly ignoring shelf -// initialization. -TEST_P(LoginUnlockThroughputRecorderLoginAnimationTest, - ReportLoginAnimationOnly) { - EnableTabletMode(GetParam()); - const std::string metrics_name = - GetParam() ? kAshLoginAnimationDurationTabletMode - : kAshLoginAnimationDurationClamshellMode; - - LoginOwner(); - RunSimpleAnimation(); - GiveItSomeTime(base::Milliseconds(100)); - - // Should not report login histogram until shelf is initialized. - EXPECT_EQ(histogram_tester_.get()->GetTotalSum(metrics_name), 0); - - // In this test case we ignore the shelf initialization. Pretend that it - // was done. - throughput_recorder()->ResetScopedThroughputReporterBlockerForTesting(); - RunSimpleAnimation(); - - MetricsWaiter(histogram_tester_.get(), - GetParam() ? kAshLoginAnimationDurationTabletMode - : kAshLoginAnimationDurationClamshellMode) - .Wait(); -} - -// Verifies that login animation metrics are reported correctly after shelf is -// initialized. -TEST_P(LoginUnlockThroughputRecorderLoginAnimationTest, - ReportLoginWithShelfInitialization) { - EnableTabletMode(GetParam()); - const std::string metrics_name = - GetParam() ? kAshLoginAnimationDurationTabletMode - : kAshLoginAnimationDurationClamshellMode; - - LoginOwner(); - GiveItSomeTime(base::Milliseconds(100)); - - // Should not report login histogram until shelf is initialized. - EXPECT_EQ(histogram_tester_.get()->GetTotalSum(metrics_name), 0); - - TestShelfModel model; - model.InitializeIconList({1, 2, 3, 4, 5, 6}); - - // None of the expected shelf items have icons loaded. - throughput_recorder()->InitShelfIconList(&model); - - RunSimpleAnimation(); - GiveItSomeTime(base::Milliseconds(100)); - EXPECT_TRUE(IsThroughputRecorderBlocked()); - - model.SetIconsLoadedFor({1, 2, 3}); - throughput_recorder()->UpdateShelfIconList(&model); - EXPECT_TRUE(IsThroughputRecorderBlocked()); - - // Remove last shelf button. - model.InitializeIconList({1, 2, 3, 4, 5}); - model.SetIconsLoadedFor({1, 2, 3}); - throughput_recorder()->UpdateShelfIconList(&model); - EXPECT_TRUE(IsThroughputRecorderBlocked()); - - // Add extra buttons. - model.InitializeIconList({4, 5, 6, 7, 8, 9}); - model.SetIconsLoadedFor({7, 8, 9}); - // Only 4 and 5 are not loaded yet. - throughput_recorder()->UpdateShelfIconList(&model); - EXPECT_TRUE(IsThroughputRecorderBlocked()); - - model.SetIconsLoadedFor({4, 5}); - // All buttons should have icons. - throughput_recorder()->UpdateShelfIconList(&model); - // All loaded icons should trigger login histograms. - EXPECT_FALSE(IsThroughputRecorderBlocked()); - EXPECT_GT(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllShelfIconsLoaded"), - 0); - - GiveItSomeTime(base::Milliseconds(100)); - // Should not report login histogram until login animation starts. - EXPECT_EQ(histogram_tester_.get()->GetTotalSum(metrics_name), 0); - // Shelf metrics should be already reported. Did not specifically start shelf - // animations, but it should be reported immediately when there are no shelf - // animation. - EXPECT_GT(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.ShelfLoginAnimationEnd"), - 0); - - // Start login animation. It should trigger metrics reporting. - RunSimpleAnimation(); - MetricsWaiter(histogram_tester_.get(), metrics_name).Wait(); -} - -TEST_P(LoginUnlockThroughputRecorderLoginAnimationTest, ReportUnlock) { - LoginOwner(); - - EnableTabletMode(GetParam()); - - LockScreenAndAnimate(); - UnlockScreenAndAnimate(); - - MetricsWaiter(histogram_tester_.get(), - GetParam() ? kAshUnlockAnimationDurationTabletMode - : kAshUnlockAnimationDurationClamshellMode) - .Wait(); -} - -using LoginUnlockThroughputRecorderWindowRestoreTest = - LoginUnlockThroughputRecorderTestBase; - -// Boolean parameter controls lacros mode. -INSTANTIATE_TEST_SUITE_P(All, - LoginUnlockThroughputRecorderWindowRestoreTest, - testing::Bool()); - -// Verifies that window restore metrics are reported correctly. -TEST_P(LoginUnlockThroughputRecorderWindowRestoreTest, - ReportWindowRestoreMetrics) { - const bool is_lacros = GetParam(); - - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); - - LoginOwner(); - GiveItSomeTime(base::Milliseconds(100)); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); - - AddScheduledRestoreBrowserWindows({1, 2, 3, 4, 5, 6}, is_lacros); - AddScheduledRestoreNonBrowserWindows({7, 8, 9, 10, 11, 12}); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); - - // The unexpected windows do not trigger the metrics. - RestoredWindowsCreated({21, 22, 23, 24, 25, 26}); - RestoredWindowsShown({21, 22, 23, 24, 25, 26}); - RestoredWindowsPresented({21, 22, 23, 24, 25, 26}); - GiveItSomeTime(base::Milliseconds(100)); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); - - // Window must go through all of the expected steps - // (Created->Shown->Presented). The non-created windows do not trigger - // metrics. - RestoredWindowsShown({1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12}); - RestoredWindowsPresented({1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12}); - GiveItSomeTime(base::Milliseconds(100)); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); - - // Only wait for the expected browser windows: expected window 1 is missing. - RestoredWindowsCreated({2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12}); - RestoredWindowsShown({2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12}); - RestoredWindowsPresented({2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12}); - GiveItSomeTime(base::Milliseconds(100)); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); - - // Last window created. - RestoredWindowsCreated({1}); - EXPECT_TRUE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsCreated")); - GiveItSomeTime(base::Milliseconds(100)); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - EXPECT_FALSE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); - - RestoredWindowsShown({1}); - EXPECT_TRUE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsShown")); - - RestoredWindowsPresented({1}); - EXPECT_TRUE(histogram_tester_.get()->GetTotalSum( - "Ash.LoginSessionRestore.AllBrowserWindowsPresented")); -} - -} // namespace ash
diff --git a/ash/projector/projector_annotation_tray.cc b/ash/projector/projector_annotation_tray.cc index d73175e6..f73f889 100644 --- a/ash/projector/projector_annotation_tray.cc +++ b/ash/projector/projector_annotation_tray.cc
@@ -24,6 +24,7 @@ #include "components/prefs/pref_service.h" #include "ui/base/l10n/l10n_util.h" #include "ui/compositor/layer.h" +#include "ui/events/event.h" #include "ui/gfx/paint_vector_icon.h" #include "ui/views/controls/image_view.h" #include "ui/views/controls/separator.h" @@ -61,7 +62,7 @@ ProjectorTool GetCurrentTool() { auto* controller = Shell::Get()->projector_controller(); - // ProjctorController may not be available yet as the ProjectorAnnotationTray + // `controller` may not be available yet as the `ProjectorAnnotationTray` // is created before it. if (!controller) return kToolNone; @@ -100,6 +101,27 @@ image_view_( tray_container()->AddChildView(std::make_unique<views::ImageView>())), pen_view_(nullptr) { + SetPressedCallback(base::BindRepeating( + [](ProjectorAnnotationTray* projector_annotation_tray, + const ui::Event& event) { + // NOTE: Long press not supported via the `views::Button` callback, it + // is handled via OnGestureEvent override. + if (event.IsMouseEvent() && + event.AsMouseEvent()->IsRightMouseButton()) { + projector_annotation_tray->ShowBubble(); + return; + } + + projector_annotation_tray->ToggleAnnotator(); + }, + base::Unretained(this))); + // Right click should show the bubble. In tablet mode, long press is + // synonymous with right click, gesture long press must be intercepted via + // `OnGestureEvent()` override, as `views::Button` forces long press to show a + // contextual menu. + SetTriggerableEventFlags(ui::EF_LEFT_MOUSE_BUTTON | + ui::EF_RIGHT_MOUSE_BUTTON); + image_view_->SetTooltipText(GetTooltip()); image_view_->SetHorizontalAlignment(views::ImageView::Alignment::kCenter); image_view_->SetVerticalAlignment(views::ImageView::Alignment::kCenter); @@ -111,28 +133,18 @@ ProjectorAnnotationTray::~ProjectorAnnotationTray() = default; -bool ProjectorAnnotationTray::PerformAction(const ui::Event& event) { - ToggleAnnotator(); - return true; -} - -void ProjectorAnnotationTray::OnMouseEvent(ui::MouseEvent* event) { - if (event->type() != ui::ET_MOUSE_PRESSED) { +void ProjectorAnnotationTray::OnGestureEvent(ui::GestureEvent* event) { + // Long Press typically is used to show a contextual menu, but because in + // tablet mode tapping the pod is used to toggle a feature, long press is the + // only available way to show the bubble. + // TODO(crbug/1374368): Put this where we handle other button activations, + // once the `views::Button` code allows it. + if (event->details().type() != ui::ET_GESTURE_LONG_PRESS) { + TrayBackgroundView::OnGestureEvent(event); return; } - if (event->IsRightMouseButton()) { - ShowBubble(); - } else if (event->IsLeftMouseButton()) { - ToggleAnnotator(); - } -} -void ProjectorAnnotationTray::OnGestureEvent(ui::GestureEvent* event) { - if (event->details().type() == ui::ET_GESTURE_LONG_PRESS) { - ShowBubble(); - } else if (event->details().type() == ui::ET_GESTURE_TAP) { - ToggleAnnotator(); - } + ShowBubble(); } void ProjectorAnnotationTray::ClickedOutsideBubble() {
diff --git a/ash/projector/projector_annotation_tray.h b/ash/projector/projector_annotation_tray.h index e11414e..442d018 100644 --- a/ash/projector/projector_annotation_tray.h +++ b/ash/projector/projector_annotation_tray.h
@@ -11,6 +11,15 @@ #include "ash/system/tray/view_click_listener.h" #include "base/scoped_observation.h" +namespace ui { +class GestureEvent; +} // namespace ui + +namespace views { +class ImageView; +class Widget; +} // namespace views + namespace ash { class HoverHighlightView; @@ -34,7 +43,7 @@ ~ProjectorAnnotationTray() override; // TrayBackgroundView: - bool PerformAction(const ui::Event& event) override; + void OnGestureEvent(ui::GestureEvent* event) override; void ClickedOutsideBubble() override; std::u16string GetAccessibleNameForTray() override; void HandleLocaleChange() override; @@ -43,8 +52,6 @@ void ShowBubble() override; TrayBubbleView* GetBubbleView() override; views::Widget* GetBubbleWidget() const override; - void OnMouseEvent(ui::MouseEvent* event) override; - void OnGestureEvent(ui::GestureEvent* event) override; void OnThemeChanged() override; // SessionObserver:
diff --git a/ash/projector/projector_ui_controller_unittest.cc b/ash/projector/projector_ui_controller_unittest.cc index 13ed3159..1e440fb 100644 --- a/ash/projector/projector_ui_controller_unittest.cc +++ b/ash/projector/projector_ui_controller_unittest.cc
@@ -24,6 +24,7 @@ #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" #include "ui/base/l10n/l10n_util.h" +#include "ui/events/gesture_detection/gesture_configuration.h" #include "ui/events/test/event_generator.h" #include "ui/message_center/message_center.h" #include "ui/message_center/message_center_observer.h" @@ -206,6 +207,77 @@ /*count=*/1); } +// Tests that right clicking the ProjectorAnnotationTray shows a bubble. +TEST_F(ProjectorUiControllerTest, RightClickShowsBubble) { + controller_->ShowAnnotationTray(Shell::GetPrimaryRootWindow()); + controller_->OnCanvasInitialized(true); + + auto* projector_annotation_tray = Shell::GetPrimaryRootWindowController() + ->GetStatusAreaWidget() + ->projector_annotation_tray(); + + // Right click the tray item, it should show a bubble. + RightClickOn(projector_annotation_tray); + EXPECT_TRUE(projector_annotation_tray->GetBubbleWidget()); +} + +// Tests that long pressing the ProjectorAnnotationTray shows a bubble. +TEST_F(ProjectorUiControllerTest, LongPressShowsBubble) { + controller_->ShowAnnotationTray(Shell::GetPrimaryRootWindow()); + controller_->OnCanvasInitialized(true); + + auto* projector_annotation_tray = Shell::GetPrimaryRootWindowController() + ->GetStatusAreaWidget() + ->projector_annotation_tray(); + + // Long press the tray item, it should show a bubble. + gfx::Point location = + projector_annotation_tray->GetBoundsInScreen().CenterPoint(); + + // Temporarily reconfigure gestures so that the long press takes 2 + // milliseconds. + ui::GestureConfiguration* gesture_config = + ui::GestureConfiguration::GetInstance(); + const int old_long_press_time_in_ms = gesture_config->long_press_time_in_ms(); + const base::TimeDelta old_short_press_time = + gesture_config->short_press_time(); + const int old_show_press_delay_in_ms = + gesture_config->show_press_delay_in_ms(); + gesture_config->set_long_press_time_in_ms(1); + gesture_config->set_short_press_time(base::Milliseconds(1)); + gesture_config->set_show_press_delay_in_ms(1); + + ui::test::EventGenerator* event_generator = GetEventGenerator(); + event_generator->set_current_screen_location(location); + event_generator->PressTouch(); + + // Hold the press down for 2 ms, to trigger a long press. + base::RunLoop run_loop; + base::ThreadTaskRunnerHandle::Get()->PostDelayedTask( + FROM_HERE, run_loop.QuitClosure(), base::Milliseconds(2)); + run_loop.Run(); + + gesture_config->set_long_press_time_in_ms(old_long_press_time_in_ms); + gesture_config->set_short_press_time(old_short_press_time); + gesture_config->set_show_press_delay_in_ms(old_show_press_delay_in_ms); + + event_generator->ReleaseTouch(); + + EXPECT_TRUE(projector_annotation_tray->GetBubbleWidget()); +} + +// Tests that tapping the ProjectorAnnotationTray enables annotation. +TEST_F(ProjectorUiControllerTest, TapEnabledAnnotation) { + controller_->ShowAnnotationTray(Shell::GetPrimaryRootWindow()); + controller_->OnCanvasInitialized(true); + + GestureTapOn(Shell::GetPrimaryRootWindowController() + ->GetStatusAreaWidget() + ->projector_annotation_tray()); + + EXPECT_TRUE(controller_->is_annotator_enabled()); +} + TEST_F(ProjectorUiControllerTest, ShowFailureNotification) { base::HistogramTester histogram_tester;
diff --git a/ash/public/cpp/app_list/app_list_color_provider.h b/ash/public/cpp/app_list/app_list_color_provider.h index 94bd32e6..cc418b8 100644 --- a/ash/public/cpp/app_list/app_list_color_provider.h +++ b/ash/public/cpp/app_list/app_list_color_provider.h
@@ -65,7 +65,6 @@ const views::Widget* app_list_widget) const = 0; virtual SkColor GetGridBackgroundCardInactiveColor( const views::Widget* app_list_widget) const = 0; - virtual ui::ColorId GetSeparatorColorId() const = 0; virtual SkColor GetFocusRingColor( const views::Widget* app_list_widget) const = 0; virtual SkColor GetInkDropBaseColor(
diff --git a/ash/public/cpp/app_list/app_list_config.cc b/ash/public/cpp/app_list/app_list_config.cc index 3d42d8e..65462e6 100644 --- a/ash/public/cpp/app_list/app_list_config.cc +++ b/ash/public/cpp/app_list/app_list_config.cc
@@ -237,14 +237,11 @@ } // namespace SharedAppListConfig& SharedAppListConfig::instance() { - static base::NoDestructor<SharedAppListConfig> shared_config; - return *shared_config; + static SharedAppListConfig shared_config; + return shared_config; } -SharedAppListConfig::SharedAppListConfig() - : search_result_title_font_style_(ui::ResourceBundle::BaseFont), - search_result_recommendation_title_font_( - ui::ResourceBundle::GetSharedInstance().GetFontListWithDelta(1)) {} +SharedAppListConfig::SharedAppListConfig() = default; int SharedAppListConfig::GetMaxNumOfItemsPerPage() const { return 20;
diff --git a/ash/public/cpp/app_list/app_list_config.h b/ash/public/cpp/app_list/app_list_config.h index b9e9e79..62e175e 100644 --- a/ash/public/cpp/app_list/app_list_config.h +++ b/ash/public/cpp/app_list/app_list_config.h
@@ -33,8 +33,6 @@ size_t max_search_results() const { return max_search_results_; } - size_t max_search_result_tiles() const { return max_search_result_tiles_; } - size_t max_search_result_list_items() const { return max_search_result_list_items_; } @@ -85,14 +83,6 @@ return suggestion_chip_icon_dimension_; } - ui::ResourceBundle::FontStyle search_result_title_font_style() const { - return search_result_title_font_style_; - } - - gfx::FontList search_result_recommendation_title_font() const { - return search_result_recommendation_title_font_; - } - int search_tile_height() const { return search_tile_height_; } // Returns the maximum number of items allowed in a page in the apps grid. @@ -102,7 +92,7 @@ int GetPreferredIconDimension(SearchResultDisplayType display_type) const; private: - friend class base::NoDestructor<SharedAppListConfig>; + friend class SharedAppListConfig; SharedAppListConfig(); // The icon dimension of tile views in apps grid view. @@ -111,9 +101,6 @@ // Maximum number of results to show in the launcher Search UI. const size_t max_search_results_ = 6; - // Max number of search result tiles in the launcher suggestion window. - const size_t max_search_result_tiles_ = 6; - // Max number of search result list items in the launcher suggestion window. const size_t max_search_result_list_items_ = 5; @@ -143,14 +130,6 @@ // The suggestion chip icon dimension. const int suggestion_chip_icon_dimension_ = 20; - // Font style for AppListSearchResultItemViews that are not suggested - // apps. - const ui::ResourceBundle::FontStyle search_result_title_font_style_; - - // Font style for AppListSearchResultTileItemViews that are suggested - // apps. - const gfx::FontList search_result_recommendation_title_font_; - // The height of tiles in search result. const int search_tile_height_ = 92; };
diff --git a/ash/public/cpp/app_list/app_list_features.cc b/ash/public/cpp/app_list/app_list_features.cc index 7fa114c..62b594d 100644 --- a/ash/public/cpp/app_list/app_list_features.cc +++ b/ash/public/cpp/app_list/app_list_features.cc
@@ -50,6 +50,9 @@ BASE_FEATURE(kLauncherPlayStoreSearch, "LauncherPlayStoreSearch", base::FEATURE_ENABLED_BY_DEFAULT); +BASE_FEATURE(kAnimateScaleOnTabletModeTransition, + "AnimateScaleOnTabletModeTransition", + base::FEATURE_DISABLED_BY_DEFAULT); bool IsAppRankerEnabled() { return base::FeatureList::IsEnabled(kEnableAppRanker); @@ -129,4 +132,8 @@ base::FeatureList::IsEnabled(kLauncherPlayStoreSearch); } +bool IsAnimateScaleOnTabletModeTransitionEnabled() { + return base::FeatureList::IsEnabled(kAnimateScaleOnTabletModeTransition); +} + } // namespace app_list_features
diff --git a/ash/public/cpp/app_list/app_list_features.h b/ash/public/cpp/app_list/app_list_features.h index a2696c56..97de35ee 100644 --- a/ash/public/cpp/app_list/app_list_features.h +++ b/ash/public/cpp/app_list/app_list_features.h
@@ -63,6 +63,10 @@ // Enables Play Store search in the launcher. ASH_PUBLIC_EXPORT BASE_DECLARE_FEATURE(kLauncherPlayStoreSearch); +// Enables new tablet <-> clamshell mode transition animation (scale + opacity +// instead of slide up/down). +ASH_PUBLIC_EXPORT BASE_DECLARE_FEATURE(kAnimateScaleOnTabletModeTransition); + ASH_PUBLIC_EXPORT bool IsAppRankerEnabled(); ASH_PUBLIC_EXPORT bool IsZeroStateAppsRankerEnabled(); ASH_PUBLIC_EXPORT bool IsZeroStateMixedTypesRankerEnabled(); @@ -79,6 +83,7 @@ ASH_PUBLIC_EXPORT base::TimeDelta DynamicSearchUpdateAnimationDuration(); ASH_PUBLIC_EXPORT bool IsCompactBubbleLauncherEnabled(); ASH_PUBLIC_EXPORT bool IsLauncherPlayStoreSearchEnabled(); +ASH_PUBLIC_EXPORT bool IsAnimateScaleOnTabletModeTransitionEnabled(); ASH_PUBLIC_EXPORT std::string AppSearchResultRankerPredictorName(); ASH_PUBLIC_EXPORT std::string CategoricalSearchType();
diff --git a/ash/public/cpp/test/test_app_list_color_provider.cc b/ash/public/cpp/test/test_app_list_color_provider.cc index 545886b..cbddd35f 100644 --- a/ash/public/cpp/test/test_app_list_color_provider.cc +++ b/ash/public/cpp/test/test_app_list_color_provider.cc
@@ -104,10 +104,6 @@ return SkColorSetA(SK_ColorWHITE, 41 /* 16% */); } -ui::ColorId TestAppListColorProvider::GetSeparatorColorId() const { - return ui::kColorAshSystemUIMenuSeparator; -} - SkColor TestAppListColorProvider::GetFocusRingColor( const views::Widget* app_list_widget) const { return gfx::kGoogleBlue600;
diff --git a/ash/public/cpp/test/test_app_list_color_provider.h b/ash/public/cpp/test/test_app_list_color_provider.h index 2ff5c222..7a002052 100644 --- a/ash/public/cpp/test/test_app_list_color_provider.h +++ b/ash/public/cpp/test/test_app_list_color_provider.h
@@ -58,7 +58,6 @@ const views::Widget* app_list_widget) const override; SkColor GetGridBackgroundCardInactiveColor( const views::Widget* app_list_widget) const override; - ui::ColorId GetSeparatorColorId() const override; SkColor GetFocusRingColor( const views::Widget* app_list_widget) const override; SkColor GetInkDropBaseColor(
diff --git a/ash/shelf/shelf.cc b/ash/shelf/shelf.cc index 5ddea72..18df6440f 100644 --- a/ash/shelf/shelf.cc +++ b/ash/shelf/shelf.cc
@@ -10,7 +10,6 @@ #include "ash/app_list/app_list_controller_impl.h" #include "ash/constants/ash_features.h" #include "ash/constants/ash_switches.h" -#include "ash/metrics/login_unlock_throughput_recorder.h" #include "ash/public/cpp/keyboard/keyboard_controller_observer.h" #include "ash/public/cpp/shelf_item_delegate.h" #include "ash/public/cpp/shelf_model.h" @@ -19,7 +18,6 @@ #include "ash/root_window_controller.h" #include "ash/shelf/hotseat_widget.h" #include "ash/shelf/login_shelf_widget.h" -#include "ash/shelf/scrollable_shelf_view.h" #include "ash/shelf/shelf_controller.h" #include "ash/shelf/shelf_focus_cycler.h" #include "ash/shelf/shelf_layout_manager.h" @@ -408,8 +406,6 @@ hotseat_transition_metrics_reporter_ = std::make_unique<HotseatWidgetAnimationMetricsReporter>( HotseatWidgetAnimationMetricsReporter::HotseatElementType::kWidget); - Shell::Get()->login_unlock_throughput_recorder()->SetShelfViewIfNotSet( - hotseat_widget_->scrollable_shelf_view()->shelf_view()); } void Shelf::CreateStatusAreaWidget(aura::Window* shelf_container) {
diff --git a/ash/shelf/shelf_view.cc b/ash/shelf/shelf_view.cc index 696ab55..c417f718 100644 --- a/ash/shelf/shelf_view.cc +++ b/ash/shelf/shelf_view.cc
@@ -1213,6 +1213,8 @@ drag_icon_bounds_in_screen_ = gfx::Rect(); drag_and_drop_shelf_id_ = ShelfID(); is_active_drag_and_drop_host_ = false; + + HandleShelfParty(); } void ShelfView::SwapButtons(views::View* button_to_swap, bool with_next) { @@ -1985,15 +1987,6 @@ return drag_icon_proxy_->GetBoundsInScreen(); } -void ShelfView::AddAnimationObserver(views::BoundsAnimatorObserver* observer) { - bounds_animator_->AddObserver(observer); -} - -void ShelfView::RemoveAnimationObserver( - views::BoundsAnimatorObserver* observer) { - bounds_animator_->RemoveObserver(observer); -} - void ShelfView::AnnounceShelfAutohideBehavior() { std::u16string announcement; switch (shelf_->auto_hide_behavior()) { @@ -2225,6 +2218,8 @@ AnnouncePinUnpinEvent(old_item, /*pinned=*/false); RecordPinUnpinUserAction(/*pinned=*/false); } + + party_.erase(old_item.id); } void ShelfView::ShelfItemChanged(int model_index, const ShelfItem& old_item) { @@ -2679,6 +2674,8 @@ } DCHECK(IsItemPinned(item)); DCHECK(!IsItemVisible(item)); + if (item.image.isNull() || item.id == drag_and_drop_shelf_id_) + continue; // Add the item if it is not already partying. const auto insertion_results = party_.try_emplace(item.id); if (insertion_results.second) {
diff --git a/ash/shelf/shelf_view.h b/ash/shelf/shelf_view.h index c75b823..3cdf669 100644 --- a/ash/shelf/shelf_view.h +++ b/ash/shelf/shelf_view.h
@@ -326,9 +326,6 @@ return current_ghost_view_index_; } - void AddAnimationObserver(views::BoundsAnimatorObserver* observer); - void RemoveAnimationObserver(views::BoundsAnimatorObserver* observer); - private: friend class ShelfViewTestAPI;
diff --git a/ash/shelf/shelf_view_unittest.cc b/ash/shelf/shelf_view_unittest.cc index 007817a..b87835c 100644 --- a/ash/shelf/shelf_view_unittest.cc +++ b/ash/shelf/shelf_view_unittest.cc
@@ -2333,39 +2333,6 @@ EXPECT_EQ(0, GetHapticTickEventsCount()); } -// Verifies that partying items are hidden from the shelf. -// TODO(crbug/1372295): This test consistently times out. -TEST_P(LtrRtlShelfViewTest, DISABLED_PartyingItemsHiddenFromShelf) { - AddAppShortcut(); - AddAppShortcut(); - AddApp(); - ShelfItem item = model_->items()[1u]; - item.status = STATUS_RUNNING; - model_->Set(1, item); - const gfx::Rect initial_bounds0 = test_api_->GetBoundsByIndex(0); - const gfx::Rect initial_bounds2 = test_api_->GetBoundsByIndex(2); - - // Start shelf party. - model_->ToggleShelfParty(); - { - const std::vector<size_t> not_partying = {1, 3}; - EXPECT_EQ(not_partying, shelf_view_->visible_views_indices()); - } - test_api_->RunMessageLoopUntilAnimationsDone(); - EXPECT_TRUE(test_api_->GetBoundsByIndex(0).IsEmpty()); - EXPECT_TRUE(test_api_->GetBoundsByIndex(2).IsEmpty()); - - // End shelf party. - model_->ToggleShelfParty(); - { - const std::vector<size_t> not_partying = {0, 1, 2, 3}; - EXPECT_EQ(not_partying, shelf_view_->visible_views_indices()); - } - test_api_->RunMessageLoopUntilAnimationsDone(); - EXPECT_EQ(initial_bounds0, test_api_->GetBoundsByIndex(0)); - EXPECT_EQ(initial_bounds2, test_api_->GetBoundsByIndex(2)); -} - // Test class that tests both context and application menus. class ShelfViewMenuTest : public ShelfViewTest, public testing::WithParamInterface<bool> { @@ -3665,4 +3632,36 @@ test_api_->RunMessageLoopUntilAnimationsDone(); } +// Verifies that partying items are hidden from the shelf. +TEST_P(ShelfPartyTest, PartyingItemsHiddenFromShelf) { + AddAppShortcut(); + AddAppShortcut(); + AddApp(); + ShelfItem item = model_->items()[1u]; + item.status = STATUS_RUNNING; + model_->Set(1, item); + const gfx::Rect initial_bounds0 = test_api_->GetBoundsByIndex(0); + const gfx::Rect initial_bounds2 = test_api_->GetBoundsByIndex(2); + + // Start shelf party. + model_->ToggleShelfParty(); + { + const std::vector<size_t> not_partying = {1, 3}; + EXPECT_EQ(not_partying, shelf_view_->visible_views_indices()); + } + task_environment()->FastForwardBy(base::Seconds(1)); + EXPECT_TRUE(test_api_->GetBoundsByIndex(0).IsEmpty()); + EXPECT_TRUE(test_api_->GetBoundsByIndex(2).IsEmpty()); + + // End shelf party. + model_->ToggleShelfParty(); + { + const std::vector<size_t> not_partying = {0, 1, 2, 3}; + EXPECT_EQ(not_partying, shelf_view_->visible_views_indices()); + } + test_api_->RunMessageLoopUntilAnimationsDone(); + EXPECT_EQ(initial_bounds0, test_api_->GetBoundsByIndex(0)); + EXPECT_EQ(initial_bounds2, test_api_->GetBoundsByIndex(2)); +} + } // namespace ash
diff --git a/ash/shell.h b/ash/shell.h index 51dfdb29..fd54b4bf 100644 --- a/ash/shell.h +++ b/ash/shell.h
@@ -738,10 +738,6 @@ void NotifyShelfAlignmentChanged(aura::Window* root_window, ShelfAlignment old_alignment); - LoginUnlockThroughputRecorder* login_unlock_throughput_recorder() { - return login_unlock_throughput_recorder_.get(); - } - private: FRIEND_TEST_ALL_PREFIXES(ExtendedDesktopTest, TestCursor); FRIEND_TEST_ALL_PREFIXES(WindowManagerTest, MouseEventCursors);
diff --git a/ash/system/tray/tray_constants.h b/ash/system/tray/tray_constants.h index aa3eb7d..8a140794 100644 --- a/ash/system/tray/tray_constants.h +++ b/ash/system/tray/tray_constants.h
@@ -74,9 +74,6 @@ // The size of the icons appearing in the material design system tray. constexpr int kTrayIconSize = 16; -// The padding on top of the time tray icon when vertically aligned. -constexpr int kTrayTimeIconTopPadding = 2; - // The padding around network tray icon in dip. constexpr int kTrayNetworkIconPadding = 2; constexpr int kUnifiedTrayNetworkIconPadding = 4;
diff --git a/ash/system/unified/unified_system_tray.cc b/ash/system/unified/unified_system_tray.cc index 7723a95..c4e048e 100644 --- a/ash/system/unified/unified_system_tray.cc +++ b/ash/system/unified/unified_system_tray.cc
@@ -222,18 +222,14 @@ notification_icons_controller_->AddNotificationTrayItems(tray_container()); for (TrayItemView* tray_item : notification_icons_controller_->tray_items()) { tray_items_.push_back(tray_item); - AddObservedTrayItem(tray_item); } AddTrayItemToContainer(screen_capture_view_); tray_items_.push_back( notification_icons_controller_->notification_counter_view()); - AddObservedTrayItem( - notification_icons_controller_->notification_counter_view()); tray_items_.push_back(notification_icons_controller_->quiet_mode_view()); - AddObservedTrayItem(notification_icons_controller_->quiet_mode_view()); if (features::IsSnoopingProtectionEnabled()) AddTrayItemToContainer(snooping_protection_view_); @@ -267,12 +263,6 @@ AddTrayItemToContainer(channel_indicator_view_); } - auto vertical_clock_padding = std::make_unique<views::View>(); - vertical_clock_padding->SetPreferredSize(gfx::Size( - 0, features::IsCalendarViewEnabled() ? 0 : kTrayTimeIconTopPadding)); - vertical_clock_padding_ = - tray_container()->AddChildView(std::move(vertical_clock_padding)); - if (!features::IsCalendarViewEnabled()) { AddTrayItemToContainer(time_view_); } @@ -284,12 +274,10 @@ set_use_bounce_in_animation(false); ShelfConfig::Get()->AddObserver(this); - Shell::Get()->AddShellObserver(this); } UnifiedSystemTray::~UnifiedSystemTray() { ShelfConfig::Get()->RemoveObserver(this); - Shell::Get()->RemoveShellObserver(this); DestroyBubbles(); @@ -308,38 +296,6 @@ observers_.RemoveObserver(observer); } -bool UnifiedSystemTray::MoreThanOneVisibleTrayItem() const { - bool one_visible_item = false; - for (TrayItemView* item : tray_items_) { - if (!item->GetVisible()) - continue; - if (one_visible_item) - return true; - one_visible_item = true; - } - return false; -} - -void UnifiedSystemTray::MaybeUpdateVerticalClockPadding() { - const bool padding_is_visible = vertical_clock_padding_->GetVisible(); - - if (shelf()->IsHorizontalAlignment()) { - if (padding_is_visible) - vertical_clock_padding_->SetVisible(false); - return; - } - - // Padding is shown when an icon besides TimeView is visible. - const bool should_show_padding = MoreThanOneVisibleTrayItem(); - if (padding_is_visible != should_show_padding) - vertical_clock_padding_->SetVisible(should_show_padding); -} - -void UnifiedSystemTray::OnViewVisibilityChanged(views::View* observed_view, - views::View* starting_view) { - MaybeUpdateVerticalClockPadding(); -} - bool UnifiedSystemTray::IsBubbleShown() const { return !!bubble_; } @@ -496,11 +452,6 @@ return absl::make_optional(TOGGLE_SYSTEM_TRAY_BUBBLE); } -void UnifiedSystemTray::OnShelfAlignmentChanged(aura::Window* root_window, - ShelfAlignment old_alignment) { - MaybeUpdateVerticalClockPadding(); -} - void UnifiedSystemTray::OnShelfConfigUpdated() { // Ensure the margin is updated correctly depending on whether dense shelf // is currently shown or not. @@ -777,11 +728,6 @@ void UnifiedSystemTray::AddTrayItemToContainer(TrayItemView* tray_item) { tray_items_.push_back(tray_item); tray_container()->AddChildView(tray_item); - AddObservedTrayItem(tray_item); -} - -void UnifiedSystemTray::AddObservedTrayItem(TrayItemView* tray_item) { - tray_items_observations_.AddObservation(tray_item); } void UnifiedSystemTray::DestroyBubbles() {
diff --git a/ash/system/unified/unified_system_tray.h b/ash/system/unified/unified_system_tray.h index 04aef94..fbfd8fc 100644 --- a/ash/system/unified/unified_system_tray.h +++ b/ash/system/unified/unified_system_tray.h
@@ -11,7 +11,6 @@ #include "ash/ash_export.h" #include "ash/public/cpp/accelerators.h" #include "ash/public/cpp/shelf_config.h" -#include "ash/shell_observer.h" #include "ash/system/status_area_widget.h" #include "ash/system/time/time_view.h" #include "ash/system/tray/tray_background_view.h" @@ -19,7 +18,6 @@ #include "ash/system/unified/unified_system_tray_model.h" #include "base/memory/scoped_refptr.h" #include "base/memory/weak_ptr.h" -#include "base/scoped_multi_source_observation.h" #include "base/time/time.h" #include "base/timer/timer.h" @@ -64,7 +62,6 @@ class ASH_EXPORT UnifiedSystemTray : public TrayBackgroundView, public ShelfConfig::Observer, - public ShellObserver, public UnifiedSystemTrayController::Observer { public: class Observer : public base::CheckedObserver { @@ -86,14 +83,6 @@ void AddObserver(Observer* observer); void RemoveObserver(Observer* observer); - // Adds a padding on top of the vertical clock if there are other visible - // icons in the tray, removes it if the clock is the only visible icon. - void MaybeUpdateVerticalClockPadding(); - - // views::ViewObserver: - void OnViewVisibilityChanged(views::View* observed_view, - views::View* starting_view) override; - // True if the bubble is shown. It does not include slider bubbles, and when // they're shown it still returns false. bool IsBubbleShown() const; @@ -200,10 +189,6 @@ const char* GetClassName() const override; absl::optional<AcceleratorAction> GetAcceleratorAction() const override; - // ShellObserver: - void OnShelfAlignmentChanged(aura::Window* root_window, - ShelfAlignment old_alignment) override; - // ShelfConfig::Observer: void OnShelfConfigUpdated() override; @@ -265,12 +250,6 @@ // The container takes the ownership of |tray_item|. void AddTrayItemToContainer(TrayItemView* tray_item); - // Returns true if there is two or more tray items that are visible. - bool MoreThanOneVisibleTrayItem() const; - - // Add observed tray item views. - void AddObservedTrayItem(TrayItemView* tray_item); - // Destroys the `bubble_` and the `message_center_bubble_`, also handles // removing bubble related observers. void DestroyBubbles(); @@ -312,13 +291,6 @@ // Contains all tray items views added to tray_container(). std::list<TrayItemView*> tray_items_; - base::ScopedMultiSourceObservation<views::View, views::ViewObserver> - tray_items_observations_{this}; - - // Padding owned by the view hierarchy used to separate vertical - // clock from other tray icons. - views::View* vertical_clock_padding_ = nullptr; - base::OneShotTimer timer_; bool first_interaction_recorded_ = false;
diff --git a/ash/system/unified/unified_system_tray_unittest.cc b/ash/system/unified/unified_system_tray_unittest.cc index d04a346c..4ce2392 100644 --- a/ash/system/unified/unified_system_tray_unittest.cc +++ b/ash/system/unified/unified_system_tray_unittest.cc
@@ -87,10 +87,6 @@ ->slider_bubble_controller_->bubble_widget_; } - bool MoreThanOneVisibleTrayItem() const { - return GetPrimaryUnifiedSystemTray()->MoreThanOneVisibleTrayItem(); - } - UnifiedSliderBubbleController::SliderType GetSliderBubbleType() { return GetPrimaryUnifiedSystemTray() ->slider_bubble_controller_->slider_type_; @@ -124,14 +120,6 @@ return GetPrimaryUnifiedSystemTray()->ime_mode_view_; } - std::list<TrayItemView*> tray_items() { - return GetPrimaryUnifiedSystemTray()->tray_items_; - } - - views::View* vertical_clock_padding() { - return GetPrimaryUnifiedSystemTray()->vertical_clock_padding_; - } - private: int id_ = 0; base::test::ScopedFeatureList feature_list_; @@ -280,42 +268,6 @@ EXPECT_EQ(time_bounds.height(), ime_bounds.height()); } -TEST_P(UnifiedSystemTrayTest, VerticalClockPadding) { - // Padding can only be visible if shelf is vertically aligned. - GetPrimaryShelf()->SetAlignment(ShelfAlignment::kLeft); - - // Sets all tray items' visibility to false except TimeView. - for (TrayItemView* item : tray_items()) { - item->SetVisible(item == time_view()); - } - - // Only one visible tray item, padding should not be visible. - EXPECT_FALSE(vertical_clock_padding()->GetVisible()); - - // Sets another tray item visibility to true. - ime_mode_view()->SetVisible(true); - - // Two visible tray items, padding should be visible. - EXPECT_TRUE(vertical_clock_padding()->GetVisible()); -} - -TEST_P(UnifiedSystemTrayTest, VerticalClockPaddingAfterAlignmentChange) { - auto* shelf = GetPrimaryShelf(); - - // Padding can only be visible if shelf is vertically aligned. - shelf->SetAlignment(ShelfAlignment::kLeft); - - // Ensure two tray items are visible, padding should be visible. - time_view()->SetVisible(true); - ime_mode_view()->SetVisible(true); - - EXPECT_TRUE(vertical_clock_padding()->GetVisible()); - - // Padding should not be visible when shelf is horizontal. - shelf->SetAlignment(ShelfAlignment::kBottom); - EXPECT_FALSE(vertical_clock_padding()->GetVisible()); -} - TEST_P(UnifiedSystemTrayTest, FocusMessageCenter) { if (IsQsRevampEnabled()) return;
diff --git a/ash/webui/os_feedback_ui/resources/confirmation_page.js b/ash/webui/os_feedback_ui/resources/confirmation_page.js index 927a7628..c980a05 100644 --- a/ash/webui/os_feedback_ui/resources/confirmation_page.js +++ b/ash/webui/os_feedback_ui/resources/confirmation_page.js
@@ -134,7 +134,7 @@ handleLinkClicked_(e) { e.stopPropagation(); - switch (e.target.id) { + switch (e.currentTarget.id) { case 'diagnostics': this.feedbackServiceProvider_.openDiagnosticsApp(); this.handleEmitMetrics_( @@ -154,7 +154,7 @@ FeedbackAppPostSubmitAction.kOpenChromebookCommunity); break; default: - console.warn('unexpected caller id: ', e.target.id); + console.warn('unexpected caller id: ', e.currentTarget.id); } }
diff --git a/ash/webui/personalization_app/mojom/personalization_app.mojom b/ash/webui/personalization_app/mojom/personalization_app.mojom index 9a67853..77c1f2c1 100644 --- a/ash/webui/personalization_app/mojom/personalization_app.mojom +++ b/ash/webui/personalization_app/mojom/personalization_app.mojom
@@ -167,9 +167,6 @@ // Represents the currently selected wallpaper image. struct CurrentWallpaper { - // URL of the currently selected wallpaper image in data scheme. - url.mojom.Url url; - // Attribution text such as "Photo by John Doe" or the file name of the local // image. array<string> attribution;
diff --git a/ash/webui/personalization_app/personalization_app_ui.cc b/ash/webui/personalization_app/personalization_app_ui.cc index 0a15295..0cba04ec 100644 --- a/ash/webui/personalization_app/personalization_app_ui.cc +++ b/ash/webui/personalization_app/personalization_app_ui.cc
@@ -6,6 +6,7 @@ #include "ash/constants/ash_features.h" #include "ash/public/cpp/ambient/ambient_client.h" +#include "ash/public/cpp/wallpaper/wallpaper_controller.h" #include "ash/rgb_keyboard/rgb_keyboard_manager.h" #include "ash/shell.h" #include "ash/webui/grit/ash_personalization_app_resources.h" @@ -17,9 +18,12 @@ #include "ash/webui/personalization_app/personalization_app_user_provider.h" #include "ash/webui/personalization_app/personalization_app_wallpaper_provider.h" #include "base/check.h" +#include "base/containers/contains.h" #include "base/strings/strcat.h" +#include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "chromeos/strings/grit/chromeos_strings.h" +#include "content/public/browser/browser_context.h" #include "content/public/browser/web_contents.h" #include "content/public/browser/web_ui_data_source.h" #include "content/public/common/url_constants.h" @@ -259,6 +263,10 @@ source->EnableReplaceI18nInJS(); } +bool ShouldHandleWebUIRequest(const std::string& path) { + return base::StartsWith(path, "wallpaper.png"); +} + } // namespace PersonalizationAppUI::PersonalizationAppUI( @@ -281,6 +289,12 @@ web_ui->GetWebContents()->GetBrowserContext(), kChromeUIPersonalizationAppHost); + // Supply a custom wallpaper image. + source->SetRequestFilter( + base::BindRepeating(&ShouldHandleWebUIRequest), + base::BindRepeating(&PersonalizationAppUI::HandleWebUIRequest, + weak_ptr_factory_.GetWeakPtr())); + source->OverrideContentSecurityPolicy( network::mojom::CSPDirectiveName::ScriptSrc, "script-src chrome://resources chrome://test chrome://webui-test " @@ -358,6 +372,14 @@ source->AddBoolean("isJellyEnabled", features::IsJellyEnabled()); } +void PersonalizationAppUI::HandleWebUIRequest( + const std::string& path, + content::WebUIDataSource::GotDataCallback callback) { + DCHECK(base::Contains(path, "?key=")) + << "wallpaper key must be provided to prevent browser cache collisions"; + wallpaper_provider_->GetWallpaperAsPngBytes(std::move(callback)); +} + WEB_UI_CONTROLLER_TYPE_IMPL(PersonalizationAppUI) } // namespace ash::personalization_app
diff --git a/ash/webui/personalization_app/personalization_app_ui.h b/ash/webui/personalization_app/personalization_app_ui.h index 71f9e91..378525d 100644 --- a/ash/webui/personalization_app/personalization_app_ui.h +++ b/ash/webui/personalization_app/personalization_app_ui.h
@@ -8,6 +8,8 @@ #include <memory> #include "ash/webui/personalization_app/mojom/personalization_app.mojom-forward.h" +#include "base/memory/weak_ptr.h" +#include "content/public/browser/web_ui_data_source.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "ui/webui/mojo_web_ui_controller.h" #include "ui/webui/resources/cr_components/color_change_listener/color_change_listener.mojom.h" @@ -69,6 +71,10 @@ private: void AddBooleans(content::WebUIDataSource* source); + + void HandleWebUIRequest(const std::string& path, + content::WebUIDataSource::GotDataCallback callback); + std::unique_ptr<PersonalizationAppAmbientProvider> ambient_provider_; std::unique_ptr<PersonalizationAppKeyboardBacklightProvider> keyboard_backlight_provider_; @@ -79,6 +85,8 @@ // changes. std::unique_ptr<ui::ColorChangeHandler> color_provider_handler_; + base::WeakPtrFactory<PersonalizationAppUI> weak_ptr_factory_{this}; + WEB_UI_CONTROLLER_TYPE_DECL(); };
diff --git a/ash/webui/personalization_app/personalization_app_wallpaper_provider.h b/ash/webui/personalization_app/personalization_app_wallpaper_provider.h index 1517aef..66e6863 100644 --- a/ash/webui/personalization_app/personalization_app_wallpaper_provider.h +++ b/ash/webui/personalization_app/personalization_app_wallpaper_provider.h
@@ -6,6 +6,7 @@ #define ASH_WEBUI_PERSONALIZATION_APP_PERSONALIZATION_APP_WALLPAPER_PROVIDER_H_ #include "ash/webui/personalization_app/mojom/personalization_app.mojom.h" +#include "content/public/browser/web_ui_data_source.h" #include "mojo/public/cpp/bindings/pending_receiver.h" namespace ash::personalization_app { @@ -17,6 +18,9 @@ virtual void BindInterface( mojo::PendingReceiver<mojom::WallpaperProvider> receiver) = 0; + virtual void GetWallpaperAsPngBytes( + content::WebUIDataSource::GotDataCallback callback) = 0; + // Not all users that can view the personalization app can also see google // photos. Users without a gaia account cannot use the photos APIs. virtual bool IsEligibleForGooglePhotos() = 0;
diff --git a/ash/webui/personalization_app/resources/js/wallpaper/utils.ts b/ash/webui/personalization_app/resources/js/wallpaper/utils.ts index e502133..4ae5566 100644 --- a/ash/webui/personalization_app/resources/js/wallpaper/utils.ts +++ b/ash/webui/personalization_app/resources/js/wallpaper/utils.ts
@@ -121,3 +121,19 @@ } return attribution; } + +/** + * Get a url to download a high quality preview of the current wallpaper. + * Responds with null in case |image| is invalid or null. + */ +export function getWallpaperSrc(image: CurrentWallpaper|null): string|null { + if (!image) { + return null; + } + if (typeof image.key !== 'string' || !image.key) { + console.warn('Invalid image key received'); + return null; + } + // Add a key query parameter to cache bust when the image changes. + return `/wallpaper.png?key=${encodeURIComponent(image.key)}`; +}
diff --git a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_actions.ts b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_actions.ts index 8d3192a..d3216693 100644 --- a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_actions.ts +++ b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_actions.ts
@@ -8,7 +8,6 @@ import {Url} from 'chrome://resources/mojo/url/mojom/url.mojom-webui.js'; import {CurrentWallpaper, GooglePhotosAlbum, GooglePhotosEnablementState, GooglePhotosPhoto, WallpaperCollection, WallpaperImage} from '../personalization_app.mojom-webui.js'; -import {isPngDataUrl} from '../utils.js'; import {DisplayableImage} from './constants.js'; @@ -449,9 +448,6 @@ */ export function setSelectedImageAction(image: CurrentWallpaper| null): SetSelectedImageAction { - assert( - image === null || isPngDataUrl(image.url), - 'only png data urls are supported'); return { image, name: WallpaperActionName.SET_SELECTED_IMAGE,
diff --git a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.html b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.html index 893ce431..45b0696 100644 --- a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.html +++ b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.html
@@ -84,16 +84,16 @@ <div id="imageContainer" class="photo-images-container clickable" on-click="onClickWallpaper_" on-keypress="onClickWallpaper_"> <div class="photo-images-border"></div> - <img src$="[[image_.url.url]]" - alt$="[[getImageAltDescription_(image_)]]"> + <img alt$="[[getImageAltDescription_(image_)]]" + src$="[[getWallpaperSrc_(image_)]]"> </div> </template> <template is="dom-if" if="[[isPolicyControlled_(image_)]]"> <div id="imageContainer" class="photo-images-container"> <div class="photo-images-border"></div> - <img src$="[[image_.url.url]]" + <img alt$="[[getImageAltDescription_(image_)]]" aria-description="$i18n{managedSetting}" - alt$="[[getImageAltDescription_(image_)]]"> + src$="[[getWallpaperSrc_(image_)]]"> </div> </template> </template>
diff --git a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.ts b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.ts index 5fa4c3e..f77a5f3 100644 --- a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.ts +++ b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_preview_element.ts
@@ -21,7 +21,7 @@ import {WithPersonalizationStore} from '../personalization_store.js'; import {isNonEmptyArray} from '../utils.js'; -import {getLocalStorageAttribution} from './utils.js'; +import {getLocalStorageAttribution, getWallpaperSrc} from './utils.js'; import {WallpaperObserver} from './wallpaper_observer.js'; import {getTemplate} from './wallpaper_preview_element.html.js'; @@ -67,6 +67,10 @@ PersonalizationRouter.instance().goToRoute(Paths.COLLECTIONS); } + private getWallpaperSrc_(image: CurrentWallpaper|null): string|null { + return getWallpaperSrc(image); + } + private getImageAltDescription_(image: CurrentWallpaper|null): string { if (!image) { return `${this.i18n('currentlySet')} ${
diff --git a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.html b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.html index c51805e..e5ac3eff 100644 --- a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.html +++ b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.html
@@ -72,7 +72,9 @@ <template is="dom-if" if="[[showImage_]]"> <div id="imageContainer" class="preview-image-container"> <div class="preview-image-border"></div> - <img class="preview-image" src$="[[image_.url.url]]" aria-hidden="true"> + <img aria-hidden="true" + class="preview-image" + src$="[[getWallpaperSrc_(image_)]]"> </div> <h2 id="textContainer" class="preview-text-container" aria-live="polite" aria-label$="[[getAriaLabel_(image_, dailyRefreshState_)]]"> <span id="currentlySet" aria-hidden="true">$i18n{currentlySet}</span>
diff --git a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.ts b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.ts index b5c52a4..b05cc8e 100644 --- a/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.ts +++ b/ash/webui/personalization_app/resources/js/wallpaper/wallpaper_selected_element.ts
@@ -21,7 +21,7 @@ import {WithPersonalizationStore} from '../personalization_store.js'; import {isNonEmptyArray} from '../utils.js'; -import {getLocalStorageAttribution, getWallpaperLayoutEnum} from './utils.js'; +import {getLocalStorageAttribution, getWallpaperLayoutEnum, getWallpaperSrc} from './utils.js'; import {getDailyRefreshState, selectGooglePhotosAlbum, setCurrentWallpaperLayout, setDailyRefreshCollectionId, updateDailyRefreshWallpaper} from './wallpaper_controller.js'; import {getWallpaperProvider} from './wallpaper_interface_provider.js'; import {WallpaperObserver} from './wallpaper_observer.js'; @@ -252,6 +252,10 @@ collectionId! || googlePhotosAlbumId!, dailyRefreshState); } + private getWallpaperSrc_(image: CurrentWallpaper|null): string|null { + return getWallpaperSrc(image); + } + private getCenterAriaPressed_(image: CurrentWallpaper): string { return (!!image && image.layout === WallpaperLayout.kCenter).toString(); }
diff --git a/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.cc b/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.cc index 38cd583..f3b26e0c 100644 --- a/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.cc +++ b/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.cc
@@ -12,6 +12,8 @@ #include "ash/webui/personalization_app/mojom/personalization_app.mojom.h" #include "ash/webui/personalization_app/proto/backdrop_wallpaper.pb.h" #include "base/check_op.h" +#include "base/memory/ref_counted_memory.h" +#include "base/memory/scoped_refptr.h" #include "base/unguessable_token.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "mojo/public/cpp/bindings/receiver.h" @@ -35,6 +37,11 @@ wallpaper_receiver_.Bind(std::move(receiver)); } +void FakePersonalizationAppWallpaperProvider::GetWallpaperAsPngBytes( + content::WebUIDataSource::GotDataCallback callback) { + std::move(callback).Run(base::MakeRefCounted<base::RefCountedBytes>()); +} + bool FakePersonalizationAppWallpaperProvider::IsEligibleForGooglePhotos() { return true; }
diff --git a/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.h b/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.h index 911fad6..11d658783 100644 --- a/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.h +++ b/ash/webui/personalization_app/test/fake_personalization_app_wallpaper_provider.h
@@ -12,6 +12,7 @@ #include "ash/public/cpp/wallpaper/wallpaper_types.h" #include "ash/webui/personalization_app/mojom/personalization_app.mojom.h" #include "base/unguessable_token.h" +#include "content/public/browser/web_ui_data_source.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "mojo/public/cpp/bindings/receiver.h" @@ -37,6 +38,10 @@ void BindInterface( mojo::PendingReceiver<ash::personalization_app::mojom::WallpaperProvider> receiver) override; + + void GetWallpaperAsPngBytes( + content::WebUIDataSource::GotDataCallback callback) override; + bool IsEligibleForGooglePhotos() override; // ash::personalization_app::mojom::WallpaperProvider:
diff --git a/ash/webui/personalization_app/test/personalization_app_mojom_banned_browsertest_fixture.cc b/ash/webui/personalization_app/test/personalization_app_mojom_banned_browsertest_fixture.cc index fcc9808..e8ebe1e 100644 --- a/ash/webui/personalization_app/test/personalization_app_mojom_banned_browsertest_fixture.cc +++ b/ash/webui/personalization_app/test/personalization_app_mojom_banned_browsertest_fixture.cc
@@ -12,6 +12,7 @@ #include "ash/webui/personalization_app/personalization_app_url_constants.h" #include "ash/webui/personalization_app/personalization_app_user_provider.h" #include "ash/webui/personalization_app/personalization_app_wallpaper_provider.h" +#include "base/memory/scoped_refptr.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "testing/gmock/include/gmock/gmock.h" @@ -118,6 +119,10 @@ ash::personalization_app::mojom::WallpaperProvider> receiver), (override)); bool IsEligibleForGooglePhotos() override { return true; } + void GetWallpaperAsPngBytes( + content::WebUIDataSource::GotDataCallback callback) override { + std::move(callback).Run(base::MakeRefCounted<base::RefCountedBytes>()); + } MOCK_METHOD(void, MakeTransparent, (), (override)); MOCK_METHOD(void, MakeOpaque, (), (override)); MOCK_METHOD(void,
diff --git a/ash/wm/desks/desks_controller.cc b/ash/wm/desks/desks_controller.cc index 71a26a0a..3b50636e 100644 --- a/ash/wm/desks/desks_controller.cc +++ b/ash/wm/desks/desks_controller.cc
@@ -1991,7 +1991,7 @@ void DesksController::ReportClosedWindowsCountPerSourceHistogram( DesksCreationRemovalSource source, int windows_closed) const { - std::string desk_removal_source_histogram; + const char* desk_removal_source_histogram = nullptr; switch (source) { case DesksCreationRemovalSource::kButton: desk_removal_source_histogram = kNumberOfWindowsClosedByButton; @@ -2011,9 +2011,9 @@ case DesksCreationRemovalSource::kDesksRestore: case DesksCreationRemovalSource::kEnsureDefaultDesk: break; - NOTREACHED(); } - base::UmaHistogramCounts100(desk_removal_source_histogram, windows_closed); + if (desk_removal_source_histogram) + base::UmaHistogramCounts100(desk_removal_source_histogram, windows_closed); } } // namespace ash
diff --git a/base/BUILD.gn b/base/BUILD.gn index a56c5fc8..d2df03e 100644 --- a/base/BUILD.gn +++ b/base/BUILD.gn
@@ -4486,7 +4486,6 @@ "test/android/javatests/src/org/chromium/base/test/util/Feature.java", "test/android/javatests/src/org/chromium/base/test/util/Features.java", "test/android/javatests/src/org/chromium/base/test/util/FeaturesBase.java", - "test/android/javatests/src/org/chromium/base/test/util/FlakyTest.java", "test/android/javatests/src/org/chromium/base/test/util/InMemorySharedPreferences.java", "test/android/javatests/src/org/chromium/base/test/util/InMemorySharedPreferencesContext.java", "test/android/javatests/src/org/chromium/base/test/util/InstrumentationUtils.java",
diff --git a/base/test/android/javatests/src/org/chromium/base/test/util/FlakyTest.java b/base/test/android/javatests/src/org/chromium/base/test/util/FlakyTest.java deleted file mode 100644 index 73d88328..0000000 --- a/base/test/android/javatests/src/org/chromium/base/test/util/FlakyTest.java +++ /dev/null
@@ -1,25 +0,0 @@ -// Copyright 2016 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -package org.chromium.base.test.util; - -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.RetentionPolicy; -import java.lang.annotation.Target; - -/** - * @deprecated Use "@DisabledTest" instead which has identical behavior. - * - * This annotation is for flaky tests. - * <p> - * Tests with this annotation will not be run on any of the normal bots. - * Please note that they might eventually run on a special bot. - */ -@Target({ElementType.METHOD, ElementType.TYPE}) -@Retention(RetentionPolicy.RUNTIME) -@Deprecated -public @interface FlakyTest { - String message() default ""; -}
diff --git a/chrome/android/javatests/src/org/chromium/chrome/browser/sync/SyncErrorCardPreferenceTest.java b/chrome/android/javatests/src/org/chromium/chrome/browser/sync/SyncErrorCardPreferenceTest.java index 95d9f46d..774ce60 100644 --- a/chrome/android/javatests/src/org/chromium/chrome/browser/sync/SyncErrorCardPreferenceTest.java +++ b/chrome/android/javatests/src/org/chromium/chrome/browser/sync/SyncErrorCardPreferenceTest.java
@@ -110,8 +110,31 @@ @LargeTest @Feature("RenderTest") @DisableFeatures({ChromeFeatureList.SYNC_ANDROID_PROMOS_WITH_TITLE}) + @EnableFeatures({ChromeFeatureList.UNIFIED_PASSWORD_MANAGER_ERROR_MESSAGES}) @ParameterAnnotations.UseMethodParameter(NightModeTestUtils.NightModeParams.class) - public void testSyncErrorCardForAuthErrorWithTitleFeatureDisabled(boolean nightModeEnabled) + public void testSyncErrorCardForAuthErrorWithTitleFeatureDisabledUpmEnabled( + boolean nightModeEnabled) throws Exception { + mFakeSyncServiceImpl.setAuthError(GoogleServiceAuthError.State.INVALID_GAIA_CREDENTIALS); + mSigninTestRule.addTestAccountThenSigninAndEnableSync(mFakeSyncServiceImpl); + TestThreadUtils.runOnUiThreadBlocking( + () + -> Assert.assertEquals("AUTH_ERROR SyncError should be set", + SyncSettingsUtils.SyncError.AUTH_ERROR, + SyncSettingsUtils.getSyncError())); + + mSettingsActivityTestRule.startSettingsActivity(); + mRenderTestRule.render( + getPersonalizedSyncPromoView(), "sync_error_card_auth_error_with_upm"); + } + + @Test + @LargeTest + @Feature("RenderTest") + @DisableFeatures({ChromeFeatureList.SYNC_ANDROID_PROMOS_WITH_TITLE, + ChromeFeatureList.UNIFIED_PASSWORD_MANAGER_ERROR_MESSAGES}) + @ParameterAnnotations.UseMethodParameter(NightModeTestUtils.NightModeParams.class) + public void + testSyncErrorCardForAuthErrorWithTitleFeatureDisabledUpmDisabled(boolean nightModeEnabled) throws Exception { mFakeSyncServiceImpl.setAuthError(GoogleServiceAuthError.State.INVALID_GAIA_CREDENTIALS); mSigninTestRule.addTestAccountThenSigninAndEnableSync(mFakeSyncServiceImpl); @@ -128,9 +151,11 @@ @Test @LargeTest @Feature("RenderTest") - @EnableFeatures({ChromeFeatureList.SYNC_ANDROID_PROMOS_WITH_TITLE}) + @EnableFeatures({ChromeFeatureList.SYNC_ANDROID_PROMOS_WITH_TITLE, + ChromeFeatureList.UNIFIED_PASSWORD_MANAGER_ERROR_MESSAGES}) @ParameterAnnotations.UseMethodParameter(NightModeTestUtils.NightModeParams.class) - public void testSyncErrorCardForAuthErrorWithTitleFeatureEnabled(boolean nightModeEnabled) + public void + testSyncErrorCardForAuthErrorWithTitleFeatureEnabledUpmEnabled(boolean nightModeEnabled) throws Exception { mFakeSyncServiceImpl.setAuthError(GoogleServiceAuthError.State.INVALID_GAIA_CREDENTIALS); mSigninTestRule.addTestAccountThenSigninAndEnableSync(mFakeSyncServiceImpl); @@ -141,6 +166,27 @@ SyncSettingsUtils.getSyncError())); mSettingsActivityTestRule.startSettingsActivity(); + mRenderTestRule.render(getPersonalizedSyncPromoView(), + "sync_error_card_auth_error_with_new_title_and_upm"); + } + + @Test + @LargeTest + @Feature("RenderTest") + @EnableFeatures({ChromeFeatureList.SYNC_ANDROID_PROMOS_WITH_TITLE}) + @DisableFeatures({ChromeFeatureList.UNIFIED_PASSWORD_MANAGER_ERROR_MESSAGES}) + @ParameterAnnotations.UseMethodParameter(NightModeTestUtils.NightModeParams.class) + public void testSyncErrorCardForAuthErrorWithTitleFeatureEnabledUpmDisabled( + boolean nightModeEnabled) throws Exception { + mFakeSyncServiceImpl.setAuthError(GoogleServiceAuthError.State.INVALID_GAIA_CREDENTIALS); + mSigninTestRule.addTestAccountThenSigninAndEnableSync(mFakeSyncServiceImpl); + TestThreadUtils.runOnUiThreadBlocking( + () + -> Assert.assertEquals("AUTH_ERROR SyncError should be set", + SyncSettingsUtils.SyncError.AUTH_ERROR, + SyncSettingsUtils.getSyncError())); + + mSettingsActivityTestRule.startSettingsActivity(); mRenderTestRule.render( getPersonalizedSyncPromoView(), "sync_error_card_auth_error_with_new_title"); }
diff --git a/chrome/browser/BUILD.gn b/chrome/browser/BUILD.gn index 4761e2d..9326ff0 100644 --- a/chrome/browser/BUILD.gn +++ b/chrome/browser/BUILD.gn
@@ -6925,6 +6925,13 @@ "//chrome/services/printing/public/mojom", "//components/crash/core/common", ] + if (is_win) { + sources += [ + "printing/printer_xml_parser_impl.cc", + "printing/printer_xml_parser_impl.h", + ] + deps += [ "//printing:printing_base" ] + } } if (is_win || enable_print_preview) { deps += [ "//chrome/services/printing/public/mojom" ]
diff --git a/chrome/browser/about_flags.cc b/chrome/browser/about_flags.cc index 3d061b0..708b17e 100644 --- a/chrome/browser/about_flags.cc +++ b/chrome/browser/about_flags.cc
@@ -3352,10 +3352,10 @@ #endif // BUILDFLAG(IS_ANDROID) const FeatureEntry::FeatureParam kUnthrottledNestedTimeout_NestingLevel = { - "nesting", "100"}; + "nesting", "15"}; const FeatureEntry::FeatureVariation kUnthrottledNestedTimeout_Variations[] = { - {"100", &kUnthrottledNestedTimeout_NestingLevel, 1, nullptr}, + {"15", &kUnthrottledNestedTimeout_NestingLevel, 1, nullptr}, }; constexpr FeatureEntry::FeatureParam kLensStandaloneWithSidePanel[] = { @@ -5016,6 +5016,9 @@ {"web-feed-sort", flag_descriptions::kWebFeedSortName, flag_descriptions::kWebFeedSortDescription, kOsAndroid, FEATURE_VALUE_TYPE(feed::kWebFeedSort)}, + {"feed-header-stick-to-top", flag_descriptions::kFeedHeaderStickToTopName, + flag_descriptions::kFeedHeaderStickToTopDescription, kOsAndroid, + FEATURE_VALUE_TYPE(feed::kFeedHeaderStickToTop)}, {"xsurface-metrics-reporting", flag_descriptions::kXsurfaceMetricsReportingName, flag_descriptions::kXsurfaceMetricsReportingDescription, kOsAndroid, @@ -8514,6 +8517,11 @@ kLensInstructionChipVariations, "LensInstructionChipImprovements")}, + {"enable-lens-region-search-static-page", + flag_descriptions::kLensRegionSearchStaticPageName, + flag_descriptions::kLensRegionSearchStaticPageDescription, kOsDesktop, + FEATURE_VALUE_TYPE(lens::features::kLensRegionSearchStaticPage)}, + {"enable-region-search-on-pdf-viewer", flag_descriptions::kEnableRegionSearchOnPdfViewerName, flag_descriptions::kEnableRegionSearchOnPdfViewerDescription, kOsDesktop, @@ -9202,7 +9210,7 @@ flag_descriptions::kIsolatedSandboxedIframesName, flag_descriptions::kIsolatedSandboxedIframesDescription, kOsDesktop, FEATURE_WITH_PARAMS_VALUE_TYPE( - features::kIsolateSandboxedIframes, + blink::features::kIsolateSandboxedIframes, kIsolateSandboxedIframesGroupingVariations, "IsolateSandboxedIframes" /* trial name */)}, #endif
diff --git a/chrome/browser/accessibility/accessibility_extension_api_chromeos.cc b/chrome/browser/accessibility/accessibility_extension_api_chromeos.cc index 940fb42..72b6edc5 100644 --- a/chrome/browser/accessibility/accessibility_extension_api_chromeos.cc +++ b/chrome/browser/accessibility/accessibility_extension_api_chromeos.cc
@@ -267,9 +267,14 @@ ::features::IsExperimentalAccessibilityDictationWithPumpkinEnabled(); break; case accessibility_private::AccessibilityFeature:: + ACCESSIBILITY_FEATURE_DICTATIONMORECOMMANDS: + enabled = + ::features::IsExperimentalAccessibilityDictationMoreCommandsEnabled(); + break; + case accessibility_private:: ACCESSIBILITY_FEATURE_SELECTTOSPEAKVOICESWITCHING: - enabled = ::features:: - IsExperimentalAccessibilitySelectToSpeakVoiceSwitchingEnabled(); + enabled = + ::features::IsExperimentalAccessibilitySelectToSpeakVoiceSwitchingEnabled(); break; case accessibility_private::AccessibilityFeature:: ACCESSIBILITY_FEATURE_NONE:
diff --git a/chrome/browser/android/chrome_backup_agent.cc b/chrome/browser/android/chrome_backup_agent.cc index a77391d..8c845a2b 100644 --- a/chrome/browser/android/chrome_backup_agent.cc +++ b/chrome/browser/android/chrome_backup_agent.cc
@@ -20,7 +20,7 @@ // TODO(crbug.com/1305213): The data type toggles shouldn't be individually // listed here. -static_assert(42 == syncer::GetNumModelTypes(), +static_assert(43 == syncer::GetNumModelTypes(), "If the new type has a corresponding pref, add it here"); const char* backed_up_preferences_[] = { autofill::prefs::kAutofillWalletImportEnabled,
diff --git a/chrome/browser/ash/accessibility/dictation_browsertest.cc b/chrome/browser/ash/accessibility/dictation_browsertest.cc index 70310bf..cee6d93 100644 --- a/chrome/browser/ash/accessibility/dictation_browsertest.cc +++ b/chrome/browser/ash/accessibility/dictation_browsertest.cc
@@ -954,6 +954,12 @@ DictationCommandsTest(const DictationCommandsTest&) = delete; DictationCommandsTest& operator=(const DictationCommandsTest&) = delete; + void SetUpCommandLine(base::CommandLine* command_line) override { + DictationTest::SetUpCommandLine(command_line); + scoped_feature_list_.InitAndEnableFeature( + ::features::kExperimentalAccessibilityDictationMoreCommands); + } + void SetUpOnMainThread() override { DictationTest::SetUpOnMainThread(); ToggleDictationWithKeystroke(); @@ -972,6 +978,9 @@ ui::ClipboardBuffer::kCopyPaste, /*data_dst=*/nullptr, &text); return base::UTF16ToUTF8(text); } + + private: + base::test::ScopedFeatureList scoped_feature_list_; }; INSTANTIATE_TEST_SUITE_P( @@ -1153,6 +1162,22 @@ "This is a t."); } +IN_PROC_BROWSER_TEST_P(DictationCommandsTest, DeleteAllTextSimple) { + SendFinalResultAndWaitForTextAreaValue("Hello, world.", "Hello, world."); + SendFinalResultAndWaitForTextAreaValue("delete all", ""); +} + +IN_PROC_BROWSER_TEST_P(DictationCommandsTest, DeleteAllTextMultiLineString) { + std::string text = " Hello, world. \n Hello, world. \n Hello, world. \n"; + SendFinalResultAndWaitForTextAreaValue(text, text); + SendFinalResultAndWaitForTextAreaValue("delete all", ""); +} + +// TODO(crbug.com/1362842) Add a test where +// you disable the MoreCommands Feature Flag +// and ensure that you can't run the Dictation +// Commands under that feature flag + IN_PROC_BROWSER_TEST_P(DictationCommandsTest, DeletePrevSentSimple) { SendFinalResultAndWaitForTextAreaValue("Hello, world.", "Hello, world."); SendFinalResultAndWaitForTextAreaValue("delete the previous sentence", "");
diff --git a/chrome/browser/ash/app_restore/full_restore_app_launch_handler.cc b/chrome/browser/ash/app_restore/full_restore_app_launch_handler.cc index 2616970..7e67100 100644 --- a/chrome/browser/ash/app_restore/full_restore_app_launch_handler.cc +++ b/chrome/browser/ash/app_restore/full_restore_app_launch_handler.cc
@@ -8,8 +8,6 @@ #include <utility> #include "ash/constants/ash_switches.h" -#include "ash/metrics/login_unlock_throughput_recorder.h" -#include "ash/shell.h" #include "base/bind.h" #include "base/command_line.h" #include "base/metrics/histogram_functions.h" @@ -451,10 +449,6 @@ return; } - LoginUnlockThroughputRecorder* throughput_recorder = - Shell::HasInstance() ? Shell::Get()->login_unlock_throughput_recorder() - : nullptr; - int arc_app_count = 0; int other_app_count = 0; for (const auto& it : restore_data()->app_id_to_launch_list()) { @@ -466,12 +460,6 @@ continue; } - if (throughput_recorder) { - for (const auto& window : it.second) { - throughput_recorder->AddScheduledRestoreWindow( - window.first, it.first, LoginUnlockThroughputRecorder::kBrowser); - } - } ++other_app_count; } VLOG(1) << "There is restore data: Browser("
diff --git a/chrome/browser/ash/crosapi/browser_data_migrator_util.h b/chrome/browser/ash/crosapi/browser_data_migrator_util.h index fed99c7..ef6b1c8 100644 --- a/chrome/browser/ash/crosapi/browser_data_migrator_util.h +++ b/chrome/browser/ash/crosapi/browser_data_migrator_util.h
@@ -302,7 +302,7 @@ }; // List of data types in Sync Data that have to stay in Ash and Ash only. -static_assert(42 == syncer::GetNumModelTypes(), +static_assert(43 == syncer::GetNumModelTypes(), "If adding a new sync data type, update the lists below if" " you want to keep the new data type in Ash only."); constexpr syncer::ModelType kAshOnlySyncDataTypes[] = {
diff --git a/chrome/browser/ash/net/network_diagnostics/network_diagnostics_util_unittest.cc b/chrome/browser/ash/net/network_diagnostics/network_diagnostics_util_unittest.cc index d2691e0..137d318 100644 --- a/chrome/browser/ash/net/network_diagnostics/network_diagnostics_util_unittest.cc +++ b/chrome/browser/ash/net/network_diagnostics/network_diagnostics_util_unittest.cc
@@ -7,6 +7,7 @@ #include <string> #include <vector> +#include "base/ranges/algorithm.h" #include "base/strings/string_util.h" #include "testing/gtest/include/gtest/gtest.h" @@ -37,8 +38,7 @@ util::GetRandomHosts(num_hosts, prefix_length); // Ensure |random_hosts| has unique entries. std::sort(random_hosts.begin(), random_hosts.end()); - EXPECT_TRUE(std::adjacent_find(random_hosts.begin(), random_hosts.end()) == - random_hosts.end()); + EXPECT_TRUE(base::ranges::adjacent_find(random_hosts) == random_hosts.end()); } TEST(NetworkDiagnosticsUtilTest, TestGetRandomHostsWithScheme) { @@ -48,8 +48,7 @@ util::GetRandomHostsWithScheme(num_hosts, prefix_length, kHttpsScheme); // Ensure |random_hosts| has unique entries. std::sort(random_hosts.begin(), random_hosts.end()); - EXPECT_TRUE(std::adjacent_find(random_hosts.begin(), random_hosts.end()) == - random_hosts.end()); + EXPECT_TRUE(base::ranges::adjacent_find(random_hosts) == random_hosts.end()); // Ensure hosts in |random_hosts| start with |kHttpsScheme|. for (const auto& host : random_hosts) { EXPECT_TRUE(host.rfind(kHttpsScheme, 0) == 0); @@ -65,8 +64,7 @@ kHttpsScheme); // Ensure |random_hosts| has unique entries. std::sort(random_hosts.begin(), random_hosts.end()); - EXPECT_TRUE(std::adjacent_find(random_hosts.begin(), random_hosts.end()) == - random_hosts.end()); + EXPECT_TRUE(base::ranges::adjacent_find(random_hosts) == random_hosts.end()); // Ensure: // (1) hosts in |random_hosts| start with |kHttpsScheme|. // (2) hosts in |random_hosts| end with |kGenerate204Path|.
diff --git a/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.cc b/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.cc index 80b0b1b..0080e20 100644 --- a/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.cc +++ b/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.cc
@@ -26,9 +26,12 @@ #include "ash/webui/personalization_app/proto/backdrop_wallpaper.pb.h" #include "base/bind.h" #include "base/files/file_path.h" +#include "base/memory/ref_counted_memory.h" #include "base/notreached.h" #include "base/rand_util.h" #include "base/strings/utf_string_conversions.h" +#include "base/task/task_traits.h" +#include "base/task/thread_pool.h" #include "base/unguessable_token.h" #include "chrome/browser/ash/profiles/profile_helper.h" #include "chrome/browser/ash/wallpaper/wallpaper_enumerator.h" @@ -47,6 +50,7 @@ #include "content/public/browser/url_data_source.h" #include "content/public/browser/web_contents.h" #include "content/public/browser/web_ui.h" +#include "content/public/browser/web_ui_data_source.h" #include "mojo/public/cpp/bindings/message.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "mojo/public/cpp/bindings/pending_remote.h" @@ -59,6 +63,7 @@ #include "ui/aura/window.h" #include "ui/base/webui/web_ui_util.h" #include "ui/gfx/geometry/size.h" +#include "ui/gfx/image/image.h" #include "ui/gfx/image/image_skia.h" #include "ui/gfx/image/image_skia_operations.h" #include "url/gurl.h" @@ -72,7 +77,7 @@ using ash::personalization_app::GetUser; constexpr int kLocalImageThumbnailSizeDip = 256; -constexpr int kCurrentWallpaperThumbnailSizeDip = 512; +constexpr int kCurrentWallpaperThumbnailSizeDip = 1024; const gfx::ImageSkia GetResizedImage(const gfx::ImageSkia& image) { // Resize the image maintaining our aspect ratio. @@ -96,6 +101,19 @@ : base::UnguessableToken::Create().ToString(); } +scoped_refptr<base::RefCountedMemory> ResizeAndEncodeWallpaperImage() { + auto* wallpaper_controller = ash::WallpaperController::Get(); + // Get wallpaper image on worker thread for performance reasons. This avoids + // having to call |image.MakeThreadSafe| in a performance critical path while + // changing wallpaper, and instead calling it in the thread pool. + auto image = wallpaper_controller->GetWallpaperImage(); + image.MakeThreadSafe(); + auto resized = GetResizedImage(image); + scoped_refptr<base::RefCountedMemory> png_bytes = + gfx::Image(resized).As1xPNGBytes(); + return png_bytes; +} + } // namespace PersonalizationAppWallpaperProviderImpl:: @@ -125,6 +143,19 @@ wallpaper_receiver_.Bind(std::move(receiver)); } +void PersonalizationAppWallpaperProviderImpl::GetWallpaperAsPngBytes( + content::WebUIDataSource::GotDataCallback callback) { + // |GetWallpaperAsPngBytes| is called in the hot path of switching wallpaper + // on the UI thread right after user makes a new selection. Make sure to do + // resizing and encoding on a task runner to avoid locking up the UI as the + // user's wallpaper is being set. + base::ThreadPool::PostTaskAndReplyWithResult( + FROM_HERE, + {base::TaskPriority::USER_VISIBLE, + base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN}, + base::BindOnce(&ResizeAndEncodeWallpaperImage), std::move(callback)); +} + bool PersonalizationAppWallpaperProviderImpl::IsEligibleForGooglePhotos() { return GetUser(profile_)->HasGaiaAccount(); } @@ -353,7 +384,6 @@ wallpaper_attribution_info_fetcher_.reset(); attribution_weak_ptr_factory_.InvalidateWeakPtrs(); - auto* controller = WallpaperController::Get(); auto* client = WallpaperControllerClientImpl::Get(); absl::optional<ash::WallpaperInfo> info = @@ -365,12 +395,6 @@ return; } - const gfx::ImageSkia& current_wallpaper = controller->GetWallpaperImage(); - const gfx::ImageSkia& current_wallpaper_resized = - GetResizedImage(current_wallpaper); - const GURL& wallpaper_data_url = - GURL(webui::GetBitmapDataUrl(*current_wallpaper_resized.bitmap())); - switch (info->type) { case ash::WallpaperType::kDaily: case ash::WallpaperType::kOnline: { @@ -380,15 +404,13 @@ // look up all collections and match URL. FetchCollections(base::BindOnce( &PersonalizationAppWallpaperProviderImpl::FindAttribution, - attribution_weak_ptr_factory_.GetWeakPtr(), *info, - wallpaper_data_url)); + attribution_weak_ptr_factory_.GetWeakPtr(), *info)); return; } backdrop::Collection collection; collection.set_collection_id(info->collection_id); - FindAttribution(*info, wallpaper_data_url, - std::vector<backdrop::Collection>{collection}); + FindAttribution(*info, std::vector<backdrop::Collection>{collection}); return; } case ash::WallpaperType::kCustomized: { @@ -405,8 +427,7 @@ NotifyWallpaperChanged( ash::personalization_app::mojom::CurrentWallpaper::New( - wallpaper_data_url, std::move(attribution), info->layout, - info->type, key)); + std::move(attribution), info->layout, info->type, key)); return; } @@ -416,8 +437,7 @@ GetAccountId(profile_), info->location, base::BindOnce(&PersonalizationAppWallpaperProviderImpl:: SendGooglePhotosAttribution, - weak_ptr_factory_.GetWeakPtr(), *info, - wallpaper_data_url)); + weak_ptr_factory_.GetWeakPtr(), *info)); return; case ash::WallpaperType::kDefault: case ash::WallpaperType::kDevice: @@ -426,8 +446,8 @@ case ash::WallpaperType::kThirdParty: NotifyWallpaperChanged( ash::personalization_app::mojom::CurrentWallpaper::New( - wallpaper_data_url, /*attribution=*/std::vector<std::string>(), - info->layout, info->type, + /*attribution=*/std::vector<std::string>(), info->layout, + info->type, /*key=*/base::UnguessableToken::Create().ToString())); return; case ash::WallpaperType::kCount: @@ -789,14 +809,13 @@ void PersonalizationAppWallpaperProviderImpl::FindAttribution( const ash::WallpaperInfo& info, - const GURL& wallpaper_data_url, const absl::optional<std::vector<backdrop::Collection>>& collections) { DCHECK(!wallpaper_attribution_info_fetcher_); if (!collections.has_value() || collections->empty()) { NotifyWallpaperChanged( ash::personalization_app::mojom::CurrentWallpaper::New( - wallpaper_data_url, /*attribution=*/std::vector<std::string>(), - info.layout, info.type, GetOnlineWallpaperKey(info))); + /*attribution=*/std::vector<std::string>(), info.layout, info.type, + GetOnlineWallpaperKey(info))); return; } @@ -808,13 +827,12 @@ wallpaper_attribution_info_fetcher_->Start(base::BindOnce( &PersonalizationAppWallpaperProviderImpl::FindAttributionInCollection, - attribution_weak_ptr_factory_.GetWeakPtr(), info, wallpaper_data_url, - current_index, collections)); + attribution_weak_ptr_factory_.GetWeakPtr(), info, current_index, + collections)); } void PersonalizationAppWallpaperProviderImpl::FindAttributionInCollection( const ash::WallpaperInfo& info, - const GURL& wallpaper_data_url, std::size_t current_index, const absl::optional<std::vector<backdrop::Collection>>& collections, bool success, @@ -843,7 +861,7 @@ attributions.push_back(attr.text()); NotifyWallpaperChanged( ash::personalization_app::mojom::CurrentWallpaper::New( - wallpaper_data_url, attributions, info.layout, info.type, + attributions, info.layout, info.type, /*key=*/base::NumberToString(backend_image->asset_id()))); wallpaper_attribution_info_fetcher_.reset(); return; @@ -854,8 +872,8 @@ if (current_index >= collections->size()) { NotifyWallpaperChanged( ash::personalization_app::mojom::CurrentWallpaper::New( - wallpaper_data_url, /*attribution=*/std::vector<std::string>(), - info.layout, info.type, GetOnlineWallpaperKey(info))); + /*attribution=*/std::vector<std::string>(), info.layout, info.type, + GetOnlineWallpaperKey(info))); wallpaper_attribution_info_fetcher_.reset(); return; } @@ -864,8 +882,8 @@ collections->at(current_index).collection_id()); fetcher->Start(base::BindOnce( &PersonalizationAppWallpaperProviderImpl::FindAttributionInCollection, - attribution_weak_ptr_factory_.GetWeakPtr(), info, wallpaper_data_url, - current_index, collections)); + attribution_weak_ptr_factory_.GetWeakPtr(), info, current_index, + collections)); // resetting the previous fetcher last because the current method is bound // to a callback owned by the previous fetcher. wallpaper_attribution_info_fetcher_ = std::move(fetcher); @@ -873,7 +891,6 @@ void PersonalizationAppWallpaperProviderImpl::SendGooglePhotosAttribution( const ash::WallpaperInfo& info, - const GURL& wallpaper_data_url, mojo::StructPtr<ash::personalization_app::mojom::GooglePhotosPhoto> photo, bool success) { // If the fetch for |photo| succeeded but |photo| does not exist, that means @@ -900,7 +917,7 @@ // NOTE: Old clients may not support |dedup_key| when setting Google Photos // wallpaper, so use |location| in such cases for backwards compatibility. NotifyWallpaperChanged(ash::personalization_app::mojom::CurrentWallpaper::New( - wallpaper_data_url, attribution, info.layout, info.type, + attribution, info.layout, info.type, /*key=*/info.dedup_key.value_or(info.location))); }
diff --git a/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.h b/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.h index e1324f63..689d6566 100644 --- a/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.h +++ b/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl.h
@@ -23,6 +23,7 @@ #include "base/memory/weak_ptr.h" #include "base/scoped_observation.h" #include "components/account_id/account_id.h" +#include "content/public/browser/web_ui_data_source.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "mojo/public/cpp/bindings/pending_remote.h" #include "mojo/public/cpp/bindings/receiver.h" @@ -77,6 +78,9 @@ mojo::PendingReceiver<ash::personalization_app::mojom::WallpaperProvider> receiver) override; + void GetWallpaperAsPngBytes( + content::WebUIDataSource::GotDataCallback callback) override; + // Not all users can see google photos. Requires a gaia account to be able to // fetch photos. bool IsEligibleForGooglePhotos() override; @@ -232,12 +236,10 @@ void FindAttribution( const ash::WallpaperInfo& info, - const GURL& wallpaper_data_url, const absl::optional<std::vector<backdrop::Collection>>& collections); void FindAttributionInCollection( const ash::WallpaperInfo& info, - const GURL& wallpaper_data_url, std::size_t current_index, const absl::optional<std::vector<backdrop::Collection>>& collections, bool success, @@ -246,7 +248,6 @@ void SendGooglePhotosAttribution( const ash::WallpaperInfo& info, - const GURL& wallpaper_data_url, mojo::StructPtr<ash::personalization_app::mojom::GooglePhotosPhoto> photo, bool success);
diff --git a/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl_unittest.cc b/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl_unittest.cc index cdf4efd..8e544e7 100644 --- a/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl_unittest.cc +++ b/chrome/browser/ash/web_applications/personalization_app/personalization_app_wallpaper_provider_impl_unittest.cc
@@ -19,6 +19,9 @@ #include "ash/webui/personalization_app/mojom/personalization_app.mojom.h" #include "base/callback_helpers.h" #include "base/json/json_reader.h" +#include "base/memory/ref_counted_memory.h" +#include "base/memory/scoped_refptr.h" +#include "base/run_loop.h" #include "base/test/bind.h" #include "base/values.h" #include "chrome/browser/ash/login/users/fake_chrome_user_manager.h" @@ -55,6 +58,7 @@ #include "third_party/skia/include/core/SkBitmap.h" #include "third_party/skia/include/core/SkColor.h" #include "ui/base/webui/web_ui_util.h" +#include "ui/gfx/codec/png_codec.h" namespace ash::personalization_app { @@ -413,7 +417,6 @@ TEST_P(PersonalizationAppWallpaperProviderImplTest, ObserveWallpaperFiresWhenBound) { - // This will create the data url referenced below in expectation. test_wallpaper_controller()->ShowWallpaperImage( CreateSolidImageSkia(/*width=*/1, /*height=*/1, SK_ColorBLACK)); @@ -443,10 +446,6 @@ EXPECT_EQ(ash::WallpaperType::kOnline, current->type); EXPECT_EQ(ash::WallpaperLayout::WALLPAPER_LAYOUT_CENTER_CROPPED, current->layout); - // Data url of a solid black image scaled up to 512x512. - EXPECT_EQ(webui::GetBitmapDataUrl( - *CreateSolidImageSkia(512, 512, SK_ColorBLACK).bitmap()), - current->url); } TEST_P(PersonalizationAppWallpaperProviderImplTest, SetCurrentWallpaperLayout) { @@ -485,6 +484,33 @@ ResetWallpaperProvider(); } +TEST_P(PersonalizationAppWallpaperProviderImplTest, GetWallpaperAsPngBytes) { + test_wallpaper_controller()->ShowWallpaperImage( + CreateSolidImageSkia(/*width=*/1, /*height=*/1, SK_ColorRED)); + + scoped_refptr<base::RefCountedMemory> png_bytes; + + base::RunLoop loop; + delegate()->GetWallpaperAsPngBytes(base::BindLambdaForTesting( + [quit = loop.QuitClosure(), + &png_bytes](scoped_refptr<base::RefCountedMemory> bytes) { + bytes.swap(png_bytes); + std::move(quit).Run(); + })); + loop.Run(); + + // Png bytes of a solid black image scaled up to 1024x1024. + scoped_refptr<base::RefCountedBytes> expected_png_bytes = + base::MakeRefCounted<base::RefCountedBytes>(); + gfx::PNGCodec::EncodeBGRASkBitmap( + CreateSolidImageSkia(1024, 1024, SK_ColorRED) + .GetRepresentation(/*scale=*/1) + .GetBitmap(), + /*discard_transparency=*/false, &expected_png_bytes->data()); + + EXPECT_TRUE(expected_png_bytes->Equals(png_bytes)); +} + class PersonalizationAppWallpaperProviderImplGooglePhotosTest : public PersonalizationAppWallpaperProviderImplTest { protected:
diff --git a/chrome/browser/creator/android/BUILD.gn b/chrome/browser/creator/android/BUILD.gn index c8afce0..cc84bdb0 100644 --- a/chrome/browser/creator/android/BUILD.gn +++ b/chrome/browser/creator/android/BUILD.gn
@@ -12,10 +12,16 @@ deps = [ ":creator_java_resources", "//base:base_java", + "//chrome/browser/feed/android:feed_java_resources", + "//chrome/browser/feed/android:java", + "//chrome/browser/xsurface:java", + "//components/browser_ui/styles/android:java", + "//components/browser_ui/widget/android:java", "//content/public/android:content_main_dex_java", "//third_party/androidx:androidx_annotation_annotation_java", "//third_party/androidx:androidx_appcompat_appcompat_java", "//third_party/androidx:androidx_appcompat_appcompat_resources_java", + "//third_party/androidx:androidx_recyclerview_recyclerview_java", "//ui/android:ui_no_recycler_view_java", "//url:gurl_java", ] @@ -27,6 +33,8 @@ sources = [ "java/res/drawable/profile_background.xml", "java/res/layout/creator_activity.xml", + "java/res/values/dimens.xml", + "java/res/values/ids.xml", "java/res/values/styles.xml", ] deps = [
diff --git a/chrome/browser/creator/android/java/res/values/dimens.xml b/chrome/browser/creator/android/java/res/values/dimens.xml new file mode 100644 index 0000000..c94fbaf --- /dev/null +++ b/chrome/browser/creator/android/java/res/values/dimens.xml
@@ -0,0 +1,11 @@ +<?xml version="1.0" encoding="utf-8"?> +<!-- +Copyright 2022 The Chromium Authors +Use of this source code is governed by a BSD-style license that can be +found in the LICENSE file. +--> + +<resources xmlns:tools="http://schemas.android.com/tools"> + <!-- Content Previews dimensions. --> + <dimen name="content_previews_padding">16dp</dimen> +</resources>
diff --git a/chrome/browser/creator/android/java/res/values/ids.xml b/chrome/browser/creator/android/java/res/values/ids.xml new file mode 100644 index 0000000..d776e90 --- /dev/null +++ b/chrome/browser/creator/android/java/res/values/ids.xml
@@ -0,0 +1,12 @@ +<?xml version="1.0" encoding="utf-8"?> + +<!-- +Copyright 2022 The Chromium Authors +Use of this source code is governed by a BSD-style license that can be +found in the LICENSE file. +--> + +<resources> + <!-- Id to use for Content Previews RecyclerView. --> + <item type="id" name="creator_feed_stream_recycler_view" /> +</resources>
diff --git a/chrome/browser/creator/android/java/src/org/chromium/chrome/browser/creator/CreatorCoordinator.java b/chrome/browser/creator/android/java/src/org/chromium/chrome/browser/creator/CreatorCoordinator.java index 291441da..f9f158f 100644 --- a/chrome/browser/creator/android/java/src/org/chromium/chrome/browser/creator/CreatorCoordinator.java +++ b/chrome/browser/creator/android/java/src/org/chromium/chrome/browser/creator/CreatorCoordinator.java
@@ -6,27 +6,95 @@ import android.app.Activity; import android.view.LayoutInflater; -import android.view.View; +import android.view.ViewGroup; + +import androidx.recyclerview.widget.RecyclerView; + +import org.chromium.chrome.browser.feed.FeedSurfaceScopeDependencyProvider; +import org.chromium.chrome.browser.feed.FeedSurfaceTracker; +import org.chromium.chrome.browser.feed.NativeViewListRenderer; +import org.chromium.chrome.browser.feed.NtpListContentManager; +import org.chromium.chrome.browser.xsurface.HybridListRenderer; +import org.chromium.chrome.browser.xsurface.ProcessScope; +import org.chromium.chrome.browser.xsurface.SurfaceScope; +import org.chromium.components.browser_ui.styles.SemanticColorUtils; + +import java.util.ArrayList; +import java.util.List; /** * Sets up the Coordinator for Cormorant Creator surface. It is based on the doc at * https://chromium.googlesource.com/chromium/src/+/HEAD/docs/ui/android/mvc_simple_list_tutorial.md */ public class CreatorCoordinator { + private static final String NATIVE_CONTENT_ID = "0"; + private CreatorMediator mMediator; private Activity mActivity; - private final View mView; + private NtpListContentManager mContentManager; + private RecyclerView mRecyclerView; + private HybridListRenderer mHybridListRenderer; + private SurfaceScope mSurfaceScope; + private FeedSurfaceScopeDependencyProvider mDependencyProvider; + + private final ViewGroup mViewGroup; public CreatorCoordinator(Activity activity) { mActivity = activity; + mRecyclerView = setUpView(); + List<NtpListContentManager.FeedContent> contentPreviewsList = new ArrayList<>(); + + // Add empty state to Content Manager + contentPreviewsList.add(new NtpListContentManager.NativeViewContent( + getLateralPaddingsPx(), NATIVE_CONTENT_ID, R.layout.no_content_v2)); + mContentManager.addContents(0, contentPreviewsList); // Inflate the XML. - mView = LayoutInflater.from(mActivity).inflate(R.layout.creator_activity, null); + mViewGroup = + (ViewGroup) LayoutInflater.from(mActivity).inflate(R.layout.creator_activity, null); + mViewGroup.addView(mRecyclerView); mMediator = new CreatorMediator(mActivity); } - public View getView() { - return mView; + public ViewGroup getView() { + return mViewGroup; + } + + private RecyclerView setUpView() { + // TODO(crbug.com/1374744): Refactor NTP naming out of the general Feed code. + mContentManager = new NtpListContentManager(); + ProcessScope processScope = FeedSurfaceTracker.getInstance().getXSurfaceProcessScope(); + + if (processScope != null) { + mDependencyProvider = + new FeedSurfaceScopeDependencyProvider(mActivity, mActivity, false); + mSurfaceScope = processScope.obtainSurfaceScope(mDependencyProvider); + } else { + mDependencyProvider = null; + mSurfaceScope = null; + } + + if (mSurfaceScope != null) { + mHybridListRenderer = mSurfaceScope.provideListRenderer(); + } else { + mHybridListRenderer = new NativeViewListRenderer(mActivity); + } + + RecyclerView view; + if (mHybridListRenderer != null) { + view = (RecyclerView) mHybridListRenderer.bind( + mContentManager, /* mViewportView */ null, /* useStaggeredLayout */ false); + view.setId(R.id.creator_feed_stream_recycler_view); + view.setClipToPadding(false); + view.setBackgroundColor(SemanticColorUtils.getDefaultBgColor(mActivity)); + } else { + view = null; + } + return view; + } + + private int getLateralPaddingsPx() { + return mActivity.getResources().getDimensionPixelSize(R.dimen.content_previews_padding); } }
diff --git a/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.cc b/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.cc index b5c8450..e85586e1 100644 --- a/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.cc +++ b/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.cc
@@ -46,10 +46,3 @@ address, port); return extensions::SocketsManifestData::CheckRequest(extension, param); } - -bool ChromeDirectSocketsDelegate::ShouldSkipPostResolveChecks( - content::RenderFrameHost* frame) const { - // chrome.sockets.* doesn't have any restrictions on resolved addresses, so we - // may skip these checks. - return IsExtensionFrame(frame); -}
diff --git a/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.h b/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.h index 3fb4421f..174b2d8 100644 --- a/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.h +++ b/chrome/browser/direct_sockets/chrome_direct_sockets_delegate.h
@@ -15,7 +15,6 @@ const std::string& address, uint16_t port, blink::mojom::DirectSocketProtocolType protocol) const override; - bool ShouldSkipPostResolveChecks(content::RenderFrameHost*) const override; }; #endif // CHROME_BROWSER_DIRECT_SOCKETS_CHROME_DIRECT_SOCKETS_DELEGATE_H_
diff --git a/chrome/browser/feed/android/java/src/org/chromium/chrome/browser/feed/FeedSurfaceScopeDependencyProvider.java b/chrome/browser/feed/android/java/src/org/chromium/chrome/browser/feed/FeedSurfaceScopeDependencyProvider.java index 564887d..c1f38013 100644 --- a/chrome/browser/feed/android/java/src/org/chromium/chrome/browser/feed/FeedSurfaceScopeDependencyProvider.java +++ b/chrome/browser/feed/android/java/src/org/chromium/chrome/browser/feed/FeedSurfaceScopeDependencyProvider.java
@@ -20,7 +20,8 @@ /** * Provides activity, darkmode and logging context for a single surface. */ -class FeedSurfaceScopeDependencyProvider implements SurfaceScopeDependencyProvider, ScrollListener { +public class FeedSurfaceScopeDependencyProvider + implements SurfaceScopeDependencyProvider, ScrollListener { private static final String TAG = "Feed"; private final Activity mActivity; private final Context mActivityContext;
diff --git a/chrome/browser/first_party_sets/first_party_sets_policy_service.cc b/chrome/browser/first_party_sets/first_party_sets_policy_service.cc index d2404db..bce556a 100644 --- a/chrome/browser/first_party_sets/first_party_sets_policy_service.cc +++ b/chrome/browser/first_party_sets/first_party_sets_policy_service.cc
@@ -51,29 +51,17 @@ : browser_context_(browser_context) { DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); DCHECK(browser_context); - Init([](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - content::FirstPartySetsHandler::GetInstance()->GetContextConfigForPolicy( - GetOverridesPolicyForProfile(prefs), std::move(callback)); - }); + Init(); } FirstPartySetsPolicyService::~FirstPartySetsPolicyService() = default; -void FirstPartySetsPolicyService::InitForTesting( - base::FunctionRef< - void(PrefService*, - base::OnceCallback<void(net::FirstPartySetsContextConfig)>)> - get_config) { +void FirstPartySetsPolicyService::InitForTesting() { DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); - Init(get_config); + Init(); } -void FirstPartySetsPolicyService::Init( - base::FunctionRef< - void(PrefService*, - base::OnceCallback<void(net::FirstPartySetsContextConfig)>)> - get_config) { +void FirstPartySetsPolicyService::Init() { DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); if (!base::FeatureList::IsEnabled(features::kFirstPartySets)) { OnReadyToNotifyDelegates(net::FirstPartySetsContextConfig(), @@ -98,14 +86,15 @@ // dynamically refresh, and all the delegates for `context` will have the same // policy and thus the same config. PrefService* prefs = profile->GetPrefs(); - get_config(prefs, base::BindOnce( - &FirstPartySetsPolicyService::OnProfileConfigReady, - weak_factory_.GetWeakPtr(), - // We should only clear site data if First-Party Sets is - // enabled when the service is created, to allow users - // to play with the FPS enabled setting without - // affecting user experience during the browser session. - GetEnabledPolicyForProfile(prefs))); + content::FirstPartySetsHandler::GetInstance()->GetContextConfigForPolicy( + GetOverridesPolicyForProfile(prefs), + base::BindOnce(&FirstPartySetsPolicyService::OnProfileConfigReady, + weak_factory_.GetWeakPtr(), + // We should only clear site data if First-Party Sets is + // enabled when the service is created, to allow users + // to play with the FPS enabled setting without + // affecting user experience during the browser session. + GetEnabledPolicyForProfile(prefs))); } void FirstPartySetsPolicyService::AddRemoteAccessDelegate(
diff --git a/chrome/browser/first_party_sets/first_party_sets_policy_service.h b/chrome/browser/first_party_sets/first_party_sets_policy_service.h index 2d995457..8ab0653 100644 --- a/chrome/browser/first_party_sets/first_party_sets_policy_service.h +++ b/chrome/browser/first_party_sets/first_party_sets_policy_service.h
@@ -14,8 +14,6 @@ #include "mojo/public/cpp/bindings/remote_set.h" #include "services/network/public/mojom/first_party_sets_access_delegate.mojom.h" -class PrefService; - namespace content { class BrowserContext; } // namespace content @@ -76,16 +74,11 @@ // factory for FirstPartySetsPolicyService instances in tests, so every // instance calls into the prod logic to eagerly initialize itself. This // method allows tests to wait for that eager initialization to complete, then - // reset state, and re-run initialization via `InitForTesting`. + // reset state, and re-run initialization via `Init`. void WaitForFirstInitCompleteForTesting(base::OnceClosure callback); - // Testing-only method that allows injecting different logic to get the - // config. - void InitForTesting( - base::FunctionRef< - void(PrefService*, - base::OnceCallback<void(net::FirstPartySetsContextConfig)>)> - get_config); + // Exposes `Init` for use in tests. + void InitForTesting(); // Returns true when this instance has received the config thus has been fully // initialized. @@ -122,11 +115,8 @@ } private: - // Initialize this instance by getting the config via `get_config` if needed. - void Init(base::FunctionRef< - void(PrefService*, - base::OnceCallback<void(net::FirstPartySetsContextConfig)>)> - get_config); + // Initialize this instance by getting the config if needed. + void Init(); // Sets the `config_` member and provides it to all delegates via NotifyReady. void OnReadyToNotifyDelegates(net::FirstPartySetsContextConfig config,
diff --git a/chrome/browser/first_party_sets/first_party_sets_policy_service_unittest.cc b/chrome/browser/first_party_sets/first_party_sets_policy_service_unittest.cc index 81278fe..02cd1fe 100644 --- a/chrome/browser/first_party_sets/first_party_sets_policy_service_unittest.cc +++ b/chrome/browser/first_party_sets/first_party_sets_policy_service_unittest.cc
@@ -19,6 +19,7 @@ #include "net/base/schemeful_site.h" #include "net/first_party_sets/first_party_set_entry.h" #include "net/first_party_sets/first_party_sets_context_config.h" +#include "net/first_party_sets/global_first_party_sets.h" #include "services/network/public/mojom/first_party_sets_access_delegate.mojom.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" @@ -146,8 +147,11 @@ void SetUp() override { DefaultFirstPartySetsPolicyServiceTest::SetUp(); - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting({}); + content::FirstPartySetsHandler::GetInstance()->SetInstanceForTesting( + &first_party_sets_handler_); + first_party_sets_handler_.SetGlobalSets({}); + SetContextConfig(net::FirstPartySetsContextConfig()); + SetCacheFilter(net::FirstPartySetsCacheFilter()); profile_manager_ = std::make_unique<TestingProfileManager>( TestingBrowserProcess::GetGlobal()); @@ -175,15 +179,30 @@ // tests if the factory has already created a service for the testing // profile being used. service_->ResetForTesting(); + content::FirstPartySetsHandler::GetInstance()->SetInstanceForTesting( + nullptr); profile_manager_->DeleteAllTestingProfiles(); profile_manager_.reset(); } + void SetContextConfig(net::FirstPartySetsContextConfig config) { + first_party_sets_handler_.SetContextConfig(std::move(config)); + } + + void SetCacheFilter(net::FirstPartySetsCacheFilter cache_filter) { + first_party_sets_handler_.SetCacheFilter(std::move(cache_filter)); + } + + void SetGlobalSets(net::GlobalFirstPartySets global_sets) { + first_party_sets_handler_.SetGlobalSets(std::move(global_sets)); + } + protected: Profile* profile() { return profile_; } FirstPartySetsPolicyService* service() { return service_; } private: + MockFirstPartySetsHandler first_party_sets_handler_; std::unique_ptr<TestingProfileManager> profile_manager_; Profile* profile_; base::test::ScopedFeatureList features_; @@ -196,15 +215,12 @@ } TEST_F(FirstPartySetsPolicyServiceTest, IsSiteInManagedSet_SiteNotInConfig) { - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig( - {{net::SchemefulSite(GURL("https://example.test")), - {net::FirstPartySetEntry( - net::SchemefulSite(GURL("https://primary.test")), - net::SiteType::kAssociated, absl::nullopt)}}})); - }); + SetContextConfig(net::FirstPartySetsContextConfig( + {{net::SchemefulSite(GURL("https://example.test")), + {net::FirstPartySetEntry( + net::SchemefulSite(GURL("https://primary.test")), + net::SiteType::kAssociated, absl::nullopt)}}})); + service()->InitForTesting(); EXPECT_FALSE(service()->IsSiteInManagedSet( net::SchemefulSite(GURL("https://not-example.test")))); @@ -215,13 +231,9 @@ IsSiteInManagedSet_SiteInConfig_AsDeletion) { net::SchemefulSite example_site = net::SchemefulSite(GURL("https://example.test")); - service()->InitForTesting( - [example_site]( - PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig( - {{example_site, {absl::nullopt}}})); - }); + SetContextConfig( + net::FirstPartySetsContextConfig({{example_site, {absl::nullopt}}})); + service()->InitForTesting(); EXPECT_FALSE(service()->IsSiteInManagedSet(example_site)); env().RunUntilIdle(); } @@ -230,16 +242,12 @@ IsSiteInManagedSet_SiteInConfig_AsModification) { net::SchemefulSite example_site = net::SchemefulSite(GURL("https://example.test")); - service()->InitForTesting( - [example_site]( - PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig( - {{example_site, - {net::FirstPartySetEntry( - net::SchemefulSite(GURL("https://primary.test")), - net::SiteType::kAssociated, absl::nullopt)}}})); - }); + SetContextConfig(net::FirstPartySetsContextConfig( + {{example_site, + {net::FirstPartySetEntry( + net::SchemefulSite(GURL("https://primary.test")), + net::SiteType::kAssociated, absl::nullopt)}}})); + service()->InitForTesting(); EXPECT_TRUE(service()->IsSiteInManagedSet(example_site)); env().RunUntilIdle(); } @@ -253,18 +261,13 @@ // Create Global First-Party Sets with the following set: // { primary: "https://primary.test", // associatedSites: ["https://associate1.test"} - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - net::GlobalFirstPartySets( - {{associate1_site, - {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, - 0)}}}, - {})); + SetGlobalSets(net::GlobalFirstPartySets( + {{associate1_site, + {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, + 0)}}}, + {})); // Simulate the profile set overrides are empty. - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); // Simulate First-Party Sets disabled by the feature. features.InitAndDisableFeature(features::kFirstPartySets); @@ -286,18 +289,13 @@ // Create Global First-Party Sets with the following set: // { primary: "https://primary.test", // associatedSites: ["https://associate1.test"} - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - net::GlobalFirstPartySets( - {{associate1_site, - {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, - 0)}}}, - {})); + SetGlobalSets(net::GlobalFirstPartySets( + {{associate1_site, + {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, + 0)}}}, + {})); // Simulate the profile set overrides are empty. - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); // Simulate First-Party Sets disabled by the preference. features.InitAndEnableFeature(features::kFirstPartySets); @@ -330,7 +328,7 @@ // Simulate the global First-Party Sets with the following set: // { primary: "https://primary.test", // associatedSites: ["https://associate1.test"} - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( + SetGlobalSets( net::GlobalFirstPartySets({{associate1_site, {associate1_entry}}}, {})); // Verify that FindEntry returns empty if both sources of sets aren't ready @@ -338,11 +336,7 @@ EXPECT_FALSE(service()->FindEntry(associate1_site)); // Simulate the profile set overrides are empty. - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); // Verify that FindEntry finally returns associate1's entry. EXPECT_EQ(service()->FindEntry(associate1_site).value(), associate1_entry); @@ -372,15 +366,11 @@ // Simulate the global First-Party Sets with the following set: // { primary: "https://primary.test", // associatedSites: ["https://associate.test"} - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( + SetGlobalSets( net::GlobalFirstPartySets({{associate_site, {associate_entry}}}, {})); // Simulate the profile set overrides are empty. - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); // The queries that occur before global sets are ready should be // counted in our metric. @@ -416,15 +406,11 @@ // Simulate the global First-Party Sets with the following set: // { primary: "https://primary.test", // associatedSites: ["https://associate.test"} - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( + SetGlobalSets( net::GlobalFirstPartySets({{associate_site, {associate_entry}}}, {})); // Simulate the profile set overrides are empty. - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); // Simulate 3 FindEntry queries which all are answered successfully. EXPECT_EQ(service()->FindEntry(associate_site).value(), associate_entry); @@ -456,12 +442,9 @@ net::FirstPartySetEntry test_entry(test_primary, net::SiteType::kPrimary, absl::nullopt); net::FirstPartySetsContextConfig test_config({{test_primary, {test_entry}}}); + SetContextConfig(test_config.Clone()); - service()->InitForTesting( - [&](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(test_config.Clone()); - }); + service()->InitForTesting(); EXPECT_CALL(mock_delegate, NotifyReady(CarryingConfig(std::ref(test_config)))) .Times(1); @@ -471,11 +454,7 @@ TEST_F(FirstPartySetsPolicyServicePrefObserverTest, OnFirstPartySetsEnabledChanged_Default_WithConfig) { - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); EXPECT_CALL(mock_delegate, SetEnabled(_)).Times(0); EXPECT_CALL(mock_delegate, NotifyReady(_)).Times(1); @@ -493,11 +472,7 @@ TEST_F(FirstPartySetsPolicyServicePrefObserverTest, OnFirstPartySetsEnabledChanged_Disables_WithConfig) { - service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); service()->OnFirstPartySetsEnabledChanged(false); EXPECT_CALL(mock_delegate, SetEnabled(false)).Times(1); @@ -522,12 +497,9 @@ net::FirstPartySetEntry test_entry(test_primary, net::SiteType::kPrimary, absl::nullopt); net::FirstPartySetsContextConfig test_config({{test_primary, {test_entry}}}); + SetContextConfig(test_config.Clone()); - service()->InitForTesting( - [&](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(test_config.Clone()); - }); + service()->InitForTesting(); service()->OnFirstPartySetsEnabledChanged(true); // Ensure access delegate is called with SetEnabled(true) and NotifyReady is @@ -551,39 +523,7 @@ env().RunUntilIdle(); } -class FirstPartySetsPolicyServiceWithMockHandlerTest - : public FirstPartySetsPolicyServiceTest { - public: - FirstPartySetsPolicyServiceWithMockHandlerTest() = default; - - void SetUp() override { - FirstPartySetsPolicyServiceTest::SetUp(); - - content::FirstPartySetsHandler::GetInstance()->SetInstanceForTesting( - &first_party_sets_handler_); - } - - void TearDown() override { - FirstPartySetsPolicyServiceTest::TearDown(); - first_party_sets_handler_.ResetForTesting(); - content::FirstPartySetsHandler::GetInstance()->SetInstanceForTesting( - nullptr); - } - - void SetContextConfig(net::FirstPartySetsContextConfig config) { - first_party_sets_handler_.SetContextConfig(std::move(config)); - } - void SetCacheFilter(net::FirstPartySetsCacheFilter cache_filter) { - first_party_sets_handler_.SetCacheFilter(std::move(cache_filter)); - } - - private: - MockFirstPartySetsHandler first_party_sets_handler_; - base::test::ScopedFeatureList features_; -}; - -TEST_F(FirstPartySetsPolicyServiceWithMockHandlerTest, - NotifiesReadyWithConfigAndCacheFilter) { +TEST_F(FirstPartySetsPolicyServiceTest, NotifiesReadyWithConfigAndCacheFilter) { net::SchemefulSite test_primary(GURL("https://a.test")); net::FirstPartySetEntry test_entry(test_primary, net::SiteType::kPrimary, absl::nullopt); @@ -592,11 +532,7 @@ /*browser_run_id=*/1); SetContextConfig(test_config.Clone()); SetCacheFilter(test_cache_filter.Clone()); - service()->InitForTesting( - [&](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(test_config.Clone()); - }); + service()->InitForTesting(); EXPECT_CALL(mock_delegate, NotifyReady(CarryingConfigAndCacheFilter( @@ -644,11 +580,7 @@ IsPrefEnabled()); base::RunLoop run_loop; service()->RegisterThrottleResumeCallback(run_loop.QuitClosure()); - service()->InitForTesting( - [&](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + service()->InitForTesting(); run_loop.Run(); }
diff --git a/chrome/browser/first_party_sets/mock_first_party_sets_handler.cc b/chrome/browser/first_party_sets/mock_first_party_sets_handler.cc index 2139a94e..06674c1 100644 --- a/chrome/browser/first_party_sets/mock_first_party_sets_handler.cc +++ b/chrome/browser/first_party_sets/mock_first_party_sets_handler.cc
@@ -7,7 +7,9 @@ #include <string> #include "base/callback.h" +#include "base/feature_list.h" #include "base/types/optional_util.h" +#include "content/public/common/content_features.h" #include "net/first_party_sets/first_party_sets_cache_filter.h" #include "net/first_party_sets/first_party_sets_context_config.h" #include "net/first_party_sets/global_first_party_sets.h" @@ -26,15 +28,11 @@ const base::Version& version, base::File sets_file) {} -void MockFirstPartySetsHandler::SetGlobalSetsForTesting( - net::GlobalFirstPartySets global_sets) { - global_sets_ = std::move(global_sets); -} - absl::optional<net::FirstPartySetEntry> MockFirstPartySetsHandler::FindEntry( const net::SchemefulSite& site, const net::FirstPartySetsContextConfig& config) const { - if (!global_sets_.has_value()) { + if (!global_sets_.has_value() || + !base::FeatureList::IsEnabled(features::kFirstPartySets)) { return absl::nullopt; } return global_sets_->FindEntry(site, config); @@ -68,9 +66,15 @@ net::FirstPartySetsContextConfig config) { config_ = std::move(config); } + void MockFirstPartySetsHandler::SetCacheFilter( net::FirstPartySetsCacheFilter cache_filter) { cache_filter_ = std::move(cache_filter); } +void MockFirstPartySetsHandler::SetGlobalSets( + net::GlobalFirstPartySets global_sets) { + global_sets_ = std::move(global_sets); +} + } // namespace first_party_sets
diff --git a/chrome/browser/first_party_sets/mock_first_party_sets_handler.h b/chrome/browser/first_party_sets/mock_first_party_sets_handler.h index a452bff..80d30c0 100644 --- a/chrome/browser/first_party_sets/mock_first_party_sets_handler.h +++ b/chrome/browser/first_party_sets/mock_first_party_sets_handler.h
@@ -40,7 +40,6 @@ void SetPublicFirstPartySets(const base::Version& version, base::File sets_file) override; void ResetForTesting() override; - void SetGlobalSetsForTesting(net::GlobalFirstPartySets global_sets) override; absl::optional<net::FirstPartySetEntry> FindEntry( const net::SchemefulSite& site, const net::FirstPartySetsContextConfig& config) const override; @@ -59,8 +58,11 @@ // Helper functions for tests to set up context. void SetContextConfig(net::FirstPartySetsContextConfig config); + void SetCacheFilter(net::FirstPartySetsCacheFilter cache_filter); + void SetGlobalSets(net::GlobalFirstPartySets global_sets); + private: absl::optional<net::GlobalFirstPartySets> global_sets_; absl::optional<net::FirstPartySetsContextConfig> config_;
diff --git a/chrome/browser/flag-metadata.json b/chrome/browser/flag-metadata.json index 60256f3..75c5e56 100644 --- a/chrome/browser/flag-metadata.json +++ b/chrome/browser/flag-metadata.json
@@ -2604,6 +2604,11 @@ "expiry_milestone": 110 }, { + "name": "enable-lens-region-search-static-page", + "owners": ["juanmojica@google.com", "stanfield@google.com", "lens-chrome@google.com"], + "expiry_milestone": 110 + }, + { "name": "enable-lens-standalone", "owners": [ "stanfield@google.com", "benwgold@google.com", "juanmojica@google.com" ], "expiry_milestone": 107 @@ -3637,6 +3642,11 @@ "expiry_milestone": 115 }, { + "name": "feed-header-stick-to-top", + "owners": [ "//chrome/android/feed/OWNERS", "chili@chromium.org" ], + "expiry_milestone": 115 + }, + { "name": "feed-interactive-refresh", "owners": [ "//chrome/android/feed/OWNERS", "jianli@chromium.org" ], "expiry_milestone": 99 @@ -3831,7 +3841,7 @@ { "name": "forced-colors", "owners": [ "almaher@microsoft.com" ], - "expiry_milestone": 108 + "expiry_milestone": 111 }, { "name": "fractional-scroll-offsets", @@ -4448,7 +4458,7 @@ { "name": "lightweight-reactions-android", "owners": [ "gujen", "sophey", "chrome-with-friends-robots@google.com" ], - "expiry_milestone": 108 + "expiry_milestone": 110 }, { "name": "link-capturing-auto-display-intent-picker",
diff --git a/chrome/browser/flag_descriptions.cc b/chrome/browser/flag_descriptions.cc index 6c64833c..2da78cd 100644 --- a/chrome/browser/flag_descriptions.cc +++ b/chrome/browser/flag_descriptions.cc
@@ -1850,6 +1850,12 @@ "Enable an entry point to Google Lens to allow users to search what they " "see using their mobile camera."; +const char kLensRegionSearchStaticPageName[] = + "Use a static page with the Lens region search feature."; +const char kLensRegionSearchStaticPageDescription[] = + "Enables use of a static page in a new tab when using the Lens region " + "search feature."; + const char kLensOnQuickActionSearchWidgetName[] = "Google Lens in Chrome's Quick Action Search Widget"; const char kLensOnQuickActionSearchWidgetDescription[] = @@ -3713,6 +3719,10 @@ "Enables showing a callout to help users return to the top of the feeds " "quickly."; +const char kFeedHeaderStickToTopName[] = "Feed header stick to top"; +const char kFeedHeaderStickToTopDescription[] = + "Stick feed header to top on scroll."; + const char kFeedInteractiveRefreshName[] = "Refresh feeds"; const char kFeedInteractiveRefreshDescription[] = "Enables refreshing feeds triggered by the users.";
diff --git a/chrome/browser/flag_descriptions.h b/chrome/browser/flag_descriptions.h index fbcb38f..3ca7adc9 100644 --- a/chrome/browser/flag_descriptions.h +++ b/chrome/browser/flag_descriptions.h
@@ -1035,6 +1035,9 @@ extern const char kLensCameraAssistedSearchName[]; extern const char kLensCameraAssistedSearchDescription[]; +extern const char kLensRegionSearchStaticPageName[]; +extern const char kLensRegionSearchStaticPageDescription[]; + extern const char kLensOnQuickActionSearchWidgetName[]; extern const char kLensOnQuickActionSearchWidgetDescription[]; @@ -2096,6 +2099,9 @@ extern const char kFeedBackToTopName[]; extern const char kFeedBackToTopDescription[]; +extern const char kFeedHeaderStickToTopName[]; +extern const char kFeedHeaderStickToTopDescription[]; + extern const char kFeedInteractiveRefreshName[]; extern const char kFeedInteractiveRefreshDescription[];
diff --git a/chrome/browser/flags/android/chrome_feature_list.cc b/chrome/browser/flags/android/chrome_feature_list.cc index eba24e5..aa9beec 100644 --- a/chrome/browser/flags/android/chrome_feature_list.cc +++ b/chrome/browser/flags/android/chrome_feature_list.cc
@@ -142,6 +142,7 @@ &feature_guide::features::kSkipCheckForLowEngagedUsers, &feed::kFeedBackToTop, &feed::kFeedClearImageMemoryCache, + &feed::kFeedHeaderStickToTop, &feed::kFeedImageMemoryCacheSizePercentage, &feed::kFeedInteractiveRefresh, &feed::kFeedLoadingPlaceholder, @@ -804,7 +805,7 @@ BASE_FEATURE(kIsVoiceSearchEnabledCache, "IsVoiceSearchEnabledCache", - base::FEATURE_DISABLED_BY_DEFAULT); + base::FEATURE_ENABLED_BY_DEFAULT); BASE_FEATURE(kKitKatSupported, "KitKatSupported",
diff --git a/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java b/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java index c50054a..0f29b42 100644 --- a/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java +++ b/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java
@@ -359,6 +359,7 @@ public static final String FEATURE_NOTIFICATION_GUIDE = "FeatureNotificationGuide"; public static final String FEED_BACK_TO_TOP = "FeedBackToTop"; public static final String FEED_CLEAR_IMAGE_MEMORY_CACHE = "FeedClearImageMemoryCache"; + public static final String FEED_HEADER_STICK_TO_TOP = "FeedHeaderStickToTop"; public static final String FEED_IMAGE_MEMORY_CACHE_SIZE_PERCENTAGE = "FeedImageMemoryCacheSizePercentage"; public static final String FEED_INTERACTIVE_REFRESH = "FeedInteractiveRefresh";
diff --git a/chrome/browser/media/router/mojo/media_router_mojo_impl.h b/chrome/browser/media/router/mojo/media_router_mojo_impl.h index ebd1e9b..9793082 100644 --- a/chrome/browser/media/router/mojo/media_router_mojo_impl.h +++ b/chrome/browser/media/router/mojo/media_router_mojo_impl.h
@@ -259,7 +259,7 @@ absl::optional<std::vector<MediaRoute>> cached_route_list_; // Per-MRP lists of routes for the query. - // TODO(crbug.com/761493): Consider making MRP ID an attribute of + // TODO(crbug.com/1374496): Consider making MRP ID an attribute of // MediaRoute, so that we can simplify these into vectors. base::flat_map<mojom::MediaRouteProviderId, std::vector<MediaRoute>> providers_to_routes_;
diff --git a/chrome/browser/performance_manager/metrics/page_timeline_monitor.cc b/chrome/browser/performance_manager/metrics/page_timeline_monitor.cc index 62f443c..cfbb1629f 100644 --- a/chrome/browser/performance_manager/metrics/page_timeline_monitor.cc +++ b/chrome/browser/performance_manager/metrics/page_timeline_monitor.cc
@@ -13,12 +13,18 @@ #include "base/rand_util.h" #include "base/time/time.h" #include "base/timer/timer.h" +#include "components/content_settings/core/common/content_settings_types.h" +#include "components/performance_manager/public/decorators/page_live_state_decorator.h" #include "components/performance_manager/public/features.h" #include "components/performance_manager/public/graph/page_node.h" #include "services/metrics/public/cpp/metrics_utils.h" #include "services/metrics/public/cpp/ukm_builders.h" #include "services/metrics/public/cpp/ukm_recorder.h" +#if !BUILDFLAG(IS_ANDROID) +#include "chrome/browser/performance_manager/public/user_tuning/user_performance_tuning_manager.h" +#endif // !BUILDFLAG(IS_ANDROID) + namespace performance_manager::metrics { PageTimelineMonitor::PageTimelineMonitor() @@ -93,8 +99,39 @@ curr_info->time_of_last_foreground_millisecond_update = now; } + bool is_active_tab = false; + bool has_notification_permission = false; + bool is_capturing_media = false; + bool is_connected_to_device = false; + + const auto* page_live_state_data = + PageLiveStateDecorator::Data::FromPageNode(page_node); + if (page_live_state_data) { + is_active_tab = page_live_state_data->IsActiveTab(); + has_notification_permission = + page_live_state_data->IsContentSettingTypeAllowed( + ContentSettingsType::NOTIFICATIONS); + is_capturing_media = page_live_state_data->IsCapturingVideo() || + page_live_state_data->IsCapturingAudio() || + page_live_state_data->IsBeingMirrored() || + page_live_state_data->IsCapturingWindow() || + page_live_state_data->IsCapturingDisplay(); + is_connected_to_device = + page_live_state_data->IsConnectedToUSBDevice() || + page_live_state_data->IsConnectedToBluetoothDevice(); + } + ukm::builders::PerformanceManager_PageTimelineState(source_id) .SetSliceId(slice_id) +#if !BUILDFLAG(IS_ANDROID) + .SetHighEfficiencyMode( + user_tuning::UserPerformanceTuningManager::GetInstance() + ->IsHighEfficiencyModeActive()) + .SetBatterySaverMode( + user_tuning::UserPerformanceTuningManager::GetInstance() + ->IsBatterySaverActive()) +#endif // !BUILDFLAG(IS_ANDROID) + .SetIsActiveTab(is_active_tab) .SetTimeSinceLastSlice(ukm::GetSemanticBucketMinForDurationTiming( time_since_last_slice.InMilliseconds())) .SetTimeSinceCreation(ukm::GetSemanticBucketMinForDurationTiming( @@ -107,6 +144,11 @@ curr_info->total_foreground_milliseconds)) .SetChangedFaviconOrTitleInBackground( curr_info->updated_title_or_favicon_in_background) + .SetHasNotificationPermission(has_notification_permission) + .SetIsCapturingMedia(is_capturing_media) + .SetIsConnectedToDevice(is_connected_to_device) + .SetIsPlayingAudio(page_node->IsAudible()) + .SetResidentSetSize(page_node->EstimateResidentSetSize()) .Record(ukm::UkmRecorder::Get()); } }
diff --git a/chrome/browser/performance_manager/metrics/page_timeline_monitor.h b/chrome/browser/performance_manager/metrics/page_timeline_monitor.h index 6d5bc1e..87b9266 100644 --- a/chrome/browser/performance_manager/metrics/page_timeline_monitor.h +++ b/chrome/browser/performance_manager/metrics/page_timeline_monitor.h
@@ -19,9 +19,6 @@ // Periodically reports tab state via UKM, to enable analysis of usage patterns // over time. -// -// TODO(crbug.com/1367052): Report whether high-efficiency mode and/or -// battery-saver mode are enabled. class PageTimelineMonitor : public PageNode::ObserverDefaultImpl, public GraphOwned { public:
diff --git a/chrome/browser/performance_manager/metrics/page_timeline_monitor_unittest.cc b/chrome/browser/performance_manager/metrics/page_timeline_monitor_unittest.cc index 7d4d956..9ab9d643 100644 --- a/chrome/browser/performance_manager/metrics/page_timeline_monitor_unittest.cc +++ b/chrome/browser/performance_manager/metrics/page_timeline_monitor_unittest.cc
@@ -8,14 +8,22 @@ #include "base/bind.h" #include "base/memory/raw_ptr.h" #include "build/build_config.h" +#include "chrome/browser/performance_manager/test_support/test_user_performance_tuning_manager_environment.h" +#include "components/performance_manager/public/decorators/page_live_state_decorator.h" #include "components/performance_manager/public/mojom/lifecycle.mojom-shared.h" +#include "components/performance_manager/public/user_tuning/prefs.h" #include "components/performance_manager/test_support/graph_test_harness.h" #include "components/performance_manager/test_support/mock_graphs.h" +#include "components/prefs/testing_pref_service.h" #include "components/ukm/test_ukm_recorder.h" #include "services/metrics/public/cpp/ukm_builders.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" +#if !BUILDFLAG(IS_ANDROID) +#include "chrome/browser/performance_manager/test_support/test_user_performance_tuning_manager_environment.h" +#endif // !BUILDFLAG(IS_ANDROID) + namespace performance_manager::metrics { class PageTimelineMonitorUnitTest : public GraphTestHarness { @@ -35,9 +43,17 @@ monitor_ = monitor.get(); graph()->PassToGraph(std::move(monitor)); test_ukm_recorder_ = std::make_unique<ukm::TestAutoSetUkmRecorder>(); +#if !BUILDFLAG(IS_ANDROID) + performance_manager::user_tuning::prefs::RegisterLocalStatePrefs( + local_state_.registry()); + environment_.SetUp(&local_state_); +#endif // !BUILDFLAG(IS_ANDROID) } void TearDown() override { +#if !BUILDFLAG(IS_ANDROID) + environment_.TearDown(); +#endif // !BUILDFLAG(IS_ANDROID) test_ukm_recorder_.reset(); GraphTestHarness::TearDown(); } @@ -53,6 +69,13 @@ private: std::unique_ptr<ukm::TestUkmRecorder> test_ukm_recorder_; + +#if !BUILDFLAG(IS_ANDROID) + // The UserPerformanceTuningManager doesn't exist on Android + TestingPrefServiceSimple local_state_; + performance_manager::user_tuning::TestUserPerformanceTuningManagerEnvironment + environment_; +#endif // !BUILDFLAG(IS_ANDROID) }; TEST_F(PageTimelineMonitorUnitTest, TestPageTimeline) { @@ -162,4 +185,157 @@ mojom::LifecycleState::kFrozen); } +TEST_F(PageTimelineMonitorUnitTest, TestHasNotificationsPermission) { + MockSinglePageInSingleProcessGraph mock_graph(graph()); + ukm::SourceId mock_source_id = ukm::NoURLSourceId(); + mock_graph.page->SetType(performance_manager::PageType::kTab); + mock_graph.page->SetUkmSourceId(mock_source_id); + mock_graph.page->SetIsVisible(true); + mock_graph.page->SetLifecycleStateForTesting(mojom::LifecycleState::kRunning); + + PageLiveStateDecorator::Data* data = + PageLiveStateDecorator::Data::GetOrCreateForPageNode( + mock_graph.page.get()); + data->SetContentSettingsForTesting( + {{ContentSettingsType::NOTIFICATIONS, CONTENT_SETTING_ALLOW}}); + + TriggerCollectSlice(); + auto entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 1UL); + test_ukm_recorder()->ExpectEntryMetric(entries[0], + "HasNotificationPermission", 1); + + data->SetContentSettingsForTesting( + {{ContentSettingsType::NOTIFICATIONS, CONTENT_SETTING_BLOCK}}); + + TriggerCollectSlice(); + entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 2UL); + + test_ukm_recorder()->ExpectEntryMetric(entries[1], + "HasNotificationPermission", 0); +} + +TEST_F(PageTimelineMonitorUnitTest, TestCapturingMedia) { + MockSinglePageInSingleProcessGraph mock_graph(graph()); + ukm::SourceId mock_source_id = ukm::NoURLSourceId(); + mock_graph.page->SetType(performance_manager::PageType::kTab); + mock_graph.page->SetUkmSourceId(mock_source_id); + mock_graph.page->SetIsVisible(true); + mock_graph.page->SetLifecycleStateForTesting(mojom::LifecycleState::kRunning); + + PageLiveStateDecorator::Data* data = + PageLiveStateDecorator::Data::GetOrCreateForPageNode( + mock_graph.page.get()); + data->SetIsCapturingVideoForTesting(false); + + TriggerCollectSlice(); + auto entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 1UL); + test_ukm_recorder()->ExpectEntryMetric(entries[0], "IsCapturingMedia", 0); + + data->SetIsCapturingVideoForTesting(true); + TriggerCollectSlice(); + entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 2UL); + test_ukm_recorder()->ExpectEntryMetric(entries[1], "IsCapturingMedia", 1); +} + +TEST_F(PageTimelineMonitorUnitTest, TestConnectedToDevice) { + MockSinglePageInSingleProcessGraph mock_graph(graph()); + ukm::SourceId mock_source_id = ukm::NoURLSourceId(); + mock_graph.page->SetType(performance_manager::PageType::kTab); + mock_graph.page->SetUkmSourceId(mock_source_id); + mock_graph.page->SetIsVisible(true); + mock_graph.page->SetLifecycleStateForTesting(mojom::LifecycleState::kRunning); + + PageLiveStateDecorator::Data* data = + PageLiveStateDecorator::Data::GetOrCreateForPageNode( + mock_graph.page.get()); + data->SetIsConnectedToUSBDeviceForTesting(false); + + TriggerCollectSlice(); + auto entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 1UL); + test_ukm_recorder()->ExpectEntryMetric(entries[0], "IsConnectedToDevice", 0); + + data->SetIsConnectedToUSBDeviceForTesting(true); + TriggerCollectSlice(); + entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 2UL); + test_ukm_recorder()->ExpectEntryMetric(entries[1], "IsConnectedToDevice", 1); +} + +TEST_F(PageTimelineMonitorUnitTest, TestAudible) { + MockSinglePageInSingleProcessGraph mock_graph(graph()); + ukm::SourceId mock_source_id = ukm::NoURLSourceId(); + mock_graph.page->SetType(performance_manager::PageType::kTab); + mock_graph.page->SetUkmSourceId(mock_source_id); + mock_graph.page->SetIsVisible(true); + mock_graph.page->SetLifecycleStateForTesting(mojom::LifecycleState::kRunning); + + mock_graph.page->SetIsAudible(false); + TriggerCollectSlice(); + auto entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 1UL); + test_ukm_recorder()->ExpectEntryMetric(entries[0], "IsPlayingAudio", 0); + + mock_graph.page->SetIsAudible(true); + TriggerCollectSlice(); + entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 2UL); + test_ukm_recorder()->ExpectEntryMetric(entries[1], "IsPlayingAudio", 1); +} + +TEST_F(PageTimelineMonitorUnitTest, TestIsActiveTab) { + MockSinglePageInSingleProcessGraph mock_graph(graph()); + ukm::SourceId mock_source_id = ukm::NoURLSourceId(); + mock_graph.page->SetType(performance_manager::PageType::kTab); + mock_graph.page->SetUkmSourceId(mock_source_id); + mock_graph.page->SetIsVisible(true); + mock_graph.page->SetLifecycleStateForTesting(mojom::LifecycleState::kRunning); + + PageLiveStateDecorator::Data* data = + PageLiveStateDecorator::Data::GetOrCreateForPageNode( + mock_graph.page.get()); + data->SetIsActiveTabForTesting(false); + + TriggerCollectSlice(); + auto entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 1UL); + test_ukm_recorder()->ExpectEntryMetric(entries[0], "IsActiveTab", 0); + + data->SetIsActiveTabForTesting(true); + TriggerCollectSlice(); + entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 2UL); + test_ukm_recorder()->ExpectEntryMetric(entries[1], "IsActiveTab", 1); +} + +TEST_F(PageTimelineMonitorUnitTest, TestResidentSetSize) { + MockSinglePageInSingleProcessGraph mock_graph(graph()); + ukm::SourceId mock_source_id = ukm::NoURLSourceId(); + mock_graph.page->SetType(performance_manager::PageType::kTab); + mock_graph.page->SetUkmSourceId(mock_source_id); + mock_graph.page->SetIsVisible(true); + mock_graph.page->SetLifecycleStateForTesting(mojom::LifecycleState::kRunning); + mock_graph.frame->SetResidentSetKbEstimate(123); + + TriggerCollectSlice(); + auto entries = test_ukm_recorder()->GetEntriesByName( + ukm::builders::PerformanceManager_PageTimelineState::kEntryName); + EXPECT_EQ(entries.size(), 1UL); + test_ukm_recorder()->ExpectEntryMetric(entries[0], "ResidentSetSize", 123); +} + } // namespace performance_manager::metrics
diff --git a/chrome/browser/performance_manager/public/user_tuning/user_performance_tuning_manager.h b/chrome/browser/performance_manager/public/user_tuning/user_performance_tuning_manager.h index 4945450..cf90cb37 100644 --- a/chrome/browser/performance_manager/public/user_tuning/user_performance_tuning_manager.h +++ b/chrome/browser/performance_manager/public/user_tuning/user_performance_tuning_manager.h
@@ -133,6 +133,9 @@ void SetTemporaryBatterySaverDisabledForSession(bool disabled); bool IsBatterySaverModeDisabledForSession() const; + // Returns true if High Efficiency mode is currently enabled. + bool IsHighEfficiencyModeActive() const; + // Returns true if Battery Saver Mode interventions are active. If any state // transitions cause an observer notification, this is guaranteed to reflect // the *new* value when the observers are notified so the UI layer can make
diff --git a/chrome/browser/performance_manager/user_tuning/user_performance_tuning_manager.cc b/chrome/browser/performance_manager/user_tuning/user_performance_tuning_manager.cc index 50e432e..6965bea5 100644 --- a/chrome/browser/performance_manager/user_tuning/user_performance_tuning_manager.cc +++ b/chrome/browser/performance_manager/user_tuning/user_performance_tuning_manager.cc
@@ -113,6 +113,11 @@ return battery_saver_mode_disabled_for_session_; } +bool UserPerformanceTuningManager::IsHighEfficiencyModeActive() const { + return pref_change_registrar_.prefs()->GetBoolean( + performance_manager::user_tuning::prefs::kHighEfficiencyModeEnabled); +} + bool UserPerformanceTuningManager::IsBatterySaverActive() const { return battery_saver_mode_enabled_; }
diff --git a/chrome/browser/printing/print_backend_service_manager.cc b/chrome/browser/printing/print_backend_service_manager.cc index bf35239a..2f590a66 100644 --- a/chrome/browser/printing/print_backend_service_manager.cc +++ b/chrome/browser/printing/print_backend_service_manager.cc
@@ -6,6 +6,7 @@ #include <memory> #include <string> +#include <utility> #include "base/bind.h" #include "base/containers/flat_map.h" @@ -29,6 +30,9 @@ #if BUILDFLAG(IS_WIN) #include "base/win/win_util.h" +#include "chrome/browser/printing/printer_xml_parser_impl.h" +#include "chrome/services/printing/public/mojom/printer_xml_parser.mojom.h" +#include "mojo/public/cpp/bindings/pending_receiver.h" #include "printing/backend/win_helper.h" #include "printing/printed_page_win.h" #include "ui/views/win/hwnd_util.h" @@ -152,6 +156,13 @@ remote_id); if (new_timeout.has_value()) UpdateServiceIdleTimeoutByRemoteId(remote_id, new_timeout.value()); + +#if BUILDFLAG(IS_WIN) + if (base::FeatureList::IsEnabled(features::kReadPrinterCapabilitiesWithXps) && + query_clients_.empty()) { + xml_parser_.reset(); + } +#endif // BUILDFLAG(IS_WIN) } void PrintBackendServiceManager::EnumeratePrinters( @@ -648,6 +659,17 @@ // Initialize the new service for the desired locale. service->Init(g_browser_process->GetApplicationLocale()); + +#if BUILDFLAG(IS_WIN) + // Bind PrintBackendService with a Remote that allows pass-through requests + // to an XML parser. + if (base::FeatureList::IsEnabled( + features::kReadPrinterCapabilitiesWithXps)) { + if (!xml_parser_) + xml_parser_ = std::make_unique<PrinterXmlParserImpl>(); + service->BindPrinterXmlParser(xml_parser_->GetRemote()); + } +#endif // BUILDFLAG(IS_WIN) } return service;
diff --git a/chrome/browser/printing/print_backend_service_manager.h b/chrome/browser/printing/print_backend_service_manager.h index 69a55088..68861abf 100644 --- a/chrome/browser/printing/print_backend_service_manager.h +++ b/chrome/browser/printing/print_backend_service_manager.h
@@ -35,6 +35,10 @@ namespace printing { +#if BUILDFLAG(IS_WIN) +class PrinterXmlParserImpl; +#endif // BUILDFLAG(IS_WIN) + class PrintedPage; class PrintBackendServiceManager { @@ -427,6 +431,12 @@ const ClientsSet& query_with_ui_clients, const PrintClientsMap& print_document_clients); +#if BUILDFLAG(IS_WIN) + // Printer XML Parser implementation used to allow Print Backend Service to + // send XML parse requests to the browser process. + std::unique_ptr<PrinterXmlParserImpl> xml_parser_; +#endif // BUILDFLAG(IS_WIN) + // Bundles of remotes for the Print Backend Service and their corresponding // wrapping hosts, to manage these sets until they disconnect. The sandboxed // and unsandboxed services are kept separate.
diff --git a/chrome/browser/printing/print_backend_service_test_impl.cc b/chrome/browser/printing/print_backend_service_test_impl.cc index 9bd6d02..65286269 100644 --- a/chrome/browser/printing/print_backend_service_test_impl.cc +++ b/chrome/browser/printing/print_backend_service_test_impl.cc
@@ -14,6 +14,14 @@ #include "chrome/browser/printing/print_backend_service_manager.h" #include "printing/backend/test_print_backend.h" +#if BUILDFLAG(IS_WIN) +#include <memory> + +#include "chrome/browser/printing/printer_xml_parser_impl.h" +#include "mojo/public/cpp/bindings/pending_remote.h" +#include "printing/printing_features.h" +#endif // BUILDFLAG(IS_WIN) + namespace printing { #if BUILDFLAG(IS_WIN) @@ -169,6 +177,15 @@ receiver_.reset(); } +#if BUILDFLAG(IS_WIN) +mojo::PendingRemote<mojom::PrinterXmlParser> +PrintBackendServiceTestImpl::GetPrinterXmlParserRemote() { + if (!xml_parser_) + xml_parser_ = std::make_unique<PrinterXmlParserImpl>(); + return xml_parser_->GetRemote(); +} +#endif // BUILDFLAG(IS_WIN) + // static std::unique_ptr<PrintBackendServiceTestImpl> PrintBackendServiceTestImpl::LaunchForTesting( @@ -183,6 +200,11 @@ new PrintBackendServiceTestImpl(std::move(receiver), std::move(backend))); service->Init(/*locale=*/std::string()); +#if BUILDFLAG(IS_WIN) + if (base::FeatureList::IsEnabled(features::kReadPrinterCapabilitiesWithXps)) + service->BindPrinterXmlParser(service->GetPrinterXmlParserRemote()); +#endif // BUILDFLAG(IS_WIN) + // Register this test version of print backend service to be used instead of // launching instances out-of-process on-demand. if (sandboxed) {
diff --git a/chrome/browser/printing/print_backend_service_test_impl.h b/chrome/browser/printing/print_backend_service_test_impl.h index 807a8f3..da877410 100644 --- a/chrome/browser/printing/print_backend_service_test_impl.h +++ b/chrome/browser/printing/print_backend_service_test_impl.h
@@ -20,6 +20,8 @@ #if BUILDFLAG(IS_WIN) #include "base/containers/queue.h" #include "base/memory/read_only_shared_memory_region.h" +#include "chrome/services/printing/public/mojom/printer_xml_parser.mojom-forward.h" +#include "mojo/public/cpp/bindings/pending_remote.h" #include "printing/mojom/print.mojom.h" #endif @@ -33,6 +35,7 @@ namespace printing { #if BUILDFLAG(IS_WIN) +class PrinterXmlParserImpl; struct RenderPrintedPageData; #endif @@ -98,6 +101,8 @@ void set_rendering_delayed_until_page(uint32_t page_number) { rendering_delayed_until_page_number_ = page_number; } + + mojo::PendingRemote<mojom::PrinterXmlParser> GetPrinterXmlParserRemote(); #endif private: @@ -127,6 +132,9 @@ // The queue of pages whose rendering processing is being delayed. base::queue<std::unique_ptr<RenderPrintedPageData>> delayed_rendering_pages_; + + // Used to parse XPS XML capabilities. + std::unique_ptr<PrinterXmlParserImpl> xml_parser_; #endif scoped_refptr<TestPrintBackend> test_print_backend_;
diff --git a/chrome/browser/printing/print_browsertest.cc b/chrome/browser/printing/print_browsertest.cc index 47fcd1e..a70cd643 100644 --- a/chrome/browser/printing/print_browsertest.cc +++ b/chrome/browser/printing/print_browsertest.cc
@@ -271,7 +271,7 @@ fail_on_use_default_settings_ = true; } -#if BUILDFLAG(IS_WIN) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) void SetCancelErrorOnAskUserForSettings() { cancel_on_ask_user_for_settings_ = true; } @@ -291,7 +291,7 @@ bool access_denied_errors_for_render_document_ = false; bool access_denied_errors_for_document_done_ = false; bool fail_on_use_default_settings_ = false; -#if BUILDFLAG(IS_WIN) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) bool cancel_on_ask_user_for_settings_ = false; #endif int new_document_called_count_ = 0; @@ -2741,7 +2741,7 @@ test_printing_context_factory()->SetFailErrorOnUseDefaultSettings(); } -#if BUILDFLAG(IS_WIN) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) void PrimeForCancelInAskUserForSettings() { test_printing_context_factory()->SetCancelErrorOnAskUserForSettings(); } @@ -2789,7 +2789,7 @@ return use_default_settings_result_; } -#if BUILDFLAG(IS_WIN) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) mojom::ResultCode ask_user_for_settings_result() const { return ask_user_for_settings_result_; } @@ -2874,7 +2874,7 @@ CheckForQuit(); } -#if BUILDFLAG(IS_WIN) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) void OnDidAskUserForSettings(mojom::ResultCode result) { ask_user_for_settings_result_ = result; CheckForQuit(); @@ -2945,7 +2945,7 @@ raw_ptr<PrintJob> print_job_ = nullptr; bool reset_errors_after_check_ = true; mojom::ResultCode use_default_settings_result_ = mojom::ResultCode::kFailed; -#if BUILDFLAG(IS_WIN) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) mojom::ResultCode ask_user_for_settings_result_ = mojom::ResultCode::kFailed; #endif mojom::ResultCode start_printing_result_ = mojom::ResultCode::kFailed; @@ -3386,9 +3386,7 @@ EXPECT_EQ(print_job_destruction_count(), 1); } -// TODO(crbug.com/809738) Extend to Linux once Wayland can be made to support -// a system be modal against an application window in the browser process. -#if BUILDFLAG(IS_WIN) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) IN_PROC_BROWSER_TEST_P(SystemAccessProcessServicePrintBrowserTest, StartBasicPrint) { AddPrinter("printer1"); @@ -3403,24 +3401,53 @@ ASSERT_TRUE(web_contents); SetUpPrintViewManager(web_contents); +#if BUILDFLAG(IS_WIN) // The test will get the default settings followed by asking the user for // settings. After that a print job will be started, with a page getting // rendered, and finally the document done notification. Wait for the one // print job to be destroyed to ensure printing finished cleanly before // completing the test. This results in a total of 6 calls. SetNumExpectedMessages(/*num=*/6); +#else + // The test sequence for this is: + // - Get the default settings. + // - Ask the user for settings. Due to issues with displaying a system dialog + // from the utility process, there is no callback to capture the request for + // user supplied settings. + // - A print job is started. + // - The document is rendered. + // - Receive document done notification. + // - Wait for the one print job to be destroyed, to ensure printing finished + // cleanly before completing the test. + // This results in a total of 5 calls. + // TODO(crbug.com/1374188) Update this expectation once + // `AskUserForSettings()` is able to be pushed OOP for Linux. + SetNumExpectedMessages(/*num=*/5); +#endif StartBasicPrint(web_contents); WaitUntilCallbackReceived(); EXPECT_EQ(use_default_settings_result(), mojom::ResultCode::kSuccess); + // macOS and Linux currently have to invoke a system dialog from within the + // browser process. There is not a callback to capture the result in these + // cases. + // TODO(crbug.com/1374188) Re-enable this check against + // `ask_user_for_settings_result()` once `AskForUserSettings()` is able to be + // pushed OOP for Linux. +#if BUILDFLAG(IS_WIN) EXPECT_EQ(ask_user_for_settings_result(), mojom::ResultCode::kSuccess); +#endif EXPECT_EQ(start_printing_result(), mojom::ResultCode::kSuccess); +#if BUILDFLAG(IS_WIN) // TODO(crbug.com/1008222) Include Windows coverage of // RenderPrintedDocument() once XPS print pipeline is added. EXPECT_EQ(render_printed_page_result(), mojom::ResultCode::kSuccess); EXPECT_EQ(render_printed_page_count(), 1); +#else + EXPECT_EQ(render_printed_document_result(), mojom::ResultCode::kSuccess); +#endif EXPECT_EQ(document_done_result(), mojom::ResultCode::kSuccess); EXPECT_EQ(print_job_destruction_count(), 1); } @@ -3459,8 +3486,18 @@ EXPECT_FALSE(print_backend_service_use_detected()); } +// macOS and Linux currently have to invoke a system dialog from within the +// browser process. There is not a callback to capture the result in these +// cases. +// TODO(crbug.com/1374188) Re-enable for Linux once `AskForUserSettings()` is +// able to be pushed OOP for Linux. +#if BUILDFLAG(IS_MAC) || BUILDFLAG(IS_LINUX) +#define MAYBE_StartBasicPrintCancel DISABLED_StartBasicPrintCancel +#else +#define MAYBE_StartBasicPrintCancel StartBasicPrintCancel +#endif IN_PROC_BROWSER_TEST_P(SystemAccessProcessServicePrintBrowserTest, - StartBasicPrintCancel) { + MAYBE_StartBasicPrintCancel) { AddPrinter("printer1"); SetPrinterNameForSubsequentContexts("printer1"); PrimeForCancelInAskUserForSettings(); @@ -3510,17 +3547,18 @@ // Now initiate a system print that would exist concurrently with that. StartBasicPrint(web_contents); - // On Windows, concurrent system print is not allowed. - // TODO(crbug.com/809738): Demonstrate that Linux allows multiple system - // prints at a time. const absl::optional<bool>& result = print_view_manager->print_now_result(); ASSERT_TRUE(result.has_value()); + // With the exception of Linux, concurrent system print is not allowed. +#if BUILDFLAG(IS_LINUX) + EXPECT_TRUE(*result); +#else EXPECT_FALSE(*result); +#endif // Cleanup before test shutdown. PrintBackendServiceManager::GetInstance().UnregisterClient(*client_id); } -#endif // BUILDFLAG(IS_WIN) // https://crbug.com/1320681 flaky. IN_PROC_BROWSER_TEST_P(SystemAccessProcessServicePrintBrowserTest, @@ -3549,6 +3587,7 @@ EXPECT_EQ(use_default_settings_result(), mojom::ResultCode::kFailed); EXPECT_EQ(print_job_construction_count(), 1); } +#endif // BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) #endif // BUILDFLAG(ENABLE_OOP_PRINTING) @@ -3757,13 +3796,11 @@ INSTANTIATE_TEST_SUITE_P(All, ContentAnalysisPrintBrowserTest, testing::Bool()); -// This test suite doesn't run on CrOS since it doesn't support non-print -// preview scripted printing. -#if !BUILDFLAG(IS_CHROMEOS) +#if BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) INSTANTIATE_TEST_SUITE_P(All, ContentAnalysisScriptedPreviewlessPrintBrowserTest, testing::Bool()); -#endif // !BUILDFLAG(IS_WIN) && !BUILDFLAG(IS_CHROMEOS) +#endif // BUILDFLAG(ENABLE_BASIC_PRINT_DIALOG) #endif // BUILDFLAG(ENABLE_PRINT_SCANNING)
diff --git a/chrome/browser/printing/printer_xml_parser_impl.cc b/chrome/browser/printing/printer_xml_parser_impl.cc new file mode 100644 index 0000000..de0efa8b --- /dev/null +++ b/chrome/browser/printing/printer_xml_parser_impl.cc
@@ -0,0 +1,60 @@ +// Copyright 2022 The Chromium Authors +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "chrome/browser/printing/printer_xml_parser_impl.h" + +#include <memory> +#include <utility> + +#include "base/callback.h" +#include "base/values.h" +#include "chrome/services/printing/public/mojom/printer_xml_parser.mojom.h" +#include "mojo/public/cpp/bindings/pending_receiver.h" +#include "mojo/public/cpp/bindings/pending_remote.h" +#include "mojo/public/cpp/bindings/receiver_set.h" +#include "printing/mojom/print.mojom.h" +#include "services/data_decoder/public/cpp/data_decoder.h" + +namespace printing { + +namespace { + +void XmlPrinterCapabilitiesParsed( + PrinterXmlParserImpl::ParseXmlCallback callback, + data_decoder::DataDecoder::ValueOrError value_or_error) { + if (!value_or_error.has_value()) { + std::move(callback).Run( + mojom::PrinterCapabilitiesValueResult::NewResultCode( + mojom::ResultCode::kFailed)); + return; + } + std::move(callback).Run( + mojom::PrinterCapabilitiesValueResult::NewCapabilities( + std::move(value_or_error.value()))); +} + +} // namespace + +PrinterXmlParserImpl::PrinterXmlParserImpl() = default; + +PrinterXmlParserImpl::~PrinterXmlParserImpl() = default; + +void PrinterXmlParserImpl::ParseXmlForPrinterCapabilities( + const std::string& capabilities_xml, + ParseXmlCallback callback) { + if (!decoder_) + decoder_ = std::make_unique<data_decoder::DataDecoder>(); + decoder_->ParseXml( + capabilities_xml, + data_decoder::mojom::XmlParser::WhitespaceBehavior::kIgnore, + base::BindOnce(&XmlPrinterCapabilitiesParsed, std::move(callback))); +} + +mojo::PendingRemote<mojom::PrinterXmlParser> PrinterXmlParserImpl::GetRemote() { + mojo::PendingRemote<mojom::PrinterXmlParser> pending_remote; + receivers_.Add(this, pending_remote.InitWithNewPipeAndPassReceiver()); + return pending_remote; +} + +} // namespace printing
diff --git a/chrome/browser/printing/printer_xml_parser_impl.h b/chrome/browser/printing/printer_xml_parser_impl.h new file mode 100644 index 0000000..a31055be --- /dev/null +++ b/chrome/browser/printing/printer_xml_parser_impl.h
@@ -0,0 +1,40 @@ +// Copyright 2022 The Chromium Authors +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#ifndef CHROME_BROWSER_PRINTING_PRINTER_XML_PARSER_IMPL_H_ +#define CHROME_BROWSER_PRINTING_PRINTER_XML_PARSER_IMPL_H_ + +#include <memory> + +#include "chrome/services/printing/public/mojom/printer_xml_parser.mojom.h" +#include "mojo/public/cpp/bindings/pending_remote.h" +#include "mojo/public/cpp/bindings/receiver_set.h" +#include "printing/mojom/print.mojom.h" +#include "services/data_decoder/public/cpp/data_decoder.h" + +namespace printing { + +class PrinterXmlParserImpl : public mojom::PrinterXmlParser { + public: + using ParseXmlCallback = + base::OnceCallback<void(mojom::PrinterCapabilitiesValueResultPtr)>; + + PrinterXmlParserImpl(); + PrinterXmlParserImpl(const PrinterXmlParserImpl&) = delete; + PrinterXmlParserImpl& operator=(const PrinterXmlParserImpl&) = delete; + ~PrinterXmlParserImpl() override; + + void ParseXmlForPrinterCapabilities(const std::string& capabilities_xml, + ParseXmlCallback callback) override; + + mojo::PendingRemote<mojom::PrinterXmlParser> GetRemote(); + + private: + mojo::ReceiverSet<mojom::PrinterXmlParser> receivers_; + std::unique_ptr<data_decoder::DataDecoder> decoder_; +}; + +} // namespace printing + +#endif // PRINTING_PRINTER_XML_PARSER_IMPL_H_
diff --git a/chrome/browser/privacy_sandbox/privacy_sandbox_service_unittest.cc b/chrome/browser/privacy_sandbox/privacy_sandbox_service_unittest.cc index a992b690..e8c5c966 100644 --- a/chrome/browser/privacy_sandbox/privacy_sandbox_service_unittest.cc +++ b/chrome/browser/privacy_sandbox/privacy_sandbox_service_unittest.cc
@@ -14,6 +14,7 @@ #include "chrome/browser/content_settings/cookie_settings_factory.h" #include "chrome/browser/content_settings/host_content_settings_map_factory.h" #include "chrome/browser/first_party_sets/first_party_sets_policy_service.h" +#include "chrome/browser/first_party_sets/mock_first_party_sets_handler.h" #include "chrome/browser/privacy_sandbox/privacy_sandbox_settings_factory.h" #include "chrome/browser/signin/identity_manager_factory.h" #include "chrome/common/chrome_features.h" @@ -683,9 +684,19 @@ void SetUp() override { CreateService(); + content::FirstPartySetsHandler::SetInstanceForTesting( + &mock_first_party_sets_handler_); + mock_first_party_sets_handler().SetCacheFilter( + net::FirstPartySetsCacheFilter()); + mock_first_party_sets_handler().SetContextConfig( + net::FirstPartySetsContextConfig()); SetGlobalFirstPartySetsAndWait(); } + void TearDown() override { + content::FirstPartySetsHandler::SetInstanceForTesting(nullptr); + } + virtual std::unique_ptr< privacy_sandbox_test_util::MockPrivacySandboxSettingsDelegate> GetMockDelegate() { @@ -752,6 +763,9 @@ browsing_topics::MockBrowsingTopicsService* mock_browsing_topics_service() { return &mock_browsing_topics_service_; } + first_party_sets::MockFirstPartySetsHandler& mock_first_party_sets_handler() { + return mock_first_party_sets_handler_; + } first_party_sets::FirstPartySetsPolicyService* first_party_sets_policy_service() { return &first_party_sets_policy_service_; @@ -763,8 +777,7 @@ #endif void SetGlobalFirstPartySetsAndWait() { - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting({}); + mock_first_party_sets_handler_.SetGlobalSets({}); base::RunLoop run_loop; first_party_sets_policy_service_.WaitForFirstInitCompleteForTesting( run_loop.QuitClosure()); @@ -779,6 +792,7 @@ base::test::ScopedFeatureList feature_list_; TestInterestGroupManager test_interest_group_manager_; browsing_topics::MockBrowsingTopicsService mock_browsing_topics_service_; + first_party_sets::MockFirstPartySetsHandler mock_first_party_sets_handler_; first_party_sets::FirstPartySetsPolicyService first_party_sets_policy_service_ = first_party_sets::FirstPartySetsPolicyService( @@ -2129,16 +2143,10 @@ content_settings::CookieControlsMode::kOff))); prefs()->SetUserPref(prefs::kPrivacySandboxFirstPartySetsEnabled, std::make_unique<base::Value>(true)); - // Reset test state to reflect required state above. - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - global_sets.Clone()); - first_party_sets_policy_service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + mock_first_party_sets_handler().SetGlobalSets(global_sets.Clone()); + + first_party_sets_policy_service()->InitForTesting(); // We shouldn't get associate1's owner since FPS is disabled. EXPECT_EQ(privacy_sandbox_service()->GetFirstPartySetOwner(associate1_gurl), absl::nullopt); @@ -2177,16 +2185,9 @@ prefs()->SetUserPref(prefs::kPrivacySandboxFirstPartySetsEnabled, std::make_unique<base::Value>(true)); - // Reset test state to reflect required state above. - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); + mock_first_party_sets_handler().SetGlobalSets(global_sets.Clone()); - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - global_sets.Clone()); - first_party_sets_policy_service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + first_party_sets_policy_service()->InitForTesting(); // We shouldn't get associate1's owner since FPS is disabled. EXPECT_EQ(privacy_sandbox_service()->GetFirstPartySetOwner(associate1_gurl), absl::nullopt); @@ -2227,16 +2228,9 @@ prefs()->SetUserPref(prefs::kPrivacySandboxFirstPartySetsEnabled, std::make_unique<base::Value>(true)); - // Reset test state to reflect required state above. - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); + mock_first_party_sets_handler().SetGlobalSets(global_sets.Clone()); - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - global_sets.Clone()); - first_party_sets_policy_service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + first_party_sets_policy_service()->InitForTesting(); // We shouldn't get associate1's owner since FPS is disabled. EXPECT_EQ(privacy_sandbox_service()->GetFirstPartySetOwner(associate1_gurl), @@ -2278,16 +2272,8 @@ prefs()->SetUserPref(prefs::kPrivacySandboxFirstPartySetsEnabled, std::make_unique<base::Value>(true)); - // Reset test state to reflect required state above. - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); - - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - global_sets.Clone()); - first_party_sets_policy_service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + mock_first_party_sets_handler().SetGlobalSets(global_sets.Clone()); + first_party_sets_policy_service()->InitForTesting(); // We shouldn't get associate1's owner since FPS is disabled. EXPECT_EQ(privacy_sandbox_service()->GetFirstPartySetOwner(associate1_gurl), @@ -2330,16 +2316,9 @@ prefs()->SetUserPref(prefs::kPrivacySandboxFirstPartySetsEnabled, std::make_unique<base::Value>(false)); - // Reset test state to reflect required state above. - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); + mock_first_party_sets_handler().SetGlobalSets(global_sets.Clone()); - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - global_sets.Clone()); - first_party_sets_policy_service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig()); - }); + first_party_sets_policy_service()->InitForTesting(); // We shouldn't get associate1's owner since FPS is disabled. EXPECT_EQ(privacy_sandbox_service()->GetFirstPartySetOwner(associate1_gurl), @@ -2413,32 +2392,26 @@ prefs()->SetUserPref(prefs::kPrivacySandboxFirstPartySetsEnabled, std::make_unique<base::Value>(true)); - // Reset test state to reflect required state above. - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); - // Simulate that the Global First-Party Sets are ready with the following set: // { primary: "https://primary.test", // associatedSites: ["https://associate1.test", "https://associate2.test"] } - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - net::GlobalFirstPartySets( - {{associate1_site, - {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, - 0)}}, - {associate2_site, - {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, - 1)}}}, - {})); + mock_first_party_sets_handler().SetGlobalSets(net::GlobalFirstPartySets( + {{associate1_site, + {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, 0)}}, + {associate2_site, + {net::FirstPartySetEntry(primary_site, net::SiteType::kAssociated, + 1)}}}, + {})); // Simulate that associate2 is removed from the Global First-Party Sets for // this profile. - first_party_sets_policy_service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run(net::FirstPartySetsContextConfig( - net::FirstPartySetsContextConfig::OverrideSets{ - {net::SchemefulSite(GURL("https://associate2.test")), - {absl::nullopt}}})); - }); + mock_first_party_sets_handler().SetContextConfig( + net::FirstPartySetsContextConfig( + net::FirstPartySetsContextConfig::OverrideSets{ + {net::SchemefulSite(GURL("https://associate2.test")), + {absl::nullopt}}})); + + first_party_sets_policy_service()->InitForTesting(); // Verify that primary owns associate1, but no longer owns associate2. EXPECT_EQ( @@ -2539,9 +2512,6 @@ prefs()->SetUserPref(prefs::kPrivacySandboxFirstPartySetsEnabled, std::make_unique<base::Value>(true)); - // Reset test state to reflect required state above. - content::FirstPartySetsHandler::GetInstance()->ResetForTesting(); - // Simulate that the Global First-Party Sets are ready with the following // set: // { primary: "https://youtube-primary.test", @@ -2551,26 +2521,23 @@ GURL youtube_gurl("https://youtube.com"); net::SchemefulSite youtube_site(youtube_gurl); - content::FirstPartySetsHandler::GetInstance()->SetGlobalSetsForTesting( - net::GlobalFirstPartySets( - {{youtube_site, - {net::FirstPartySetEntry(youtube_primary_site, - net::SiteType::kAssociated, 0)}}}, - {})); + mock_first_party_sets_handler().SetGlobalSets(net::GlobalFirstPartySets( + {{youtube_site, + {net::FirstPartySetEntry(youtube_primary_site, + net::SiteType::kAssociated, 0)}}}, + {})); // Simulate that https://google.de is moved into a new First-Party Set for // this profile. - first_party_sets_policy_service()->InitForTesting( - [](PrefService* prefs, - base::OnceCallback<void(net::FirstPartySetsContextConfig)> callback) { - std::move(callback).Run( - net::FirstPartySetsContextConfig(net::FirstPartySetsContextConfig( - net::FirstPartySetsContextConfig::OverrideSets{ - {net::SchemefulSite(GURL("https://google.de")), - {net::FirstPartySetEntry( - net::SchemefulSite(GURL("https://new-primary.test")), - net::SiteType::kAssociated, 0)}}}))); - }); + mock_first_party_sets_handler().SetContextConfig( + net::FirstPartySetsContextConfig( + net::FirstPartySetsContextConfig::OverrideSets{ + {net::SchemefulSite(GURL("https://google.de")), + {net::FirstPartySetEntry( + net::SchemefulSite(GURL("https://new-primary.test")), + net::SiteType::kAssociated, 0)}}})); + + first_party_sets_policy_service()->InitForTesting(); // Expect queries to be resolved based on the FPS sample sets. EXPECT_GT(privacy_sandbox_service()->GetSampleFirstPartySets().size(), 0u);
diff --git a/chrome/browser/renderer_context_menu/render_view_context_menu.cc b/chrome/browser/renderer_context_menu/render_view_context_menu.cc index a4c41a8..6760b10 100644 --- a/chrome/browser/renderer_context_menu/render_view_context_menu.cc +++ b/chrome/browser/renderer_context_menu/render_view_context_menu.cc
@@ -247,6 +247,7 @@ #if BUILDFLAG(GOOGLE_CHROME_BRANDING) #include "chrome/browser/lens/region_search/lens_region_search_controller.h" +#include "chrome/browser/ui/lens/lens_side_panel_helper.h" #include "chrome/grit/theme_resources.h" #include "ui/base/resource/resource_bundle.h" #endif @@ -3638,9 +3639,14 @@ int event_flags, bool is_google_default_search_provider) { #if BUILDFLAG(GOOGLE_CHROME_BRANDING) + Browser* browser = GetBrowser(); + CHECK(browser); + if (lens::features::IsLensRegionSearchStaticPageEnabled()) { + lens::OpenLensStaticPage(browser); + return; + } + if (!lens_region_search_controller_) { - Browser* browser = GetBrowser(); - CHECK(browser); WebContents* web_contents = source_web_contents_; if (base::FeatureList::IsEnabled( lens::features::kEnableRegionSearchOnPdfViewer)) {
diff --git a/chrome/browser/resources/BUILD.gn b/chrome/browser/resources/BUILD.gn index f185ec8..e3fdecd2 100644 --- a/chrome/browser/resources/BUILD.gn +++ b/chrome/browser/resources/BUILD.gn
@@ -53,7 +53,10 @@ "whats_new:resources", ] if (is_chrome_branded) { - public_deps += [ "media_router/cast_feedback:resources" ] + public_deps += [ + "lens:resources", + "media_router/cast_feedback:resources", + ] } }
diff --git a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/macro_names.js b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/macro_names.js index b6ffdee..3696654 100644 --- a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/macro_names.js +++ b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/macro_names.js
@@ -100,5 +100,8 @@ // Move the cursor to the previous sentence. NAV_PREV_SENT: 26, + // Deletes all text in the input box. + DELETE_ALL_TEXT: 27, + // Any new actions should match with Voice Access's semantic tags. };
diff --git a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/repeatable_key_press_macro.js b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/repeatable_key_press_macro.js index 9722f96..73e06f97 100644 --- a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/repeatable_key_press_macro.js +++ b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/macros/repeatable_key_press_macro.js
@@ -247,3 +247,16 @@ LocaleInfo.isRTLLocale() ? KeyCode.RIGHT : KeyCode.LEFT, {ctrl: true}); } } + +/** Macro to delete all text in input field. */ +export class DeleteAllText extends RepeatableKeyPressMacro { + constructor() { + super(MacroName.DELETE_ALL_TEXT, 1); + } + + /** @override */ + doKeyPress() { + EventGenerator.sendKeyPress(KeyCode.A, {ctrl: true}); + EventGenerator.sendKeyPress(KeyCode.BACK); + } +}
diff --git a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/pumpkin_parse_strategy.js b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/pumpkin_parse_strategy.js index e5854b2f..a3e63cf 100644 --- a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/pumpkin_parse_strategy.js +++ b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/pumpkin_parse_strategy.js
@@ -200,6 +200,8 @@ text = argument.value; } } + // TODO(crbug.com/1362842) Add all macros under the DictationMoreCommands + // to this switch statement. switch (tag) { case MacroName.INPUT_TEXT_VIEW: return new InputTextViewMacro(text, this.getInputController()); @@ -255,6 +257,9 @@ // Try to get results from Pumpkin. // TODO(crbug.com/1264544): Could increase the hypotheses count from 1 // when we are ready to implement disambiguation. + // TODO(crbug.com/1362842) Add logic to check whether + // DictationMoreCommands is enabled or not before + // running the macros hidden within that flag. const taggerResults = this.pumpkinTagger_.tagAndGetNBestHypotheses(text, 1); if (taggerResults && taggerResults.hypothesisList.length > 0) {
diff --git a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/simple_parse_strategy.js b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/simple_parse_strategy.js index eaa1a5d0..2dabfbc 100644 --- a/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/simple_parse_strategy.js +++ b/chrome/browser/resources/chromeos/accessibility/accessibility_common/dictation/parse/simple_parse_strategy.js
@@ -237,9 +237,17 @@ messageId: 'dictation_command_nav_prev_sent', build: NavPrevSentMacro, }, + [MacroName.DELETE_ALL_TEXT]: { + messageId: 'dictation_command_delete_all_text', + build: RepeatableKeyPress.DeleteAllText, + }, }; } } +// TODO(crbug.com/1362842) Do not release any Macros +// hidden under the DictationMoreCommands Flag +// using the simple_parse_strategy, rather +// move them over to the pumpkin_parse_strategy /** A parsing strategy that utilizes SimpleMacroFactory. */ export class SimpleParseStrategy extends ParseStrategy { @@ -252,6 +260,13 @@ * @private {!Map<MacroName, !SimpleMacroFactory>} */ this.macroFactoryMap_ = new Map(); + + /** @private {boolean} */ + this.isMoreCommandsFeatureEnabled_ = false; + + /** @private {!Array<!MacroName>}*/ + this.moreCommandsSet_ = [MacroName.DELETE_ALL_TEXT]; + this.initialize_(); } @@ -267,6 +282,13 @@ this.macroFactoryMap_.set( name, new SimpleMacroFactory(name, this.getInputController())); } + + const moreCommandsFeature = chrome.accessibilityPrivate.AccessibilityFeature + .DICTATION_MORE_COMMANDS; + chrome.accessibilityPrivate.isFeatureEnabled( + moreCommandsFeature, enabled => { + this.isMoreCommandsFeatureEnabled_ = enabled; + }); } /** @override */ @@ -280,12 +302,30 @@ this.initialize_(); } + /** + * @param {Macro} macro + * @return {boolean} + * @private + */ + shouldAddMacro_(macro) { + if (!macro) { + return false; + } + + const isNewCommand = this.moreCommandsSet_.includes(macro.getMacroName()); + if (!isNewCommand) { + return true; + } + + return this.isMoreCommandsFeatureEnabled_; + } + /** @override */ async parse(text) { const macros = []; for (const [name, factory] of this.macroFactoryMap_) { const macro = factory.createMacro(text); - if (macro) { + if (this.shouldAddMacro_(macro)) { macros.push(macro); } }
diff --git a/chrome/browser/resources/chromeos/accessibility/strings/dictation_strings.grdp b/chrome/browser/resources/chromeos/accessibility/strings/dictation_strings.grdp index 9b5ccab..6fb98ded 100644 --- a/chrome/browser/resources/chromeos/accessibility/strings/dictation_strings.grdp +++ b/chrome/browser/resources/chromeos/accessibility/strings/dictation_strings.grdp
@@ -78,4 +78,7 @@ <message name="DICTATION_COMMAND_NAV_PREV_SENT" desc="A spoken command to move the cursor to the previous sentence." is_accessibility_with_no_ui="true"> move to the previous sentence </message> + <message name="DICTATION_COMMAND_DELETE_ALL_TEXT" desc="A spoken command to delete all text in the input field." is_accessibility_with_no_ui="true"> + delete all + </message> </grit-part>
diff --git a/chrome/browser/resources/lens/BUILD.gn b/chrome/browser/resources/lens/BUILD.gn new file mode 100644 index 0000000..8ae57c93 --- /dev/null +++ b/chrome/browser/resources/lens/BUILD.gn
@@ -0,0 +1,15 @@ +# Copyright 2022 The Chromium Authors +# Use of this source code is governed by a BSD-style license that can be +# found in the LICENSE file. +import("//build/config/chrome_build.gni") +import("//chrome/browser/resources/tools/build_webui.gni") + +assert(toolkit_views) +assert(is_chrome_branded) + +build_webui("build") { + grd_prefix = "lens" + static_files = [ "lens.html" ] + non_web_component_files = [ "app.ts" ] + ts_use_local_config = false +}
diff --git a/chrome/browser/resources/lens/DIR_METADATA b/chrome/browser/resources/lens/DIR_METADATA new file mode 100644 index 0000000..749bdad --- /dev/null +++ b/chrome/browser/resources/lens/DIR_METADATA
@@ -0,0 +1,4 @@ +monorail { + component: "UI>Browser" +} +team_email: "lens-chrome-eng+bugs@google.com"
diff --git a/chrome/browser/resources/lens/OWNERS b/chrome/browser/resources/lens/OWNERS new file mode 100644 index 0000000..f6009d23 --- /dev/null +++ b/chrome/browser/resources/lens/OWNERS
@@ -0,0 +1 @@ +file://components/lens/OWNERS
diff --git a/chrome/browser/resources/lens/app.ts b/chrome/browser/resources/lens/app.ts new file mode 100644 index 0000000..d2e09d1 --- /dev/null +++ b/chrome/browser/resources/lens/app.ts
@@ -0,0 +1,7 @@ +// Copyright 2022 The Chromium Authors +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +document.addEventListener('DOMContentLoaded', function() { + console.info('Content loaded.'); +});
diff --git a/chrome/browser/resources/lens/lens.html b/chrome/browser/resources/lens/lens.html new file mode 100644 index 0000000..339a59e --- /dev/null +++ b/chrome/browser/resources/lens/lens.html
@@ -0,0 +1,12 @@ +<!doctype html> +<html dir="$i18n{textdirection}" lang="$i18n{language}"> +<head> + <meta charset="utf-8"> +</head> +<body> + <pre> + Placeholder text + </pre> + <script type="module" src="app.js"></script> +</body> +</html>
diff --git a/chrome/browser/resources/settings/chromeos/BUILD.gn b/chrome/browser/resources/settings/chromeos/BUILD.gn index 6785c150..4a27d27 100644 --- a/chrome/browser/resources/settings/chromeos/BUILD.gn +++ b/chrome/browser/resources/settings/chromeos/BUILD.gn
@@ -405,6 +405,7 @@ ":deep_linking_behavior", ":find_shortcut_behavior", ":global_scroll_target_behavior", + ":icon", ":lazy_load", ":metrics_recorder", ":os_icons", @@ -526,6 +527,9 @@ ] } +js_library("icon") { +} + js_library("personalization_search_handler") { deps = [ "//ash/webui/personalization_app/search:mojo_bindings_webui_js" ] }
diff --git a/chrome/browser/resources/settings/chromeos/date_time_page/date_time_page.ts b/chrome/browser/resources/settings/chromeos/date_time_page/date_time_page.ts index 5585ffe..8ca94560 100644 --- a/chrome/browser/resources/settings/chromeos/date_time_page/date_time_page.ts +++ b/chrome/browser/resources/settings/chromeos/date_time_page/date_time_page.ts
@@ -24,11 +24,10 @@ import {loadTimeData} from '../../i18n_setup.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; import {PrefsBehavior, PrefsBehaviorInterface} from '../prefs_behavior.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './date_time_page.html.js'; import {TimeZoneBrowserProxy, TimeZoneBrowserProxyImpl} from './timezone_browser_proxy.js'; @@ -38,11 +37,10 @@ [ DeepLinkingBehavior, PrefsBehavior, - RouteObserverBehavior, ], - I18nMixin(WebUIListenerMixin(PolymerElement))) as { + RouteObserverMixin(I18nMixin(WebUIListenerMixin(PolymerElement)))) as { new (): PolymerElement & DeepLinkingBehaviorInterface & - PrefsBehaviorInterface & RouteObserverBehaviorInterface & + PrefsBehaviorInterface & RouteObserverMixinInterface & I18nMixinInterface & WebUIListenerMixinInterface, };
diff --git a/chrome/browser/resources/settings/chromeos/date_time_page/timezone_subpage.ts b/chrome/browser/resources/settings/chromeos/date_time_page/timezone_subpage.ts index 01f2e46..0f55cf5 100644 --- a/chrome/browser/resources/settings/chromeos/date_time_page/timezone_subpage.ts +++ b/chrome/browser/resources/settings/chromeos/date_time_page/timezone_subpage.ts
@@ -19,11 +19,10 @@ import {SettingsDropdownMenuElement} from '../../controls/settings_dropdown_menu.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; import {PrefsBehavior, PrefsBehaviorInterface} from '../prefs_behavior.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {TimeZoneAutoDetectMethod} from './date_time_types.js'; import {TimeZoneBrowserProxy, TimeZoneBrowserProxyImpl} from './timezone_browser_proxy.js'; @@ -37,17 +36,17 @@ }; } -const TimezoneSubpageElementBase = mixinBehaviors( - [ - DeepLinkingBehavior, - PrefsBehavior, - RouteObserverBehavior, - ], - WebUIListenerMixin(PolymerElement)) as { - new (): PolymerElement & DeepLinkingBehaviorInterface & - PrefsBehaviorInterface & RouteObserverBehaviorInterface & - WebUIListenerMixinInterface, -}; +const TimezoneSubpageElementBase = + mixinBehaviors( + [ + DeepLinkingBehavior, + PrefsBehavior, + ], + RouteObserverMixin(WebUIListenerMixin(PolymerElement))) as { + new (): PolymerElement & DeepLinkingBehaviorInterface & + PrefsBehaviorInterface & RouteObserverMixinInterface & + WebUIListenerMixinInterface, + }; class TimezoneSubpageElement extends TimezoneSubpageElementBase { static get is() { @@ -87,7 +86,7 @@ } /** - * RouteObserverBehavior + * RouteObserverMixin * Called when the timezone subpage is hit. Child accounts need parental * approval to modify their timezone, this method starts this process on the * C++ side, and timezone setting will be disable. Once it is complete the
diff --git a/chrome/browser/resources/settings/chromeos/device_page/device_page.ts b/chrome/browser/resources/settings/chromeos/device_page/device_page.ts index 1f93fa7..db28856 100644 --- a/chrome/browser/resources/settings/chromeos/device_page/device_page.ts +++ b/chrome/browser/resources/settings/chromeos/device_page/device_page.ts
@@ -24,11 +24,10 @@ import {I18nMixin, I18nMixinInterface} from 'chrome://resources/cr_elements/i18n_mixin.js'; import {WebUIListenerMixin, WebUIListenerMixinInterface} from 'chrome://resources/cr_elements/web_ui_listener_mixin.js'; import {loadTimeData} from 'chrome://resources/js/load_time_data.m.js'; -import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; +import {PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; -import {Router} from '../../router.js'; +import {RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './device_page.html.js'; import {DevicePageBrowserProxy, DevicePageBrowserProxyImpl} from './device_page_browser_proxy.js'; @@ -39,12 +38,12 @@ }; } +// TODO(crbug/1315757) Remove need to typecast and intersect mixin interfaces +// once RouteObserverMixin is converted to TS const SettingsDevicePageElementBase = - mixinBehaviors( - [RouteObserverBehavior], - I18nMixin(WebUIListenerMixin(PolymerElement))) as { - new (): PolymerElement & I18nMixinInterface & - WebUIListenerMixinInterface & RouteObserverBehaviorInterface, + RouteObserverMixin(I18nMixin(WebUIListenerMixin(PolymerElement))) as { + new (): PolymerElement & WebUIListenerMixinInterface & + I18nMixinInterface & RouteObserverMixinInterface, }; class SettingsDevicePageElement extends SettingsDevicePageElementBase {
diff --git a/chrome/browser/resources/settings/chromeos/device_page/pointers.ts b/chrome/browser/resources/settings/chromeos/device_page/pointers.ts index 6558eabe..47e3ac5 100644 --- a/chrome/browser/resources/settings/chromeos/device_page/pointers.ts +++ b/chrome/browser/resources/settings/chromeos/device_page/pointers.ts
@@ -19,21 +19,19 @@ import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; import {PrefsBehavior, PrefsBehaviorInterface} from '../prefs_behavior.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './pointers.html.js'; -const SettingsPointersElementBase = - mixinBehaviors( - [DeepLinkingBehavior, PrefsBehavior, RouteObserverBehavior], - PolymerElement) as { - new (): PolymerElement & DeepLinkingBehaviorInterface & - PrefsBehaviorInterface & RouteObserverBehaviorInterface, - }; +const SettingsPointersElementBase = mixinBehaviors( + [DeepLinkingBehavior, PrefsBehavior], + RouteObserverMixin(PolymerElement)) as { + new (): PolymerElement & DeepLinkingBehaviorInterface & + PrefsBehaviorInterface & RouteObserverMixinInterface, +}; class SettingsPointersElement extends SettingsPointersElementBase { static get is() {
diff --git a/chrome/browser/resources/settings/chromeos/device_page/stylus.ts b/chrome/browser/resources/settings/chromeos/device_page/stylus.ts index 8987d02..9495f8f0 100644 --- a/chrome/browser/resources/settings/chromeos/device_page/stylus.ts +++ b/chrome/browser/resources/settings/chromeos/device_page/stylus.ts
@@ -20,12 +20,11 @@ import {microTask, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface} from '../../router.js'; import {assertExists} from '../assert_extras.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {recordSettingChange} from '../metrics_recorder.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {DevicePageBrowserProxy, DevicePageBrowserProxyImpl, NoteAppInfo, NoteAppLockScreenSupport} from './device_page_browser_proxy.js'; import {getTemplate} from './stylus.html.js'; @@ -40,10 +39,10 @@ 'collection/promotion_30023cb_stylus_apps'; const SettingsStylusElementBase = - mixinBehaviors( - [DeepLinkingBehavior, RouteObserverBehavior], PolymerElement) as { + mixinBehaviors([DeepLinkingBehavior], RouteObserverMixin(PolymerElement)) as + { new (): PolymerElement & DeepLinkingBehaviorInterface & - RouteObserverBehaviorInterface, + RouteObserverMixinInterface, }; class SettingsStylusElement extends SettingsStylusElementBase {
diff --git a/chrome/browser/resources/settings/chromeos/google_assistant_page/google_assistant_page.ts b/chrome/browser/resources/settings/chromeos/google_assistant_page/google_assistant_page.ts index 973e5cd3..9eb716e 100644 --- a/chrome/browser/resources/settings/chromeos/google_assistant_page/google_assistant_page.ts +++ b/chrome/browser/resources/settings/chromeos/google_assistant_page/google_assistant_page.ts
@@ -22,13 +22,12 @@ import {SettingsToggleButtonElement} from '../../controls/settings_toggle_button.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface} from '../../router.js'; import {cast, castExists} from '../assert_extras.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {recordSettingChange} from '../metrics_recorder.js'; import {routes} from '../os_route.js'; import {PrefsBehavior, PrefsBehaviorInterface} from '../prefs_behavior.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {GoogleAssistantBrowserProxy, GoogleAssistantBrowserProxyImpl} from './google_assistant_browser_proxy.js'; import {getTemplate} from './google_assistant_page.html.js'; @@ -68,12 +67,11 @@ [ DeepLinkingBehavior, PrefsBehavior, - RouteObserverBehavior, ], - WebUIListenerMixin(I18nMixin(PolymerElement))) as { + RouteObserverMixin(WebUIListenerMixin(I18nMixin(PolymerElement)))) as { new (): PolymerElement & I18nMixinInterface & WebUIListenerMixinInterface & DeepLinkingBehaviorInterface & - PrefsBehaviorInterface & RouteObserverBehaviorInterface, + PrefsBehaviorInterface & RouteObserverMixinInterface, }; class SettingsGoogleAssistantPageElement extends
diff --git a/chrome/browser/resources/settings/chromeos/icon.js b/chrome/browser/resources/settings/chromeos/icon.js new file mode 100644 index 0000000..ee53d5e --- /dev/null +++ b/chrome/browser/resources/settings/chromeos/icon.js
@@ -0,0 +1,85 @@ +// Copyright 2022 The Chromium Authors +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +/** + * Note: This file is deprecated in favor of the TypeScript version at + * ui/webui/resources/js/icon.ts. icon.ts should be used by all TypeScript + * files, and can also be used by any JS file that is not using closure + * compiler for typechecking. + */ + +/** + * @return {!Array<number>} The scale factors supported by this platform for + * webui resources. + */ +function getSupportedScaleFactors() { + // 1 is supported to match code in ResourceBundle::InitSharedInstance() that + // supports SCALE_FACTOR_100P on all non-iOS platforms. + // All desktop platforms including Ash support zooming which also updates the + // renderer's device scale factors (a.k.a devicePixelRatio), and these + // platforms have high DPI assets for 2x. Let the renderer pick the closest + // image for the current device scale factor. + return [1, 2]; +} + +/** + * Generates a CSS url string. + * @param {string} s The URL to generate the CSS url for. + * @return {string} The CSS url string. + */ +export function getUrlForCss(s) { + // http://www.w3.org/TR/css3-values/#uris + // Parentheses, commas, whitespace characters, single quotes (') and double + // quotes (") appearing in a URI must be escaped with a backslash + const s2 = s.replace(/(\(|\)|\,|\s|\'|\"|\\)/g, '\\$1'); + return `url("${s2}")`; +} + +/** + * Generates a CSS -webkit-image-set for a chrome:// url. + * An entry in the image set is added for each of getSupportedScaleFactors(). + * The scale-factor-specific url is generated by replacing the first instance + * of 'scalefactor' in |path| with the numeric scale factor. + * + * @param {string} path The URL to generate an image set for. + * 'scalefactor' should be a substring of |path|. + * @return {string} The CSS -webkit-image-set. + */ +function getImageSet(path) { + const supportedScaleFactors = getSupportedScaleFactors(); + + const replaceStartIndex = path.indexOf('SCALEFACTOR'); + if (replaceStartIndex < 0) { + return getUrlForCss(path); + } + + let s = ''; + for (let i = 0; i < supportedScaleFactors.length; ++i) { + const scaleFactor = supportedScaleFactors[i]; + const pathWithScaleFactor = path.substr(0, replaceStartIndex) + + scaleFactor + path.substr(replaceStartIndex + 'scalefactor'.length); + + s += getUrlForCss(pathWithScaleFactor) + ' ' + scaleFactor + 'x'; + + if (i !== supportedScaleFactors.length - 1) { + s += ', '; + } + } + return '-webkit-image-set(' + s + ')'; +} + +/** + * Returns the URL of the image, or an image set of URLs for the provided + * path. Resources in chrome://theme have multiple supported scale factors. + * + * @param {string} path The path of the image. + * @return {string} The url, or an image set of URLs. + */ +export function getImage(path) { + const chromeThemePath = 'chrome://theme'; + const isChromeThemeUrl = + (path.slice(0, chromeThemePath.length) === chromeThemePath); + return isChromeThemeUrl ? getImageSet(path + '@SCALEFACTORx') : + getUrlForCss(path); +}
diff --git a/chrome/browser/resources/settings/chromeos/kerberos_page/BUILD.gn b/chrome/browser/resources/settings/chromeos/kerberos_page/BUILD.gn index bd13cbd..8bb804f 100644 --- a/chrome/browser/resources/settings/chromeos/kerberos_page/BUILD.gn +++ b/chrome/browser/resources/settings/chromeos/kerberos_page/BUILD.gn
@@ -21,6 +21,7 @@ deps = [ ":kerberos_accounts_browser_proxy", "..:deep_linking_behavior", + "..:icon", "..:metrics_recorder", "..:os_route", "..:route_observer_behavior", @@ -30,7 +31,6 @@ "//ash/webui/common/resources:web_ui_listener_behavior", "//third_party/polymer/v3_0/components-chromium/polymer:polymer_bundled", "//ui/webui/resources/js:cr.m", - "//ui/webui/resources/js:icon", ] externs_list = [
diff --git a/chrome/browser/resources/settings/chromeos/kerberos_page/kerberos_accounts.js b/chrome/browser/resources/settings/chromeos/kerberos_page/kerberos_accounts.js index 767cf48..1e4ea58 100644 --- a/chrome/browser/resources/settings/chromeos/kerberos_page/kerberos_accounts.js +++ b/chrome/browser/resources/settings/chromeos/kerberos_page/kerberos_accounts.js
@@ -19,7 +19,6 @@ import './kerberos_add_account_dialog.js'; import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js'; -import {getImage} from 'chrome://resources/js/icon.js'; import {WebUIListenerBehavior, WebUIListenerBehaviorInterface} from 'chrome://resources/ash/common/web_ui_listener_behavior.js'; import {html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; @@ -27,6 +26,7 @@ import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; import {Route, Router} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; +import {getImage} from '../icon.js'; import {recordSettingChange} from '../metrics_recorder.js'; import {Account} from '../os_people_page/account_manager_browser_proxy.js'; import {routes} from '../os_route.js';
diff --git a/chrome/browser/resources/settings/chromeos/os_about_page/detailed_build_info.ts b/chrome/browser/resources/settings/chromeos/os_about_page/detailed_build_info.ts index 6297a7b..a6d4165b 100644 --- a/chrome/browser/resources/settings/chromeos/os_about_page/detailed_build_info.ts +++ b/chrome/browser/resources/settings/chromeos/os_about_page/detailed_build_info.ts
@@ -25,12 +25,11 @@ import {loadTimeData} from '../../i18n_setup.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface} from '../../router.js'; import {castExists} from '../assert_extras.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; import {PrefsBehavior, PrefsBehaviorInterface} from '../prefs_behavior.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {AboutPageBrowserProxy, AboutPageBrowserProxyImpl, browserChannelToI18nId, ChannelInfo, VersionInfo} from './about_page_browser_proxy.js'; import {getTemplate} from './detailed_build_info.html.js'; @@ -48,12 +47,11 @@ [ DeepLinkingBehavior, PrefsBehavior, - RouteObserverBehavior, ], - I18nMixin(WebUIListenerMixin(PolymerElement))) as { + RouteObserverMixin(I18nMixin(WebUIListenerMixin(PolymerElement)))) as { new (): PolymerElement & DeepLinkingBehaviorInterface & WebUIListenerMixinInterface & I18nMixinInterface & - PrefsBehaviorInterface & RouteObserverBehaviorInterface, + PrefsBehaviorInterface & RouteObserverMixinInterface, }; class SettingsDetailedBuildInfoElement extends SettingsDetailedBuildInfoBase {
diff --git a/chrome/browser/resources/settings/chromeos/os_files_page/os_files_page.ts b/chrome/browser/resources/settings/chromeos/os_files_page/os_files_page.ts index 0643230..243b19e0 100644 --- a/chrome/browser/resources/settings/chromeos/os_files_page/os_files_page.ts +++ b/chrome/browser/resources/settings/chromeos/os_files_page/os_files_page.ts
@@ -16,18 +16,17 @@ import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './os_files_page.html.js'; const OsSettingsFilesPageElementBase = - mixinBehaviors( - [DeepLinkingBehavior, RouteObserverBehavior], PolymerElement) as { - new (): DeepLinkingBehaviorInterface & RouteObserverBehaviorInterface & - PolymerElement, + mixinBehaviors([DeepLinkingBehavior], RouteObserverMixin(PolymerElement)) as + { + new (): PolymerElement & DeepLinkingBehaviorInterface & + RouteObserverMixinInterface, }; class OsSettingsFilesPageElement extends OsSettingsFilesPageElementBase {
diff --git a/chrome/browser/resources/settings/chromeos/os_files_page/smb_shares_page.ts b/chrome/browser/resources/settings/chromeos/os_files_page/smb_shares_page.ts index 7b5a618..9202a6e 100644 --- a/chrome/browser/resources/settings/chromeos/os_files_page/smb_shares_page.ts +++ b/chrome/browser/resources/settings/chromeos/os_files_page/smb_shares_page.ts
@@ -11,18 +11,18 @@ import '../../settings_shared.css.js'; import '../../settings_vars.css.js'; -import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; +import {PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './smb_shares_page.html.js'; -const SettingsSmbSharesPageElementBase = - mixinBehaviors([RouteObserverBehavior], PolymerElement) as { - new (): PolymerElement & RouteObserverBehaviorInterface, - }; +// TODO(crbug/1315757) Remove need to typecast and intersect mixin interfaces +// once RouteObserverMixin is converted to TS +const SettingsSmbSharesPageElementBase = RouteObserverMixin(PolymerElement) as { + new (): PolymerElement & RouteObserverMixinInterface, +}; class SettingsSmbSharesPageElement extends SettingsSmbSharesPageElementBase { static get is() { @@ -51,7 +51,7 @@ private showAddSmbDialog_: boolean; /** - * Overridden from RouteObserverBehavior. + * Overridden from RouteObserverMixin. */ override currentRouteChanged(route: Route) { if (route === routes.SMB_SHARES) {
diff --git a/chrome/browser/resources/settings/chromeos/os_people_page/BUILD.gn b/chrome/browser/resources/settings/chromeos/os_people_page/BUILD.gn index 944d364..6429132 100644 --- a/chrome/browser/resources/settings/chromeos/os_people_page/BUILD.gn +++ b/chrome/browser/resources/settings/chromeos/os_people_page/BUILD.gn
@@ -36,6 +36,7 @@ deps = [ ":account_manager_browser_proxy", "..:deep_linking_behavior", + "..:icon", "..:metrics_recorder", "..:os_route", "..:route_observer_behavior", @@ -44,7 +45,6 @@ "//ash/webui/common/resources:web_ui_listener_behavior", "//third_party/polymer/v3_0/components-chromium/polymer:polymer_bundled", "//ui/webui/resources/js:cr.m", - "//ui/webui/resources/js:icon", ] externs_list = [ @@ -128,6 +128,7 @@ ":lock_state_behavior", ":os_sync_controls", "..:deep_linking_behavior", + "..:icon", "..:os_page_visibility", "..:os_route", "..:route_observer_behavior", @@ -139,7 +140,6 @@ "//ash/webui/common/resources/quick_unlock:lock_screen_constants", "//third_party/polymer/v3_0/components-chromium/polymer:polymer_bundled", "//ui/webui/resources/js:assert", - "//ui/webui/resources/js:icon", "//ui/webui/resources/js:load_time_data.m", ] externs_list =
diff --git a/chrome/browser/resources/settings/chromeos/os_people_page/account_manager.js b/chrome/browser/resources/settings/chromeos/os_people_page/account_manager.js index 53c5a53..5443be0 100644 --- a/chrome/browser/resources/settings/chromeos/os_people_page/account_manager.js +++ b/chrome/browser/resources/settings/chromeos/os_people_page/account_manager.js
@@ -18,7 +18,6 @@ import '../../settings_shared.css.js'; import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js'; -import {getImage} from 'chrome://resources/js/icon.js'; import {WebUIListenerBehavior, WebUIListenerBehaviorInterface} from 'chrome://resources/ash/common/web_ui_listener_behavior.js'; import {html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; @@ -26,6 +25,7 @@ import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; import {Route} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; +import {getImage} from '../icon.js'; import {recordSettingChange} from '../metrics_recorder.js'; import {routes} from '../os_route.js'; import {ParentalControlsBrowserProxy, ParentalControlsBrowserProxyImpl} from '../parental_controls_page/parental_controls_browser_proxy.js';
diff --git a/chrome/browser/resources/settings/chromeos/os_people_page/os_people_page.js b/chrome/browser/resources/settings/chromeos/os_people_page/os_people_page.js index 1d46ef6..30a3096 100644 --- a/chrome/browser/resources/settings/chromeos/os_people_page/os_people_page.js +++ b/chrome/browser/resources/settings/chromeos/os_people_page/os_people_page.js
@@ -33,7 +33,6 @@ import {WebUIListenerBehavior, WebUIListenerBehaviorInterface} from 'chrome://resources/ash/common/web_ui_listener_behavior.js'; import {assert} from 'chrome://resources/js/assert.js'; import {sendWithPromise} from 'chrome://resources/js/cr.m.js'; -import {getImage} from 'chrome://resources/js/icon.js'; import {afterNextRender, flush, html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {loadTimeData} from '../../i18n_setup.js'; @@ -42,6 +41,7 @@ import {SyncBrowserProxyImpl} from '../../people_page/sync_browser_proxy.js'; import {Route, Router} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; +import {getImage} from '../icon.js'; import {OSPageVisibility} from '../os_page_visibility.js'; import {routes} from '../os_route.js'; import {SettingsParentalControlsPageElement} from '../parental_controls_page/parental_controls_page.js';
diff --git a/chrome/browser/resources/settings/chromeos/os_reset_page/os_reset_page.ts b/chrome/browser/resources/settings/chromeos/os_reset_page/os_reset_page.ts index d6248cf..46042da 100644 --- a/chrome/browser/resources/settings/chromeos/os_reset_page/os_reset_page.ts +++ b/chrome/browser/resources/settings/chromeos/os_reset_page/os_reset_page.ts
@@ -16,10 +16,9 @@ import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './os_reset_page.html.js'; @@ -30,10 +29,10 @@ } const OsSettingsResetPageElementBase = - mixinBehaviors( - [DeepLinkingBehavior, RouteObserverBehavior], PolymerElement) as { + mixinBehaviors([DeepLinkingBehavior], RouteObserverMixin(PolymerElement)) as + { new (): PolymerElement & DeepLinkingBehaviorInterface & - RouteObserverBehaviorInterface, + RouteObserverMixinInterface, }; class OsSettingsResetPageElement extends OsSettingsResetPageElementBase {
diff --git a/chrome/browser/resources/settings/chromeos/os_search_page/os_search_page.ts b/chrome/browser/resources/settings/chromeos/os_search_page/os_search_page.ts index 4470b02..b06f9569 100644 --- a/chrome/browser/resources/settings/chromeos/os_search_page/os_search_page.ts +++ b/chrome/browser/resources/settings/chromeos/os_search_page/os_search_page.ts
@@ -27,19 +27,18 @@ import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './os_search_page.html.js'; const OsSettingsSearchPageElementBase = mixinBehaviors( - [DeepLinkingBehavior, RouteObserverBehavior], - I18nMixin(PolymerElement)) as { + [DeepLinkingBehavior], RouteObserverMixin(I18nMixin(PolymerElement))) as + { new (): PolymerElement & I18nMixinInterface & - DeepLinkingBehaviorInterface & RouteObserverBehaviorInterface, + DeepLinkingBehaviorInterface & RouteObserverMixinInterface, }; class OsSettingsSearchPageElement extends OsSettingsSearchPageElementBase {
diff --git a/chrome/browser/resources/settings/chromeos/os_search_page/search_subpage.ts b/chrome/browser/resources/settings/chromeos/os_search_page/search_subpage.ts index 414753e8..edef082 100644 --- a/chrome/browser/resources/settings/chromeos/os_search_page/search_subpage.ts +++ b/chrome/browser/resources/settings/chromeos/os_search_page/search_subpage.ts
@@ -25,22 +25,21 @@ import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {Setting} from '../../mojom-webui/setting.mojom-webui.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {castExists} from '../assert_extras.js'; import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js'; import {routes} from '../os_route.js'; import {PrefsBehavior, PrefsBehaviorInterface} from '../prefs_behavior.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './search_subpage.html.js'; const SettingsSearchSubpageElementBase = mixinBehaviors( - [DeepLinkingBehavior, PrefsBehavior, RouteObserverBehavior], - I18nMixin(PolymerElement)) as { + [DeepLinkingBehavior, PrefsBehavior], + RouteObserverMixin(I18nMixin(PolymerElement))) as { new (): PolymerElement & DeepLinkingBehaviorInterface & I18nMixinInterface & PrefsBehaviorInterface & - RouteObserverBehaviorInterface, + RouteObserverMixinInterface, }; class SettingsSearchSubpageElement extends SettingsSearchSubpageElementBase {
diff --git a/chrome/browser/resources/settings/chromeos/os_settings.gni b/chrome/browser/resources/settings/chromeos/os_settings.gni index 1ad40de8f..dc1b84c 100644 --- a/chrome/browser/resources/settings/chromeos/os_settings.gni +++ b/chrome/browser/resources/settings/chromeos/os_settings.gni
@@ -137,6 +137,7 @@ "chromeos/global_scroll_target_behavior.js", "chromeos/google_assistant_page/google_assistant_browser_proxy.ts", "chromeos/guest_os/guest_os_browser_proxy.js", + "chromeos/icon.js", "chromeos/internet_page/cellular_setup_settings_delegate.js", "chromeos/internet_page/internet_page_browser_proxy.js", "chromeos/kerberos_page/kerberos_accounts_browser_proxy.js",
diff --git a/chrome/browser/resources/settings/chromeos/os_settings_main/os_settings_main.ts b/chrome/browser/resources/settings/chromeos/os_settings_main/os_settings_main.ts index f372cc33..7341116 100644 --- a/chrome/browser/resources/settings/chromeos/os_settings_main/os_settings_main.ts +++ b/chrome/browser/resources/settings/chromeos/os_settings_main/os_settings_main.ts
@@ -17,14 +17,13 @@ import '../../settings_shared.css.js'; import '../../settings_vars.css.js'; -import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; +import {PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {loadTimeData} from '../../i18n_setup.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {assertExists} from '../assert_extras.js'; import {OSPageVisibility} from '../os_page_visibility.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './os_settings_main.html.js'; @@ -39,10 +38,11 @@ }; } -const OsSettingsMainElementBase = - mixinBehaviors([RouteObserverBehavior], PolymerElement) as { - new (): PolymerElement & RouteObserverBehaviorInterface, - }; +// TODO(crbug/1315757) Remove need to typecast and intersect mixin interfaces +// once RouteObserverMixin is converted to TS +const OsSettingsMainElementBase = RouteObserverMixin(PolymerElement) as { + new (): PolymerElement & RouteObserverMixinInterface, +}; class OsSettingsMainElement extends OsSettingsMainElementBase { static get is() {
diff --git a/chrome/browser/resources/settings/chromeos/os_settings_menu/os_settings_menu.ts b/chrome/browser/resources/settings/chromeos/os_settings_menu/os_settings_menu.ts index 385b0edc..457a6ef 100644 --- a/chrome/browser/resources/settings/chromeos/os_settings_menu/os_settings_menu.ts +++ b/chrome/browser/resources/settings/chromeos/os_settings_menu/os_settings_menu.ts
@@ -17,12 +17,11 @@ import {loadTimeData} from 'chrome://resources/js/load_time_data.m.js'; import {IronCollapseElement} from 'chrome://resources/polymer/v3_0/iron-collapse/iron-collapse.js'; import {IronSelectorElement} from 'chrome://resources/polymer/v3_0/iron-selector/iron-selector.js'; -import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; +import {PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {castExists} from '../assert_extras.js'; import {routes} from '../os_route.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './os_settings_menu.html.js'; @@ -34,10 +33,11 @@ }; } -const OsSettingsMenuElementBase = - mixinBehaviors([RouteObserverBehavior], PolymerElement) as { - new (): PolymerElement & RouteObserverBehaviorInterface, - }; +// TODO(crbug/1315757) Remove need to typecast and intersect mixin interfaces +// once RouteObserverMixin is converted to TS +const OsSettingsMenuElementBase = RouteObserverMixin(PolymerElement) as { + new (): PolymerElement & RouteObserverMixinInterface, +}; class OsSettingsMenuElement extends OsSettingsMenuElementBase { static get is() {
diff --git a/chrome/browser/resources/settings/chromeos/os_settings_ui/os_settings_ui.ts b/chrome/browser/resources/settings/chromeos/os_settings_ui/os_settings_ui.ts index 8ab15d404..ceae6a7 100644 --- a/chrome/browser/resources/settings/chromeos/os_settings_ui/os_settings_ui.ts +++ b/chrome/browser/resources/settings/chromeos/os_settings_ui/os_settings_ui.ts
@@ -22,23 +22,22 @@ import '../../prefs/prefs.js'; import '../../settings_vars.css.js'; -import {CrContainerShadowBehavior} from 'chrome://resources/ash/common/cr_container_shadow_behavior.js'; +import {CrContainerShadowMixin, CrContainerShadowMixinInterface} from 'chrome://resources/cr_elements/cr_container_shadow_mixin.js'; import {CrDrawerElement} from 'chrome://resources/cr_elements/cr_drawer/cr_drawer.js'; import {FindShortcutMixin, FindShortcutMixinInterface} from 'chrome://resources/cr_elements/find_shortcut_mixin.js'; import {assert} from 'chrome://resources/js/assert_ts.js'; import {listenOnce} from 'chrome://resources/js/util.js'; -import {Debouncer, DomIf, microTask, mixinBehaviors, PolymerElement, timeOut} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; +import {Debouncer, DomIf, microTask, PolymerElement, timeOut} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; import {loadTimeData} from '../../i18n_setup.js'; import {SettingsPrefsElement} from '../../prefs/prefs.js'; -import {Route, Router} from '../../router.js'; +import {Route, RouteObserverMixin, RouteObserverMixinInterface, Router} from '../../router.js'; import {castExists} from '../assert_extras.js'; import {setGlobalScrollTarget} from '../global_scroll_target_behavior.js'; import {recordClick, recordNavigation, recordPageBlur, recordPageFocus, recordSettingChange} from '../metrics_recorder.js'; import {OSPageVisibility, osPageVisibility} from '../os_page_visibility.js'; import {OsToolbarElement} from '../os_toolbar/os_toolbar.js'; import {PrefToSettingMetricConverter} from '../pref_to_setting_metric_converter.js'; -import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js'; import {getTemplate} from './os_settings_ui.html.js'; @@ -72,17 +71,16 @@ }; } +// TODO(crbug/1315757) Remove need to typecast and intersect mixin interfaces +// once RouteObserverMixin is converted to TS const OsSettingsUiElementBase = - mixinBehaviors( - [ - CrContainerShadowBehavior, - // Calls currentRouteChanged() in attached(),so ensure other behaviors - // run their attached() first. - RouteObserverBehavior, - ], - FindShortcutMixin(PolymerElement)) as { - new (): PolymerElement & CrContainerShadowBehavior & - FindShortcutMixinInterface & RouteObserverBehaviorInterface, + // RouteObserverMixin calls currentRouteChanged() in + // connectedCallback(), so ensure other mixins/behaviors run their + // connectedCallback() first. + RouteObserverMixin( + FindShortcutMixin(CrContainerShadowMixin(PolymerElement))) as { + new (): PolymerElement & FindShortcutMixinInterface & + CrContainerShadowMixinInterface & RouteObserverMixinInterface, }; class OsSettingsUiElement extends OsSettingsUiElementBase {
diff --git a/chrome/browser/resources/side_panel/read_anything/app.ts b/chrome/browser/resources/side_panel/read_anything/app.ts index 28124e1a..bfda629 100644 --- a/chrome/browser/resources/side_panel/read_anything/app.ts +++ b/chrome/browser/resources/side_panel/read_anything/app.ts
@@ -82,6 +82,10 @@ } const element = document.createElement(htmlTag); + const direction = chrome.readAnything.getTextDirection(nodeId); + if (direction) { + element.setAttribute('dir', direction); + } const url = chrome.readAnything.getUrl(nodeId); if (url) { element.setAttribute('href', url);
diff --git a/chrome/browser/resources/side_panel/read_anything/read_anything.d.ts b/chrome/browser/resources/side_panel/read_anything/read_anything.d.ts index 7a8d604a..04ae40a 100644 --- a/chrome/browser/resources/side_panel/read_anything/read_anything.d.ts +++ b/chrome/browser/resources/side_panel/read_anything/read_anything.d.ts
@@ -38,6 +38,9 @@ // the selected text. function getTextContent(nodeId: number): string; + // Returns the text direction of the AXNode for the provided AXNodeID. + function getTextDirection(nodeId: number): string; + // Returns the url of the AXNode for the provided AXNodeID. function getUrl(nodeId: number): string;
diff --git a/chrome/browser/resources/tab_strip/drag_manager.ts b/chrome/browser/resources/tab_strip/drag_manager.ts index 05d7e6e9..06c2ae71 100644 --- a/chrome/browser/resources/tab_strip/drag_manager.ts +++ b/chrome/browser/resources/tab_strip/drag_manager.ts
@@ -69,7 +69,7 @@ placeTabGroupElement(element: TabGroupElement, index: number): void; - shouldPreventDrag(): boolean; + shouldPreventDrag(isDraggingTab: boolean): boolean; } type DragManagerDelegateElement = DragManagerDelegate&HTMLElement; @@ -470,7 +470,7 @@ return; } - if (this.delegate_.shouldPreventDrag()) { + if (this.delegate_.shouldPreventDrag(isTabElement(draggedItem))) { event.preventDefault(); return; }
diff --git a/chrome/browser/resources/tab_strip/tab_list.ts b/chrome/browser/resources/tab_strip/tab_list.ts index 7909f39..4b9ce7b 100644 --- a/chrome/browser/resources/tab_strip/tab_list.ts +++ b/chrome/browser/resources/tab_strip/tab_list.ts
@@ -778,12 +778,17 @@ this.animateScrollPosition_(scrollBy); } - shouldPreventDrag(): boolean { - // Do not allow dragging if there's only 1 tab with no tab group, or only 1 - // tab group with no other tabs outside of the tab group. - return (this.pinnedTabsElement_.childElementCount + - this.unpinnedTabsElement_.childElementCount) === - 1; + shouldPreventDrag(isDraggingTab: boolean): boolean { + if (isDraggingTab) { + // Do not allow dragging a tab if there's only 1 tab. + return this.$all('tabstrip-tab').length === 1; + } else { + // Do not allow dragging the tab group with no others outside of the tab + // group. In this case there is only 1 pinned and unpinned top level + // element, which is the dragging tab group itself. + return (this.pinnedTabsElement_.childElementCount + + this.unpinnedTabsElement_.childElementCount) === 1; + } } private tabThumbnailUpdated_(tabId: number, imgData: string) {
diff --git a/chrome/browser/safe_browsing/chrome_password_protection_service_unittest.cc b/chrome/browser/safe_browsing/chrome_password_protection_service_unittest.cc index 1a5bd4b..0c9daa6 100644 --- a/chrome/browser/safe_browsing/chrome_password_protection_service_unittest.cc +++ b/chrome/browser/safe_browsing/chrome_password_protection_service_unittest.cc
@@ -962,6 +962,26 @@ EXPECT_EQ("token2", reuse_lookup.verdict_token()) << t; t++; } + + { + auto response = std::make_unique<LoginReputationClientResponse>(); + response->set_verdict_token("token3"); + response->set_verdict_type(LoginReputationClientResponse::PHISHING); + service_->MaybeLogPasswordReuseLookupEvent( + web_contents(), RequestOutcome::RESPONSE_ALREADY_CACHED, + PasswordType::PRIMARY_ACCOUNT_PASSWORD, response.get()); + ASSERT_EQ(t + 1, GetUserEventService()->GetRecordedUserEvents().size()) + << t; + PasswordReuseLookup reuse_lookup = GetUserEventService() + ->GetRecordedUserEvents()[t] + .gaia_password_reuse_event() + .reuse_lookup(); + EXPECT_EQ(PasswordReuseLookup::CACHE_HIT, reuse_lookup.lookup_result()) + << t; + EXPECT_EQ(PasswordReuseLookup::PHISHING, reuse_lookup.verdict()) << t; + EXPECT_EQ("token3", reuse_lookup.verdict_token()) << t; + t++; + } } TEST_F(ChromePasswordProtectionServiceTest, VerifyGetDefaultChangePasswordURL) {
diff --git a/chrome/browser/segmentation_platform/segmentation_platform_config_unittest.cc b/chrome/browser/segmentation_platform/segmentation_platform_config_unittest.cc index 981be8bf..d08dece 100644 --- a/chrome/browser/segmentation_platform/segmentation_platform_config_unittest.cc +++ b/chrome/browser/segmentation_platform/segmentation_platform_config_unittest.cc
@@ -17,27 +17,14 @@ namespace segmentation_platform { -using ::base::test::ScopedFeatureList; - class SegmentationPlatformConfigTest : public testing::Test { public: SegmentationPlatformConfigTest() = default; ~SegmentationPlatformConfigTest() override = default; - void SetUp() override { - Test::SetUp(); - scoped_feature_list_ = std::make_unique<ScopedFeatureList>(); - } - - void TearDown() override { - Test::TearDown(); - scoped_feature_list_.reset(); - } - void EnableFeaturesWithParams( - const std::vector<ScopedFeatureList::FeatureAndParams>& - enabled_features) { - scoped_feature_list_->InitWithFeaturesAndParameters(enabled_features, {}); + const std::vector<base::test::FeatureRefAndParams>& enabled_features) { + scoped_feature_list_.InitWithFeaturesAndParameters(enabled_features, {}); // Activate each field trial so that these trials are considered as Active // groups. the FieldTrialList currently does not consider force enabled // trials as "active" by default. @@ -51,7 +38,7 @@ } protected: - std::unique_ptr<ScopedFeatureList> scoped_feature_list_; + base::test::ScopedFeatureList scoped_feature_list_; }; TEST_F(SegmentationPlatformConfigTest, GetSegmentationPlatformConfig) { @@ -63,15 +50,14 @@ } TEST_F(SegmentationPlatformConfigTest, EmptyFeatures) { - ScopedFeatureList scoped_feature_list_; std::vector<std::unique_ptr<Config>> configs; AppendConfigsFromExperiments(configs); EXPECT_TRUE(configs.empty()); } TEST_F(SegmentationPlatformConfigTest, BadFormat) { - std::vector<ScopedFeatureList::FeatureAndParams> features; - features.push_back(ScopedFeatureList::FeatureAndParams( + std::vector<base::test::FeatureRefAndParams> features; + features.push_back(base::test::FeatureRefAndParams( features::kSegmentationPlatformLowEngagementFeature, {{"segmentation_platform_add_config_param", "bad_json"}})); EnableFeaturesWithParams(features); @@ -119,14 +105,14 @@ proto::SegmentId:: OPTIMIZATION_TARGET_NOTIFICATION_PERMISSION_PREDICTIONS}; - std::vector<ScopedFeatureList::FeatureAndParams> features; - features.push_back(ScopedFeatureList::FeatureAndParams( + std::vector<base::test::FeatureRefAndParams> features; + features.push_back(base::test::FeatureRefAndParams( features::kSegmentationPlatformLowEngagementFeature, {{"segmentation_platform_add_config_param", "bad_json"}})); - features.push_back(ScopedFeatureList::FeatureAndParams( + features.push_back(base::test::FeatureRefAndParams( features::kSegmentationPlatformFeedSegmentFeature, {{"segmentation_platform_add_config_param", kValidConfig1}})); - features.push_back(ScopedFeatureList::FeatureAndParams( + features.push_back(base::test::FeatureRefAndParams( features::kShoppingUserSegmentFeature, {{"segmentation_platform_add_config_param", kValidConfig2}})); EnableFeaturesWithParams(features);
diff --git a/chrome/browser/sessions/DEPS b/chrome/browser/sessions/DEPS index ee77d5b..7878abe 100644 --- a/chrome/browser/sessions/DEPS +++ b/chrome/browser/sessions/DEPS
@@ -8,8 +8,4 @@ # Tests may depend upon views. "+chrome/browser/ui/views", ], - "session_restore\.cc": [ - "+ash/shell.h", - "+ash/metrics/login_unlock_throughput_recorder.h", - ], }
diff --git a/chrome/browser/sessions/session_restore.cc b/chrome/browser/sessions/session_restore.cc index f49871b..f71a402e 100644 --- a/chrome/browser/sessions/session_restore.cc +++ b/chrome/browser/sessions/session_restore.cc
@@ -92,11 +92,7 @@ #include "extensions/common/extension_set.h" #if BUILDFLAG(IS_CHROMEOS_ASH) -#include "ash/metrics/login_unlock_throughput_recorder.h" -#include "ash/shell.h" #include "chrome/browser/ash/boot_times_recorder.h" -#include "components/app_restore/window_properties.h" -#include "ui/compositor/layer.h" #endif using content::NavigationController; @@ -118,50 +114,6 @@ // Pointers to SessionRestoreImpls which are currently restoring the session. std::set<SessionRestoreImpl*>* active_session_restorers = nullptr; -#if BUILDFLAG(IS_CHROMEOS_ASH) -void StartRecordingRestoredWindowsMetrics( - const std::vector<std::unique_ptr<sessions::SessionWindow>>& windows) { - // Ash is not always initialized in unit tests. - if (!ash::Shell::HasInstance()) - return; - - ash::LoginUnlockThroughputRecorder* throughput_recorder = - ash::Shell::Get()->login_unlock_throughput_recorder(); - - for (const auto& w : windows) { - if (w->type == sessions::SessionWindow::TYPE_NORMAL) { - throughput_recorder->AddScheduledRestoreWindow( - w->window_id.id(), w->app_name, - ash::LoginUnlockThroughputRecorder::kBrowser); - } - } -} - -void ReportRestoredWindowCreated(aura::Window* window) { - // Ash is not always initialized in unit tests. - if (!ash::Shell::HasInstance()) - return; - - const int32_t restore_window_id = - window->GetProperty(app_restore::kRestoreWindowIdKey); - - // Restored window IDs are always non-zero. - if (restore_window_id == 0) - return; - - ash::LoginUnlockThroughputRecorder* throughput_recorder = - ash::Shell::Get()->login_unlock_throughput_recorder(); - throughput_recorder->OnRestoredWindowCreated(restore_window_id); - aura::Window* root_window = window->GetRootWindow(); - if (root_window) { - ui::Compositor* compositor = root_window->layer()->GetCompositor(); - throughput_recorder->OnBeforeRestoredWindowShown(restore_window_id, - compositor); - } -} - -#endif - } // namespace // SessionRestoreImpl --------------------------------------------------------- @@ -463,11 +415,6 @@ if (!read_error_) read_error_ = read_error; -#if BUILDFLAG(IS_CHROMEOS_ASH) - if (!read_error_) - StartRecordingRestoredWindowsMetrics(windows); -#endif - // Copy windows into windows_ so that we can combine both app and browser // windows together before doing a one-pass restore. std::copy(std::make_move_iterator(windows.begin()), @@ -629,7 +576,6 @@ #if BUILDFLAG(IS_CHROMEOS_ASH) ash::BootTimesRecorder::Get()->AddLoginTimeMarker( "SessionRestore-CreateRestoredBrowser-End", false); - ReportRestoredWindowCreated(browser->window()->GetNativeWindow()); #endif }
diff --git a/chrome/browser/site_isolation/isolated_sandboxed_iframe_browsertest.cc b/chrome/browser/site_isolation/isolated_sandboxed_iframe_browsertest.cc index ceb6f0c..9244c679 100644 --- a/chrome/browser/site_isolation/isolated_sandboxed_iframe_browsertest.cc +++ b/chrome/browser/site_isolation/isolated_sandboxed_iframe_browsertest.cc
@@ -20,6 +20,7 @@ #include "content/public/test/test_navigation_observer.h" #include "net/dns/mock_host_resolver.h" #include "net/test/embedded_test_server/embedded_test_server.h" +#include "third_party/blink/public/common/features.h" #include "url/gurl.h" class TestMemoryDetails : public MetricsMemoryDetails { @@ -75,13 +76,13 @@ // BrowsingInstances. if (enable_isolate_sandboxed_iframes_) { feature_list_.InitWithFeatures( - /* enable_features */ {features::kIsolateSandboxedIframes, + /* enable_features */ {blink::features::kIsolateSandboxedIframes, features::kDisableProcessReuse}, /* disable_features */ {features::kSpareRendererForSitePerProcess}); } else { feature_list_.InitWithFeatures( /* enable_features */ {features::kDisableProcessReuse}, - /* disable_features */ {features::kIsolateSandboxedIframes, + /* disable_features */ {blink::features::kIsolateSandboxedIframes, features::kSpareRendererForSitePerProcess}); } }
diff --git a/chrome/browser/sync/sync_service_factory_unittest.cc b/chrome/browser/sync/sync_service_factory_unittest.cc index 7e22ca7b..33714acf 100644 --- a/chrome/browser/sync/sync_service_factory_unittest.cc +++ b/chrome/browser/sync/sync_service_factory_unittest.cc
@@ -84,7 +84,7 @@ // Returns the collection of default datatypes. syncer::ModelTypeSet DefaultDatatypes() { - static_assert(42 == syncer::GetNumModelTypes(), + static_assert(43 == syncer::GetNumModelTypes(), "When adding a new type, you probably want to add it here as " "well (assuming it is already enabled)."); @@ -96,6 +96,9 @@ // ChromeSyncClient types. datatypes.Put(syncer::READING_LIST); datatypes.Put(syncer::SECURITY_EVENTS); + if (base::FeatureList::IsEnabled(syncer::kSyncSegmentationDataType)) { + datatypes.Put(syncer::SEGMENTATION); + } #if BUILDFLAG(ENABLE_SUPERVISED_USERS) datatypes.Put(syncer::SUPERVISED_USER_SETTINGS);
diff --git a/chrome/browser/sync/test/integration/single_client_standalone_transport_sync_test.cc b/chrome/browser/sync/test/integration/single_client_standalone_transport_sync_test.cc index b943f862..01bc140d 100644 --- a/chrome/browser/sync/test/integration/single_client_standalone_transport_sync_test.cc +++ b/chrome/browser/sync/test/integration/single_client_standalone_transport_sync_test.cc
@@ -28,7 +28,7 @@ namespace { syncer::ModelTypeSet AllowedTypesInStandaloneTransportMode() { - static_assert(42 == syncer::GetNumModelTypes(), + static_assert(43 == syncer::GetNumModelTypes(), "Add new types below if they run in transport mode"); // Only some special allowlisted types (and control types) are allowed in // standalone transport mode.
diff --git a/chrome/browser/touch_to_fill/android/internal/java/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewBinder.java b/chrome/browser/touch_to_fill/android/internal/java/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewBinder.java index 5770f64..be61137 100644 --- a/chrome/browser/touch_to_fill/android/internal/java/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewBinder.java +++ b/chrome/browser/touch_to_fill/android/internal/java/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewBinder.java
@@ -171,6 +171,11 @@ TextView passwordText = view.findViewById(R.id.password); passwordText.setText(credential.getPassword()); passwordText.setTransformationMethod(new PasswordTransformationMethod()); + + String contentDescription = view.getContext().getString( + R.string.touch_to_fill_password_credential_accessibility_description, + credential.getFormattedUsername()); + view.setContentDescription(contentDescription); } else if (propertyKey == SHOW_SUBMIT_BUTTON) { // Whether Touch To Fill should auto-submit a form doesn't affect the credentials list. } else { @@ -202,6 +207,10 @@ usernameText.setText(credential.getUsername()); TextView descriptionText = view.findViewById(R.id.webauthn_credential_context); descriptionText.setText(R.string.touch_to_fill_sheet_webauthn_credential_context); + String contentDescription = view.getContext().getString( + R.string.touch_to_fill_passkey_credential_accessibility_description, + credential.getUsername()); + view.setContentDescription(contentDescription); } else if (propertyKey == SHOW_WEBAUTHN_SUBMIT_BUTTON) { // Ignore. } else {
diff --git a/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings.grd b/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings.grd index 7de9939f..654e3a8 100644 --- a/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings.grd +++ b/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings.grd
@@ -232,6 +232,12 @@ <message name="IDS_TOUCH_TO_FILL_SHEET_WEBAUTHN_CREDENTIAL_CONTEXT" desc="Context string for second line of a Passkey on the Touch To Fill sheet, beneath the username."> Use your screen lock </message> + <message name="IDS_TOUCH_TO_FILL_PASSWORD_CREDENTIAL_ACCESSIBILITY_DESCRIPTION" desc="Content description for a password credential item on the sheet list. This is not visually displayed, but is audibly read by screen readers for accessibility purposes."> + Password for <ph name="USERNAME">%1$s<ex>John.Doe@example.com</ex></ph>. + </message> + <message name="IDS_TOUCH_TO_FILL_PASSKEY_CREDENTIAL_ACCESSIBILITY_DESCRIPTION" desc="Content description for a passkey credential item on the sheet list. This is not visually displayed, but is audibly read by screen readers for accessibility purposes."> + Passkey for <ph name="USERNAME">%1$s<ex>John.Doe@example.com</ex></ph>, use your screen lock. + </message> </messages> </release> </grit>
diff --git a/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings_grd/IDS_TOUCH_TO_FILL_PASSKEY_CREDENTIAL_ACCESSIBILITY_DESCRIPTION.png.sha1 b/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings_grd/IDS_TOUCH_TO_FILL_PASSKEY_CREDENTIAL_ACCESSIBILITY_DESCRIPTION.png.sha1 new file mode 100644 index 0000000..4cac228 --- /dev/null +++ b/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings_grd/IDS_TOUCH_TO_FILL_PASSKEY_CREDENTIAL_ACCESSIBILITY_DESCRIPTION.png.sha1
@@ -0,0 +1 @@ +dfa6a8d96285248a2f2ffd64dc81dfd446a411d3 \ No newline at end of file
diff --git a/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings_grd/IDS_TOUCH_TO_FILL_PASSWORD_CREDENTIAL_ACCESSIBILITY_DESCRIPTION.png.sha1 b/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings_grd/IDS_TOUCH_TO_FILL_PASSWORD_CREDENTIAL_ACCESSIBILITY_DESCRIPTION.png.sha1 new file mode 100644 index 0000000..05e94f0 --- /dev/null +++ b/chrome/browser/touch_to_fill/android/internal/java/strings/android_touch_to_fill_strings_grd/IDS_TOUCH_TO_FILL_PASSWORD_CREDENTIAL_ACCESSIBILITY_DESCRIPTION.png.sha1
@@ -0,0 +1 @@ +ebf05b0512c6da1827e834248575e35ef980c870 \ No newline at end of file
diff --git a/chrome/browser/touch_to_fill/android/javatests/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewTest.java b/chrome/browser/touch_to_fill/android/javatests/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewTest.java index c8c9dce..b4fb53259 100644 --- a/chrome/browser/touch_to_fill/android/javatests/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewTest.java +++ b/chrome/browser/touch_to_fill/android/javatests/src/org/chromium/chrome/browser/touch_to_fill/TouchToFillViewTest.java
@@ -6,6 +6,7 @@ import static org.hamcrest.Matchers.instanceOf; import static org.hamcrest.Matchers.is; +import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertNotNull; import static org.junit.Assert.assertThat; import static org.mockito.ArgumentMatchers.eq; @@ -26,6 +27,7 @@ import static org.chromium.chrome.browser.touch_to_fill.TouchToFillProperties.ON_CLICK_MANAGE; import static org.chromium.chrome.browser.touch_to_fill.TouchToFillProperties.SHEET_ITEMS; import static org.chromium.chrome.browser.touch_to_fill.TouchToFillProperties.VISIBLE; +import static org.chromium.chrome.browser.touch_to_fill.TouchToFillProperties.WebAuthnCredentialProperties.WEBAUTHN_CREDENTIAL; import static java.util.Arrays.asList; @@ -53,6 +55,7 @@ import org.chromium.chrome.browser.flags.ChromeSwitches; import org.chromium.chrome.browser.touch_to_fill.TouchToFillProperties.HeaderProperties; import org.chromium.chrome.browser.touch_to_fill.data.Credential; +import org.chromium.chrome.browser.touch_to_fill.data.WebAuthnCredential; import org.chromium.chrome.test.ChromeJUnit4ClassRunner; import org.chromium.chrome.test.ChromeTabbedActivityTestRule; import org.chromium.chrome.test.util.browser.Features.DisableFeatures; @@ -81,6 +84,7 @@ new Credential("", "***", "No Username", "m.example.xyz", true, false, 0); private static final Credential BOB = new Credential("Bob", "***", "Bob", "mobile.example.xyz", true, false, 0); + private static final WebAuthnCredential CAM = new WebAuthnCredential("Cam", "12345"); private static final Credential NIK = new Credential("Nik", "***", "Nik", "group.xyz", false, true, 0); @@ -435,6 +439,41 @@ verify(mDismissHandler).onResult(BottomSheetController.StateChangeReason.NONE); } + @Test + @MediumTest + public void testPasswordCredentialAccessibilityDescription() { + TestThreadUtils.runOnUiThreadBlocking(() -> { + mModel.get(SHEET_ITEMS).addAll(Collections.singletonList(buildCredentialItem(ANA))); + mModel.set(VISIBLE, true); + }); + + BottomSheetTestSupport.waitForOpen(mBottomSheetController); + + assertNotNull(getCredentials().getChildAt(0)); + assertEquals(getCredentials().getChildAt(0).getContentDescription(), + getActivity().getString( + R.string.touch_to_fill_password_credential_accessibility_description, + ANA.getFormattedUsername())); + } + + @Test + @MediumTest + public void testPasskeyCredentialAccessibilityDescription() { + TestThreadUtils.runOnUiThreadBlocking(() -> { + mModel.get(SHEET_ITEMS) + .addAll(Collections.singletonList(buildWebAuthnCredentialItem(CAM))); + mModel.set(VISIBLE, true); + }); + + BottomSheetTestSupport.waitForOpen(mBottomSheetController); + + assertNotNull(getCredentials().getChildAt(0)); + assertEquals(getCredentials().getChildAt(0).getContentDescription(), + getActivity().getString( + R.string.touch_to_fill_passkey_credential_accessibility_description, + CAM.getUsername())); + } + private ChromeActivity getActivity() { return mActivityTestRule.getActivity(); } @@ -469,6 +508,14 @@ TouchToFillProperties.ItemType.CREDENTIAL, credential, mCredentialCallback, false); } + private MVCListAdapter.ListItem buildWebAuthnCredentialItem(WebAuthnCredential credential) { + return new MVCListAdapter.ListItem(TouchToFillProperties.ItemType.WEBAUTHN_CREDENTIAL, + new PropertyModel + .Builder(TouchToFillProperties.WebAuthnCredentialProperties.ALL_KEYS) + .with(WEBAUTHN_CREDENTIAL, credential) + .build()); + } + private MVCListAdapter.ListItem buildConfirmationButton( Credential credential, boolean showSubmitButton) { return buildSheetItem(TouchToFillProperties.ItemType.FILL_BUTTON, credential,
diff --git a/chrome/browser/ui/BUILD.gn b/chrome/browser/ui/BUILD.gn index 4a3990d..9ed442c5 100644 --- a/chrome/browser/ui/BUILD.gn +++ b/chrome/browser/ui/BUILD.gn
@@ -1762,6 +1762,13 @@ ] } + if (toolkit_views && is_chrome_branded) { + sources += [ + "webui/lens/lens_ui.cc", + "webui/lens/lens_ui.h", + ] + } + deps += [ "//base", "//build:chromeos_buildflags",
diff --git a/chrome/browser/ui/android/toolbar/java/res/layout/optional_button_layout.xml b/chrome/browser/ui/android/toolbar/java/res/layout/optional_button_layout.xml index 8337c5de..14606bf 100644 --- a/chrome/browser/ui/android/toolbar/java/res/layout/optional_button_layout.xml +++ b/chrome/browser/ui/android/toolbar/java/res/layout/optional_button_layout.xml
@@ -35,6 +35,8 @@ android:layout_marginStart="40dp" android:gravity="center" android:visibility="gone" + android:maxLines="1" + android:ellipsize="end" android:textAppearance="@style/TextAppearance.TextLarge.Primary"/> <org.chromium.ui.widget.ChromeImageView android:layout_width="40dp"
diff --git a/chrome/browser/ui/android/toolbar/java/res/values/dimens.xml b/chrome/browser/ui/android/toolbar/java/res/values/dimens.xml index 4471383..1a3aa3c 100644 --- a/chrome/browser/ui/android/toolbar/java/res/values/dimens.xml +++ b/chrome/browser/ui/android/toolbar/java/res/values/dimens.xml
@@ -19,6 +19,7 @@ <!-- Toolbar Phone - padding of the optional button when menu button is present --> <dimen name="toolbar_phone_optional_button_padding">12dp</dimen> + <dimen name="toolbar_phone_optional_button_action_chip_max_width">175dp</dimen> <!-- Home Button Menu dimensions--> <dimen name="home_button_list_menu_width">222dp</dimen>
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/BaseButtonDataProvider.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/BaseButtonDataProvider.java index 5877f91e..c97ed1f 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/BaseButtonDataProvider.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/BaseButtonDataProvider.java
@@ -36,6 +36,7 @@ private ModalDialogManagerObserver mModalDialogObserver; private boolean mShouldShowOnIncognitoTabs; + private @StringRes int mActionChipResourceId; /** * Creates a new instance of {@code BaseButtonDataProvider}. @@ -122,6 +123,16 @@ mButtonData.updateIPHCommandBuilder(getIphCommandBuilder(tab)); } + private void maybeSetActionChipResourceId() { + if (!mButtonData.getButtonSpec().isDynamicAction() || !FeatureList.isInitialized() + || !AdaptiveToolbarFeatures.shouldShowActionChip() + || mButtonData.getButtonSpec().getActionChipLabelResId() != Resources.ID_NULL) { + return; + } + + mButtonData.updateActionChipResourceId(mActionChipResourceId); + } + /** * Sets whether the button should be shown on incognito tabs, default is false. */ @@ -130,6 +141,15 @@ } /** + * Sets a string resource ID to be used when the action chip variant is enabled, only used on + * dynamic actions. + * @param actionChipResourceId A string resource to use as the action chip label. + */ + protected void setActionChipResourceId(@StringRes int actionChipResourceId) { + mActionChipResourceId = actionChipResourceId; + } + + /** * Gets an {@link IPHCommandBuilder} builder instance to use on this button. Only called when * native is initialized and when there's no IPHCommandBuilder set. * @param tab Current tab. @@ -155,6 +175,7 @@ public ButtonData get(Tab tab) { mButtonData.setCanShow(shouldShowButton(tab)); maybeSetIphCommandBuilder(tab); + maybeSetActionChipResourceId(); return mButtonData; }
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsChecker.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsChecker.java index e1e8544d..11382b94 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsChecker.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsChecker.java
@@ -4,6 +4,8 @@ package org.chromium.chrome.browser.toolbar; +import android.os.Handler; + import androidx.annotation.NonNull; import org.chromium.base.Callback; @@ -19,15 +21,19 @@ private final ViewResourceAdapter mViewResourceAdapter; @NonNull private final ObservableSupplier<Integer> mConstraintsSupplier; + @NonNull + private final Handler mHandler; /** * @param viewResourceAdapter The target to notify when a capture is needed. * @param constraintsSupplier The underlying supplier for the state of constraints. + * @param handler Handler to post deferred tasks to. */ public ConstraintsChecker(@NonNull ViewResourceAdapter viewResourceAdapter, - @NonNull ObservableSupplier<Integer> constraintsSupplier) { + @NonNull ObservableSupplier<Integer> constraintsSupplier, @NonNull Handler handler) { mViewResourceAdapter = viewResourceAdapter; mConstraintsSupplier = constraintsSupplier; + mHandler = handler; } /** @@ -53,7 +59,7 @@ public void onResult(Integer result) { if (!areControlsLocked()) { mConstraintsSupplier.removeObserver(this); - mViewResourceAdapter.onResourceRequested(); + mHandler.post(() -> mViewResourceAdapter.onResourceRequested()); } } } \ No newline at end of file
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsCheckerTest.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsCheckerTest.java index 673ecf8e..ebaa559 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsCheckerTest.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/ConstraintsCheckerTest.java
@@ -5,9 +5,13 @@ package org.chromium.chrome.browser.toolbar; import static org.junit.Assert.assertEquals; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.doAnswer; import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; +import android.os.Handler; + import org.junit.Rule; import org.junit.Test; import org.junit.runner.RunWith; @@ -30,13 +34,22 @@ @Mock private ViewResourceAdapter mViewResourceAdapter; + @Mock + private Handler mHandler; private ObservableSupplierImpl mConstraintsSupplier = new ObservableSupplierImpl(); @Test public void testScheduleRequestResourceOnUnlock() { + doAnswer((invocation) -> { + Runnable runnable = (Runnable) invocation.getArguments()[0]; + runnable.run(); + return null; + }) + .when(mHandler) + .post(any(Runnable.class)); ConstraintsChecker constraintsChecker = - new ConstraintsChecker(mViewResourceAdapter, mConstraintsSupplier); + new ConstraintsChecker(mViewResourceAdapter, mConstraintsSupplier, mHandler); constraintsChecker.scheduleRequestResourceOnUnlock(); mConstraintsSupplier.set(BrowserControlsState.SHOWN); verify(mViewResourceAdapter, times(0)).onResourceRequested(); @@ -51,7 +64,7 @@ @Test public void testAreControlsLocked() { ConstraintsChecker constraintsChecker = - new ConstraintsChecker(mViewResourceAdapter, mConstraintsSupplier); + new ConstraintsChecker(mViewResourceAdapter, mConstraintsSupplier, mHandler); assertEquals(true, constraintsChecker.areControlsLocked()); mConstraintsSupplier.set(BrowserControlsState.SHOWN);
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/bottom/ScrollingBottomViewResourceFrameLayout.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/bottom/ScrollingBottomViewResourceFrameLayout.java index 9651936..32f1b93 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/bottom/ScrollingBottomViewResourceFrameLayout.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/bottom/ScrollingBottomViewResourceFrameLayout.java
@@ -8,6 +8,8 @@ import android.graphics.Canvas; import android.graphics.PorterDuff; import android.graphics.Rect; +import android.os.Handler; +import android.os.Looper; import android.util.AttributeSet; import androidx.annotation.NonNull; @@ -120,6 +122,7 @@ */ public void setConstraintsSupplier(ObservableSupplier<Integer> constraintsSupplier) { assert mConstraintsChecker == null; - mConstraintsChecker = new ConstraintsChecker(getResourceAdapter(), constraintsSupplier); + mConstraintsChecker = new ConstraintsChecker( + getResourceAdapter(), constraintsSupplier, new Handler(Looper.getMainLooper())); } }
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonView.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonView.java index 3aefd75..3deb1485 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonView.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonView.java
@@ -584,12 +584,16 @@ mActionChipLabel.setVisibility(VISIBLE); mBackground.setVisibility(VISIBLE); - // TODO(salg): Add a check to avoid expanding too much. float actionChipLabelTextWidth = mActionChipLabel.getPaint().measureText(mActionChipLabelString); - int expandedStateWidthPx = - (int) (mCollapsedStateWidthPx + actionChipLabelTextWidth + mExpandedStatePaddingPx); + int maxExpandedStateWidthPx = getResources().getDimensionPixelSize( + R.dimen.toolbar_phone_optional_button_action_chip_max_width); + + int expandedStateWidthPx = Math.min( + (int) (mCollapsedStateWidthPx + actionChipLabelTextWidth + mExpandedStatePaddingPx), + maxExpandedStateWidthPx); + setWidth(expandedStateWidthPx); mState = State.RUNNING_ACTION_CHIP_EXPANSION_TRANSITION;
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonViewTest.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonViewTest.java index b81bf609..92a7b1f 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonViewTest.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/optional_button/OptionalButtonViewTest.java
@@ -36,6 +36,7 @@ import androidx.appcompat.content.res.AppCompatResources; +import org.hamcrest.Matchers; import org.junit.Assert; import org.junit.Before; import org.junit.Test; @@ -666,4 +667,27 @@ // visibility many times should begin a new animation. verify(mMockBeginDelayedTransition, times(2)).onResult(any()); } + + @Test + public void testUpdateButton_actionChipWidthIsRestricted() { + ButtonDataImpl buttonData = getDataForPriceTrackingActionChip(); + // Set a string that's too long as an action chip label. Real code measures the number of + // pixels this will take on screen, but robolectric just uses the character count, so use + // any string with more than 150 characters. + buttonData.updateActionChipResourceId(R.string.sign_in_managed_account_description); + + int maxActionChipWidth = mOptionalButtonView.getResources().getDimensionPixelSize( + R.dimen.toolbar_phone_optional_button_action_chip_max_width); + + ViewGroup transitionRoot = mock(ViewGroup.class); + mOptionalButtonView.setTransitionRoot(transitionRoot); + + mOptionalButtonView.updateButtonWithAnimation(buttonData); + mOptionalButtonView.onTransitionStart(null); + mOptionalButtonView.onTransitionEnd(null); + + // Button shouldn't be wider than the established maximum. + Assert.assertThat(mOptionalButtonView.getLayoutParams().width, + Matchers.lessThanOrEqualTo(maxActionChipWidth)); + } }
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainer.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainer.java index 876408f..51308db 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainer.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainer.java
@@ -11,6 +11,8 @@ import android.graphics.Region; import android.graphics.drawable.Drawable; import android.os.Build; +import android.os.Handler; +import android.os.Looper; import android.util.AttributeSet; import android.view.MotionEvent; import android.view.View; @@ -237,7 +239,8 @@ assert mConstraintsObserver == null; if (constraintsSupplier != null) { - mConstraintsObserver = new ConstraintsChecker(this, constraintsSupplier); + mConstraintsObserver = new ConstraintsChecker( + this, constraintsSupplier, new Handler(Looper.getMainLooper())); } assert mTabSupplier == null;
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainerTest.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainerTest.java index 955cfc81..c28a563 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainerTest.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarControlContainerTest.java
@@ -18,6 +18,7 @@ import org.mockito.junit.MockitoJUnit; import org.mockito.junit.MockitoRule; import org.robolectric.annotation.Config; +import org.robolectric.shadows.ShadowLooper; import org.chromium.base.metrics.RecordHistogram; import org.chromium.base.metrics.UmaRecorderHolder; @@ -218,6 +219,7 @@ // BOTH should cause a new onResourceRequested call. mConstraintsSupplier.set(BrowserControlsState.BOTH); + ShadowLooper.idleMainLooper(); Assert.assertEquals(1, onResourceRequestedCount.get()); // The constraints should no longer block isDirty/captures.
diff --git a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarPhone.java b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarPhone.java index 5c92c4f..d9b5e3ec 100644 --- a/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarPhone.java +++ b/chrome/browser/ui/android/toolbar/java/src/org/chromium/chrome/browser/toolbar/top/ToolbarPhone.java
@@ -1522,6 +1522,7 @@ @Override public void draw(Canvas canvas) { + if (mDestroyChecker.isDestroyed()) return; // If capturing a texture of the toolbar, ensure the alpha is set prior to draw(...) being // called. The alpha is being used prior to getting to draw(...), so updating the value // after this point was having no affect.
diff --git a/chrome/browser/ui/ash/shelf/chrome_shelf_controller.cc b/chrome/browser/ui/ash/shelf/chrome_shelf_controller.cc index 568ca547..589f64e8 100644 --- a/chrome/browser/ui/ash/shelf/chrome_shelf_controller.cc +++ b/chrome/browser/ui/ash/shelf/chrome_shelf_controller.cc
@@ -12,7 +12,6 @@ #include "ash/components/arc/arc_util.h" #include "ash/constants/app_types.h" #include "ash/constants/ash_pref_names.h" -#include "ash/metrics/login_unlock_throughput_recorder.h" #include "ash/public/cpp/multi_user_window_manager.h" #include "ash/public/cpp/shelf_item.h" #include "ash/public/cpp/shelf_model.h" @@ -130,26 +129,6 @@ return standard_image; } -// Report shelf buttons initialized to LoginUnlockThroughputRecorder. -void ReportInitShelfIconList(const ash::ShelfModel* model) { - // Shell is not always initializaed in tests. - if (!ash::Shell::HasInstance()) - return; - - ash::Shell::Get()->login_unlock_throughput_recorder()->InitShelfIconList( - model); -} - -// Report shelf buttons updated to LoginUnlockThroughputRecorder. -void ReportUpdateShelfIconList(const ash::ShelfModel* model) { - // Shell is not always initializaed in tests. - if (!ash::Shell::HasInstance()) - return; - - ash::Shell::Get()->login_unlock_throughput_recorder()->UpdateShelfIconList( - model); -} - } // namespace // A class to get events from ChromeOS when a user gets changed or added. @@ -337,7 +316,6 @@ if (browser_status_monitor_) { browser_status_monitor_->Initialize(); } - ReportInitShelfIconList(model_); } ash::ShelfID ChromeShelfController::CreateAppItem( @@ -487,8 +465,6 @@ new_item.id.app_id, image); model_->Set(model_->ItemIndexByID(shelf_id), new_item); } - - ReportUpdateShelfIconList(model_); } void ChromeShelfController::UpdateItemImage(const std::string& app_id) { @@ -711,7 +687,6 @@ model_->GetItemIndexForType(ash::TYPE_UNPINNED_BROWSER_SHORTCUT); if (item_index >= 0) { model_->RemoveItemAt(item_index); - ReportUpdateShelfIconList(model_); return; } } @@ -737,8 +712,6 @@ browser_item.status = browser_status; model_->Set(browser_index, browser_item); - - ReportUpdateShelfIconList(model_); } void ChromeShelfController::SetShelfIDForBrowserWindowContents( @@ -823,8 +796,6 @@ model_->RemoveItemAt(index); model_->AddAt(index, item, std::make_unique<AppShortcutShelfItemController>(item.id)); - - ReportUpdateShelfIconList(model_); } void ChromeShelfController::PinAppAtIndex(const std::string& app_id, @@ -839,8 +810,6 @@ model_->AddAt(target_index, item, std::make_unique<AppShortcutShelfItemController>(item.id)); - - ReportUpdateShelfIconList(model_); } int ChromeShelfController::PinnedItemIndexByAppID(const std::string& app_id) { @@ -1131,8 +1100,6 @@ model_->Set(index, item); // It's possible we're waiting on more than one item, so don't break. } - - ReportUpdateShelfIconList(model_); } /////////////////////////////////////////////////////////////////////////////// @@ -1174,8 +1141,6 @@ const int index = model_->ItemIndexByID(id); if (index >= 0 && index < model_->item_count()) model_->RemoveItemAt(index); - - ReportUpdateShelfIconList(model_); } void ChromeShelfController::PinRunningAppInternal( @@ -1308,8 +1273,6 @@ } UpdatePolicyPinnedAppsFromPrefs(); - - ReportUpdateShelfIconList(model_); } bool ChromeShelfController::EnsureAppPinnedInModelAtIndex( @@ -1367,8 +1330,6 @@ item.pinned_by_policy = pinned_by_policy; model_->Set(model_index, item); } - - ReportUpdateShelfIconList(model_); } ash::ShelfItemStatus ChromeShelfController::GetAppState( @@ -1412,8 +1373,6 @@ item.app_status = ShelfControllerHelper::GetAppStatus( latest_active_profile_, item_delegate->shelf_id().app_id); model_->AddAt(index, item, std::move(item_delegate)); - - ReportUpdateShelfIconList(model_); return item.id; } @@ -1443,8 +1402,6 @@ model_->Add(browser_shortcut, std::make_unique<BrowserShortcutShelfItemController>(model_)); } - - ReportUpdateShelfIconList(model_); } int ChromeShelfController::FindInsertionPoint() { @@ -1611,8 +1568,6 @@ // Update the pin position preference as needed. if (ShouldSyncItemWithReentrancy(item)) SyncPinPosition(item.id); - - ReportUpdateShelfIconList(model_); } void ChromeShelfController::ShelfItemRemoved(int index, @@ -1639,6 +1594,4 @@ SyncPinPosition(item.id); else if (ShouldSyncItemWithReentrancy(old_item) && !ItemTypeIsPinned(item)) shelf_prefs_->RemovePinPosition(profile(), old_item.id); - - ReportUpdateShelfIconList(model_); }
diff --git a/chrome/browser/ui/lens/lens_side_panel_helper.h b/chrome/browser/ui/lens/lens_side_panel_helper.h index 9637f6a..d87a812 100644 --- a/chrome/browser/ui/lens/lens_side_panel_helper.h +++ b/chrome/browser/ui/lens/lens_side_panel_helper.h
@@ -25,6 +25,9 @@ void OpenLensSidePanel(Browser* browser, const content::OpenURLParams& url_params); +// Opens the Lens region search feature in a new tab with a WebUI page. +void OpenLensStaticPage(Browser* browser); + // Check if the lens URL is a valid results page. This is done by checking if // the URL has a payload parameter. bool IsValidLensResultUrl(const GURL& url);
diff --git a/chrome/browser/ui/tabs/tab_strip_model.cc b/chrome/browser/ui/tabs/tab_strip_model.cc index 1a3e5076..09144bcb 100644 --- a/chrome/browser/ui/tabs/tab_strip_model.cc +++ b/chrome/browser/ui/tabs/tab_strip_model.cc
@@ -1038,7 +1038,7 @@ // Ensure that the indices are nonempty, sorted, and unique. DCHECK_GT(indices.size(), 0u); DCHECK(base::ranges::is_sorted(indices)); - DCHECK(std::adjacent_find(indices.begin(), indices.end()) == indices.end()); + DCHECK(base::ranges::adjacent_find(indices) == indices.end()); // The odds of |new_group| colliding with an existing group are astronomically // low. If there is a collision, a DCHECK will fail in |AddToNewGroupImpl()|, @@ -1058,7 +1058,7 @@ // Ensure that the indices are sorted and unique. DCHECK(base::ranges::is_sorted(indices)); - DCHECK(std::adjacent_find(indices.begin(), indices.end()) == indices.end()); + DCHECK(base::ranges::adjacent_find(indices) == indices.end()); CHECK(ContainsIndex(*(indices.begin()))); CHECK(ContainsIndex(*(indices.rbegin())));
diff --git a/chrome/browser/ui/views/lens/lens_side_panel_helper.cc b/chrome/browser/ui/views/lens/lens_side_panel_helper.cc index 7c142ab..9746fddc 100644 --- a/chrome/browser/ui/views/lens/lens_side_panel_helper.cc +++ b/chrome/browser/ui/views/lens/lens_side_panel_helper.cc
@@ -11,6 +11,7 @@ #include "chrome/browser/ui/views/lens/lens_side_panel_controller.h" #include "chrome/browser/ui/views/side_panel/lens/lens_side_panel_coordinator.h" #include "chrome/browser/ui/views/side_panel/side_panel_coordinator.h" +#include "chrome/common/webui_url_constants.h" #include "components/lens/lens_entrypoints.h" #include "components/lens/lens_features.h" #include "components/lens/lens_rendering_environment.h" @@ -119,4 +120,16 @@ return web_contents; } +void OpenLensStaticPage(Browser* browser) { + // TODO(juanmojica): Expand this function to simulate the current region + // search experience in the new tab. + DCHECK(browser); + GURL url(chrome::kChromeUILensURL); + content::OpenURLParams params( + url, content::Referrer(), WindowOpenDisposition::NEW_FOREGROUND_TAB, + ui::PAGE_TRANSITION_LINK, /*is_renderer_initiated=*/false); + params.initiator_origin = url::Origin::Create(url); + browser->OpenURL(params); +} + } // namespace lens
diff --git a/chrome/browser/ui/views/performance_controls/high_efficiency_chip_view_browsertest.cc b/chrome/browser/ui/views/performance_controls/high_efficiency_chip_view_browsertest.cc index 31e2d57e..621c7a1 100644 --- a/chrome/browser/ui/views/performance_controls/high_efficiency_chip_view_browsertest.cc +++ b/chrome/browser/ui/views/performance_controls/high_efficiency_chip_view_browsertest.cc
@@ -65,7 +65,7 @@ return promo_controller; } - PageActionIconView* GetPageActionIconView() { + PageActionIconView* GetHighEfficiencyChipView() { BrowserView* browser_view = BrowserView::GetBrowserViewForBrowser(browser()); return browser_view->GetLocationBarView() @@ -73,9 +73,10 @@ ->GetIconView(PageActionIconType::kHighEfficiency); } - void PressButton(views::Button* button) { + void ClickHighEfficiencyChip() { views::test::InteractionTestUtilSimulatorViews::PressButton( - button, ui::test::InteractionTestUtil::InputType::kMouse); + GetHighEfficiencyChipView(), + ui::test::InteractionTestUtil::InputType::kMouse); } void SetTabDiscardState(bool is_discarded) { @@ -100,8 +101,31 @@ waiter.WaitIfNeededAndGet(); } + user_education::HelpBubbleView* GetHelpBubbleView() { + return GetFeaturePromoController() + ->promo_bubble_for_testing() + ->AsA<user_education::HelpBubbleViews>() + ->bubble_view(); + } + + void ClickIPHCancelButton() { + views::test::WidgetDestroyedWaiter waiter(GetHelpBubbleView()->GetWidget()); + views::test::InteractionTestUtilSimulatorViews::PressButton( + GetHelpBubbleView()->GetDefaultButtonForTesting(), + ui::test::InteractionTestUtil::InputType::kMouse); + waiter.Wait(); + } + + void ClickIPHSettingsButton() { + views::test::WidgetDestroyedWaiter waiter(GetHelpBubbleView()->GetWidget()); + views::test::InteractionTestUtilSimulatorViews::PressButton( + GetHelpBubbleView()->GetNonDefaultButtonForTesting(0), + ui::test::InteractionTestUtil::InputType::kMouse); + waiter.Wait(); + } + views::InkDropState GetInkDropState() { - return views::InkDrop::Get(GetPageActionIconView()) + return views::InkDrop::Get(GetHighEfficiencyChipView()) ->GetInkDrop() ->GetTargetInkDropState(); } @@ -111,17 +135,13 @@ }; IN_PROC_BROWSER_TEST_F(HighEfficiencyChipViewBrowserTest, - PromoCustomActionClicked) { + NavigatesOnIPHSettingsLinkClicked) { auto lock = BrowserFeaturePromoController::BlockActiveWindowCheckForTesting(); - auto* const promo_controller = GetFeaturePromoController(); EXPECT_FALSE(GetFeaturePromoController()->IsPromoActive( feature_engagement::kIPHHighEfficiencyInfoModeFeature)); SetTabDiscardState(true); - PageActionIconView* icon = GetPageActionIconView(); - EXPECT_TRUE(icon->GetVisible()); - WaitForIPHToShow(); EXPECT_TRUE(GetFeaturePromoController()->IsPromoActive( @@ -129,11 +149,7 @@ content::TestNavigationObserver navigation_observer( browser()->tab_strip_model()->GetWebContentsAt(0)); - auto* promo_bubble = promo_controller->promo_bubble_for_testing() - ->AsA<user_education::HelpBubbleViews>() - ->bubble_view(); - auto* custom_action_button = promo_bubble->GetNonDefaultButtonForTesting(0); - PressButton(custom_action_button); + ClickIPHSettingsButton(); navigation_observer.Wait(); GURL expected(chrome::kChromeUIPerformanceSettingsURL); @@ -141,56 +157,40 @@ } IN_PROC_BROWSER_TEST_F(HighEfficiencyChipViewBrowserTest, - PromoDismissesOnChipClick) { + PromoDismissesOnCancelClick) { auto lock = BrowserFeaturePromoController::BlockActiveWindowCheckForTesting(); SetTabDiscardState(true); - PageActionIconView* icon = GetPageActionIconView(); WaitForIPHToShow(); EXPECT_TRUE(GetFeaturePromoController()->IsPromoActive( feature_engagement::kIPHHighEfficiencyInfoModeFeature)); - PressButton(icon); + ClickHighEfficiencyChip(); // Expect the bubble to be open and the promo to be closed. EXPECT_FALSE(GetFeaturePromoController()->IsPromoActive( feature_engagement::kIPHHighEfficiencyInfoModeFeature)); - EXPECT_NE(icon->GetBubble(), nullptr); + EXPECT_NE(GetHighEfficiencyChipView()->GetBubble(), nullptr); } IN_PROC_BROWSER_TEST_F(HighEfficiencyChipViewBrowserTest, ShowAndHideInkDropWithPromo) { auto lock = BrowserFeaturePromoController::BlockActiveWindowCheckForTesting(); - auto* const promo_controller = GetFeaturePromoController(); EXPECT_FALSE(GetFeaturePromoController()->IsPromoActive( feature_engagement::kIPHHighEfficiencyInfoModeFeature)); SetTabDiscardState(true); - PageActionIconView* icon = GetPageActionIconView(); - EXPECT_TRUE(icon->GetVisible()); - WaitForIPHToShow(); EXPECT_TRUE(GetFeaturePromoController()->IsPromoActive( feature_engagement::kIPHHighEfficiencyInfoModeFeature)); - EXPECT_EQ(GetInkDropState(), views::InkDropState::ACTIVATED); - auto* promo_bubble = promo_controller->promo_bubble_for_testing() - ->AsA<user_education::HelpBubbleViews>() - ->bubble_view(); + ClickIPHCancelButton(); - views::test::WidgetDestroyedWaiter waiter(promo_bubble->GetWidget()); - auto* default_action_button = promo_bubble->GetDefaultButtonForTesting(); - PressButton(default_action_button); - waiter.Wait(); - - EXPECT_FALSE(browser()->window()->IsFeaturePromoActive( + EXPECT_FALSE(GetFeaturePromoController()->IsPromoActive( feature_engagement::kIPHHighEfficiencyInfoModeFeature)); - - views::InkDropState current_state = GetInkDropState(); - EXPECT_TRUE(current_state == views::InkDropState::HIDDEN || - current_state == views::InkDropState::DEACTIVATED); + EXPECT_TRUE(GetInkDropState() == views::InkDropState::DEACTIVATED); }
diff --git a/chrome/browser/ui/views/side_panel/lens/lens_side_panel_coordinator_browsertest.cc b/chrome/browser/ui/views/side_panel/lens/lens_side_panel_coordinator_browsertest.cc index 455907a..646a985 100644 --- a/chrome/browser/ui/views/side_panel/lens/lens_side_panel_coordinator_browsertest.cc +++ b/chrome/browser/ui/views/side_panel/lens/lens_side_panel_coordinator_browsertest.cc
@@ -432,6 +432,7 @@ IN_PROC_BROWSER_TEST_F(SearchImageWithSidePanel3PDseDisabled, ImageSearchFor3PDSEWithValidImageOpensInNewTab) { + SetupImageSearchEngine(); SetupAndLoadValidImagePage(); // Ensures that the lens side panel coordinator is open and is valid when @@ -458,6 +459,7 @@ IN_PROC_BROWSER_TEST_F(SearchImageWithSidePanel3PDseDisabled, ImageSearchForLensWithValidImageOpensInSidePanel) { + SetupImageSearchEngine(); SetupUnifiedSidePanel(); EXPECT_TRUE(GetUnifiedSidePanel()->GetVisible());
diff --git a/chrome/browser/ui/webui/chrome_web_ui_controller_factory.cc b/chrome/browser/ui/webui/chrome_web_ui_controller_factory.cc index ede9f17..1f0c418 100644 --- a/chrome/browser/ui/webui/chrome_web_ui_controller_factory.cc +++ b/chrome/browser/ui/webui/chrome_web_ui_controller_factory.cc
@@ -396,6 +396,10 @@ #include "chrome/browser/ui/webui/media_router/cast_feedback_ui.h" #endif +#if !BUILDFLAG(IS_ANDROID) && BUILDFLAG(GOOGLE_CHROME_BRANDING) +#include "chrome/browser/ui/webui/lens/lens_ui.h" +#endif + #if BUILDFLAG(PLATFORM_CFM) #include "chrome/browser/ui/webui/ash/chromebox_for_meetings/network_settings_dialog.h" #endif // BUILDFLAG(PLATFORM_CFM) @@ -1354,6 +1358,12 @@ return &NewWebUI<PrivacySandboxDialogUI>; #endif // !BUILDFLAG(IS_ANDROID) +#if !BUILDFLAG(IS_ANDROID) && BUILDFLAG(GOOGLE_CHROME_BRANDING) + if (url.host_piece() == chrome::kChromeUILensHost) { + return &NewWebUI<LensUI>; + } +#endif + return nullptr; }
diff --git a/chrome/browser/ui/webui/lens/DIR_METADATA b/chrome/browser/ui/webui/lens/DIR_METADATA new file mode 100644 index 0000000..749bdad --- /dev/null +++ b/chrome/browser/ui/webui/lens/DIR_METADATA
@@ -0,0 +1,4 @@ +monorail { + component: "UI>Browser" +} +team_email: "lens-chrome-eng+bugs@google.com"
diff --git a/chrome/browser/ui/webui/lens/OWNERS b/chrome/browser/ui/webui/lens/OWNERS new file mode 100644 index 0000000..6e913aa --- /dev/null +++ b/chrome/browser/ui/webui/lens/OWNERS
@@ -0,0 +1,2 @@ +file://components/lens/OWNERS +
diff --git a/chrome/browser/ui/webui/lens/lens_ui.cc b/chrome/browser/ui/webui/lens/lens_ui.cc new file mode 100644 index 0000000..fa20353 --- /dev/null +++ b/chrome/browser/ui/webui/lens/lens_ui.cc
@@ -0,0 +1,25 @@ +// Copyright 2022 The Chromium Authors +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "chrome/browser/ui/webui/lens/lens_ui.h" + +#include "chrome/browser/profiles/profile.h" +#include "chrome/browser/ui/webui/webui_util.h" +#include "chrome/common/webui_url_constants.h" +#include "chrome/grit/lens_resources.h" +#include "chrome/grit/lens_resources_map.h" +#include "content/public/browser/web_ui.h" +#include "content/public/browser/web_ui_data_source.h" + +LensUI::LensUI(content::WebUI* web_ui) : WebUIController(web_ui) { + // Set up the chrome://lens source. + content::WebUIDataSource* html_source = + content::WebUIDataSource::CreateAndAdd(Profile::FromWebUI(web_ui), + chrome::kChromeUILensHost); + webui::SetupWebUIDataSource( + html_source, base::make_span(kLensResources, kLensResourcesSize), + IDR_LENS_LENS_HTML); +} + +LensUI::~LensUI() = default;
diff --git a/chrome/browser/ui/webui/lens/lens_ui.h b/chrome/browser/ui/webui/lens/lens_ui.h new file mode 100644 index 0000000..0bc6a6fa --- /dev/null +++ b/chrome/browser/ui/webui/lens/lens_ui.h
@@ -0,0 +1,16 @@ +// Copyright 2022 The Chromium Authors +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#ifndef CHROME_BROWSER_UI_WEBUI_LENS_LENS_UI_H_ +#define CHROME_BROWSER_UI_WEBUI_LENS_LENS_UI_H_ + +#include "content/public/browser/web_ui_controller.h" + +// The WebUI for chrome://lens +class LensUI : public content::WebUIController { + public: + explicit LensUI(content::WebUI* web_ui); + ~LensUI() override; +}; +#endif // CHROME_BROWSER_UI_WEBUI_LENS_LENS_UI_H_
diff --git a/chrome/browser/web_applications/isolated_web_apps/install_isolated_app_from_command_line_browsertest.cc b/chrome/browser/web_applications/isolated_web_apps/install_isolated_app_from_command_line_browsertest.cc index bd70ac5..346f478 100644 --- a/chrome/browser/web_applications/isolated_web_apps/install_isolated_app_from_command_line_browsertest.cc +++ b/chrome/browser/web_applications/isolated_web_apps/install_isolated_app_from_command_line_browsertest.cc
@@ -76,5 +76,6 @@ const WebApp* web_app = GetWebAppRegistrar().GetAppById(app_id); EXPECT_THAT(web_app->isolation_data().has_value(), IsTrue()); } + } // namespace } // namespace web_app
diff --git a/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory.cc b/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory.cc index 45e2a23..cee7da70 100644 --- a/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory.cc +++ b/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory.cc
@@ -19,6 +19,7 @@ #include "chrome/browser/web_applications/isolated_web_apps/isolated_web_app_reader_registry.h" #include "chrome/browser/web_applications/isolated_web_apps/isolated_web_app_reader_registry_factory.h" #include "chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_info.h" +#include "chrome/browser/web_applications/isolated_web_apps/pending_install_info.h" #include "chrome/browser/web_applications/isolation_data.h" #include "chrome/browser/web_applications/web_app_provider.h" #include "chrome/common/url_constants.h" @@ -140,6 +141,7 @@ LOG(ERROR) << error_message; return; } + web_contents->ForEachRenderFrameHostWithAction( [frame_tree_node_id, &error_message](content::RenderFrameHost* render_frame_host) { @@ -194,8 +196,8 @@ IsolatedWebAppURLLoader(const IsolatedWebAppURLLoader&) = delete; IsolatedWebAppURLLoader& operator=(const IsolatedWebAppURLLoader&) = delete; - - ~IsolatedWebAppURLLoader() override = default; + IsolatedWebAppURLLoader(IsolatedWebAppURLLoader&&) = delete; + IsolatedWebAppURLLoader& operator=(IsolatedWebAppURLLoader&&) = delete; private: void OnResponseRead( @@ -348,63 +350,79 @@ return; } + auto forward_request_to_isolation_data_content = + [&](const IsolationData& isolation_data) { + if (!IsSupportedHttpMethod(resource_request.method)) { + CompleteWithGeneratedHtmlResponse( + mojo::Remote<network::mojom::URLLoaderClient>( + std::move(loader_client)), + net::HTTP_METHOD_NOT_ALLOWED, /*body=*/absl::nullopt); + return; + } + + absl::visit( + base::Overloaded{ + [&](const IsolationData::InstalledBundle& content) { + auto* isolated_web_app_reader_registry = + IsolatedWebAppReaderRegistryFactory::GetForProfile( + profile_); + if (!isolated_web_app_reader_registry) { + LogErrorAndFail( + "Support for Isolated Web Apps is not enabled.", + std::move(loader_client)); + return; + } + mojo::MakeSelfOwnedReceiver( + std::make_unique<IsolatedWebAppURLLoader>( + isolated_web_app_reader_registry, content.path, + *web_bundle_id, std::move(loader_client), + resource_request, frame_tree_node_id_), + mojo::PendingReceiver<network::mojom::URLLoader>( + std::move(loader_receiver))); + }, + [&](const IsolationData::DevModeBundle& content) { + // TODO: Implement dev mode bundles. + CompleteWithGeneratedHtmlResponse( + mojo::Remote<network::mojom::URLLoaderClient>( + std::move(loader_client)), + net::HTTP_NOT_FOUND, /*body=*/absl::nullopt); + }, + [&](const IsolationData::DevModeProxy& content) { + HandleDevModeProxy(*url_info, content, + std::move(loader_receiver), + resource_request, std::move(loader_client), + traffic_annotation); + }}, + isolation_data.content); + }; + + absl::optional<IsolationData> pending_install_isolation_data = + IsolatedWebAppPendingInstallInfo::FromWebContents( + *content::WebContents::FromFrameTreeNodeId(frame_tree_node_id_)) + .isolation_data(); + if (pending_install_isolation_data.has_value()) { + if (resource_request.url.path() == kInstallPagePath && + IsSupportedHttpMethod(resource_request.method)) { + CompleteWithGeneratedHtmlResponse( + mojo::Remote<network::mojom::URLLoaderClient>( + std::move(loader_client)), + net::HTTP_OK, kInstallPageContent); + return; + } + + forward_request_to_isolation_data_content(*pending_install_isolation_data); + return; + } + base::expected<std::reference_wrapper<const WebApp>, std::string> iwa = FindIsolatedWebApp(profile_, *url_info); + if (!iwa.has_value()) { LogErrorAndFail(iwa.error(), std::move(loader_client)); return; } - if (!IsSupportedHttpMethod(resource_request.method)) { - CompleteWithGeneratedHtmlResponse( - mojo::Remote<network::mojom::URLLoaderClient>(std::move(loader_client)), - net::HTTP_METHOD_NOT_ALLOWED, /*body=*/absl::nullopt); - return; - } - - // TODO(crbug.com/1333966): Check that the app is being installed before - // returning the auto-generated installation page. - if (resource_request.url.path() == kInstallPagePath) { - CompleteWithGeneratedHtmlResponse( - mojo::Remote<network::mojom::URLLoaderClient>(std::move(loader_client)), - net::HTTP_OK, kInstallPageContent); - return; - } - - IsolationData isolation_data = iwa->get().isolation_data().value(); - absl::visit( - base::Overloaded{ - [&](const IsolationData::InstalledBundle& content) { - auto* isolated_web_app_reader_registry = - IsolatedWebAppReaderRegistryFactory::GetForProfile(profile_); - if (!isolated_web_app_reader_registry) { - LogErrorAndFail("Support for Isolated Web Apps is not enabled.", - std::move(loader_client)); - return; - } - - auto loader = std::make_unique<IsolatedWebAppURLLoader>( - isolated_web_app_reader_registry, content.path, *web_bundle_id, - std::move(loader_client), resource_request, - frame_tree_node_id_); - mojo::MakeSelfOwnedReceiver( - std::move(std::move(loader)), - mojo::PendingReceiver<network::mojom::URLLoader>( - std::move(loader_receiver))); - }, - [&loader_client](const IsolationData::DevModeBundle& content) { - // TODO: Implement dev mode bundles. - CompleteWithGeneratedHtmlResponse( - mojo::Remote<network::mojom::URLLoaderClient>( - std::move(loader_client)), - net::HTTP_NOT_FOUND, /*body=*/absl::nullopt); - }, - [&](const IsolationData::DevModeProxy& content) { - HandleDevModeProxy(*url_info, content, std::move(loader_receiver), - resource_request, std::move(loader_client), - traffic_annotation); - }}, - isolation_data.content); + forward_request_to_isolation_data_content(*iwa->get().isolation_data()); } void IsolatedWebAppURLLoaderFactory::OnProfileWillBeDestroyed( @@ -439,19 +457,20 @@ network::ResourceRequest proxy_request(resource_request); proxy_request.url = proxy_url; - content::StoragePartition* iwa_partition = profile_->GetStoragePartition( + content::StoragePartition* storage_partition = profile_->GetStoragePartition( url_info.storage_partition_config(profile_), /*can_create=*/false); - if (iwa_partition == nullptr) { + if (storage_partition == nullptr) { LogErrorAndFail(base::StrCat({"Storage not found for Isolated Web App: ", resource_request.url.spec()}), std::move(loader_client)); return; } - iwa_partition->GetURLLoaderFactoryForBrowserProcess()->CreateLoaderAndStart( - std::move(loader_receiver), - /*request_id=*/0, network::mojom::kURLLoadOptionNone, proxy_request, - std::move(loader_client), traffic_annotation); + storage_partition->GetURLLoaderFactoryForBrowserProcess() + ->CreateLoaderAndStart(std::move(loader_receiver), + /*request_id=*/0, + network::mojom::kURLLoadOptionNone, proxy_request, + std::move(loader_client), traffic_annotation); } void IsolatedWebAppURLLoaderFactory::LogErrorAndFail(
diff --git a/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory_unittest.cc b/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory_unittest.cc index 4f183eb7..4f818ec 100644 --- a/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory_unittest.cc +++ b/chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_loader_factory_unittest.cc
@@ -12,6 +12,7 @@ #include "base/files/file_util.h" #include "base/files/scoped_temp_dir.h" #include "chrome/browser/web_applications/isolated_web_apps/isolated_web_app_url_info.h" +#include "chrome/browser/web_applications/isolated_web_apps/pending_install_info.h" #include "chrome/browser/web_applications/isolation_data.h" #include "chrome/browser/web_applications/test/fake_web_app_provider.h" #include "chrome/browser/web_applications/test/web_app_test.h" @@ -23,6 +24,7 @@ #include "components/web_package/test_support/signed_web_bundles/web_bundle_signer.h" #include "components/web_package/web_bundle_builder.h" #include "content/public/browser/storage_partition_config.h" +#include "content/public/browser/web_contents.h" #include "content/public/common/content_features.h" #include "content/public/test/simple_url_loader_test_helper.h" #include "content/public/test/url_loader_interceptor.h" @@ -149,10 +151,7 @@ void RegisterWebApp(std::unique_ptr<WebApp> web_app, bool create_storage_partition = true) { if (create_storage_partition) { - auto url_info = IsolatedWebAppUrlInfo::Create(web_app->scope()); - profile()->GetStoragePartition( - url_info->storage_partition_config(profile()), - /*can_create=*/true); + CreateStoragePartitionForUrl(web_app->scope()); } provider()->GetRegistrarMutable().registry().emplace(web_app->app_id(), @@ -160,9 +159,9 @@ } void CreateFactory() { - int dummy_frame_tree_node_id = 42; factory_.Bind(IsolatedWebAppURLLoaderFactory::Create( - dummy_frame_tree_node_id, profile())); + web_contents()->GetPrimaryMainFrame()->GetFrameTreeNodeId(), + profile())); } int CreateLoaderAndRun(std::unique_ptr<network::ResourceRequest> request) { @@ -187,6 +186,30 @@ return loader->NetError(); } + void CreateStoragePartitionForUrl(const GURL& url) { + base::expected<IsolatedWebAppUrlInfo, std::string> url_info = + IsolatedWebAppUrlInfo::Create(url); + if (!url_info.has_value()) { + CHECK(false) << "Can't create url info for url " << url + << ", error: " << url_info.error(); + } + + content::StoragePartition* current_storage_partition = + profile()->GetStoragePartition( + url_info->storage_partition_config(profile()), + /*can_create=*/false); + + CHECK(current_storage_partition == nullptr) + << "Partition already exists for url: " << url; + + content::StoragePartition* new_storage_partition = + profile()->GetStoragePartition( + url_info->storage_partition_config(profile()), + /*can_create=*/true); + + CHECK(new_storage_partition != nullptr); + } + const ScopedUrlHandler& url_handler() { CHECK(url_handler_); return *url_handler_; @@ -304,7 +327,7 @@ } TEST_F(IsolatedWebAppURLLoaderFactoryTest, - PostRequestsReturnMethodNotSupported) { + PostRequestsReturnMethodNotSupportedWhenAppIsInstalled) { RegisterWebApp(CreateIsolatedWebApp(kAppStartUrl, IsolationData{IsolationData::DevModeProxy{ .proxy_url = kProxyUrl.spec()}})); @@ -322,12 +345,39 @@ IsHttpStatusCode(net::HTTP_METHOD_NOT_ALLOWED)); } -TEST_F(IsolatedWebAppURLLoaderFactoryTest, - PostRequestsFailWithErrFailedIfAppNotInstalled) { +TEST_F( + IsolatedWebAppURLLoaderFactoryTest, + PostRequestsReturnMethodNotSupportedWhenAppIsInstalledAndThereIsPendingInstall) { + RegisterWebApp(CreateIsolatedWebApp( + kAppStartUrl, IsolationData{IsolationData::DevModeProxy{ + .proxy_url = "http://installed-app-proxy-url.com"}})); + + IsolatedWebAppPendingInstallInfo::FromWebContents(*web_contents()) + .set_isolation_data(IsolationData{IsolationData::DevModeProxy{ + .proxy_url = "http://pending-install-proxy-url.com"}}); + CreateFactory(); auto request = std::make_unique<network::ResourceRequest>(); - request->method = net::HttpRequestHeaders::kPostMethod; + const char* kUnsupportedHttpMethod = net::HttpRequestHeaders::kPostMethod; + request->method = kUnsupportedHttpMethod; + request->url = kAppStartUrl; + int status = CreateLoaderAndRun(std::move(request)); + + EXPECT_THAT(status, IsNetError(net::OK)); + ASSERT_THAT(ResponseInfo(), NotNull()); + EXPECT_THAT(ResponseInfo()->headers->response_code(), + IsHttpStatusCode(net::HTTP_METHOD_NOT_ALLOWED)); +} + +TEST_F(IsolatedWebAppURLLoaderFactoryTest, + RequestWithUnsupportedHttpMethodFailWithErrFailedIfAppNotInstalled) { + CreateFactory(); + + auto request = std::make_unique<network::ResourceRequest>(); + + const char* kUnsupportedHttpMethod = net::HttpRequestHeaders::kPostMethod; + request->method = kUnsupportedHttpMethod; request->url = kAppStartUrl; int status = CreateLoaderAndRun(std::move(request)); @@ -455,7 +505,8 @@ Eq(GURL("http://example.com/foo/bar.html"))); } -TEST_F(IsolatedWebAppURLLoaderFactoryTest, GeneratedInstallPageIsReturned) { +TEST_F(IsolatedWebAppURLLoaderFactoryTest, + DoNotReturnGeneratedPageWhenNotInstallingApplication) { RegisterWebApp(CreateIsolatedWebApp(kAppStartUrl, IsolationData{IsolationData::DevModeProxy{ .proxy_url = "http://example.com"}})); @@ -465,6 +516,31 @@ auto request = std::make_unique<network::ResourceRequest>(); request->url = GURL("isolated-app://" + kWebBundleId + "/.well-known/_generated_install_page.html"); + + int status = CreateLoaderAndRun(std::move(request)); + + EXPECT_THAT(status, IsNetError(net::OK)); + EXPECT_THAT( + url_handler().intercepted_url(), + Eq("http://example.com/.well-known/_generated_install_page.html")); +} + +TEST_F(IsolatedWebAppURLLoaderFactoryTest, + ReturnGeneratedPageWhenInstallingApplication) { + IsolatedWebAppPendingInstallInfo::FromWebContents(*web_contents()) + .set_isolation_data(IsolationData{IsolationData::DevModeProxy{ + .proxy_url = "http://some-proxy-url.com"}}); + + RegisterWebApp(CreateIsolatedWebApp(kAppStartUrl, + IsolationData{IsolationData::DevModeProxy{ + .proxy_url = "http://example.com"}})); + + CreateFactory(); + + auto request = std::make_unique<network::ResourceRequest>(); + request->url = GURL("isolated-app://" + kWebBundleId + + "/.well-known/_generated_install_page.html"); + int status = CreateLoaderAndRun(std::move(request)); EXPECT_THAT(status, IsNetError(net::OK)); @@ -475,7 +551,53 @@ } TEST_F(IsolatedWebAppURLLoaderFactoryTest, - GeneratedInstallPageIsNotReturnedForNonIwa) { + RequestsRedirectedToPendingInstallIsolationDataWhenAppIsInstalled) { + IsolatedWebAppPendingInstallInfo::FromWebContents(*web_contents()) + .set_isolation_data(IsolationData{IsolationData::DevModeProxy{ + .proxy_url = "http://some-proxy-url.com"}}); + + RegisterWebApp(CreateIsolatedWebApp(kAppStartUrl, + IsolationData{IsolationData::DevModeProxy{ + .proxy_url = "http://example.com"}})); + + CreateFactory(); + + auto request = std::make_unique<network::ResourceRequest>(); + request->url = GURL("isolated-app://" + kWebBundleId + + "/some-resource-for-testing.html"); + + int status = CreateLoaderAndRun(std::move(request)); + + EXPECT_THAT(status, IsNetError(net::OK)); + EXPECT_THAT( + url_handler().intercepted_url(), + Eq(GURL("http://some-proxy-url.com/some-resource-for-testing.html"))); +} + +TEST_F(IsolatedWebAppURLLoaderFactoryTest, + RequestsRedirectedToPendingInstallIsolationDataWhenAppIsNotInstalled) { + CreateStoragePartitionForUrl(GURL("isolated-app://" + kWebBundleId)); + + IsolatedWebAppPendingInstallInfo::FromWebContents(*web_contents()) + .set_isolation_data(IsolationData{IsolationData::DevModeProxy{ + .proxy_url = "http://some-proxy-url.com"}}); + + CreateFactory(); + + auto request = std::make_unique<network::ResourceRequest>(); + request->url = GURL("isolated-app://" + kWebBundleId + + "/some-resource-for-testing.html"); + + int status = CreateLoaderAndRun(std::move(request)); + + EXPECT_THAT(status, IsNetError(net::OK)); + EXPECT_THAT( + url_handler().intercepted_url(), + Eq(GURL("http://some-proxy-url.com/some-resource-for-testing.html"))); +} + +TEST_F(IsolatedWebAppURLLoaderFactoryTest, + GeneratedInstallPageIsNotReturnedForNonInstallingApp) { RegisterWebApp(CreateWebApp(kAppStartUrl)); CreateFactory();
diff --git a/chrome/browser/win/conflicts/installed_applications_unittest.cc b/chrome/browser/win/conflicts/installed_applications_unittest.cc index f1b6d7c..ae494e1 100644 --- a/chrome/browser/win/conflicts/installed_applications_unittest.cc +++ b/chrome/browser/win/conflicts/installed_applications_unittest.cc
@@ -4,9 +4,9 @@ #include "chrome/browser/win/conflicts/installed_applications.h" -#include <algorithm> #include <map> +#include "base/ranges/algorithm.h" #include "base/strings/stringprintf.h" #include "base/test/test_reg_util_win.h" #include "base/win/registry.h" @@ -397,13 +397,10 @@ auto applications = installed_applications().applications_; std::sort(std::begin(applications), std::end(applications)); EXPECT_EQ(std::end(applications), - std::adjacent_find(std::begin(applications), std::end(applications), - [](const auto& lhs, const auto& rhs) { - return std::tie(lhs.name, lhs.registry_root, - lhs.registry_key_path, - lhs.registry_wow64_access) == - std::tie(rhs.name, rhs.registry_root, - rhs.registry_key_path, - rhs.registry_wow64_access); - })); + base::ranges::adjacent_find( + applications, std::equal<>(), [](const auto& app) { + return std::tie(app.name, app.registry_root, + app.registry_key_path, + app.registry_wow64_access); + })); }
diff --git a/chrome/browser/win/conflicts/module_blocklist_cache_util_unittest.cc b/chrome/browser/win/conflicts/module_blocklist_cache_util_unittest.cc index 0af2cdd09..2db9d15 100644 --- a/chrome/browser/win/conflicts/module_blocklist_cache_util_unittest.cc +++ b/chrome/browser/win/conflicts/module_blocklist_cache_util_unittest.cc
@@ -52,8 +52,8 @@ // Sort the entries and make sure each module is unique. std::sort(entries.begin(), entries.end(), internal::ModuleLess()); - CHECK(std::adjacent_find(entries.begin(), entries.end(), - internal::ModuleEqual()) == entries.end()); + CHECK(base::ranges::adjacent_find(entries, internal::ModuleEqual()) == + entries.end()); return entries; }
diff --git a/chrome/build/linux.pgo.txt b/chrome/build/linux.pgo.txt index 01d48837..883d8a80 100644 --- a/chrome/build/linux.pgo.txt +++ b/chrome/build/linux.pgo.txt
@@ -1 +1 @@ -chrome-linux-main-1665748375-b2b65b3edd4aa54d39584ec37f17a02b534c6a88.profdata +chrome-linux-main-1665770398-17d4f7bea9fcc7d745ba4f47c559f834a80ec320.profdata
diff --git a/chrome/build/mac-arm.pgo.txt b/chrome/build/mac-arm.pgo.txt index a6075393..324fd7d 100644 --- a/chrome/build/mac-arm.pgo.txt +++ b/chrome/build/mac-arm.pgo.txt
@@ -1 +1 @@ -chrome-mac-arm-main-1664819315-34a4cf7009d4c6f71f01d59b264c8857d33d762e.profdata +chrome-mac-arm-main-1665770398-c2ca4d7140879c329be52f41540880b88c5e6fea.profdata
diff --git a/chrome/build/mac.pgo.txt b/chrome/build/mac.pgo.txt index 069b275..40e18e1 100644 --- a/chrome/build/mac.pgo.txt +++ b/chrome/build/mac.pgo.txt
@@ -1 +1 @@ -chrome-mac-main-1665748375-fdaca138baf43e64fc055e9a634fed17cf608280.profdata +chrome-mac-main-1665770398-3e2cbb4cbec67c24aa069dcf24f9a5316ce169a2.profdata
diff --git a/chrome/build/win32.pgo.txt b/chrome/build/win32.pgo.txt index 2484d07..b704373 100644 --- a/chrome/build/win32.pgo.txt +++ b/chrome/build/win32.pgo.txt
@@ -1 +1 @@ -chrome-win32-main-1665725711-7a5ece51c21f206817e2bbc0fdf1b1588c007956.profdata +chrome-win32-main-1665770398-e5491636ea29ef2e4a2a0d4123cb4c9b39977e3f.profdata
diff --git a/chrome/build/win64.pgo.txt b/chrome/build/win64.pgo.txt index 9fadd49..c0e6f11 100644 --- a/chrome/build/win64.pgo.txt +++ b/chrome/build/win64.pgo.txt
@@ -1 +1 @@ -chrome-win64-main-1665748375-fe5d46e806b458fcf6d095059fa61765fe504a88.profdata +chrome-win64-main-1665770398-60ea39db78e2e9990263a15c0195420700de507e.profdata
diff --git a/chrome/chrome_paks.gni b/chrome/chrome_paks.gni index c298d35..c5eb50b 100644 --- a/chrome/chrome_paks.gni +++ b/chrome/chrome_paks.gni
@@ -217,6 +217,11 @@ "//chrome/browser/resources/media_router/cast_feedback:resources", ] } + + if (is_chrome_branded && toolkit_views) { + sources += [ "$root_gen_dir/chrome/lens_resources.pak" ] + deps += [ "//chrome/browser/resources/lens:resources" ] + } } if (is_chromeos_ash) { sources += [
diff --git a/chrome/common/extensions/api/accessibility_private.json b/chrome/common/extensions/api/accessibility_private.json index 28393334..8d119adc 100644 --- a/chrome/common/extensions/api/accessibility_private.json +++ b/chrome/common/extensions/api/accessibility_private.json
@@ -234,7 +234,7 @@ { "id": "AccessibilityFeature", "type": "string", - "enum": [ "enhancedNetworkVoices", "googleTtsLanguagePacks", "dictationPumpkinParsing", "selectToSpeakVoiceSwitching"], + "enum": [ "enhancedNetworkVoices", "googleTtsLanguagePacks", "dictationPumpkinParsing", "selectToSpeakVoiceSwitching", "dictationMoreCommands"], "description": "Subset of accessibility features." }, {
diff --git a/chrome/common/webui_url_constants.cc b/chrome/common/webui_url_constants.cc index cc945d1..c7f9465 100644 --- a/chrome/common/webui_url_constants.cc +++ b/chrome/common/webui_url_constants.cc
@@ -619,6 +619,11 @@ const char kChromeUICastFeedbackHost[] = "cast-feedback"; #endif +#if !BUILDFLAG(IS_ANDROID) && BUILDFLAG(GOOGLE_CHROME_BRANDING) +const char kChromeUILensURL[] = "chrome://lens/"; +const char kChromeUILensHost[] = "lens"; +#endif + // Extension sub pages. const char kExtensionConfigureCommandsSubPage[] = "configureCommands";
diff --git a/chrome/common/webui_url_constants.h b/chrome/common/webui_url_constants.h index 276f7a7..31a4c4c 100644 --- a/chrome/common/webui_url_constants.h +++ b/chrome/common/webui_url_constants.h
@@ -542,6 +542,11 @@ extern const char kChromeUICastFeedbackHost[]; #endif +#if !BUILDFLAG(IS_ANDROID) && BUILDFLAG(GOOGLE_CHROME_BRANDING) +extern const char kChromeUILensURL[]; +extern const char kChromeUILensHost[]; +#endif + // Extensions sub pages. extern const char kExtensionConfigureCommandsSubPage[];
diff --git a/chrome/renderer/accessibility/read_anything_app_controller.cc b/chrome/renderer/accessibility/read_anything_app_controller.cc index 533da31c..e3f4235 100644 --- a/chrome/renderer/accessibility/read_anything_app_controller.cc +++ b/chrome/renderer/accessibility/read_anything_app_controller.cc
@@ -105,6 +105,17 @@ html_tag); } +void SetAXNodeDataTextDirection(v8::Isolate* isolate, + gin::Dictionary* v8_dict, + ui::AXNodeData* ax_node_data) { + v8::Local<v8::Value> v8_direction; + v8_dict->Get("direction", &v8_direction); + int direction; + gin::ConvertFromV8(isolate, v8_direction, &direction); + ax_node_data->AddIntAttribute(ax::mojom::IntAttribute::kTextDirection, + direction); +} + void SetAXNodeDataUrl(v8::Isolate* isolate, gin::Dictionary* v8_dict, ui::AXNodeData* ax_node_data) { @@ -183,6 +194,7 @@ SetAXNodeDataChildIds(isolate, &v8_node_dict, &ax_node_data); SetAXNodeDataHtmlTag(isolate, &v8_node_dict, &ax_node_data); SetAXNodeDataLanguage(isolate, &v8_node_dict, &ax_node_data); + SetAXNodeDataTextDirection(isolate, &v8_node_dict, &ax_node_data); SetAXNodeDataUrl(isolate, &v8_node_dict, &ax_node_data); snapshot.nodes.push_back(ax_node_data); } @@ -213,13 +225,13 @@ ReadAnythingAppController* ReadAnythingAppController::Install( content::RenderFrame* render_frame) { v8::Isolate* isolate = blink::MainThreadIsolate(); - v8::MicrotasksScope microtask_scope(isolate, - v8::MicrotasksScope::kDoNotRunMicrotasks); v8::HandleScope handle_scope(isolate); v8::Local<v8::Context> context = render_frame->GetWebFrame()->MainWorldScriptContext(); if (context.IsEmpty()) return nullptr; + v8::MicrotasksScope microtask_scope(isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Context::Scope context_scope(context); @@ -325,6 +337,8 @@ .SetProperty("letterSpacing", &ReadAnythingAppController::LetterSpacing) .SetProperty("lineSpacing", &ReadAnythingAppController::LineSpacing) .SetMethod("getChildren", &ReadAnythingAppController::GetChildren) + .SetMethod("getTextDirection", + &ReadAnythingAppController::GetTextDirection) .SetMethod("getHtmlTag", &ReadAnythingAppController::GetHtmlTag) .SetMethod("getLanguage", &ReadAnythingAppController::GetLanguage) .SetMethod("getTextContent", &ReadAnythingAppController::GetTextContent) @@ -415,6 +429,30 @@ return text_content; } +std::string ReadAnythingAppController::GetTextDirection( + ui::AXNodeID ax_node_id) { + ui::AXNode* ax_node = GetAXNode(ax_node_id); + if (!ax_node) + return std::string(); + + auto text_direction = static_cast<ax::mojom::WritingDirection>( + ax_node->GetIntAttribute(ax::mojom::IntAttribute::kTextDirection)); + + // Vertical writing is displayed horizontally with "auto". + switch (text_direction) { + case ax::mojom::WritingDirection::kLtr: + return "ltr"; + case ax::mojom::WritingDirection::kRtl: + return "rtl"; + case ax::mojom::WritingDirection::kTtb: + return "auto"; + case ax::mojom::WritingDirection::kBtt: + return "auto"; + default: + return std::string(); + } +} + std::string ReadAnythingAppController::GetUrl(ui::AXNodeID ax_node_id) { ui::AXNode* ax_node = GetAXNode(ax_node_id); if (!ax_node)
diff --git a/chrome/renderer/accessibility/read_anything_app_controller.h b/chrome/renderer/accessibility/read_anything_app_controller.h index 527cc5d..2a33fea3 100644 --- a/chrome/renderer/accessibility/read_anything_app_controller.h +++ b/chrome/renderer/accessibility/read_anything_app_controller.h
@@ -82,6 +82,7 @@ std::string GetHtmlTag(ui::AXNodeID ax_node_id); std::string GetLanguage(ui::AXNodeID ax_node_id); std::string GetTextContent(ui::AXNodeID ax_node_id); + std::string GetTextDirection(ui::AXNodeID ax_node_id); std::string GetUrl(ui::AXNodeID ax_node_id); void OnConnected();
diff --git a/chrome/renderer/cart/commerce_hint_agent.cc b/chrome/renderer/cart/commerce_hint_agent.cc index 878bbc3..7d9ed7e 100644 --- a/chrome/renderer/cart/commerce_hint_agent.cc +++ b/chrome/renderer/cart/commerce_hint_agent.cc
@@ -501,6 +501,9 @@ const GURL& navigation_url(frame->GetDocument().Url()); const GURL& url = request.Url(); + if (CommerceHintAgent::ShouldSkipAddToCartRequest(navigation_url, url)) { + return false; + } bool is_add_to_cart = false; if (navigation_url.DomainIs("dickssportinggoods.com")) { is_add_to_cart = CommerceHintAgent::IsAddToCart(url.spec()); @@ -532,10 +535,6 @@ return true; } - if (CommerceHintAgent::ShouldSkipAddToCartRequest(navigation_url, url)) { - return false; - } - if (IsCartHeuristicsImprovementEnabled()) { if (navigation_url.DomainIs("abebooks.com")) return false; @@ -1170,11 +1169,17 @@ bool CommerceHintAgent::ShouldSkipAddToCartRequest(const GURL& navigation_url, const GURL& request_url) { + const std::string& navigation_domain = eTLDPlusOne(navigation_url); + const re2::RE2* pattern = + commerce_heuristics::CommerceHeuristicsData::GetInstance() + .GetSkipAddToCartPatternForDomain(navigation_domain); + if (pattern) { + return PartialMatch(request_url.spec().substr(0, kLengthLimit), *pattern); + } const std::map<std::string, std::string>& skip_string_map = GetSkipAddToCartMapping(); static base::NoDestructor<std::map<std::string, std::unique_ptr<re2::RE2>>> skip_regex_map; - const std::string& navigation_domain = eTLDPlusOne(navigation_url); if (skip_string_map.find(navigation_domain) == skip_string_map.end()) { return false; }
diff --git a/chrome/renderer/cart/commerce_hint_agent_browsertest.cc b/chrome/renderer/cart/commerce_hint_agent_browsertest.cc index 5687c26..b08ec0c6 100644 --- a/chrome/renderer/cart/commerce_hint_agent_browsertest.cc +++ b/chrome/renderer/cart/commerce_hint_agent_browsertest.cc
@@ -504,6 +504,27 @@ ExpectUKMCount(XHREntry::kEntryName, "IsAddToCart", 1); } +IN_PROC_BROWSER_TEST_F(CommerceHintAgentTest, SkipAddToCart_FromComponent) { + bool is_populated = + commerce_hint_service_->InitializeCommerceHeuristicsForTesting( + base::Version("0.0.0.1"), R"###( + { + "guitarcenter.com": { + "skip_add_to_cart_regex": "dummy-request" + } + } + )###", + "{}", "", ""); + DCHECK(is_populated); + + NavigateToURL("https://www.guitarcenter.com/"); + SendXHR("/add-to-cart", "product: 123"); + WaitForUmaCount("Commerce.Carts.AddToCartByURL", 1); + + SendXHR("/add-to-cart/dummy-request-url", "product: 123"); + WaitForUmaCount("Commerce.Carts.AddToCartByURL", 1); +} + // TODO(https://crbug/1310497, https://crbug.com/1362442): This test is flaky // on ChromeOS and Linux Asan. #if BUILDFLAG(IS_CHROMEOS)
diff --git a/chrome/renderer/net/net_error_page_controller.cc b/chrome/renderer/net/net_error_page_controller.cc index 20d0993..6424315 100644 --- a/chrome/renderer/net/net_error_page_controller.cc +++ b/chrome/renderer/net/net_error_page_controller.cc
@@ -24,13 +24,14 @@ base::WeakPtr<Delegate> delegate) { v8::Isolate* isolate = blink::MainThreadIsolate(); v8::HandleScope handle_scope(isolate); - v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Context> context = render_frame->GetWebFrame()->MainWorldScriptContext(); if (context.IsEmpty()) return; + v8::MicrotasksScope microtasks_scope( + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Context::Scope context_scope(context); gin::Handle<NetErrorPageController> controller = gin::CreateHandle(
diff --git a/chrome/renderer/supervised_user/supervised_user_error_page_controller.cc b/chrome/renderer/supervised_user/supervised_user_error_page_controller.cc index da71ced..f5da744d 100644 --- a/chrome/renderer/supervised_user/supervised_user_error_page_controller.cc +++ b/chrome/renderer/supervised_user/supervised_user_error_page_controller.cc
@@ -33,13 +33,14 @@ base::WeakPtr<SupervisedUserErrorPageControllerDelegate> delegate) { v8::Isolate* isolate = blink::MainThreadIsolate(); v8::HandleScope handle_scope(isolate); - v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Context> context = render_frame->GetWebFrame()->MainWorldScriptContext(); if (context.IsEmpty()) return; + v8::MicrotasksScope microtasks_scope( + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Context::Scope context_scope(context); gin::Handle<SupervisedUserErrorPageController> controller = gin::CreateHandle(
diff --git a/chrome/services/printing/print_backend_service_impl.cc b/chrome/services/printing/print_backend_service_impl.cc index 4c3b12b..7589844 100644 --- a/chrome/services/printing/print_backend_service_impl.cc +++ b/chrome/services/printing/print_backend_service_impl.cc
@@ -51,6 +51,9 @@ #if BUILDFLAG(IS_WIN) #include "base/containers/queue.h" #include "base/win/win_util.h" +#include "chrome/services/printing/public/mojom/printer_xml_parser.mojom.h" +#include "mojo/public/cpp/bindings/pending_remote.h" +#include "mojo/public/cpp/bindings/remote.h" #include "printing/emf_win.h" #include "printing/printed_page_win.h" #include "ui/gfx/geometry/rect.h" @@ -734,6 +737,13 @@ std::move(callback))); } +#if BUILDFLAG(IS_WIN) +void PrintBackendServiceImpl::BindPrinterXmlParser( + mojo::PendingRemote<mojom::PrinterXmlParser> remote) { + xml_parser_remote_.Bind(std::move(remote)); +} +#endif // BUILDFLAG(IS_WIN) + void PrintBackendServiceImpl::OnDidStartPrintingReadyDocument( DocumentHelper& document_helper, mojom::ResultCode result) {
diff --git a/chrome/services/printing/print_backend_service_impl.h b/chrome/services/printing/print_backend_service_impl.h index ca339999..cfd95ee 100644 --- a/chrome/services/printing/print_backend_service_impl.h +++ b/chrome/services/printing/print_backend_service_impl.h
@@ -27,6 +27,12 @@ #include "printing/printing_context.h" #include "ui/gfx/native_widget_types.h" +#if BUILDFLAG(IS_WIN) +#include "chrome/services/printing/public/mojom/printer_xml_parser.mojom.h" +#include "mojo/public/cpp/bindings/pending_remote.h" +#include "mojo/public/cpp/bindings/remote.h" +#endif // BUILDFLAG(IS_WIN) + #if !BUILDFLAG(ENABLE_OOP_PRINTING) #error "Out-of-process printing must be enabled." #endif @@ -122,6 +128,10 @@ // mojom::PrintBackendService implementation: void Init(const std::string& locale) override; +#if BUILDFLAG(IS_WIN) + void BindPrinterXmlParser( + mojo::PendingRemote<mojom::PrinterXmlParser> remote) override; +#endif // BUILDFLAG(IS_WIN) void Poke() override; void EnumeratePrinters( mojom::PrintBackendService::EnumeratePrintersCallback callback) override; @@ -218,6 +228,10 @@ std::vector<std::unique_ptr<DocumentHelper>> documents_; mojo::Receiver<mojom::PrintBackendService> receiver_; + +#if BUILDFLAG(IS_WIN) + mojo::Remote<mojom::PrinterXmlParser> xml_parser_remote_; +#endif // BUILDFLAG(IS_WIN) }; } // namespace printing
diff --git a/chrome/services/printing/public/mojom/BUILD.gn b/chrome/services/printing/public/mojom/BUILD.gn index 5c17796..759524f1 100644 --- a/chrome/services/printing/public/mojom/BUILD.gn +++ b/chrome/services/printing/public/mojom/BUILD.gn
@@ -46,6 +46,9 @@ "//printing/mojom:printing_context", "//ui/gfx/geometry/mojom", ] + if (is_win) { + sources += [ "printer_xml_parser.mojom" ] + } } cpp_typemaps = [
diff --git a/chrome/services/printing/public/mojom/print_backend_service.mojom b/chrome/services/printing/public/mojom/print_backend_service.mojom index 912330b..bf5e1de0 100644 --- a/chrome/services/printing/public/mojom/print_backend_service.mojom +++ b/chrome/services/printing/public/mojom/print_backend_service.mojom
@@ -13,6 +13,9 @@ import "sandbox/policy/mojom/sandbox.mojom"; import "ui/gfx/geometry/mojom/geometry.mojom"; +[EnableIf=is_win] +import "chrome/services/printing/public/mojom/printer_xml_parser.mojom"; + // The default printer name, or the `ResultCode` if there was an error when // trying to retrieve this data. union DefaultPrinterNameResult { @@ -91,6 +94,11 @@ // interface to the underlying data source. Init(string locale); + // Binds an interface that is used to communicate with the browser process + // to parse XML in a separate process. + [EnableIf=is_win] + BindPrinterXmlParser(pending_remote<PrinterXmlParser> remote); + // TODO(crbug.com/1225111) Message with no arguments and no reply that is // useful to ensure that an idle timeout change takes effect. Poke();
diff --git a/chrome/services/printing/public/mojom/printer_xml_parser.mojom b/chrome/services/printing/public/mojom/printer_xml_parser.mojom new file mode 100644 index 0000000..18abced4 --- /dev/null +++ b/chrome/services/printing/public/mojom/printer_xml_parser.mojom
@@ -0,0 +1,27 @@ +// Copyright 2022 The Chromium Authors +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +module printing.mojom; + +import "mojo/public/mojom/base/values.mojom"; +import "printing/mojom/print.mojom"; + +// The capabilities of a printer as a `base::Value`, or the `ResultCode` if +// there was an error when trying to retrieve this data. +union PrinterCapabilitiesValueResult { + mojo_base.mojom.Value capabilities; + ResultCode result_code; +}; + +// An interface that is used by printing utility services to send XML parse +// requests to the browser process. The browser process can then facilitate +// safely parsing the XML. +interface PrinterXmlParser { + // Receives an XML string of printer capabilities and responds with a + // base::Value of the parsed XML or ResultCode if there was an error + // retrieving the data. + [Sync] + ParseXmlForPrinterCapabilities(string capabilities_xml) => + (PrinterCapabilitiesValueResult value_result); +};
diff --git a/chrome/test/data/webui/chromeos/os_feedback_ui/confirmation_page_test.js b/chrome/test/data/webui/chromeos/os_feedback_ui/confirmation_page_test.js index ff265ae5..2c5784e 100644 --- a/chrome/test/data/webui/chromeos/os_feedback_ui/confirmation_page_test.js +++ b/chrome/test/data/webui/chromeos/os_feedback_ui/confirmation_page_test.js
@@ -264,6 +264,15 @@ assertEquals(1, feedbackServiceProvider.getOpenDiagnosticsAppCallCount()); verifyRecordPostSubmitActionCalled( true, FeedbackAppPostSubmitAction.kOpenDiagnosticsApp); + + // Make sure that the label and the sub-label are clickable too. + const label = link.querySelector('.label'); + label.click(); + assertEquals(2, feedbackServiceProvider.getOpenDiagnosticsAppCallCount()); + + const subLabel = link.querySelector('.sub-label'); + subLabel.click(); + assertEquals(3, feedbackServiceProvider.getOpenDiagnosticsAppCallCount()); }); // Test clicking explore app link. @@ -280,6 +289,15 @@ assertEquals(1, feedbackServiceProvider.getOpenExploreAppCallCount()); verifyRecordPostSubmitActionCalled( true, FeedbackAppPostSubmitAction.kOpenExploreApp); + + // Make sure that the label and the sub-label are clickable too. + const label = link.querySelector('.label'); + label.click(); + assertEquals(2, feedbackServiceProvider.getOpenExploreAppCallCount()); + + const subLabel = link.querySelector('.sub-label'); + subLabel.click(); + assertEquals(3, feedbackServiceProvider.getOpenExploreAppCallCount()); }); // Test clicking openChromebookHelp link. @@ -312,6 +330,15 @@ url, 'https://support.google.com/chromebook/?hl=en#topic=3399709'); verifyRecordPostSubmitActionCalled( true, FeedbackAppPostSubmitAction.kOpenChromebookCommunity); + + // Make sure that the label and the sub-label are clickable too. + const label = link.querySelector('.label'); + label.click(); + assertEquals(2, windowOpenCalled); + + const subLabel = link.querySelector('.sub-label'); + subLabel.click(); + assertEquals(3, windowOpenCalled); }); // Test that we only record the user's first action on confirmation page.
diff --git a/chrome/test/data/webui/chromeos/personalization_app/test_wallpaper_interface_provider.ts b/chrome/test/data/webui/chromeos/personalization_app/test_wallpaper_interface_provider.ts index acbbb1e4..49911a4 100644 --- a/chrome/test/data/webui/chromeos/personalization_app/test_wallpaper_interface_provider.ts +++ b/chrome/test/data/webui/chromeos/personalization_app/test_wallpaper_interface_provider.ts
@@ -104,7 +104,6 @@ layout: WallpaperLayout.kCenter, key: '1', type: WallpaperType.kOnline, - url: {url: 'data:image/png;base64somedataurl/0'}, }; }
diff --git a/chrome/test/data/webui/chromeos/personalization_app/wallpaper_fullscreen_element_test.ts b/chrome/test/data/webui/chromeos/personalization_app/wallpaper_fullscreen_element_test.ts index bf1479c..e51df833 100644 --- a/chrome/test/data/webui/chromeos/personalization_app/wallpaper_fullscreen_element_test.ts +++ b/chrome/test/data/webui/chromeos/personalization_app/wallpaper_fullscreen_element_test.ts
@@ -26,7 +26,6 @@ layout: WallpaperLayout.kCenter, key: 'testing', type: WallpaperType.kCustomized, - url: {url: 'data://testing'}, }; const pendingSelectedCustomImage:
diff --git a/chrome/test/data/webui/chromeos/personalization_app/wallpaper_preview_element_test.ts b/chrome/test/data/webui/chromeos/personalization_app/wallpaper_preview_element_test.ts index 3a76ca9..d1e341f 100644 --- a/chrome/test/data/webui/chromeos/personalization_app/wallpaper_preview_element_test.ts +++ b/chrome/test/data/webui/chromeos/personalization_app/wallpaper_preview_element_test.ts
@@ -86,7 +86,11 @@ await waitAfterNextRender(wallpaperPreviewElement); const img = wallpaperPreviewElement.shadowRoot!.querySelector('img'); - assertEquals(wallpaperProvider.currentWallpaper.url.url, img!.src); + assertEquals( + `chrome://personalization/wallpaper.png?key=${ + wallpaperProvider.currentWallpaper.key}`, + img!.src, + 'current wallpaper key is appended to url as query parameter'); }); test('shows placeholders when image fails to load', async () => {
diff --git a/chrome/test/data/webui/chromeos/personalization_app/wallpaper_selected_element_test.ts b/chrome/test/data/webui/chromeos/personalization_app/wallpaper_selected_element_test.ts index f5e6c31b..73fc488 100644 --- a/chrome/test/data/webui/chromeos/personalization_app/wallpaper_selected_element_test.ts +++ b/chrome/test/data/webui/chromeos/personalization_app/wallpaper_selected_element_test.ts
@@ -91,7 +91,10 @@ await waitAfterNextRender(wallpaperSelectedElement); const img = wallpaperSelectedElement.shadowRoot!.querySelector('img'); - assertEquals(wallpaperProvider.currentWallpaper.url.url, img!.src); + assertEquals( + `chrome://personalization/wallpaper.png?key=${ + wallpaperProvider.currentWallpaper.key}`, + img!.src, 'sets current wallpaper key appended to url'); const textContainerElements = wallpaperSelectedElement.shadowRoot!.querySelectorAll( @@ -142,17 +145,22 @@ const img = wallpaperSelectedElement.shadowRoot!.querySelector('img') as HTMLImageElement; - assertEquals(wallpaperProvider.currentWallpaper.url.url, img.src); + assertEquals( + `chrome://personalization/wallpaper.png?key=${ + wallpaperProvider.currentWallpaper.key}`, + img!.src, 'sets current wallpaper key appended to url'); + personalizationStore.data.wallpaper.currentSelected = { - url: {url: 'data:image/png;base64_some_new_data'}, - attribution: ['New attribution'], - assetId: BigInt(100), + ...personalizationStore.data.wallpaper.currentSelected, + key: 'new_key', }; personalizationStore.notifyObservers(); await waitAfterNextRender(wallpaperSelectedElement); - assertEquals('data:image/png;base64_some_new_data', img.src); + assertEquals( + `chrome://personalization/wallpaper.png?key=new_key`, img.src, + 'updates wallpaper key query parameter'); }); test('shows placeholders when image fails to load', async () => { @@ -181,20 +189,6 @@ null, wallpaperSelectedElement.shadowRoot!.querySelector('img')); }); - test('shows image url with data scheme', async () => { - personalizationStore.data.wallpaper.currentSelected = { - url: {url: ''}, - attribution: [], - assetId: BigInt(100), - }; - personalizationStore.data.wallpaper.loading.selected = false; - wallpaperSelectedElement = initElement(WallpaperSelected); - await waitAfterNextRender(wallpaperSelectedElement); - - const img = wallpaperSelectedElement.shadowRoot!.querySelector('img'); - assertEquals('', img!.src); - }); - test('shows daily refresh option on the collection view', async () => { personalizationStore.data.wallpaper.currentSelected = { url: {url: ''}, @@ -335,7 +329,6 @@ test('shows attribution for device default wallpaper', async () => { const currentSelected: CurrentWallpaper = { - url: {url: 'url'}, attribution: ['testing attribution'], layout: WallpaperLayout.kStretch, type: WallpaperType.kDefault,
diff --git a/chrome/test/data/webui/side_panel/read_anything/read_anything_app_test.ts b/chrome/test/data/webui/side_panel/read_anything/read_anything_app_test.ts index 4d6777d..63161a01 100644 --- a/chrome/test/data/webui/side_panel/read_anything/read_anything_app_test.ts +++ b/chrome/test/data/webui/side_panel/read_anything/read_anything_app_test.ts
@@ -608,4 +608,146 @@ const expected: string = '<div><p>ello</p><p>World</p><p>Fr</p></div>'; assertContainerInnerHTML(expected); }); + + test('updateContent textDirection', () => { + // root htmlTag='#document' id=1 + // ++paragraph htmlTag='p' id=2 direction='ltr' + // ++++staticText name='This is left to right writing' id=3 + // ++paragraph htmlTag='p' id=4 direction='rtl' + // ++++staticText name='This is right to left writing' id=4 + const axTree = { + rootId: 1, + nodes: [ + { + id: 1, + role: 'rootWebArea', + htmlTag: '#document', + childIds: [2, 4], + }, + { + id: 2, + role: 'paragraph', + htmlTag: 'p', + direction: 1, + childIds: [3], + }, + { + id: 3, + role: 'staticText', + name: 'This is left to right writing', + }, + { + id: 4, + role: 'paragraph', + htmlTag: 'p', + direction: 2, + childIds: [5], + }, + { + id: 5, + role: 'staticText', + name: 'This is right to left writing', + }, + ], + }; + chrome.readAnything.setContentForTesting(axTree, [2, 4]); + const expected: string = '<p dir="ltr">This is left to right writing</p>' + + '<p dir="rtl">This is right to left writing</p>'; + assertContainerInnerHTML(expected); + }); + + test('updateContent textDirection parentNodeDiffDir', () => { + // root htmlTag='#document' id=1 + // ++paragraph htmlTag='p' id=2 direction='ltr' + // ++++staticText name='This is ltr' id=3 + // ++++link htmlTag='a' url='http://www.google.com/' id=4 direction='rtl' + // ++++++staticText name='This link is rtl' id=5 + const axTree = { + rootId: 1, + nodes: [ + { + id: 1, + role: 'rootWebArea', + htmlTag: '#document', + childIds: [2], + }, + { + id: 2, + role: 'paragraph', + htmlTag: 'p', + direction: 1, + childIds: [3, 4], + }, + { + id: 3, + role: 'staticText', + name: 'This is ltr', + }, + { + id: 4, + role: 'link', + htmlTag: 'a', + direction: 2, + url: 'http://www.google.com/', + childIds: [5], + }, + { + id: 5, + role: 'staticText', + name: 'This link is rtl', + }, + ], + }; + chrome.readAnything.setContentForTesting(axTree, [2]); + const expected: string = + '<p dir="ltr">This is ltr<a dir="rtl" href="http://www.google.com/">This link is rtl</a></p>'; + assertContainerInnerHTML(expected); + }); + + test('updateContent textDirection verticalDir', () => { + // root htmlTag='#document' id=1 + // ++paragraph htmlTag='p' id=2 direction='ttb' + // ++++staticText name='This should be auto' id=3 + // ++paragraph htmlTag='p' id=4 direction='btt' + // ++++staticText name='This should be also be auto' id=4 + const axTree = { + rootId: 1, + nodes: [ + { + id: 1, + role: 'rootWebArea', + htmlTag: '#document', + childIds: [2, 4], + }, + { + id: 2, + role: 'paragraph', + htmlTag: 'p', + direction: 3, + childIds: [3], + }, + { + id: 3, + role: 'staticText', + name: 'This should be auto', + }, + { + id: 4, + role: 'paragraph', + htmlTag: 'p', + direction: 4, + childIds: [5], + }, + { + id: 5, + role: 'staticText', + name: 'This should be also be auto', + }, + ], + }; + chrome.readAnything.setContentForTesting(axTree, [2, 4]); + const expected: string = '<p dir="auto">This should be auto</p>' + + '<p dir="auto">This should be also be auto</p>'; + assertContainerInnerHTML(expected); + }); });
diff --git a/chrome/test/data/webui/tab_strip/tab_list_test.ts b/chrome/test/data/webui/tab_strip/tab_list_test.ts index d60ba37..e33c7cd 100644 --- a/chrome/test/data/webui/tab_strip/tab_list_test.ts +++ b/chrome/test/data/webui/tab_strip/tab_list_test.ts
@@ -994,15 +994,15 @@ }); test('PreventsDraggingWhenOnlyOneTab', () => { - assertFalse(tabList.shouldPreventDrag()); + assertFalse(tabList.shouldPreventDrag(/*isDraggingTab=*/ true)); const tabElements = getUnpinnedTabs(); tabElements[1]!.remove(); tabElements[2]!.remove(); - assertTrue(tabList.shouldPreventDrag()); + assertTrue(tabList.shouldPreventDrag(/*isDraggingTab=*/ true)); }); test('PreventsDraggingWhenOnlyOneTabGroup', async () => { - // Create a tab group with 2 tabs. + // Create a tab group with 1 tab. const appendedTab = createTab({ groupId: 'group0', id: 3, @@ -1011,6 +1011,22 @@ }); callbackRouter.tabCreated(appendedTab); await flushTasks(); + + // Remove all tabs outside the tab group. + const tabElements = getUnpinnedTabs(); + tabElements[0]!.remove(); + tabElements[1]!.remove(); + tabElements[2]!.remove(); + + // At this point there's only 1 tab in 1 tab group. + + // Dragging a tab is not allowed. + assertTrue(tabList.shouldPreventDrag(/*isDraggingTab=*/ true)); + + // Dragging a tab group is not allowed. + assertTrue(tabList.shouldPreventDrag(/*isDraggingTab=*/ false)); + + // Add another tab in the same tab group. const appendedTabInSameGroup = createTab({ groupId: 'group0', id: 4, @@ -1019,13 +1035,13 @@ }); callbackRouter.tabCreated(appendedTabInSameGroup); await flushTasks(); - assertFalse(tabList.shouldPreventDrag()); - // Remove all tabs outside the tab group. - const tabElements = getUnpinnedTabs(); - tabElements[0]!.remove(); - tabElements[1]!.remove(); - tabElements[2]!.remove(); - assertTrue(tabList.shouldPreventDrag()); + // At this point there are 2 tabs in 1 tab group; + + // Dragging a tab is still allowed. + assertFalse(tabList.shouldPreventDrag(/*isDraggingTab=*/ true)); + + // Dragging a tab group is not allowed. + assertTrue(tabList.shouldPreventDrag(/*isDraggingTab=*/ false)); }); });
diff --git a/chrome/test/media_router/media_router_integration_browsertest.cc b/chrome/test/media_router/media_router_integration_browsertest.cc index 281a8b27..49f74e8f 100644 --- a/chrome/test/media_router/media_router_integration_browsertest.cc +++ b/chrome/test/media_router/media_router_integration_browsertest.cc
@@ -214,10 +214,11 @@ if (!test_provider_->HasRoutes()) return; - // FIXME: There can't be a good reason to use the observer API to check for - // routes asynchronously, which is fragile. However, some browser tests rely - // on this behavior. Either add a callback parameter to TerminateRoute, or - // add pass callback to the TestProvider to run when all routes are gone. + // TODO(crbug.com/1374499): There can't be a good reason to use the observer + // API to check for routes asynchronously, which is fragile. However, some + // browser tests rely on this behavior. Either add a callback parameter to + // TerminateRoute, or add pass callback to the TestProvider to run when all + // routes are gone. base::RunLoop run_loop; NoRoutesObserver no_routes_observer( MediaRouterFactory::GetApiForBrowserContext(
diff --git a/chrome/updater/test/integration_tests.cc b/chrome/updater/test/integration_tests.cc index 91305ef..292dfef 100644 --- a/chrome/updater/test/integration_tests.cc +++ b/chrome/updater/test/integration_tests.cc
@@ -369,9 +369,7 @@ Uninstall(); } -// TODO(crbug.com/1345407): this test is disabled temporarily. Reenable after -// the build that adds `IUpdater::FetchPolicies` is published to CIPD. -TEST_F(IntegrationTest, DISABLED_OverinstallWorking) { +TEST_F(IntegrationTest, OverinstallWorking) { SetupRealUpdaterLowerVersion(); EXPECT_TRUE(WaitForUpdaterExit()); ExpectVersionNotActive(kUpdaterVersion); @@ -385,13 +383,7 @@ Uninstall(); } -// TODO(crbug.com/1359334): Flaky on Win10. -#if BUILDFLAG(IS_WIN) -#define MAYBE_OverinstallBroken DISABLED_OverinstallBroken -#else -#define MAYBE_OverinstallBroken OverinstallBroken -#endif -TEST_F(IntegrationTest, MAYBE_OverinstallBroken) { +TEST_F(IntegrationTest, OverinstallBroken) { SetupRealUpdaterLowerVersion(); EXPECT_TRUE(WaitForUpdaterExit()); DeleteUpdaterDirectory();
diff --git a/chromeos/metrics/login_event_recorder.cc b/chromeos/metrics/login_event_recorder.cc index 388f6c339..b6cf00b 100644 --- a/chromeos/metrics/login_event_recorder.cc +++ b/chromeos/metrics/login_event_recorder.cc
@@ -106,10 +106,6 @@ total_hist->AddTime(total); std::string output = base::StringPrintf("%s: %.2f", uma_name.c_str(), total.InSecondsF()); - if (uma_name == "BootTime.Login2" || uma_name == "BootTime.LoginNewUser") { - UMA_HISTOGRAM_CUSTOM_TIMES("Ash.Tast.BootTime.Login2", total, - base::Milliseconds(1), base::Seconds(300), 100); - } base::Time prev = first; // Convert base::Time to base::TimeTicks for tracing. auto time2timeticks = [](const base::Time& ts) {
diff --git a/chromeos/profiles/atom.afdo.newest.txt b/chromeos/profiles/atom.afdo.newest.txt index 0fc7416..59a3cae8 100644 --- a/chromeos/profiles/atom.afdo.newest.txt +++ b/chromeos/profiles/atom.afdo.newest.txt
@@ -1 +1 @@ -chromeos-chrome-amd64-atom-108-5335.0-1665395930-benchmark-108.0.5355.0-r1-redacted.afdo.xz +chromeos-chrome-amd64-atom-108-5335.0-1665395930-benchmark-108.0.5359.0-r1-redacted.afdo.xz
diff --git a/chromeos/profiles/bigcore.afdo.newest.txt b/chromeos/profiles/bigcore.afdo.newest.txt index b8215a2..e3fc160 100644 --- a/chromeos/profiles/bigcore.afdo.newest.txt +++ b/chromeos/profiles/bigcore.afdo.newest.txt
@@ -1 +1 @@ -chromeos-chrome-amd64-bigcore-108-5335.0-1665395359-benchmark-108.0.5355.0-r1-redacted.afdo.xz +chromeos-chrome-amd64-bigcore-108-5335.0-1665395359-benchmark-108.0.5359.0-r1-redacted.afdo.xz
diff --git a/components/breadcrumbs/DIR_METADATA b/components/breadcrumbs/DIR_METADATA new file mode 100644 index 0000000..030c855 --- /dev/null +++ b/components/breadcrumbs/DIR_METADATA
@@ -0,0 +1,3 @@ +monorail { + component: "Internals>CrashReporting" +}
diff --git a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/AllSiteSettings.java b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/AllSiteSettings.java index 2b9eb67f..df41bc1 100644 --- a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/AllSiteSettings.java +++ b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/AllSiteSettings.java
@@ -398,7 +398,6 @@ private boolean isNewAllSitesUiEnabled() { // Only in the "All sites" mode and with the flag enabled. return mCategory.getType() == SiteSettingsCategory.Type.ALL_SITES - && SiteSettingsFeatureList.isEnabled( - SiteSettingsFeatureList.SITE_DATA_IMPROVEMENTS); + && SiteSettingsUtil.isSiteDataImprovementEnabled(); } }
diff --git a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/GroupedWebsitesSettings.java b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/GroupedWebsitesSettings.java index 0d72530..7cd3d5c 100644 --- a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/GroupedWebsitesSettings.java +++ b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/GroupedWebsitesSettings.java
@@ -4,9 +4,7 @@ package org.chromium.components.browser_ui.site_settings; -import android.content.Context; import android.os.Bundle; -import android.text.format.Formatter; import androidx.preference.Preference; import androidx.preference.PreferenceCategory; @@ -81,12 +79,11 @@ private void setUpClearDataPreference() { ClearWebsiteStorage preference = findPreference(PREF_CLEAR_DATA); - long usage = mSiteGroup.getTotalUsage(); - if (usage > 0) { - Context context = preference.getContext(); - preference.setTitle( - String.format(context.getString(R.string.origin_settings_storage_usage_brief), - Formatter.formatShortFileSize(context, usage))); + long storage = mSiteGroup.getTotalUsage(); + int cookies = mSiteGroup.getNumberOfCookies(); + if (storage > 0 || cookies > 0) { + preference.setTitle(SiteSettingsUtil.generateStorageUsageText( + preference.getContext(), storage, cookies)); // TODO(crbug.com/1342991): Get clearingApps information from underlying sites. preference.setDataForDisplay(mSiteGroup.getDomainAndRegistry(), /*clearingApps=*/false); // TODO(crbug.com/1342991): Disable the preference if all underlying origins have
diff --git a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SingleWebsiteSettings.java b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SingleWebsiteSettings.java index 85fb1f7..70235b23 100644 --- a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SingleWebsiteSettings.java +++ b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SingleWebsiteSettings.java
@@ -524,13 +524,20 @@ private void setUpClearDataPreference() { ClearWebsiteStorage preference = findPreference(PREF_CLEAR_DATA); long usage = mSite.getTotalUsage(); - if (usage > 0) { + int cookies = mSite.getNumberOfCookies(); + // Only take cookies into account when the new UI is enabled. + if (usage > 0 || (SiteSettingsUtil.isSiteDataImprovementEnabled() && cookies > 0)) { boolean appFound = getSiteSettingsDelegate().getOriginsWithInstalledApp().contains( mSite.getAddress().getOrigin()); Context context = preference.getContext(); - preference.setTitle( - String.format(context.getString(R.string.origin_settings_storage_usage_brief), - Formatter.formatShortFileSize(context, usage))); + if (SiteSettingsUtil.isSiteDataImprovementEnabled()) { + preference.setTitle( + SiteSettingsUtil.generateStorageUsageText(context, usage, cookies)); + } else { + preference.setTitle(String.format( + context.getString(R.string.origin_settings_storage_usage_brief), + Formatter.formatShortFileSize(context, usage))); + } preference.setDataForDisplay(mSite.getTitle(), appFound); if (WebsitePreferenceBridge.isCookieDeletionDisabled( getSiteSettingsDelegate().getBrowserContextHandle(), @@ -545,7 +552,7 @@ private void setupResetSitePreference() { Preference preference = findPreference(PREF_RESET_SITE); int titleResId; - if (SiteSettingsFeatureList.isEnabled(SiteSettingsFeatureList.SITE_DATA_IMPROVEMENTS)) { + if (SiteSettingsUtil.isSiteDataImprovementEnabled()) { titleResId = mHideNonPermissionPreferences ? R.string.page_info_permissions_reset : R.string.website_reset_full; } else {
diff --git a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SiteSettingsUtil.java b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SiteSettingsUtil.java index aa5b7f5..3090b6a 100644 --- a/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SiteSettingsUtil.java +++ b/components/browser_ui/site_settings/android/java/src/org/chromium/components/browser_ui/site_settings/SiteSettingsUtil.java
@@ -4,6 +4,9 @@ package org.chromium.components.browser_ui.site_settings; +import android.content.Context; +import android.text.format.Formatter; + import androidx.annotation.NonNull; import androidx.annotation.VisibleForTesting; @@ -76,4 +79,34 @@ } return ContentSettingsType.DEFAULT; } + + /** + * @return whether the flag for the improved UI for "All sites" and "Site settings" is enabled. + */ + public static boolean isSiteDataImprovementEnabled() { + return SiteSettingsFeatureList.isEnabled(SiteSettingsFeatureList.SITE_DATA_IMPROVEMENTS); + } + + /** + * @param context A {@link Context} object to pull strings out of. + * @param storage The amount of storage (in bytes) used by the entry. + * @param cookies The number of cookies associated with the entry. + * @return A string to display in the UI to show and clear storage and cookies. + */ + public static String generateStorageUsageText(Context context, long storage, int cookies) { + String result = ""; + if (storage > 0) { + result = String.format(context.getString(R.string.origin_settings_storage_usage_brief), + Formatter.formatShortFileSize(context, storage)); + } + if (cookies > 0) { + String cookie_str = context.getResources().getQuantityString( + R.plurals.cookies_count, cookies, cookies); + result = result.isEmpty() + ? cookie_str + : String.format(context.getString(R.string.summary_with_one_bullet), result, + cookie_str); + } + return result; + } }
diff --git a/components/commerce/core/commerce_heuristics_data.cc b/components/commerce/core/commerce_heuristics_data.cc index a07ddf14..d64b12e 100644 --- a/components/commerce/core/commerce_heuristics_data.cc +++ b/components/commerce/core/commerce_heuristics_data.cc
@@ -18,6 +18,7 @@ constexpr char kMerchantCartURLRegexType[] = "cart_url_regex"; constexpr char kMerchantCheckoutURLRegexType[] = "checkout_url_regex"; constexpr char kMerchantPurchaseURLRegexType[] = "purchase_url_regex"; +constexpr char kSkipAddToCartRegexType[] = "skip_add_to_cart_regex"; // CommerceGlobalHeuristics types. constexpr char kSkipProductPatternType[] = "sensitive_product_regex"; @@ -90,6 +91,7 @@ domain_cart_url_pattern_mapping_.clear(); domain_checkout_url_pattern_mapping_.clear(); domain_purchase_url_pattern_mapping_.clear(); + domain_skip_add_to_cart_pattern_mapping_.clear(); return true; } @@ -177,6 +179,13 @@ kMerchantPurchaseURLRegexType, domain); } +const re2::RE2* CommerceHeuristicsData::GetSkipAddToCartPatternForDomain( + const std::string& domain) { + return GetCommerceHintHeuristicsRegex( + domain_skip_add_to_cart_pattern_mapping_, kSkipAddToCartRegexType, + domain); +} + std::string CommerceHeuristicsData::GetProductIDExtractionJSON() { return product_id_json_; }
diff --git a/components/commerce/core/commerce_heuristics_data.h b/components/commerce/core/commerce_heuristics_data.h index ae9356c..49f4d9ab 100644 --- a/components/commerce/core/commerce_heuristics_data.h +++ b/components/commerce/core/commerce_heuristics_data.h
@@ -88,6 +88,10 @@ // `domain`. const re2::RE2* GetPurchasePageURLPatternForDomain(const std::string& domain); + // Try to get the pattern regex used to match against XHR request URL to see + // if the request should be ignored for AddToCart detection in `domain`. + const re2::RE2* GetSkipAddToCartPatternForDomain(const std::string& domain); + // Get the JSON data with product ID extraction heuristics. std::string GetProductIDExtractionJSON(); @@ -132,6 +136,8 @@ domain_checkout_url_pattern_mapping_; std::map<std::string, std::unique_ptr<re2::RE2>> domain_purchase_url_pattern_mapping_; + std::map<std::string, std::unique_ptr<re2::RE2>> + domain_skip_add_to_cart_pattern_mapping_; std::string product_id_json_; std::string cart_extraction_script_; };
diff --git a/components/commerce/core/commerce_heuristics_data_unittest.cc b/components/commerce/core/commerce_heuristics_data_unittest.cc index 0ef8836f..04197e8e 100644 --- a/components/commerce/core/commerce_heuristics_data_unittest.cc +++ b/components/commerce/core/commerce_heuristics_data_unittest.cc
@@ -21,7 +21,8 @@ "checkout_url_regex" : "bar.com/([^/]+/)?checkout" }, "baz.com": { - "purchase_url_regex" : "baz.com/([^/]+/)?purchase" + "purchase_url_regex" : "baz.com/([^/]+/)?purchase", + "skip_add_to_cart_regex": "dummy-request-url" } } )###"; @@ -95,6 +96,9 @@ EXPECT_EQ( *hint_heuristics->FindDict("baz.com")->FindString("purchase_url_regex"), "baz.com/([^/]+/)?purchase"); + EXPECT_EQ(*hint_heuristics->FindDict("baz.com")->FindString( + "skip_add_to_cart_regex"), + "dummy-request-url"); auto* global_heuristics = GetGlobalHeuristics(); EXPECT_EQ(global_heuristics->size(), 9u); EXPECT_TRUE(global_heuristics->contains("sensitive_product_regex")); @@ -271,6 +275,16 @@ "baz.com/([^/]+/)?purchase"); } +TEST_F(CommerceHeuristicsDataTest, TestGetSkipAddToCartPatternForDomain) { + auto& data = commerce_heuristics::CommerceHeuristicsData::GetInstance(); + + ASSERT_TRUE(data.PopulateDataFromComponent( + kHintHeuristicsJSONData, kGlobalHeuristicsJSONData, "", "")); + + EXPECT_EQ(data.GetSkipAddToCartPatternForDomain("baz.com")->pattern(), + "dummy-request-url"); +} + TEST_F(CommerceHeuristicsDataTest, TestRepopulateHintData) { auto& data = commerce_heuristics::CommerceHeuristicsData::GetInstance();
diff --git a/components/drive/OWNERS b/components/drive/OWNERS index 4ad89d4..6f2bdb8f 100644 --- a/components/drive/OWNERS +++ b/components/drive/OWNERS
@@ -1,5 +1 @@ -# Primary file://chrome/browser/sync_file_system/OWNERS - -# Secondary -file://ash/projector/OWNERS
diff --git a/components/exo/pointer.cc b/components/exo/pointer.cc index 9ee52ae0..067fe06 100644 --- a/components/exo/pointer.cc +++ b/components/exo/pointer.cc
@@ -67,13 +67,8 @@ bool SameLocation(const gfx::PointF& location_in_target, const gfx::PointF& location) { - // In general, it is good practice to compare floats using an epsilon. - // In particular, the mouse location_f() could differ between the - // MOUSE_PRESSED and MOUSE_RELEASED events. At MOUSE_RELEASED, it will have a - // targeter() already cached, while at MOUSE_PRESSED, it will have to - // calculate it passing through all the hierarchy of windows, and that could - // generate rounding error. std::numeric_limits<float>::epsilon() is not big - // enough to catch this rounding error. + // TODO(crbug.com/1354573): This is no longer necessary. Switch to + // std::numeric_limits<float>::eplison(). gfx::Vector2dF offset = location_in_target - location; return offset.LengthSquared() < (2 * kLocatedEventEpsilonSquared); } @@ -381,7 +376,8 @@ aura::Env::GetInstance()->AddPreTargetHandler( this, ui::EventTarget::Priority::kSystem); - location_when_pointer_capture_enabled_ = gfx::ToRoundedPoint(location_); + location_when_pointer_capture_enabled_ = + gfx::ToRoundedPoint(location_in_root_); if (ShouldMoveToCenter()) MoveCursorToCenterOfActiveDisplay(); @@ -464,7 +460,7 @@ } if (surface == focus_surface_) { - SetFocus(nullptr, gfx::PointF(), 0); + SetFocus(nullptr, gfx::PointF(), gfx::PointF(), 0); was_correctly_subscribed = true; } else if (surface == root_surface()) { UpdatePointerSurface(nullptr); @@ -488,12 +484,13 @@ gfx::PointF location_in_target; Surface* target = GetEffectiveTargetForEvent(event, &location_in_target); + gfx::PointF location_in_root = event->root_location_f(); // Update focus if target is different than the current pointer focus. - if (target != focus_surface_) - SetFocus(target, location_in_target, event->button_flags()); - - gfx::PointF location_in_root = GetLocationInRoot(target, location_in_target); + if (target != focus_surface_) { + SetFocus(target, location_in_root, location_in_target, + event->button_flags()); + } if (!focus_surface_) return; @@ -516,9 +513,9 @@ // so to avoid generating mouse event jitter we consider the location of // these events to be the same as |location| if floored values match. bool same_location = !event->IsSynthesized() - ? SameLocation(location_in_root, location_) + ? SameLocation(location_in_root, location_in_root_) : gfx::ToFlooredPoint(location_in_root) == - gfx::ToFlooredPoint(location_); + gfx::ToFlooredPoint(location_in_root_); // Ordinal motion is sent only on platforms that support it, which is // indicated by the presence of a flag. @@ -561,7 +558,7 @@ } if (needs_frame) delegate_->OnPointerFrame(); - location_ = location_in_root; + location_in_root_ = location_in_root; } } switch (event->type()) { @@ -717,7 +714,6 @@ //////////////////////////////////////////////////////////////////////////////// // aura::client::DragDropClientObserver overrides: void Pointer::OnDragStarted() { -#if BUILDFLAG(IS_CHROMEOS_ASH) // Drag 'n drop operations driven by sources different than pointer/mouse // should have not effect here. WMHelper* helper = WMHelper::GetInstance(); @@ -727,12 +723,10 @@ return; } - SetFocus(nullptr, gfx::PointF(), 0); -#endif + SetFocus(nullptr, gfx::PointF(), gfx::PointF(), 0); } void Pointer::OnDragCompleted(const ui::DropTargetEvent& event) { -#if BUILDFLAG(IS_CHROMEOS_ASH) // Drag 'n drop operations driven by sources different than pointer/mouse // should have not effect here. WMHelper* helper = WMHelper::GetInstance(); @@ -757,9 +751,10 @@ gfx::PointF location_in_target; auto* target = GetEffectiveTargetForEvent(&event, &location_in_target); - if (target) - SetFocus(target, location_in_target, /*button_flags=*/0); -#endif + if (target) { + SetFocus(target, event.root_location_f(), location_in_target, + /*button_flags=*/0); + } } //////////////////////////////////////////////////////////////////////////////// @@ -836,7 +831,8 @@ } void Pointer::SetFocus(Surface* surface, - const gfx::PointF& location, + const gfx::PointF& root_location, + const gfx::PointF& surface_location, int button_flags) { DCHECK(!surface || delegate_->CanAcceptPointerEventsForSurface(surface)); // First generate a leave event if we currently have a target in focus. @@ -852,9 +848,9 @@ } // Second generate an enter event if focus moved to a new surface. if (surface) { - delegate_->OnPointerEnter(surface, location, button_flags); + delegate_->OnPointerEnter(surface, surface_location, button_flags); delegate_->OnPointerFrame(); - location_ = GetLocationInRoot(surface, location); + location_in_root_ = root_location; focus_surface_ = surface; if (!focus_surface_->HasSurfaceObserver(this)) focus_surface_->AddSurfaceObserver(this); @@ -996,23 +992,13 @@ } } -gfx::PointF Pointer::GetLocationInRoot(Surface* target, - gfx::PointF location_in_target) { - if (!target || !target->window()) - return location_in_target; - aura::Window* w = target->window(); - gfx::PointF p(location_in_target.x(), location_in_target.y()); - aura::Window::ConvertPointToTarget(w, w->GetRootWindow(), &p); - return gfx::PointF(p.x(), p.y()); -} - bool Pointer::ShouldMoveToCenter() { if (!capture_window_) return false; gfx::Rect rect = capture_window_->GetRootWindow()->bounds(); rect.Inset(gfx::Insets::VH(rect.height() / 6, rect.width() / 6)); - return !rect.Contains(location_.x(), location_.y()); + return !rect.Contains(location_in_root_.x(), location_in_root_.y()); } void Pointer::MoveCursorToCenterOfActiveDisplay() { @@ -1031,7 +1017,7 @@ if (!relative_pointer_delegate_) return false; - gfx::Vector2dF delta = location_in_root - location_; + gfx::Vector2dF delta = location_in_root - location_in_root_; relative_pointer_delegate_->OnPointerRelativeMotion( time_stamp, delta, ordinal_motion.has_value() ? ordinal_motion.value() : delta);
diff --git a/components/exo/pointer.h b/components/exo/pointer.h index 20f92d9..099bc00 100644 --- a/components/exo/pointer.h +++ b/components/exo/pointer.h
@@ -157,7 +157,8 @@ // Change pointer focus to |surface|. void SetFocus(Surface* surface, - const gfx::PointF& location, + const gfx::PointF& root_location, + const gfx::PointF& surface_location, int button_flags); // Updates the root_surface in |SurfaceTreeHost| from which the cursor @@ -175,10 +176,6 @@ // Update |cursor_| to |cursor_bitmap_| transformed for the current display. void UpdateCursor(); - // Convert the given |location_in_target| to coordinates in the root window. - gfx::PointF GetLocationInRoot(Surface* target, - gfx::PointF location_in_target); - // Called to check if cursor should be moved to the center of the window when // sending relative movements. bool ShouldMoveToCenter(); @@ -225,8 +222,8 @@ // The current focus surface for the pointer. Surface* focus_surface_ = nullptr; - // The location of the pointer in the current focus surface. - gfx::PointF location_; + // The location of the pointer in the root window. + gfx::PointF location_in_root_; // The location of the pointer when pointer capture is first enabled. absl::optional<gfx::Point> location_when_pointer_capture_enabled_;
diff --git a/components/exo/shell_surface_base.cc b/components/exo/shell_surface_base.cc index 63c5e6e2..d5b4a9fc 100644 --- a/components/exo/shell_surface_base.cc +++ b/components/exo/shell_surface_base.cc
@@ -9,7 +9,6 @@ #include "ash/constants/ash_constants.h" #include "ash/display/screen_orientation_controller.h" #include "ash/frame/non_client_frame_view_ash.h" -#include "ash/metrics/login_unlock_throughput_recorder.h" #include "ash/public/cpp/ash_constants.h" #include "ash/public/cpp/rounded_corner_utils.h" #include "ash/public/cpp/shelf_types.h" @@ -653,9 +652,6 @@ DCHECK(!widget_); restore_session_id_.emplace(restore_session_id); restore_window_id_.emplace(restore_window_id); - ash::LoginUnlockThroughputRecorder* throughput_recorder = - ash::Shell::Get()->login_unlock_throughput_recorder(); - throughput_recorder->OnRestoredWindowCreated(restore_window_id); } void ShellSurfaceBase::SetRestoreInfoWithWindowIdSource( @@ -1812,17 +1808,6 @@ needs_layout_on_show_ = false; } - if (restore_window_id_.has_value()) { - ash::LoginUnlockThroughputRecorder* throughput_recorder = - ash::Shell::Get()->login_unlock_throughput_recorder(); - - aura::Window* root_window = host_window()->GetRootWindow(); - if (root_window) { - ui::Compositor* compositor = root_window->layer()->GetCompositor(); - throughput_recorder->OnBeforeRestoredWindowShown( - restore_window_id_.value(), compositor); - } - } widget_->Show(); if (has_grab_) StartCapture();
diff --git a/components/feed/core/v2/api_test/feed_api_subscriptions_unittest.cc b/components/feed/core/v2/api_test/feed_api_subscriptions_unittest.cc index f8e6392a..e3c03194 100644 --- a/components/feed/core/v2/api_test/feed_api_subscriptions_unittest.cc +++ b/components/feed/core/v2/api_test/feed_api_subscriptions_unittest.cc
@@ -224,6 +224,9 @@ "ContentSuggestions.Feed.WebFeed.FollowCount.AfterFollow", 1, 1); histograms.ExpectUniqueSample( "ContentSuggestions.Feed.WebFeed.NewFollow.IsRecommended", 0, 1); + histograms.ExpectUniqueSample( + "ContentSuggestions.Feed.WebFeed.NewFollow.ChangeReason", + WebFeedChangeReason::WEB_PAGE_MENU, 1); } TEST_F(FeedApiSubscriptionsTest, FollowWebFeedAbortOnClearAll) {
diff --git a/components/feed/core/v2/metrics_reporter.cc b/components/feed/core/v2/metrics_reporter.cc index c0dc933..941d52e2 100644 --- a/components/feed/core/v2/metrics_reporter.cc +++ b/components/feed/core/v2/metrics_reporter.cc
@@ -1102,6 +1102,14 @@ base::UmaHistogramBoolean( "ContentSuggestions.Feed.WebFeed.NewFollow.IsRecommended", result.web_feed_metadata.is_recommended); + if (result.change_reason) { + // Because WebFeedChangeReason_MAX is not an enum value, we can't use + // UmaHistogramEnumeration, but UmaHistogramExactLinear is equivalent. + base::UmaHistogramExactLinear( + "ContentSuggestions.Feed.WebFeed.NewFollow.ChangeReason", + static_cast<int>(result.change_reason), + feedwire::webfeed::WebFeedChangeReason_MAX + 1); + } } }
diff --git a/components/feed/core/v2/public/web_feed_subscriptions.h b/components/feed/core/v2/public/web_feed_subscriptions.h index 033f6211..02cac4c5 100644 --- a/components/feed/core/v2/public/web_feed_subscriptions.h +++ b/components/feed/core/v2/public/web_feed_subscriptions.h
@@ -24,6 +24,8 @@ WebFeedMetadata web_feed_metadata; // Number of subscriptions the user has after the Follow operation. int subscription_count = 0; + // The change reason from the request. + feedwire::webfeed::WebFeedChangeReason change_reason; }; // Follow a web feed given information about a web page. Calls `callback` when // complete. The callback parameter reports whether the url is now considered
diff --git a/components/feed/core/v2/web_feed_subscription_coordinator.cc b/components/feed/core/v2/web_feed_subscription_coordinator.cc index 0770071..f6eaafb 100644 --- a/components/feed/core/v2/web_feed_subscription_coordinator.cc +++ b/components/feed/core/v2/web_feed_subscription_coordinator.cc
@@ -304,6 +304,7 @@ index_.IsRecommended(result.followed_web_feed_id); callback_result.request_status = result.request_status; callback_result.subscription_count = index_.SubscriptionCount(); + callback_result.change_reason = result.change_reason; feed_stream_->GetMetricsReporter().OnFollowAttempt(followed_with_id, callback_result); std::move(callback).Run(std::move(callback_result));
diff --git a/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.cc b/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.cc index 8785fa0b..fcf87c74 100644 --- a/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.cc +++ b/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.cc
@@ -108,6 +108,7 @@ result.request_status = status; result.web_feed_info = subscribed_web_feed_info_; result.followed_web_feed_id = subscribed_web_feed_info_.web_feed_id(); + result.change_reason = request_.change_reason; std::move(callback_).Run(std::move(result)); TaskComplete(); }
diff --git a/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.h b/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.h index fb362bd8..b2ff2fb 100644 --- a/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.h +++ b/components/feed/core/v2/web_feed_subscriptions/subscribe_to_web_feed_task.h
@@ -26,7 +26,7 @@ WebFeedPageInformation page_info; std::string web_feed_id; feedwire::webfeed::WebFeedChangeReason change_reason = feedwire::webfeed:: - WebFeedChangeReason ::WEB_FEED_CHANGE_REASON_UNSPECIFIED; + WebFeedChangeReason::WEB_FEED_CHANGE_REASON_UNSPECIFIED; // Whether the subscription request will be stored and retried if failed. bool durable = false; }; @@ -37,6 +37,9 @@ // is its ID. std::string followed_web_feed_id; feedstore::WebFeedInfo web_feed_info; + // The change reason from the request. + feedwire::webfeed::WebFeedChangeReason change_reason = feedwire::webfeed:: + WebFeedChangeReason::WEB_FEED_CHANGE_REASON_UNSPECIFIED; }; SubscribeToWebFeedTask(FeedStream* stream, const OperationToken& operation_token,
diff --git a/components/feed/feed_feature_list.cc b/components/feed/feed_feature_list.cc index e3fbe7a..527974c 100644 --- a/components/feed/feed_feature_list.cc +++ b/components/feed/feed_feature_list.cc
@@ -73,6 +73,9 @@ BASE_FEATURE(kReliabilityLogging, "FeedReliabilityLogging", base::FEATURE_ENABLED_BY_DEFAULT); +BASE_FEATURE(kFeedHeaderStickToTop, + "FeedHeaderStickToTop", + base::FEATURE_DISABLED_BY_DEFAULT); BASE_FEATURE(kFeedInteractiveRefresh, "FeedInteractiveRefresh", base::FEATURE_ENABLED_BY_DEFAULT);
diff --git a/components/feed/feed_feature_list.h b/components/feed/feed_feature_list.h index 332bf065..edce9f13 100644 --- a/components/feed/feed_feature_list.h +++ b/components/feed/feed_feature_list.h
@@ -57,6 +57,9 @@ // Whether to log reliability events. BASE_DECLARE_FEATURE(kReliabilityLogging); +// Feature that enables sticky header when users scroll down. +BASE_DECLARE_FEATURE(kFeedHeaderStickToTop); + // Feature that enables refreshing feeds triggered by the users. BASE_DECLARE_FEATURE(kFeedInteractiveRefresh);
diff --git a/components/guest_view/renderer/guest_view_container.cc b/components/guest_view/renderer/guest_view_container.cc index 6a214a9..889d4c0 100644 --- a/components/guest_view/renderer/guest_view_container.cc +++ b/components/guest_view/renderer/guest_view_container.cc
@@ -173,8 +173,9 @@ return; v8::Context::Scope context_scope(context); - v8::MicrotasksScope microtasks( - destruction_isolate_, v8::MicrotasksScope::kDoNotRunMicrotasks); + v8::MicrotasksScope microtasks(destruction_isolate_, + context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); callback->Call(context, context->Global(), 0 /* argc */, nullptr) .FromMaybe(v8::Local<v8::Value>()); @@ -242,8 +243,9 @@ v8::Integer::New(element_resize_isolate_, new_size.height())}; v8::Context::Scope context_scope(context); - v8::MicrotasksScope microtasks( - element_resize_isolate_, v8::MicrotasksScope::kDoNotRunMicrotasks); + v8::MicrotasksScope microtasks(element_resize_isolate_, + context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); callback->Call(context, context->Global(), argc, argv) .FromMaybe(v8::Local<v8::Value>());
diff --git a/components/guest_view/renderer/guest_view_request.cc b/components/guest_view/renderer/guest_view_request.cc index 00443c3..2816577 100644 --- a/components/guest_view/renderer/guest_view_request.cc +++ b/components/guest_view/renderer/guest_view_request.cc
@@ -80,7 +80,7 @@ return; v8::Context::Scope context_scope(context); - v8::MicrotasksScope microtasks(isolate_, + v8::MicrotasksScope microtasks(isolate_, context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); callback->Call(context, context->Global(), argc, argv.get())
diff --git a/components/history/core/browser/expire_history_backend.cc b/components/history/core/browser/expire_history_backend.cc index 785d36e..6a286cad 100644 --- a/components/history/core/browser/expire_history_backend.cc +++ b/components/history/core/browser/expire_history_backend.cc
@@ -6,7 +6,6 @@ #include <stddef.h> -#include <algorithm> #include <functional> #include <limits> #include <memory> @@ -21,6 +20,7 @@ #include "base/location.h" #include "base/metrics/histogram_macros.h" #include "base/numerics/safe_conversions.h" +#include "base/ranges/algorithm.h" #include "base/task/sequenced_task_runner.h" #include "build/build_config.h" #include "components/favicon/core/favicon_database.h" @@ -262,10 +262,8 @@ // `times` must be in reverse chronological order and have no // duplicates, i.e. each member must be earlier than the one before // it. - DCHECK( - std::adjacent_find( - times.begin(), times.end(), std::less_equal<base::Time>()) == - times.end()); + DCHECK(base::ranges::adjacent_find(times, std::less_equal<base::Time>()) == + times.end()); if (!main_db_) return;
diff --git a/components/lens/lens_features.cc b/components/lens/lens_features.cc index 4751673..37f9fd86 100644 --- a/components/lens/lens_features.cc +++ b/components/lens/lens_features.cc
@@ -50,6 +50,10 @@ "EnableImageSearchSidePanelFor3PDse", base::FEATURE_ENABLED_BY_DEFAULT); +BASE_FEATURE(kLensRegionSearchStaticPage, + "LensRegionSearchStaticPage", + base::FEATURE_DISABLED_BY_DEFAULT); + const base::FeatureParam<bool> kEnableUKMLoggingForRegionSearch{ &kLensStandalone, "region-search-enable-ukm-logging", true}; @@ -221,5 +225,9 @@ return kEnablePersistentBubble.Get(); } +bool IsLensRegionSearchStaticPageEnabled() { + return base::FeatureList::IsEnabled(kLensRegionSearchStaticPage); +} + } // namespace features } // namespace lens
diff --git a/components/lens/lens_features.h b/components/lens/lens_features.h index 5ec19da..cb31a61 100644 --- a/components/lens/lens_features.h +++ b/components/lens/lens_features.h
@@ -44,6 +44,9 @@ // engines BASE_DECLARE_FEATURE(kEnableImageSearchSidePanelFor3PDse); +// Enables launching the region search experience in a new tab with WebUI. +BASE_DECLARE_FEATURE(kLensRegionSearchStaticPage); + // Enables using `Google` as the visual search provider instead of `Google // Lens`. extern const base::FeatureParam<bool> kUseGoogleAsVisualSearchProvider; @@ -174,6 +177,9 @@ // Returns whether we should use an alternative instruction chip string. extern bool UseAltChipString(); + +// Returns whether we should use a WebUI static page for region search. +extern bool IsLensRegionSearchStaticPageEnabled(); } // namespace features } // namespace lens
diff --git a/components/local_state/DIR_METADATA b/components/local_state/DIR_METADATA new file mode 100644 index 0000000..1c5b27e --- /dev/null +++ b/components/local_state/DIR_METADATA
@@ -0,0 +1,5 @@ +monorail { + component: "Internals>Metrics>Variations" +} + +team_email: "chromium-dev@chromium.org"
diff --git a/components/omnibox/browser/in_memory_url_index_unittest.cc b/components/omnibox/browser/in_memory_url_index_unittest.cc index f7cdcbd..6347206 100644 --- a/components/omnibox/browser/in_memory_url_index_unittest.cc +++ b/components/omnibox/browser/in_memory_url_index_unittest.cc
@@ -7,7 +7,6 @@ #include <stddef.h> #include <stdint.h> -#include <algorithm> #include <fstream> #include <memory> #include <numeric> @@ -21,6 +20,7 @@ #include "base/i18n/case_conversion.h" #include "base/memory/raw_ptr.h" #include "base/path_service.h" +#include "base/ranges/algorithm.h" #include "base/run_loop.h" #include "base/strings/string_split.h" #include "base/strings/string_util.h" @@ -723,9 +723,8 @@ // Each next group should fill almost everything, while the previous group // should occupy what's left. - auto* error_position = std::adjacent_find( - std::begin(item_groups), std::end(item_groups), - [&](const ItemGroup& previous, const ItemGroup& current) { + auto* error_position = base::ranges::adjacent_find( + item_groups, [&](const ItemGroup& previous, const ItemGroup& current) { auto ids = GetHistoryIdsUpTo(current.max_id); EXPECT_TRUE(GetPrivateData()->TrimHistoryIdsPool(&ids));
diff --git a/components/omnibox/browser/scored_history_match.cc b/components/omnibox/browser/scored_history_match.cc index f853e91d..bb2c625f 100644 --- a/components/omnibox/browser/scored_history_match.cc +++ b/components/omnibox/browser/scored_history_match.cc
@@ -6,13 +6,13 @@ #include <math.h> -#include <algorithm> #include <utility> #include <vector> #include "base/check_op.h" #include "base/no_destructor.h" #include "base/numerics/safe_conversions.h" +#include "base/ranges/algorithm.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_split.h" #include "base/strings/string_util.h" @@ -634,11 +634,8 @@ auto visits_end = visits.begin() + std::min(visits.size(), max_visits_to_score_); // Visits should be in newest to oldest order. - DCHECK(std::adjacent_find( - visits.begin(), visits_end, - [](const history::VisitInfo& a, const history::VisitInfo& b) { - return a.first < b.first; - }) == visits_end); + DCHECK(base::ranges::adjacent_find(visits.begin(), visits_end, std::less<>(), + &history::VisitInfo::first) == visits_end); for (auto i = visits.begin(); i != visits_end; ++i) { const bool is_page_transition_typed = ui::PageTransitionCoreTypeIs(i->second, ui::PAGE_TRANSITION_TYPED);
diff --git a/components/password_manager/core/browser/generation/password_generator.cc b/components/password_manager/core/browser/generation/password_generator.cc index 5aec52d..d6544e0f 100644 --- a/components/password_manager/core/browser/generation/password_generator.cc +++ b/components/password_manager/core/browser/generation/password_generator.cc
@@ -4,7 +4,6 @@ #include "components/password_manager/core/browser/generation/password_generator.h" -#include <algorithm> #include <limits> #include <map> #include <utility> @@ -12,6 +11,7 @@ #include "base/check.h" #include "base/rand_util.h" +#include "base/ranges/algorithm.h" #include "base/strings/utf_string_conversions.h" #include "components/autofill/core/browser/proto/password_requirements.pb.h" @@ -80,10 +80,9 @@ // sequences of '-' or '_' that are joined into long strokes on the screen // in many fonts. bool IsDifficultToRead(const std::u16string& password) { - return std::adjacent_find(password.begin(), password.end(), - [](auto a, auto b) { - return a == b && (a == '-' || a == '_'); - }) != password.end(); + return base::ranges::adjacent_find(password, [](auto a, auto b) { + return a == b && (a == '-' || a == '_'); + }) != password.end(); } // Generates a password according to |spec| and tries to maximze the entropy
diff --git a/components/plugins/renderer/webview_plugin.cc b/components/plugins/renderer/webview_plugin.cc index 24294f8..f33819af 100644 --- a/components/plugins/renderer/webview_plugin.cc +++ b/components/plugins/renderer/webview_plugin.cc
@@ -384,10 +384,11 @@ v8::Isolate* isolate = blink::MainThreadIsolate(); v8::HandleScope handle_scope(isolate); - v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Context> context = frame_->MainWorldScriptContext(); DCHECK(!context.IsEmpty()); + v8::MicrotasksScope microtasks_scope( + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Context::Scope context_scope(context); v8::Local<v8::Object> global = context->Global();
diff --git a/components/security_interstitials/content/renderer/security_interstitial_page_controller.cc b/components/security_interstitials/content/renderer/security_interstitial_page_controller.cc index 79ecbcbc..fc0d72f4 100644 --- a/components/security_interstitials/content/renderer/security_interstitial_page_controller.cc +++ b/components/security_interstitials/content/renderer/security_interstitial_page_controller.cc
@@ -24,13 +24,14 @@ content::RenderFrame* render_frame) { v8::Isolate* isolate = blink::MainThreadIsolate(); v8::HandleScope handle_scope(isolate); - v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Context> context = render_frame->GetWebFrame()->MainWorldScriptContext(); if (context.IsEmpty()) return; + v8::MicrotasksScope microtasks_scope( + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Context::Scope context_scope(context); gin::Handle<SecurityInterstitialPageController> controller =
diff --git a/components/sync/base/features.cc b/components/sync/base/features.cc index 42e0b316..41f925f25 100644 --- a/components/sync/base/features.cc +++ b/components/sync/base/features.cc
@@ -56,6 +56,10 @@ "SyncResetPollIntervalOnStart", base::FEATURE_DISABLED_BY_DEFAULT); +BASE_FEATURE(kSyncSegmentationDataType, + "SyncSegmentationDataType", + base::FEATURE_DISABLED_BY_DEFAULT); + BASE_FEATURE(kSyncSendInterestedDataTypes, "SyncSendInterestedDataTypes", base::FEATURE_ENABLED_BY_DEFAULT);
diff --git a/components/sync/base/features.h b/components/sync/base/features.h index ab9dc96..fbdeac4 100644 --- a/components/sync/base/features.h +++ b/components/sync/base/features.h
@@ -72,6 +72,9 @@ BASE_DECLARE_FEATURE(kSyncResetPollIntervalOnStart); +// If enabled, Segmentation data type will be synced. +BASE_DECLARE_FEATURE(kSyncSegmentationDataType); + // If enabled, interested data types, excluding Wallet and Offer, will be sent // to the Sync Server as part of DeviceInfo. BASE_DECLARE_FEATURE(kSyncSendInterestedDataTypes);
diff --git a/components/sync/base/model_type.cc b/components/sync/base/model_type.cc index a9ceb99..c0eaae1 100644 --- a/components/sync/base/model_type.cc +++ b/components/sync/base/model_type.cc
@@ -73,8 +73,8 @@ "Autofill Wallet Offer", sync_pb::EntitySpecifics::kAutofillOfferFieldNumber, ModelTypeForHistograms::kAutofillWalletOffer}, - {AUTOFILL_WALLET_USAGE, "AUTOFILL_WALLET_USAGE", - "autofill_wallet_usage", "Autofill Wallet Usage", + {AUTOFILL_WALLET_USAGE, "AUTOFILL_WALLET_USAGE", "autofill_wallet_usage", + "Autofill Wallet Usage", sync_pb::EntitySpecifics::kAutofillWalletUsageFieldNumber, ModelTypeForHistograms::kAutofillWalletUsage}, {THEMES, "THEME", "themes", "Themes", @@ -137,6 +137,9 @@ {USER_CONSENTS, "USER_CONSENT", "user_consent", "User Consents", sync_pb::EntitySpecifics::kUserConsentFieldNumber, ModelTypeForHistograms::kUserConsents}, + {SEGMENTATION, "SEGMENTATION", "segmentation", "Segmentation", + sync_pb::EntitySpecifics::kSegmentationFieldNumber, + ModelTypeForHistograms::kSegmentation}, {SEND_TAB_TO_SELF, "SEND_TAB_TO_SELF", "send_tab_to_self", "Send Tab To Self", sync_pb::EntitySpecifics::kSendTabToSelfFieldNumber, ModelTypeForHistograms::kSendTabToSelf}, @@ -184,7 +187,7 @@ static_assert(std::size(kModelTypeInfoMap) == GetNumModelTypes(), "kModelTypeInfoMap should have GetNumModelTypes() elements"); -static_assert(42 == syncer::GetNumModelTypes(), +static_assert(43 == syncer::GetNumModelTypes(), "When adding a new type, update enum SyncModelTypes in enums.xml " "and suffix SyncModelType in histograms.xml."); @@ -318,6 +321,9 @@ case CONTACT_INFO: specifics->mutable_contact_info(); break; + case SEGMENTATION: + specifics->mutable_segmentation(); + break; } } @@ -348,7 +354,7 @@ } ModelType GetModelTypeFromSpecifics(const sync_pb::EntitySpecifics& specifics) { - static_assert(42 == syncer::GetNumModelTypes(), + static_assert(43 == syncer::GetNumModelTypes(), "When adding new protocol types, the following type lookup " "logic must be updated."); if (specifics.has_bookmark()) @@ -431,6 +437,8 @@ return CONTACT_INFO; if (specifics.has_autofill_wallet_usage()) return AUTOFILL_WALLET_USAGE; + if (specifics.has_segmentation()) + return SEGMENTATION; // This client version doesn't understand |specifics|. DVLOG(1) << "Unknown datatype in sync proto."; @@ -438,7 +446,7 @@ } ModelTypeSet EncryptableUserTypes() { - static_assert(42 == syncer::GetNumModelTypes(), + static_assert(43 == syncer::GetNumModelTypes(), "If adding an unencryptable type, remove from " "encryptable_user_types below."); ModelTypeSet encryptable_user_types = UserTypes();
diff --git a/components/sync/base/model_type.h b/components/sync/base/model_type.h index ed145e8..0e2e0149 100644 --- a/components/sync/base/model_type.h +++ b/components/sync/base/model_type.h
@@ -112,6 +112,8 @@ USER_EVENTS, // Commit only user consents. USER_CONSENTS, + // Segmentation data. + SEGMENTATION, // Tabs sent between devices. SEND_TAB_TO_SELF, // Commit only security events. @@ -238,7 +240,8 @@ kPrintersAuthorizationServers = 52, kContactInfo = 53, kAutofillWalletUsage = 54, - kMaxValue = kAutofillWalletUsage + kSegmentation = 55, + kMaxValue = kSegmentation }; // Used to mark the type of EntitySpecifics that has no actual data.
diff --git a/components/sync/base/user_selectable_type.cc b/components/sync/base/user_selectable_type.cc index 76bbe59..96cf28bd 100644 --- a/components/sync/base/user_selectable_type.cc +++ b/components/sync/base/user_selectable_type.cc
@@ -33,7 +33,7 @@ constexpr char kWifiConfigurationsTypeName[] = "wifiConfigurations"; UserSelectableTypeInfo GetUserSelectableTypeInfo(UserSelectableType type) { - static_assert(42 == syncer::GetNumModelTypes(), + static_assert(43 == syncer::GetNumModelTypes(), "Almost always when adding a new ModelType, you must tie it to " "a UserSelectableType below (new or existing) so the user can " "disable syncing of that data. Today you must also update the " @@ -45,9 +45,12 @@ case UserSelectableType::kBookmarks: return {kBookmarksTypeName, BOOKMARKS, {BOOKMARKS}}; case UserSelectableType::kPreferences: + // TODO(crbug.com/1369259): Add GetPreconditionState() logic to check + // history state as a precondition for SEGMENTATION. return {kPreferencesTypeName, PREFERENCES, - {PREFERENCES, DICTIONARY, PRIORITY_PREFERENCES, SEARCH_ENGINES}}; + {PREFERENCES, DICTIONARY, PRIORITY_PREFERENCES, SEARCH_ENGINES, + SEGMENTATION}}; case UserSelectableType::kPasswords: return {kPasswordsTypeName, PASSWORDS, {PASSWORDS}}; case UserSelectableType::kAutofill:
diff --git a/components/sync/driver/sync_user_settings_impl.cc b/components/sync/driver/sync_user_settings_impl.cc index 57254a2..3c1b2c87 100644 --- a/components/sync/driver/sync_user_settings_impl.cc +++ b/components/sync/driver/sync_user_settings_impl.cc
@@ -268,7 +268,7 @@ #endif types.RetainAll(registered_model_types_); - static_assert(42 == GetNumModelTypes(), + static_assert(43 == GetNumModelTypes(), "If adding a new sync data type, update the list below below if" " you want to disable the new data type for local sync."); types.PutAll(ControlTypes()); @@ -277,6 +277,7 @@ types.Remove(AUTOFILL_WALLET_OFFER); types.Remove(AUTOFILL_WALLET_USAGE); types.Remove(SECURITY_EVENTS); + types.Remove(SEGMENTATION); types.Remove(SEND_TAB_TO_SELF); types.Remove(SHARING_MESSAGE); types.Remove(USER_CONSENTS);
diff --git a/components/sync/engine/cycle/data_type_tracker.cc b/components/sync/engine/cycle/data_type_tracker.cc index 790a92a..106f205 100644 --- a/components/sync/engine/cycle/data_type_tracker.cc +++ b/components/sync/engine/cycle/data_type_tracker.cc
@@ -30,6 +30,9 @@ constexpr base::TimeDelta kDefaultLocalChangeNudgeDelayForSessions = base::Seconds(11); +constexpr base::TimeDelta kDefaultLocalChangeNudgeDelayForSegmentations = + base::Seconds(11); + // Nudge delay for remote invalidations. Common to all data types. constexpr base::TimeDelta kRemoteInvalidationDelay = base::Milliseconds(250); @@ -56,6 +59,11 @@ // custom nudge delay, tuned for a reasonable trade-off between traffic // and freshness. return kDefaultLocalChangeNudgeDelayForSessions; + case SEGMENTATION: + // There are multiple segmentations computed during start-up within + // seconds. Applies a custom nudge delay, so that they are batched into + // one commit. + return kDefaultLocalChangeNudgeDelayForSegmentations; case BOOKMARKS: case PREFERENCES: // Types with sometimes automatic changes get longer delays to allow more @@ -146,6 +154,7 @@ case PRINTERS_AUTHORIZATION_SERVERS: case READING_LIST: case USER_CONSENTS: + case SEGMENTATION: case SEND_TAB_TO_SELF: case SECURITY_EVENTS: case WIFI_CONFIGURATIONS:
diff --git a/components/sync/protocol/proto_value_conversions.cc b/components/sync/protocol/proto_value_conversions.cc index 26361e5..f4bb0d8 100644 --- a/components/sync/protocol/proto_value_conversions.cc +++ b/components/sync/protocol/proto_value_conversions.cc
@@ -138,12 +138,14 @@ void VisitBytes(const P& parent_proto, const char* field_name, const std::string& field) { - value_->Set(field_name, BytesToValue(field)); + value_->GetDict().Set( + field_name, + base::Base64Encode(base::as_bytes(base::make_span(field)))); } template <class P, class E> void VisitEnum(const P& parent_proto, const char* field_name, E field) { - value_->Set(field_name, EnumToValue(field)); + value_->GetDict().Set(field_name, ProtoEnumToString(field)); } template <class P, class F> @@ -155,7 +157,7 @@ for (const auto& field : repeated_field) { list.Append(base::Value::FromUniquePtrValue(ToValue(field))); } - value_->Set(field_name, std::make_unique<base::Value>(std::move(list))); + value_->GetDict().Set(field_name, std::move(list)); } } @@ -168,7 +170,7 @@ for (const auto& field : repeated_field) { list.Append(base::Value::FromUniquePtrValue(ToValue(field))); } - value_->Set(field_name, std::make_unique<base::Value>(std::move(list))); + value_->GetDict().Set(field_name, std::move(list)); } } @@ -260,17 +262,6 @@ return value; } - static std::unique_ptr<base::Value> BytesToValue(const std::string& bytes) { - std::string bytes_base64; - base::Base64Encode(bytes, &bytes_base64); - return std::make_unique<base::Value>(bytes_base64); - } - - template <class E> - static std::unique_ptr<base::Value> EnumToValue(E value) { - return std::make_unique<base::Value>(ProtoEnumToString(value)); - } - std::unique_ptr<base::Value> ToValue(const std::string& value) const { return std::make_unique<base::Value>(value); } @@ -301,7 +292,8 @@ // Needs to be here to see all ToValue() overloads above. template <class P, class F> void VisitImpl(P&, const char* field_name, const F& field) { - value_->Set(field_name, ToValue(field)); + value_->GetDict().Set(field_name, + base::Value::FromUniquePtrValue(ToValue(field))); } const ProtoValueConversionOptions options_;
diff --git a/components/sync/protocol/proto_value_conversions_unittest.cc b/components/sync/protocol/proto_value_conversions_unittest.cc index 409fc51..30d9c11 100644 --- a/components/sync/protocol/proto_value_conversions_unittest.cc +++ b/components/sync/protocol/proto_value_conversions_unittest.cc
@@ -64,7 +64,7 @@ DEFINE_SPECIFICS_TO_VALUE_TEST(encrypted) -static_assert(42 == syncer::GetNumModelTypes(), +static_assert(43 == syncer::GetNumModelTypes(), "When adding a new field, add a DEFINE_SPECIFICS_TO_VALUE_TEST " "for your field below, and optionally a test for the specific " "conversions.");
diff --git a/components/sync/protocol/proto_visitors.h b/components/sync/protocol/proto_visitors.h index b47302d..1f1f9166 100644 --- a/components/sync/protocol/proto_visitors.h +++ b/components/sync/protocol/proto_visitors.h
@@ -534,7 +534,7 @@ } VISIT_PROTO_FIELDS(const sync_pb::EntitySpecifics& proto) { - static_assert(42 == GetNumModelTypes(), + static_assert(43 == GetNumModelTypes(), "When adding a new protocol type, you will likely need to add " "it here as well."); VISIT(encrypted);
diff --git a/components/translate/core/browser/translate_language_list.cc b/components/translate/core/browser/translate_language_list.cc index 088cc236..1e1753e 100644 --- a/components/translate/core/browser/translate_language_list.cc +++ b/components/translate/core/browser/translate_language_list.cc
@@ -6,7 +6,6 @@ #include <stddef.h> -#include <algorithm> #include <iterator> #include "base/bind.h" @@ -14,6 +13,7 @@ #include "base/json/json_reader.h" #include "base/lazy_instance.h" #include "base/notreached.h" +#include "base/ranges/algorithm.h" #include "base/strings/string_util.h" #include "base/strings/stringprintf.h" #include "base/values.h" @@ -185,8 +185,7 @@ DCHECK( std::is_sorted(supported_languages_.begin(), supported_languages_.end())); DCHECK(supported_languages_.end() == - std::adjacent_find(supported_languages_.begin(), - supported_languages_.end())); + base::ranges::adjacent_find(supported_languages_)); if (update_is_disabled) return; @@ -362,8 +361,7 @@ DCHECK( std::is_sorted(supported_languages_.begin(), supported_languages_.end())); DCHECK(supported_languages_.end() == - std::adjacent_find(supported_languages_.begin(), - supported_languages_.end())); + base::ranges::adjacent_find(supported_languages_)); NotifyEvent(__LINE__, base::JoinString(supported_languages_, ", ")); return true;
diff --git a/components/viz/service/display/direct_renderer.cc b/components/viz/service/display/direct_renderer.cc index b71d40d..e4cc6d8 100644 --- a/components/viz/service/display/direct_renderer.cc +++ b/components/viz/service/display/direct_renderer.cc
@@ -314,16 +314,11 @@ OverlayProcessorInterface::OutputSurfaceOverlayPlane* primary_plane = nullptr; if (output_surface_->IsDisplayedAsOverlayPlane()) { - // OutputSurface::GetOverlayMailbox() returns the mailbox for the last - // used buffer, which is most likely different from the one being used - // this frame. However, for the purpose of testing the overlay - // configuration, the mailbox for ANY buffer from BufferQueue is good - // enough because they're all created with identical properties. current_frame()->output_surface_plane = overlay_processor_->ProcessOutputSurfaceAsOverlay( device_viewport_size, surface_resource_size, frame_buffer_format, frame_color_space, frame_has_alpha, 1.0f /*opacity*/, - output_surface_->GetOverlayMailbox()); + GetPrimaryPlaneOverlayTestingMailbox()); primary_plane = &(current_frame()->output_surface_plane.value()); } @@ -1121,4 +1116,9 @@ return gfx::Rect(); } +gpu::Mailbox DirectRenderer::GetPrimaryPlaneOverlayTestingMailbox() { + NOTREACHED(); + return gpu::Mailbox(); +} + } // namespace viz
diff --git a/components/viz/service/display/direct_renderer.h b/components/viz/service/display/direct_renderer.h index 5b47646..733a7d6e8 100644 --- a/components/viz/service/display/direct_renderer.h +++ b/components/viz/service/display/direct_renderer.h
@@ -180,6 +180,10 @@ // 0 < n <= capabilities_.number_of_buffers. virtual void EnsureMinNumberOfBuffers(int n) {} + // Gets a mailbox that can be used for overlay testing the primary plane. This + // does not need to be the next mailbox that will be swapped. + virtual gpu::Mailbox GetPrimaryPlaneOverlayTestingMailbox(); + // Return the bounding rect of previously drawn delegated ink trail. gfx::Rect GetDelegatedInkTrailDamageRect();
diff --git a/components/viz/service/display/skia_renderer.cc b/components/viz/service/display/skia_renderer.cc index daab4044..758999e 100644 --- a/components/viz/service/display/skia_renderer.cc +++ b/components/viz/service/display/skia_renderer.cc
@@ -2718,6 +2718,9 @@ #elif defined(USE_OZONE) // Only Wayland uses this code path. for (auto& overlay : current_frame()->overlay_list) { + if (overlay.is_root_render_pass) { + continue; + } if (overlay.rpdq) { PrepareRenderPassOverlay(&overlay); locks.emplace_back(overlay.mailbox); @@ -3711,6 +3714,23 @@ } } +gpu::Mailbox SkiaRenderer::GetPrimaryPlaneOverlayTestingMailbox() { + // For the purpose of testing the overlay configuration, the mailbox for ANY + // buffer from BufferQueue is good enough because they're all created with + // identical properties. + // At the time we're testing overlays we don't yet know which mailbox will be + // presented this frame so we'll just use the last swapped buffer. (We might + // present a new frame's mailbox, or if we empty-swap we'll present the + // previous frame's mailbox.) + if (buffer_queue_) { + return buffer_queue_->GetLastSwappedBuffer(); + } else { + // OutputSurface::GetOverlayMailbox() returns the mailbox for the last + // swapped buffer. + return skia_output_surface_->GetOverlayMailbox(); + } +} + #if defined(USE_OZONE) const gpu::Mailbox SkiaRenderer::GetImageMailboxForColor( const SkColor4f& color) {
diff --git a/components/viz/service/display/skia_renderer.h b/components/viz/service/display/skia_renderer.h index c6f95a3..b38120b 100644 --- a/components/viz/service/display/skia_renderer.h +++ b/components/viz/service/display/skia_renderer.h
@@ -75,6 +75,7 @@ gfx::Rect GetCurrentFramebufferDamage() const override; void Reshape(const OutputSurface::ReshapeParams& reshape_params) override; void EnsureMinNumberOfBuffers(int n) override; + gpu::Mailbox GetPrimaryPlaneOverlayTestingMailbox() override; protected: bool CanPartialSwap() override;
diff --git a/components/viz/service/display_embedder/buffer_queue.cc b/components/viz/service/display_embedder/buffer_queue.cc index 6f2f0c6..2b760087 100644 --- a/components/viz/service/display_embedder/buffer_queue.cc +++ b/components/viz/service/display_embedder/buffer_queue.cc
@@ -158,6 +158,29 @@ return buffer; } +gpu::Mailbox BufferQueue::GetLastSwappedBuffer() { + // The last swapped buffer will generally be in displayed_buffer_, as long as + // SwapBuffersComplete() has been called at least once for a non-empty swap + // since the last Reshape(). + if (displayed_buffer_) { + return displayed_buffer_->mailbox; + } + + // If displayed_buffer_ is null then any available buffer will do. + if (!available_buffers_.empty()) { + return available_buffers_.back()->mailbox; + } + + // If there's nothing displayed or available, then we should have no buffers + // allocated because Reshape() hasn't been called yet, so a zero-mailbox is + // returned. + // If any buffers are in flight at this point then BufferQueue is being used + // incorrectly. We should not be Swap()ing all available buffers before + // receiving any SwapBuffersComplete() calls. + DCHECK(in_flight_buffers_.empty()); + return gpu::Mailbox(); +} + void BufferQueue::EnsureMinNumberOfBuffers(size_t n) { if (n <= number_of_buffers_) { return;
diff --git a/components/viz/service/display_embedder/buffer_queue.h b/components/viz/service/display_embedder/buffer_queue.h index 78843b1..4381440 100644 --- a/components/viz/service/display_embedder/buffer_queue.h +++ b/components/viz/service/display_embedder/buffer_queue.h
@@ -49,6 +49,8 @@ // target for compositing). gpu::Mailbox GetCurrentBuffer(); + gpu::Mailbox GetLastSwappedBuffer(); + // Returns a rectangle whose contents may have changed since the current // buffer was last submitted and needs to be redrawn. For partial swap, // only the contents outside this rectangle can be considered valid and do not
diff --git a/components/viz/service/display_embedder/buffer_queue_unittest.cc b/components/viz/service/display_embedder/buffer_queue_unittest.cc index 908eb944..cce12c8f 100644 --- a/components/viz/service/display_embedder/buffer_queue_unittest.cc +++ b/components/viz/service/display_embedder/buffer_queue_unittest.cc
@@ -504,4 +504,53 @@ SendDamagedFrame(small_damage); } +TEST_F(BufferQueueTest, GetLastSwappedBuffer) { + // No images allocated, so zero-mailbox is returned. + EXPECT_TRUE(buffer_queue_->GetLastSwappedBuffer().IsZero()); + + // After reshape we'll get the last buffer in the queue. + EXPECT_TRUE(buffer_queue_->Reshape(screen_size, kBufferQueueColorSpace, + kBufferQueueFormat)); + gpu::Mailbox last_swapped1 = buffer_queue_->GetLastSwappedBuffer(); + EXPECT_FALSE(last_swapped1.IsZero()); + + // The last swapped buffer won't change until calling SwapBuffersComplete. + gpu::Mailbox mailbox1 = buffer_queue_->GetCurrentBuffer(); + EXPECT_NE(last_swapped1, mailbox1); + EXPECT_EQ(last_swapped1, buffer_queue_->GetLastSwappedBuffer()); + buffer_queue_->SwapBuffers(screen_rect); + EXPECT_EQ(last_swapped1, buffer_queue_->GetLastSwappedBuffer()); + buffer_queue_->SwapBuffersComplete(); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox1); + + // Swap another frame. Last swapped only updates after SwapBuffersComplete(). + gpu::Mailbox mailbox2 = buffer_queue_->GetCurrentBuffer(); + buffer_queue_->SwapBuffers(screen_rect); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox1); + buffer_queue_->SwapBuffersComplete(); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox2); + + // Swap a third frame. Last swapped only updates after SwapBuffersComplete(). + gpu::Mailbox mailbox3 = buffer_queue_->GetCurrentBuffer(); + // The third mailbox is the first one we got from GetLastSwappedBuffer(). + EXPECT_EQ(mailbox3, last_swapped1); + buffer_queue_->SwapBuffers(screen_rect); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox2); + buffer_queue_->SwapBuffersComplete(); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox3); + + // Empty swap, Last swapped stays the same. + buffer_queue_->SwapBuffers(gfx::Rect()); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox3); + buffer_queue_->SwapBuffersComplete(); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox3); + + // Swap a fourth frame. Last swapped only updates after SwapBuffersComplete(). + EXPECT_EQ(buffer_queue_->GetCurrentBuffer(), mailbox1); + buffer_queue_->SwapBuffers(screen_rect); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox3); + buffer_queue_->SwapBuffersComplete(); + EXPECT_EQ(buffer_queue_->GetLastSwappedBuffer(), mailbox1); +} + } // namespace viz
diff --git a/components/webapps/browser/features.cc b/components/webapps/browser/features.cc index b18c71b..8222b62 100644 --- a/components/webapps/browser/features.cc +++ b/components/webapps/browser/features.cc
@@ -62,7 +62,7 @@ // Enables showing a detailed install dialog for user installs. BASE_FEATURE(kDesktopPWAsDetailedInstallDialog, "DesktopPWAsDetailedInstallDialog", - base::FEATURE_DISABLED_BY_DEFAULT); + base::FEATURE_ENABLED_BY_DEFAULT); // Enables sending the beforeinstallprompt without a service worker check. BASE_FEATURE(kSkipServiceWorkerForInstallPrompt,
diff --git a/components/zucchini/imposed_ensemble_matcher.cc b/components/zucchini/imposed_ensemble_matcher.cc index 36afad6..09c09f0 100644 --- a/components/zucchini/imposed_ensemble_matcher.cc +++ b/components/zucchini/imposed_ensemble_matcher.cc
@@ -4,12 +4,12 @@ #include "components/zucchini/imposed_ensemble_matcher.h" -#include <algorithm> #include <sstream> #include <utility> #include "base/bind.h" #include "base/logging.h" +#include "base/ranges/algorithm.h" #include "components/zucchini/io_utils.h" namespace zucchini { @@ -67,9 +67,8 @@ }); // Check for overlaps in "new" file. - if (std::adjacent_find( - matches_.begin(), matches_.end(), - [](const ElementMatch& match1, const ElementMatch& match2) { + if (base::ranges::adjacent_find( + matches_, [](const ElementMatch& match1, const ElementMatch& match2) { return match1.new_element.hi() > match2.new_element.lo(); }) != matches_.end()) { return kOverlapInNew;
diff --git a/content/browser/direct_sockets/direct_sockets_open_browsertest.cc b/content/browser/direct_sockets/direct_sockets_open_browsertest.cc index 4e27e196..2f919ab 100644 --- a/content/browser/direct_sockets/direct_sockets_open_browsertest.cc +++ b/content/browser/direct_sockets/direct_sockets_open_browsertest.cc
@@ -5,12 +5,10 @@ #include <algorithm> #include <vector> -#include "base/bind.h" #include "base/command_line.h" #include "base/memory/raw_ptr.h" #include "base/memory/scoped_refptr.h" #include "base/metrics/histogram.h" -#include "base/strings/strcat.h" #include "base/strings/stringprintf.h" #include "base/test/metrics/histogram_tester.h" #include "content/browser/direct_sockets/direct_sockets_service_impl.h" @@ -68,81 +66,12 @@ constexpr char kLocalhostAddress[] = "127.0.0.1"; -constexpr char kPermissionDeniedHistogramName[] = - "DirectSockets.PermissionDeniedFailures"; - constexpr char kTCPNetworkFailuresHistogramName[] = "DirectSockets.TCPNetworkFailures"; constexpr char kUDPNetworkFailuresHistogramName[] = "DirectSockets.UDPNetworkFailures"; -const std::string kIPv4_tests[] = { - // 0.0.0.0/8 - "0.0.0.0", "0.255.255.255", - // 10.0.0.0/8 - "10.0.0.0", "10.255.255.255", - // 100.64.0.0/10 - "100.64.0.0", "100.127.255.255", - // 127.0.0.0/8 - "127.0.0.0", "127.255.255.255", - // 169.254.0.0/16 - "169.254.0.0", "169.254.255.255", - // 172.16.0.0/12 - "172.16.0.0", "172.31.255.255", - // 192.0.2.0/24 - "192.0.2.0", "192.0.2.255", - // 192.88.99.0/24 - "192.88.99.0", "192.88.99.255", - // 192.168.0.0/16 - "192.168.0.0", "192.168.255.255", - // 198.18.0.0/15 - "198.18.0.0", "198.19.255.255", - // 198.51.100.0/24 - "198.51.100.0", "198.51.100.255", - // 203.0.113.0/24 - "203.0.113.0", "203.0.113.255", - // 224.0.0.0/8 - 255.0.0.0/8 - "224.0.0.0", "255.255.255.255"}; - -const std::string kIPv6_tests[] = { - // 0000::/8. - // Skip testing ::ffff:/96 explicitly since it will be tested through - // mapping Ipv4 Addresses. - "0:0:0:0:0:0:0:0", "ff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 0100::/8 - "100:0:0:0:0:0:0:0", "1ff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 0200::/7 - "200:0:0:0:0:0:0:0", "3ff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 0400::/6 - "400:0:0:0:0:0:0:0", "7ff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 0800::/5 - "800:0:0:0:0:0:0:0", "fff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 1000::/4 - "1000:0:0:0:0:0:0:0", "1fff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 4000::/3 - "4000:0:0:0:0:0:0:0", "5fff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 6000::/3 - "6000:0:0:0:0:0:0:0", "7fff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // 8000::/3 - "8000:0:0:0:0:0:0:0", "9fff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // c000::/3 - "c000:0:0:0:0:0:0:0", "dfff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // e000::/4 - "e000:0:0:0:0:0:0:0", "efff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // f000::/5 - "f000:0:0:0:0:0:0:0", "f7ff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // f800::/6 - "f800:0:0:0:0:0:0:0", "fbff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // fc00::/7 - "fc00:0:0:0:0:0:0:0", "fdff:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // fe00::/9 - "fe00:0:0:0:0:0:0:0", "fe7f:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // fe80::/10 - "fe80:0:0:0:0:0:0:0", "febf:ffff:ffff:ffff:ffff:ffff:ffff:ffff", - // fec0::/10 - "fec0:0:0:0:0:0:0:0", "feff:ffff:ffff:ffff:ffff:ffff:ffff:ffff"}; - class MockOpenNetworkContext : public content::test::MockNetworkContext { public: explicit MockOpenNetworkContext(net::Error result) : result_(result) {} @@ -291,78 +220,6 @@ ::testing::HasSubstr("keepAliveDelay must be no less than")); } -class DirectSocketsOpenCannotConnectBrowserTest - : public DirectSocketsOpenBrowserTest, - public testing::WithParamInterface<ProtocolType> { - public: - static std::vector<std::string> ProduceAllTestParams() { - std::vector<std::string> params; - std::copy(std::begin(kIPv4_tests), std::end(kIPv4_tests), - std::back_inserter(params)); - std::transform(std::begin(kIPv4_tests), std::end(kIPv4_tests), - std::back_inserter(params), - [](const std::string& ip_address) { - net::IPAddress address; - EXPECT_TRUE(address.AssignFromIPLiteral(ip_address)); - net::IPAddress mapped_address = - net::ConvertIPv4ToIPv4MappedIPv6(address); - return base::StrCat({"[", mapped_address.ToString(), "]"}); - }); - std::transform(std::begin(kIPv6_tests), std::end(kIPv6_tests), - std::back_inserter(params), - [](const std::string& ip_address) { - return base::StrCat({"[", ip_address, "]"}); - }); - return params; - } - - void RunTest() { - const auto protocol = GetParam(); - const std::string type = protocol == ProtocolType::kTcp ? "Tcp" : "Udp"; - const std::string expected_result = base::StringPrintf( - "open%s failed: NetworkError: Network Error.", type.c_str()); - - const std::string example_hostname = "mail.example.com"; - const std::string script = - protocol == ProtocolType::kTcp - ? base::StringPrintf("openTcp('%s', 993)", example_hostname.c_str()) - : base::StringPrintf( - "openUdp({ remoteAddress: '%s', remotePort: 993 })", - example_hostname.c_str()); - - for (const auto& address : ProduceAllTestParams()) { - const std::string mapping_rules = base::StringPrintf( - "MAP %s %s", example_hostname.c_str(), address.c_str()); - - MockOpenNetworkContext mock_network_context(net::OK); - mock_network_context.set_host_mapping_rules(mapping_rules); - DirectSocketsServiceImpl::SetNetworkContextForTesting( - &mock_network_context); - - base::HistogramTester histogram_tester; - histogram_tester.ExpectBucketCount( - kPermissionDeniedHistogramName, - blink::mojom::DirectSocketFailureType::kResolvingToNonPublic, 0); - - EXPECT_EQ(expected_result, EvalJs(shell(), script)); - - histogram_tester.ExpectBucketCount( - kPermissionDeniedHistogramName, - blink::mojom::DirectSocketFailureType::kResolvingToNonPublic, 1); - } - } -}; - -IN_PROC_BROWSER_TEST_P(DirectSocketsOpenCannotConnectBrowserTest, - Open_CannotConnectNonPublic) { - RunTest(); -} - -INSTANTIATE_TEST_SUITE_P( - /*empty*/, - DirectSocketsOpenCannotConnectBrowserTest, - testing::Values(ProtocolType::kTcp, ProtocolType::kUdp)); - IN_PROC_BROWSER_TEST_F(DirectSocketsOpenBrowserTest, OpenTcp_OptionsOne) { base::HistogramTester histogram_tester; histogram_tester.ExpectUniqueSample(kTCPNetworkFailuresHistogramName,
diff --git a/content/browser/direct_sockets/direct_sockets_service_impl.cc b/content/browser/direct_sockets/direct_sockets_service_impl.cc index f9777e33..62bfde2 100644 --- a/content/browser/direct_sockets/direct_sockets_service_impl.cc +++ b/content/browser/direct_sockets/direct_sockets_service_impl.cc
@@ -5,27 +5,25 @@ #include "content/browser/direct_sockets/direct_sockets_service_impl.h" #include "build/build_config.h" -#include "build/chromeos_buildflags.h" #include "content/browser/direct_sockets/direct_udp_socket_impl.h" #include "content/browser/direct_sockets/resolve_host_and_open_socket.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/content_browser_client.h" #include "content/public/browser/direct_sockets_delegate.h" -#include "content/public/browser/render_frame_host.h" +#include "content/public/browser/document_service.h" #include "content/public/browser/storage_partition.h" -#include "content/public/browser/web_contents.h" #include "content/public/common/content_client.h" #include "mojo/public/cpp/bindings/message.h" -#include "mojo/public/cpp/bindings/self_owned_receiver.h" +#include "mojo/public/cpp/bindings/pending_receiver.h" #include "mojo/public/cpp/system/data_pipe.h" #include "net/base/ip_address.h" #include "net/base/ip_endpoint.h" #include "net/traffic_annotation/network_traffic_annotation.h" -#include "services/network/public/cpp/simple_url_loader.h" #include "services/network/public/mojom/network_context.mojom.h" #include "services/network/public/mojom/tcp_socket.mojom.h" #include "third_party/abseil-cpp/absl/types/optional.h" #include "third_party/blink/public/mojom/direct_sockets/direct_sockets.mojom-shared.h" +#include "third_party/blink/public/mojom/direct_sockets/direct_sockets.mojom.h" #include "third_party/blink/public/mojom/permissions_policy/permissions_policy_feature.mojom-shared.h" namespace content { @@ -57,11 +55,13 @@ } bool IsFrameSufficientlyIsolated(content::RenderFrameHost* frame) { - return frame->GetWebExposedIsolationLevel() >= - content::RenderFrameHost::WebExposedIsolationLevel:: - kMaybeIsolatedApplication && - frame->IsFeatureEnabled( - blink::mojom::PermissionsPolicyFeature::kDirectSockets); + if (frame->GetWebExposedIsolationLevel() >= + content::RenderFrameHost::WebExposedIsolationLevel:: + kMaybeIsolatedApplication) { + return true; + } + + return false; } network::mojom::TCPConnectedSocketOptionsPtr CreateTCPConnectedSocketOptions( @@ -112,23 +112,29 @@ } // namespace -DirectSocketsServiceImpl::DirectSocketsServiceImpl(RenderFrameHost* frame_host) - : WebContentsObserver(WebContents::FromRenderFrameHost(frame_host)), - frame_host_(frame_host) {} +DirectSocketsServiceImpl::DirectSocketsServiceImpl( + RenderFrameHost* render_frame_host, + mojo::PendingReceiver<blink::mojom::DirectSocketsService> receiver) + : DocumentService(*render_frame_host, std::move(receiver)) {} DirectSocketsServiceImpl::~DirectSocketsServiceImpl() = default; // static void DirectSocketsServiceImpl::CreateForFrame( - RenderFrameHost* frame, + RenderFrameHost* render_frame_host, mojo::PendingReceiver<blink::mojom::DirectSocketsService> receiver) { DCHECK_CURRENTLY_ON(BrowserThread::UI); - if (!IsFrameSufficientlyIsolated(frame)) { + if (!render_frame_host->IsFeatureEnabled( + blink::mojom::PermissionsPolicyFeature::kDirectSockets)) { + mojo::ReportBadMessage( + "Permissions policy blocks access to Direct Sockets."); + } + if (!IsFrameSufficientlyIsolated(render_frame_host)) { + mojo::ReportBadMessage( + "Frame is not sufficiently isolated to use Direct Sockets."); return; } - mojo::MakeSelfOwnedReceiver( - base::WrapUnique(new DirectSocketsServiceImpl(frame)), - std::move(receiver)); + new DirectSocketsServiceImpl(render_frame_host, std::move(receiver)); } content::DirectSocketsDelegate* DirectSocketsServiceImpl::GetDelegate() { @@ -140,37 +146,26 @@ mojo::PendingReceiver<network::mojom::TCPConnectedSocket> receiver, mojo::PendingRemote<network::mojom::SocketObserver> observer, OpenTcpSocketCallback callback) { - if (!IsFrameSufficientlyIsolated(frame_host_)) { - mojo::ReportBadMessage("Insufficient isolation to open socket"); - return; - } - - if (!GetNetworkContext()) { - mojo::ReportBadMessage("Invalid request to open socket"); - return; - } + const std::string remote_host = options->remote_hostname; + const uint16_t remote_port = options->remote_port; if (auto* delegate = GetDelegate(); - delegate && - !delegate->ValidateAddressAndPort( - frame_host_, options->remote_hostname, options->remote_port, - blink::mojom::DirectSocketProtocolType::kTcp)) { + delegate && !delegate->ValidateAddressAndPort( + &render_frame_host(), remote_host, remote_port, + blink::mojom::DirectSocketProtocolType::kTcp)) { std::move(callback).Run(net::ERR_ACCESS_DENIED, absl::nullopt, absl::nullopt, mojo::ScopedDataPipeConsumerHandle(), mojo::ScopedDataPipeProducerHandle()); return; } - const std::string remote_host = options->remote_hostname; - const uint16_t remote_port = options->remote_port; - - auto weak_ptr = weak_ptr_factory_.GetWeakPtr(); ResolveHostAndOpenSocket::Create( - weak_ptr, remote_host, remote_port, + remote_host, remote_port, base::BindOnce(&DirectSocketsServiceImpl::OnResolveCompleteForTcpSocket, - weak_ptr, std::move(options), std::move(receiver), - std::move(observer), std::move(callback))) - ->Start(); + weak_ptr_factory_.GetWeakPtr(), std::move(options), + std::move(receiver), std::move(observer), + std::move(callback))) + ->Start(GetNetworkContext()); } void DirectSocketsServiceImpl::OpenUdpSocket( @@ -178,42 +173,25 @@ mojo::PendingReceiver<blink::mojom::DirectUDPSocket> receiver, mojo::PendingRemote<network::mojom::UDPSocketListener> listener, OpenUdpSocketCallback callback) { - if (!IsFrameSufficientlyIsolated(frame_host_)) { - mojo::ReportBadMessage("Insufficient isolation to open socket"); - return; - } - - if (!GetNetworkContext()) { - mojo::ReportBadMessage("Invalid request to open socket"); - return; - } + const std::string remote_host = options->remote_hostname; + const uint16_t remote_port = options->remote_port; if (auto* delegate = GetDelegate(); - delegate && - !delegate->ValidateAddressAndPort( - frame_host_, options->remote_hostname, options->remote_port, - blink::mojom::DirectSocketProtocolType::kUdp)) { + delegate && !delegate->ValidateAddressAndPort( + &render_frame_host(), remote_host, remote_port, + blink::mojom::DirectSocketProtocolType::kUdp)) { std::move(callback).Run(net::ERR_ACCESS_DENIED, absl::nullopt, absl::nullopt); return; } - const std::string remote_host = options->remote_hostname; - const uint16_t remote_port = options->remote_port; - - auto weak_ptr = weak_ptr_factory_.GetWeakPtr(); ResolveHostAndOpenSocket::Create( - weak_ptr, remote_host, remote_port, + remote_host, remote_port, base::BindOnce(&DirectSocketsServiceImpl::OnResolveCompleteForUdpSocket, - weak_ptr, std::move(options), std::move(receiver), - std::move(listener), std::move(callback))) - ->Start(); -} - -// static -net::MutableNetworkTrafficAnnotationTag -DirectSocketsServiceImpl::MutableTrafficAnnotation() { - return net::MutableNetworkTrafficAnnotationTag{TrafficAnnotation()}; + weak_ptr_factory_.GetWeakPtr(), std::move(options), + std::move(receiver), std::move(listener), + std::move(callback))) + ->Start(GetNetworkContext()); } // static @@ -227,38 +205,12 @@ GetNetworkContextForTesting() = network_context; } -void DirectSocketsServiceImpl::RenderFrameDeleted( - RenderFrameHost* render_frame_host) { - if (render_frame_host == frame_host_) { - frame_host_ = nullptr; - } -} - -void DirectSocketsServiceImpl::RenderFrameHostChanged( - RenderFrameHost* old_host, - RenderFrameHost* new_host) { - if (old_host == frame_host_) { - frame_host_ = nullptr; - } -} - -void DirectSocketsServiceImpl::WebContentsDestroyed() { - frame_host_ = nullptr; -} - network::mojom::NetworkContext* DirectSocketsServiceImpl::GetNetworkContext() const { - if (GetNetworkContextForTesting()) - return GetNetworkContextForTesting(); - - if (!frame_host_) - return nullptr; - - return frame_host_->GetStoragePartition()->GetNetworkContext(); -} - -RenderFrameHost* DirectSocketsServiceImpl::GetFrameHost() const { - return frame_host_; + if (auto* network_context = GetNetworkContextForTesting()) { + return network_context; + } + return render_frame_host().GetStoragePartition()->GetNetworkContext(); } void DirectSocketsServiceImpl::OnResolveCompleteForTcpSocket( @@ -275,19 +227,15 @@ return; } - auto* network_context = GetNetworkContext(); - if (!network_context) { - return; - } - DCHECK(resolved_addresses && !resolved_addresses->empty()); absl::optional<net::IPEndPoint> local_addr = GetLocalAddress(*options); - network_context->CreateTCPConnectedSocket( + GetNetworkContext()->CreateTCPConnectedSocket( std::move(local_addr), *resolved_addresses, CreateTCPConnectedSocketOptions(std::move(options)), - DirectSocketsServiceImpl::MutableTrafficAnnotation(), std::move(socket), - std::move(observer), std::move(callback)); + net::MutableNetworkTrafficAnnotationTag{ + DirectSocketsServiceImpl::TrafficAnnotation()}, + std::move(socket), std::move(observer), std::move(callback)); } void DirectSocketsServiceImpl::OnResolveCompleteForUdpSocket( @@ -303,16 +251,11 @@ return; } - auto* network_context = GetNetworkContext(); - if (!network_context) { - return; - } - DCHECK(resolved_addresses && !resolved_addresses->empty()); net::IPEndPoint peer_addr = resolved_addresses->front(); auto direct_udp_socket = std::make_unique<DirectUDPSocketImpl>( - network_context, std::move(listener)); + GetNetworkContext(), std::move(listener)); direct_udp_socket->Connect( resolved_addresses->front(), CreateUDPSocketOptions(std::move(options)),
diff --git a/content/browser/direct_sockets/direct_sockets_service_impl.h b/content/browser/direct_sockets/direct_sockets_service_impl.h index 9c2837a..6d2fad8 100644 --- a/content/browser/direct_sockets/direct_sockets_service_impl.h +++ b/content/browser/direct_sockets/direct_sockets_service_impl.h
@@ -8,9 +8,8 @@ #include "base/memory/raw_ptr.h" #include "base/memory/weak_ptr.h" #include "content/common/content_export.h" -#include "content/public/browser/direct_sockets_delegate.h" +#include "content/public/browser/document_service.h" #include "content/public/browser/render_frame_host.h" -#include "content/public/browser/web_contents_observer.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "mojo/public/cpp/bindings/pending_remote.h" #include "mojo/public/cpp/bindings/unique_receiver_set.h" @@ -24,10 +23,11 @@ namespace content { +class DirectSocketsDelegate; + // Implementation of the DirectSocketsService Mojo service. class CONTENT_EXPORT DirectSocketsServiceImpl - : public blink::mojom::DirectSocketsService, - public WebContentsObserver { + : public DocumentService<blink::mojom::DirectSocketsService> { public: ~DirectSocketsServiceImpl() override; @@ -49,23 +49,17 @@ mojo::PendingRemote<network::mojom::UDPSocketListener> listener, OpenUdpSocketCallback callback) override; - // WebContentsObserver: - void RenderFrameDeleted(RenderFrameHost* render_frame_host) override; - void RenderFrameHostChanged(RenderFrameHost* old_host, - RenderFrameHost* new_host) override; - void WebContentsDestroyed() override; - - network::mojom::NetworkContext* GetNetworkContext() const; - RenderFrameHost* GetFrameHost() const; - - static net::MutableNetworkTrafficAnnotationTag MutableTrafficAnnotation(); static net::NetworkTrafficAnnotationTag TrafficAnnotation(); // Testing: static void SetNetworkContextForTesting(network::mojom::NetworkContext*); private: - explicit DirectSocketsServiceImpl(RenderFrameHost*); + DirectSocketsServiceImpl( + RenderFrameHost*, + mojo::PendingReceiver<blink::mojom::DirectSocketsService> receiver); + + network::mojom::NetworkContext* GetNetworkContext() const; void OnResolveCompleteForTcpSocket( blink::mojom::DirectSocketOptionsPtr, @@ -83,7 +77,6 @@ int result, const absl::optional<net::AddressList>& resolved_addresses); - raw_ptr<RenderFrameHost> frame_host_; mojo::UniqueReceiverSet<blink::mojom::DirectUDPSocket> direct_udp_socket_receivers_;
diff --git a/content/browser/direct_sockets/direct_udp_socket_impl.cc b/content/browser/direct_sockets/direct_udp_socket_impl.cc index f19414a..f975091 100644 --- a/content/browser/direct_sockets/direct_udp_socket_impl.cc +++ b/content/browser/direct_sockets/direct_udp_socket_impl.cc
@@ -5,6 +5,7 @@ #include "content/browser/direct_sockets/direct_udp_socket_impl.h" #include "content/browser/direct_sockets/direct_sockets_service_impl.h" +#include "net/traffic_annotation/network_traffic_annotation.h" namespace content { @@ -40,7 +41,8 @@ return; } remote_->Send(std::move(data), - DirectSocketsServiceImpl::MutableTrafficAnnotation(), + net::MutableNetworkTrafficAnnotationTag{ + DirectSocketsServiceImpl::TrafficAnnotation()}, std::move(callback)); }
diff --git a/content/browser/direct_sockets/resolve_host_and_open_socket.cc b/content/browser/direct_sockets/resolve_host_and_open_socket.cc index bb23d60..7d9908e 100644 --- a/content/browser/direct_sockets/resolve_host_and_open_socket.cc +++ b/content/browser/direct_sockets/resolve_host_and_open_socket.cc
@@ -5,24 +5,15 @@ #include "content/browser/direct_sockets/resolve_host_and_open_socket.h" #include "base/functional/bind.h" -#include "base/memory/weak_ptr.h" -#include "base/metrics/histogram_functions.h" #include "build/build_config.h" -#include "content/browser/direct_sockets/direct_sockets_service_impl.h" -#include "content/public/browser/direct_sockets_delegate.h" #include "content/public/browser/render_frame_host.h" #include "net/base/address_list.h" -#include "net/base/ip_address.h" -#include "net/base/ip_endpoint.h" #include "net/net_buildflags.h" #include "services/network/public/mojom/network_context.mojom.h" namespace content { namespace { -constexpr char kPermissionDeniedHistogramName[] = - "DirectSockets.PermissionDeniedFailures"; - #if BUILDFLAG(ENABLE_MDNS) bool ResemblesMulticastDNSName(const std::string& hostname) { return base::EndsWith(hostname, ".local") || @@ -30,64 +21,37 @@ } #endif // !BUILDFLAG(ENABLE_MDNS) -bool IsRawIPAddress(const std::string& address) { - net::IPAddress ip; - return ip.AssignFromIPLiteral(address); -} - -bool ContainNonPubliclyRoutableAddress(const net::AddressList& addresses) { - DCHECK(!addresses.empty()); - return !base::ranges::all_of(addresses, &net::IPAddress::IsPubliclyRoutable, - &net::IPEndPoint::address); -} - -RenderFrameHost* GetFrameHostFromService( - base::WeakPtr<DirectSocketsServiceImpl> service) { - if (!service) { - return nullptr; - } - return service->GetFrameHost(); -} - } // namespace -ResolveHostAndOpenSocket::ResolveHostAndOpenSocket( - base::WeakPtr<DirectSocketsServiceImpl> service, - const std::string& host, - uint16_t port, - OpenSocketCallback callback) - : service_(service), - host_(host), - port_(port), - callback_(std::move(callback)) {} +ResolveHostAndOpenSocket::ResolveHostAndOpenSocket(const std::string& host, + uint16_t port, + OpenSocketCallback callback) + : host_(host), port_(port), callback_(std::move(callback)) {} ResolveHostAndOpenSocket::~ResolveHostAndOpenSocket() = default; // static ResolveHostAndOpenSocket* ResolveHostAndOpenSocket::Create( - base::WeakPtr<DirectSocketsServiceImpl> service, - const std::string& host, + const std::string& address, uint16_t port, OpenSocketCallback callback) { - return new ResolveHostAndOpenSocket(std::move(service), host, port, - std::move(callback)); + return new ResolveHostAndOpenSocket(address, port, std::move(callback)); } -void ResolveHostAndOpenSocket::Start() { - auto* network_context = service_->GetNetworkContext(); - DCHECK(network_context); +void ResolveHostAndOpenSocket::Start( + network::mojom::NetworkContext* network_context) { DCHECK(!receiver_.is_bound()); DCHECK(!resolver_.is_bound()); - network_context->CreateHostResolver(/*config_overrides=*/absl::nullopt, - resolver_.BindNewPipeAndPassReceiver()); + network_context->CreateHostResolver( + /*config_overrides=*/absl::nullopt, + resolver_.BindNewPipeAndPassReceiver()); network::mojom::ResolveHostParametersPtr parameters = network::mojom::ResolveHostParameters::New(); #if BUILDFLAG(ENABLE_MDNS) if (ResemblesMulticastDNSName(host_)) { parameters->source = net::HostResolverSource::MULTICAST_DNS; - is_mdns_name_ = true; } #endif // !BUILDFLAG(ENABLE_MDNS) // Intentionally using a HostPortPair because scheme isn't specified. @@ -111,36 +75,8 @@ endpoint_results_with_metadata) { DCHECK(receiver_.is_bound()); receiver_.reset(); + resolver_.reset(); - auto* frame = GetFrameHostFromService(service_); - if (!frame) { - OpenSocket(net::ERR_UNEXPECTED, {}); - return; - } - - if (auto* delegate = DirectSocketsServiceImpl::GetDelegate(); - delegate && delegate->ShouldSkipPostResolveChecks(frame)) { - OpenSocket(result, resolved_addresses); - return; - } - - // Reject hostnames that resolve to non-public exception unless a raw IP - // address or a *.local hostname is entered by the user. - if (!IsRawIPAddress(host_) && !is_mdns_name_ && resolved_addresses && - ContainNonPubliclyRoutableAddress(*resolved_addresses)) { - base::UmaHistogramEnumeration( - kPermissionDeniedHistogramName, - blink::mojom::DirectSocketFailureType::kResolvingToNonPublic); - OpenSocket(net::ERR_NETWORK_ACCESS_DENIED, {}); - return; - } - - OpenSocket(result, resolved_addresses); -} - -void ResolveHostAndOpenSocket::OpenSocket( - int result, - const absl::optional<net::AddressList>& resolved_addresses) { std::move(callback_).Run(result, resolved_addresses); delete this; }
diff --git a/content/browser/direct_sockets/resolve_host_and_open_socket.h b/content/browser/direct_sockets/resolve_host_and_open_socket.h index 0d778ce..7f4b414 100644 --- a/content/browser/direct_sockets/resolve_host_and_open_socket.h +++ b/content/browser/direct_sockets/resolve_host_and_open_socket.h
@@ -5,7 +5,6 @@ #ifndef CONTENT_BROWSER_DIRECT_SOCKETS_RESOLVE_HOST_AND_OPEN_SOCKET_H_ #define CONTENT_BROWSER_DIRECT_SOCKETS_RESOLVE_HOST_AND_OPEN_SOCKET_H_ -#include "base/memory/weak_ptr.h" #include "content/common/content_export.h" #include "mojo/public/cpp/bindings/receiver.h" #include "mojo/public/cpp/bindings/remote.h" @@ -13,12 +12,14 @@ #include "net/dns/public/host_resolver_results.h" #include "services/network/public/cpp/resolve_host_client_base.h" +namespace network::mojom { +class NetworkContext; +} // namespace network::mojom + namespace content { -class DirectSocketsServiceImpl; - -// Resolves the host/port pair provided in options. Upon completion invokes the -// supplied callback and deletes |this|. +// Resolves the host/port pair provided on creation. After resolver signals +// completion via OnComplete(), fires the supplied callback and deletes itself. class CONTENT_EXPORT ResolveHostAndOpenSocket : public network::ResolveHostClientBase { public: @@ -28,17 +29,14 @@ ~ResolveHostAndOpenSocket() override; - static ResolveHostAndOpenSocket* Create( - base::WeakPtr<DirectSocketsServiceImpl>, - const std::string& host, - uint16_t port, - OpenSocketCallback); + static ResolveHostAndOpenSocket* Create(const std::string& host, + uint16_t port, + OpenSocketCallback); - void Start(); + void Start(network::mojom::NetworkContext*); private: - ResolveHostAndOpenSocket(base::WeakPtr<DirectSocketsServiceImpl>, - const std::string& host, + ResolveHostAndOpenSocket(const std::string& host, uint16_t port, OpenSocketCallback); @@ -48,18 +46,11 @@ const absl::optional<net::HostResolverEndpointResults>& endpoint_results_with_metadata) override; - void OpenSocket(int result, - const absl::optional<net::AddressList>& resolved_addresses); - - base::WeakPtr<DirectSocketsServiceImpl> service_; - const std::string host_; uint16_t port_; OpenSocketCallback callback_; - bool is_mdns_name_ = false; - mojo::Receiver<network::mojom::ResolveHostClient> receiver_{this}; mojo::Remote<network::mojom::HostResolver> resolver_; };
diff --git a/content/browser/first_party_sets/first_party_sets_handler_impl.cc b/content/browser/first_party_sets/first_party_sets_handler_impl.cc index ee262ed..5568647 100644 --- a/content/browser/first_party_sets/first_party_sets_handler_impl.cc +++ b/content/browser/first_party_sets/first_party_sets_handler_impl.cc
@@ -222,12 +222,6 @@ db_helper_.Reset(); } -void FirstPartySetsHandlerImpl::SetGlobalSetsForTesting( - net::GlobalFirstPartySets global_sets) { - DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); - global_sets_ = std::move(global_sets); -} - void FirstPartySetsHandlerImpl::GetPersistedGlobalSetsForTesting( const std::string& browser_context_id, base::OnceCallback<void(absl::optional<net::GlobalFirstPartySets>)>
diff --git a/content/browser/first_party_sets/first_party_sets_handler_impl.h b/content/browser/first_party_sets/first_party_sets_handler_impl.h index 5ea1af2..bc69f225 100644 --- a/content/browser/first_party_sets/first_party_sets_handler_impl.h +++ b/content/browser/first_party_sets/first_party_sets_handler_impl.h
@@ -111,7 +111,6 @@ void SetPublicFirstPartySets(const base::Version& version, base::File sets_file) override; void ResetForTesting() override; - void SetGlobalSetsForTesting(net::GlobalFirstPartySets global_sets) override; absl::optional<net::FirstPartySetEntry> FindEntry( const net::SchemefulSite& site, const net::FirstPartySetsContextConfig& config) const override;
diff --git a/content/browser/interest_group/interest_group_auction.cc b/content/browser/interest_group/interest_group_auction.cc index 29c2078..fc279464 100644 --- a/content/browser/interest_group/interest_group_auction.cc +++ b/content/browser/interest_group/interest_group_auction.cc
@@ -1091,7 +1091,7 @@ DCHECK(!bidding_and_scoring_phase_callback_); DCHECK(!reporting_phase_callback_); DCHECK(!final_auction_result_); - DCHECK(!top_bid_); + DCHECK(!auction_leader_.top_bid); DCHECK_EQ(pending_component_seller_worklet_requests_, 0u); TRACE_EVENT_NESTABLE_ASYNC_BEGIN0("fledge", "bidding_and_scoring_phase", @@ -1139,16 +1139,17 @@ DCHECK(!bidding_and_scoring_phase_callback_); DCHECK(!reporting_phase_callback_); DCHECK(!final_auction_result_); - DCHECK(top_bid_); + DCHECK(auction_leader_.top_bid); // This should only be called on top-level auctions. DCHECK(!parent_); TRACE_EVENT_NESTABLE_ASYNC_BEGIN0("fledge", "reporting_phase", trace_id_); InterestGroupAuctionReporter::WinningBidInfo winning_bid_info; - winning_bid_info.storage_interest_group = &top_bid_->bid->bid_state->bidder; - winning_bid_info.render_url = top_bid_->bid->render_url; - winning_bid_info.ad_components = top_bid_->bid->ad_components; + winning_bid_info.storage_interest_group = + &auction_leader_.top_bid->bid->bid_state->bidder; + winning_bid_info.render_url = auction_leader_.top_bid->bid->render_url; + winning_bid_info.ad_components = auction_leader_.top_bid->bid->ad_components; // Need the bid from the bidder itself. If the bid was from a component // auction, then `top_bid_->bid` will be the bid from the component auction, // which the component seller worklet may have modified, and thus the wrong @@ -1157,22 +1158,23 @@ // the bid was from the top-level auction, and the original top bid from the // component auction, otherwise, so will always be the bid returned by the // winning bidder's generateBid() method. - winning_bid_info.bid = top_bid_->bid->auction->top_bid()->bid->bid; - winning_bid_info.bid_duration = top_bid_->bid->bid_duration; + winning_bid_info.bid = + auction_leader_.top_bid->bid->auction->top_bid()->bid->bid; + winning_bid_info.bid_duration = auction_leader_.top_bid->bid->bid_duration; winning_bid_info.bidding_signals_data_version = - top_bid_->bid->bidding_signals_data_version; + auction_leader_.top_bid->bid->bidding_signals_data_version; InterestGroupAuctionReporter::SellerWinningBidInfo top_level_seller_winning_bid_info; top_level_seller_winning_bid_info.auction_config = config_; - top_level_seller_winning_bid_info.bid = top_bid_->bid->bid; - top_level_seller_winning_bid_info.score = top_bid_->score; + top_level_seller_winning_bid_info.bid = auction_leader_.top_bid->bid->bid; + top_level_seller_winning_bid_info.score = auction_leader_.top_bid->score; top_level_seller_winning_bid_info.highest_scoring_other_bid = - highest_scoring_other_bid_; + auction_leader_.highest_scoring_other_bid; top_level_seller_winning_bid_info.highest_scoring_other_bid_owner = - highest_scoring_other_bid_owner_; + auction_leader_.highest_scoring_other_bid_owner; top_level_seller_winning_bid_info.scoring_signals_data_version = - top_bid_->scoring_signals_data_version; + auction_leader_.top_bid->scoring_signals_data_version; top_level_seller_winning_bid_info.trace_id = trace_id_; // Populate the SellerWinningBidInfo for the component auction that the @@ -1187,25 +1189,27 @@ // SellerWinningBidInfos. absl::optional<InterestGroupAuctionReporter::SellerWinningBidInfo> component_seller_winning_bid_info; - if (top_bid_->bid->auction != this) { - const InterestGroupAuction* component_auction = top_bid_->bid->auction; + if (auction_leader_.top_bid->bid->auction != this) { + const InterestGroupAuction* component_auction = + auction_leader_.top_bid->bid->auction; component_seller_winning_bid_info.emplace(); component_seller_winning_bid_info->auction_config = component_auction->config_; component_seller_winning_bid_info->bid = - component_auction->top_bid_->bid->bid; + component_auction->auction_leader_.top_bid->bid->bid; component_seller_winning_bid_info->score = - component_auction->top_bid_->score; + component_auction->auction_leader_.top_bid->score; component_seller_winning_bid_info->highest_scoring_other_bid = - component_auction->highest_scoring_other_bid_; + component_auction->auction_leader_.highest_scoring_other_bid; component_seller_winning_bid_info->highest_scoring_other_bid_owner = - component_auction->highest_scoring_other_bid_owner_; + component_auction->auction_leader_.highest_scoring_other_bid_owner; component_seller_winning_bid_info->scoring_signals_data_version = - component_auction->top_bid_->scoring_signals_data_version; + component_auction->auction_leader_.top_bid + ->scoring_signals_data_version; component_seller_winning_bid_info->trace_id = component_auction->trace_id_; component_seller_winning_bid_info->component_auction_modified_bid_params = - component_auction->top_bid_->component_auction_modified_bid_params - ->Clone(); + component_auction->auction_leader_.top_bid + ->component_auction_modified_bid_params->Clone(); } reporting_phase_callback_ = std::move(reporting_phase_callback); @@ -1369,14 +1373,15 @@ // `debug_win_report_urls`. BidState* winner = nullptr; if (final_auction_result_ == AuctionResult::kSuccess && - top_bid_->bid->auction == this) { - winner = top_bid_->bid->bid_state; + auction_leader_.top_bid->bid->auction == this) { + winner = auction_leader_.top_bid->bid->bid_state; } // `signals` includes post auction signals from current auction. PostAuctionSignals signals; - signals.winning_bid = top_bid_ ? top_bid_->bid->bid : 0.0; - signals.highest_scoring_other_bid = highest_scoring_other_bid_; + signals.winning_bid = + auction_leader_.top_bid ? auction_leader_.top_bid->bid->bid : 0.0; + signals.highest_scoring_other_bid = auction_leader_.highest_scoring_other_bid; // `top_level_signals` includes post auction signals from top-level auction. // Will only will be used in debug report URLs of top-level seller and // component sellers. @@ -1386,27 +1391,31 @@ if (parent_) { top_level_signals = PostAuctionSignals(); top_level_signals->winning_bid = - parent_->top_bid_ ? parent_->top_bid_->bid->bid : 0.0; + parent_->auction_leader_.top_bid + ? parent_->auction_leader_.top_bid->bid->bid + : 0.0; } - if (!top_bid_) { - DCHECK_EQ(highest_scoring_other_bid_, 0); - DCHECK(!highest_scoring_other_bid_owner_.has_value()); + if (!auction_leader_.top_bid) { + DCHECK_EQ(auction_leader_.highest_scoring_other_bid, 0); + DCHECK(!auction_leader_.highest_scoring_other_bid_owner.has_value()); } for (const auto& buyer_helper : buyer_helpers_) { const url::Origin& owner = buyer_helper->owner(); - if (top_bid_) - signals.made_winning_bid = owner == top_bid_->bid->interest_group->owner; - - if (highest_scoring_other_bid_owner_.has_value()) { - DCHECK_GT(highest_scoring_other_bid_, 0); - signals.made_highest_scoring_other_bid = - owner == highest_scoring_other_bid_owner_.value(); + if (auction_leader_.top_bid) { + signals.made_winning_bid = + owner == auction_leader_.top_bid->bid->interest_group->owner; } - if (parent_ && parent_->top_bid_) { + + if (auction_leader_.highest_scoring_other_bid_owner.has_value()) { + DCHECK_GT(auction_leader_.highest_scoring_other_bid, 0); + signals.made_highest_scoring_other_bid = + owner == auction_leader_.highest_scoring_other_bid_owner.value(); + } + if (parent_ && parent_->auction_leader_.top_bid) { top_level_signals->made_winning_bid = - owner == parent_->top_bid_->bid->interest_group->owner; + owner == parent_->auction_leader_.top_bid->bid->interest_group->owner; } buyer_helper->TakeDebugReportUrls(winner, signals, top_level_signals, @@ -1463,8 +1472,8 @@ InterestGroupAuction::ScoredBid* InterestGroupAuction::top_bid() { DCHECK(all_bids_scored_); - DCHECK(top_bid_); - return top_bid_.get(); + DCHECK(auction_leader_.top_bid); + return auction_leader_.top_bid.get(); } absl::optional<uint16_t> InterestGroupAuction::GetBuyerExperimentId( @@ -1489,6 +1498,9 @@ return absl::nullopt; } +InterestGroupAuction::LeaderInfo::LeaderInfo() = default; +InterestGroupAuction::LeaderInfo::~LeaderInfo() = default; + void InterestGroupAuction::OnInterestGroupRead( std::vector<StorageInterestGroup> interest_groups) { ++num_owners_loaded_; @@ -1848,8 +1860,8 @@ errors_.insert(errors_.end(), errors.begin(), errors.end()); - // Use separate fields for component and top-level seller reports, so both can - // send debug reports. + // Use separate fields for component and top-level seller reports, so both + // can send debug reports. if (bid->auction == this) { bid->bid_state->seller_debug_loss_report_url = std::move(debug_loss_report_url); @@ -1866,88 +1878,101 @@ } // A score <= 0 means the seller rejected the bid. - if (score <= 0) { - // Need to delete `bid` because OnBiddingAndScoringComplete() may delete - // this, which leaves danging pointers on the stack. While this is safe to - // do (nothing has access to `bid` to dereference them), it makes the - // dangling pointer tooling sad. - bid.reset(); - MaybeCompleteBiddingAndScoringPhase(); - return; + if (score > 0) { + UpdateAuctionLeaders(std::move(bid), score, + std::move(component_auction_modified_bid_params), + data_version, has_data_version, auction_leader_); } + // Need to delete `bid` because OnBiddingAndScoringComplete() may delete + // this, which leaves danging pointers on the stack. While this is safe to + // do (nothing has access to `bid` to dereference them), it makes the + // dangling pointer tooling sad. + bid.reset(); + MaybeCompleteBiddingAndScoringPhase(); +} + +void InterestGroupAuction::UpdateAuctionLeaders( + std::unique_ptr<Bid> bid, + double score, + auction_worklet::mojom::ComponentAuctionModifiedBidParamsPtr + component_auction_modified_bid_params, + uint32_t data_version, + bool has_data_version, + LeaderInfo& leader_info) { bool is_top_bid = false; const url::Origin& owner = bid->interest_group->owner; - if (!top_bid_ || score > top_bid_->score) { + if (!leader_info.top_bid || score > leader_info.top_bid->score) { // If there's no previous top bidder, or the bidder has the highest score, // need to replace the previous top bidder. is_top_bid = true; - if (top_bid_) { - OnNewHighestScoringOtherBid(top_bid_->score, top_bid_->bid->bid, - &top_bid_->bid->interest_group->owner); + if (leader_info.top_bid) { + OnNewHighestScoringOtherBid( + leader_info.top_bid->score, leader_info.top_bid->bid->bid, + &leader_info.top_bid->bid->interest_group->owner, leader_info); } - num_top_bids_ = 1; - at_most_one_top_bid_owner_ = true; - } else if (score == top_bid_->score) { + leader_info.num_top_bids = 1; + leader_info.at_most_one_top_bid_owner = true; + } else if (score == leader_info.top_bid->score) { // If there's a tie, replace the top-bidder with 1-in-`num_top_bids_` // chance. This is the select random value from a stream with fixed // storage problem. - ++num_top_bids_; - if (1 == base::RandInt(1, num_top_bids_)) + ++leader_info.num_top_bids; + if (1 == base::RandInt(1, leader_info.num_top_bids)) is_top_bid = true; - if (owner != top_bid_->bid->interest_group->owner) - at_most_one_top_bid_owner_ = false; + if (owner != leader_info.top_bid->bid->interest_group->owner) + leader_info.at_most_one_top_bid_owner = false; // If the top bid is being replaced, need to add the old top bid as a second // highest bid. Otherwise, need to add the current bid as a second highest // bid. double new_highest_scoring_other_bid = - is_top_bid ? top_bid_->bid->bid : bid->bid; - OnNewHighestScoringOtherBid( - score, new_highest_scoring_other_bid, - at_most_one_top_bid_owner_ ? &bid->interest_group->owner : nullptr); - } else if (score >= second_highest_score_) { + is_top_bid ? leader_info.top_bid->bid->bid : bid->bid; + OnNewHighestScoringOtherBid(score, new_highest_scoring_other_bid, + leader_info.at_most_one_top_bid_owner + ? &bid->interest_group->owner + : nullptr, + leader_info); + } else if (score >= leader_info.second_highest_score) { // Also use this bid (the most recent one) as highest scoring other bid if // there's a tie for second highest score. - OnNewHighestScoringOtherBid(score, bid->bid, &owner); + OnNewHighestScoringOtherBid(score, bid->bid, &owner, leader_info); } if (is_top_bid) { - top_bid_ = std::make_unique<ScoredBid>( + leader_info.top_bid = std::make_unique<ScoredBid>( score, has_data_version ? data_version : absl::optional<uint32_t>(), std::move(bid), std::move(component_auction_modified_bid_params)); } - - bid.reset(); - MaybeCompleteBiddingAndScoringPhase(); } void InterestGroupAuction::OnNewHighestScoringOtherBid( double score, double bid_value, - const url::Origin* owner) { + const url::Origin* owner, + LeaderInfo& leader_info) { // Current (the most recent) bid becomes highest scoring other bid. - if (score > second_highest_score_) { - highest_scoring_other_bid_ = bid_value; - num_second_highest_bids_ = 1; + if (score > leader_info.second_highest_score) { + leader_info.highest_scoring_other_bid = bid_value; + leader_info.num_second_highest_bids = 1; // Owner may be false if this is one of the bids tied for first place. if (!owner) { - highest_scoring_other_bid_owner_.reset(); + leader_info.highest_scoring_other_bid_owner.reset(); } else { - highest_scoring_other_bid_owner_ = *owner; + leader_info.highest_scoring_other_bid_owner = *owner; } - second_highest_score_ = score; + leader_info.second_highest_score = score; return; } - DCHECK_EQ(score, second_highest_score_); - if (!owner || *owner != highest_scoring_other_bid_owner_) - highest_scoring_other_bid_owner_.reset(); - ++num_second_highest_bids_; + DCHECK_EQ(score, leader_info.second_highest_score); + if (!owner || *owner != leader_info.highest_scoring_other_bid_owner) + leader_info.highest_scoring_other_bid_owner.reset(); + ++leader_info.num_second_highest_bids; // In case of a tie, randomly pick one. This is the select random value from a // stream with fixed storage problem. - if (1 == base::RandInt(1, num_second_highest_bids_)) - highest_scoring_other_bid_ = bid_value; + if (1 == base::RandInt(1, leader_info.num_second_highest_bids)) + leader_info.highest_scoring_other_bid = bid_value; } absl::optional<base::TimeDelta> InterestGroupAuction::PerBuyerTimeout( @@ -1983,7 +2008,7 @@ // If there's no winning bid, fail with kAllBidsRejected if there were any // bids. Otherwise, fail with kNoBids. - if (!top_bid_) { + if (!auction_leader_.top_bid) { if (any_bid_made_) { OnBiddingAndScoringComplete(AuctionResult::kAllBidsRejected); } else { @@ -2036,8 +2061,10 @@ for (auto& component_auction : component_auctions_) { // Leave the state of the winning component auction alone, if the winning // bid is from a component auction. - if (top_bid_ && top_bid_->bid->auction == component_auction.get()) + if (auction_leader_.top_bid && + auction_leader_.top_bid->bid->auction == component_auction.get()) { continue; + } if (component_auction->final_auction_result_) continue; component_auction->final_auction_result_ = @@ -2069,8 +2096,10 @@ // TODO(mmenke): Extract relevant data from `this` when creating the Reporter, // and have it handle reporting only if auction results are loaded in a frame, // or if there's no result. - if (top_bid_) - top_bid_->bid->auction->final_auction_result_ = AuctionResult::kSuccess; + if (auction_leader_.top_bid) { + auction_leader_.top_bid->bid->auction->final_auction_result_ = + AuctionResult::kSuccess; + } // Close all pipes, as they're no longer needed. ClosePipes();
diff --git a/content/browser/interest_group/interest_group_auction.h b/content/browser/interest_group/interest_group_auction.h index cc91bd24..7096451 100644 --- a/content/browser/interest_group/interest_group_auction.h +++ b/content/browser/interest_group/interest_group_auction.h
@@ -467,6 +467,36 @@ // interest groups of a particular buyer, etc). class BuyerHelper; + struct LeaderInfo { + LeaderInfo(); + ~LeaderInfo(); + + LeaderInfo(const LeaderInfo&) = delete; + LeaderInfo& operator=(LeaderInfo&) = delete; + + // The highest scoring bid so far. Null if no bid has been accepted yet. + std::unique_ptr<ScoredBid> top_bid; + // Number of bidders with the same score as `top_bidder`. + size_t num_top_bids = 0; + // Number of bidders with the same score as `second_highest_score`. If the + // second highest score matches the highest score, this does not include the + // top bid. + size_t num_second_highest_bids = 0; + + // The numeric value of the bid that got the second highest score. When + // there's a tie for the second highest score, one of the second highest + // scoring bids is randomly chosen. + double highest_scoring_other_bid = 0.0; + double second_highest_score = 0.0; + // Whether all bids of the highest score are from the same interest group + // owner. + bool at_most_one_top_bid_owner = true; + // Will be null in the end if there are interest groups having the second + // highest score with different owners. That includes the top bid itself, in + // the case there's a tie for the top bid. + absl::optional<url::Origin> highest_scoring_other_bid_owner; + }; + // --------------------------------- // Load interest group phase methods // --------------------------------- @@ -567,12 +597,24 @@ PrivateAggregationRequests pa_requests, const std::vector<std::string>& errors) override; + // Compares `bid` with current auction leaders in `leader_info`, updating + // `leader_info` if needed. + void UpdateAuctionLeaders( + std::unique_ptr<Bid> bid, + double score, + auction_worklet::mojom::ComponentAuctionModifiedBidParamsPtr + component_auction_modified_bid_params, + uint32_t data_version, + bool has_data_version, + LeaderInfo& leader_info); + // Invoked when the bid becomes the new highest scoring other bid, to handle // calculation of post auction signals. `owner` is nullptr in the event the // bid is tied with the top bid, and they have different origins. void OnNewHighestScoringOtherBid(double score, double bid_value, - const url::Origin* owner); + const url::Origin* owner, + LeaderInfo& leader_info); absl::optional<base::TimeDelta> PerBuyerTimeout(const BidState* state); absl::optional<base::TimeDelta> SellerTimeout(); @@ -727,27 +769,7 @@ std::vector<std::pair<blink::InterestGroupKey, double>> post_auction_priority_updates_; - // The highest scoring bid so far. Null if no bid has been accepted yet. - std::unique_ptr<ScoredBid> top_bid_; - // Number of bidders with the same score as `top_bidder`. - size_t num_top_bids_ = 0; - // Number of bidders with the same score as `second_highest_score_`. If the - // second highest score matches the highest score, this does not include the - // top bid. - size_t num_second_highest_bids_ = 0; - - // The numeric value of the bid that got the second highest score. When - // there's a tie for the second highest score, one of the second highest - // scoring bids is randomly chosen. - double highest_scoring_other_bid_ = 0.0; - double second_highest_score_ = 0.0; - // Whether all bids of the highest score are from the same interest group - // owner. - bool at_most_one_top_bid_owner_ = true; - // Will be null in the end if there are interest groups having the second - // highest score with different owners. That includes the top bid itself, in - // the case there's a tie for the top bid. - absl::optional<url::Origin> highest_scoring_other_bid_owner_; + LeaderInfo auction_leader_; // Holds a reference to the SellerWorklet used by the auction. std::unique_ptr<AuctionWorkletManager::WorkletHandle> seller_worklet_handle_;
diff --git a/content/browser/renderer_host/navigation_request.cc b/content/browser/renderer_host/navigation_request.cc index 14dfb45..25a9534 100644 --- a/content/browser/renderer_host/navigation_request.cc +++ b/content/browser/renderer_host/navigation_request.cc
@@ -3495,8 +3495,8 @@ // if two or more such documents share a site/origin. Using // navigation_id_ means that each new NavigationRequest (and thus each // document) will get a different value. - if (features::kIsolateSandboxedIframesGroupingParam.Get() == - features::IsolateSandboxedIframesGrouping::kPerDocument) { + if (blink::features::kIsolateSandboxedIframesGroupingParam.Get() == + blink::features::IsolateSandboxedIframesGrouping::kPerDocument) { url_info_init.WithUniqueSandboxId(navigation_id_); } }
diff --git a/content/browser/site_info.cc b/content/browser/site_info.cc index 1314683..ff5cf87b 100644 --- a/content/browser/site_info.cc +++ b/content/browser/site_info.cc
@@ -24,6 +24,7 @@ #include "content/public/common/content_switches.h" #include "content/public/common/url_constants.h" #include "net/base/registry_controlled_domains/registry_controlled_domain.h" +#include "third_party/blink/public/common/features.h" namespace content { @@ -845,8 +846,8 @@ // origin, as we should be using the full origin for the SiteInstance, but // we don't need to track the origin like we do for OriginAgentCluster. if (real_url_info.is_sandboxed && - features::kIsolateSandboxedIframesGroupingParam.Get() == - features::IsolateSandboxedIframesGrouping::kPerOrigin) { + blink::features::kIsolateSandboxedIframesGroupingParam.Get() == + blink::features::IsolateSandboxedIframesGrouping::kPerOrigin) { return origin.GetURL(); }
diff --git a/content/browser/site_instance_impl.cc b/content/browser/site_instance_impl.cc index f591ec3e..3c57b5c 100644 --- a/content/browser/site_instance_impl.cc +++ b/content/browser/site_instance_impl.cc
@@ -31,6 +31,7 @@ #include "content/public/common/url_utils.h" #include "net/base/registry_controlled_domains/registry_controlled_domain.h" #include "third_party/blink/public/common/chrome_debug_urls.h" +#include "third_party/blink/public/common/features.h" #include "third_party/perfetto/include/perfetto/tracing/traced_value.h" #include "url/origin.h" @@ -1117,8 +1118,8 @@ // known to have the same value. if (SiteIsolationPolicy::IsStrictOriginIsolationEnabled() || (real_src_url_info.is_sandboxed && - features::kIsolateSandboxedIframesGroupingParam.Get() == - features::IsolateSandboxedIframesGrouping::kPerOrigin)) { + blink::features::kIsolateSandboxedIframesGroupingParam.Get() == + blink::features::IsolateSandboxedIframesGrouping::kPerOrigin)) { return src_origin == dest_origin; }
diff --git a/content/browser/site_per_process_oopsif_browsertest.cc b/content/browser/site_per_process_oopsif_browsertest.cc index 9196ca1de..e901391 100644 --- a/content/browser/site_per_process_oopsif_browsertest.cc +++ b/content/browser/site_per_process_oopsif_browsertest.cc
@@ -12,6 +12,7 @@ #include "content/public/test/test_frame_navigation_observer.h" #include "content/test/render_document_feature.h" #include "net/dns/mock_host_resolver.h" +#include "third_party/blink/public/common/features.h" namespace content { @@ -24,8 +25,8 @@ public ::testing::WithParamInterface<bool> { public: SrcdocIsolatedSandboxedIframeTest() { - feature_list_.InitWithFeatureState(features::kIsolateSandboxedIframes, - GetParam()); + feature_list_.InitWithFeatureState( + blink::features::kIsolateSandboxedIframes, GetParam()); } void SetUpOnMainThread() override { @@ -50,7 +51,8 @@ : public SitePerProcessBrowserTest { public: SitePerProcessIsolatedSandboxedIframeTest() { - feature_list_.InitAndEnableFeature(features::kIsolateSandboxedIframes); + feature_list_.InitAndEnableFeature( + blink::features::kIsolateSandboxedIframes); } private: @@ -61,7 +63,8 @@ : public SitePerProcessBrowserTest { public: SitePerProcessNotIsolatedSandboxedIframeTest() { - feature_list_.InitAndDisableFeature(features::kIsolateSandboxedIframes); + feature_list_.InitAndDisableFeature( + blink::features::kIsolateSandboxedIframes); } private: @@ -75,7 +78,8 @@ public: SitePerProcessPerOriginIsolatedSandboxedIframeTest() { feature_list_.InitWithFeaturesAndParameters( - {{features::kIsolateSandboxedIframes, {{"grouping", "per-origin"}}}}, + {{blink::features::kIsolateSandboxedIframes, + {{"grouping", "per-origin"}}}}, {/* disabled_features */}); } @@ -87,7 +91,8 @@ : public SitePerProcessIsolatedSandboxedIframeTest { public: SitePerProcessIsolatedSandboxWithoutStrictSiteIsolationBrowserTest() { - feature_list_.InitAndEnableFeature(features::kIsolateSandboxedIframes); + feature_list_.InitAndEnableFeature( + blink::features::kIsolateSandboxedIframes); } void SetUpCommandLine(base::CommandLine* command_line) override { @@ -150,7 +155,8 @@ public: SitePerProcessPerDocumentIsolatedSandboxedIframeTest() { feature_list_.InitWithFeaturesAndParameters( - {{features::kIsolateSandboxedIframes, {{"grouping", "per-document"}}}}, + {{blink::features::kIsolateSandboxedIframes, + {{"grouping", "per-document"}}}}, {/* disabled_features */}); }
diff --git a/content/browser/storage_partition_impl.cc b/content/browser/storage_partition_impl.cc index fe77d59..f0048908 100644 --- a/content/browser/storage_partition_impl.cc +++ b/content/browser/storage_partition_impl.cc
@@ -2613,7 +2613,8 @@ attribution_manager->ClearData( begin, end, generic_filter, filter_builder, remove_mask_ & REMOVE_DATA_MASK_ATTRIBUTION_REPORTING_INTERNAL, - CreateTaskCompletionClosure(TracingDataType::kConversions)); + mojo::WrapCallbackWithDefaultInvokeIfNotRun( + CreateTaskCompletionClosure(TracingDataType::kConversions))); } else if (storage_key.IsFirstPartyContext()) { // Attribution Reporting API doesn't support cross-site data deletion. std::unique_ptr<BrowsingDataFilterBuilder> effective_filter_builder = @@ -2623,7 +2624,8 @@ attribution_manager->ClearData( begin, end, generic_filter, effective_filter_builder.get(), remove_mask_ & REMOVE_DATA_MASK_ATTRIBUTION_REPORTING_INTERNAL, - CreateTaskCompletionClosure(TracingDataType::kConversions)); + mojo::WrapCallbackWithDefaultInvokeIfNotRun( + CreateTaskCompletionClosure(TracingDataType::kConversions))); } } @@ -2638,7 +2640,8 @@ // `CookiesTreeModel`. aggregation_service->ClearData( begin, end, generic_filter, - CreateTaskCompletionClosure(TracingDataType::kAggregationService)); + mojo::WrapCallbackWithDefaultInvokeIfNotRun( + CreateTaskCompletionClosure(TracingDataType::kAggregationService))); } if (private_aggregation_manager &&
diff --git a/content/browser/web_contents/web_contents_impl_browsertest.cc b/content/browser/web_contents/web_contents_impl_browsertest.cc index 860e005..d8cd60b 100644 --- a/content/browser/web_contents/web_contents_impl_browsertest.cc +++ b/content/browser/web_contents/web_contents_impl_browsertest.cc
@@ -18,6 +18,7 @@ #include "base/memory/raw_ptr.h" #include "base/memory/ref_counted.h" #include "base/path_service.h" +#include "base/ranges/algorithm.h" #include "base/run_loop.h" #include "base/strings/pattern.h" #include "base/strings/stringprintf.h" @@ -1004,8 +1005,8 @@ const std::vector<double>& progresses = delegate->progresses; // All updates should be in order ... - if (std::adjacent_find(progresses.begin(), progresses.end(), - std::greater<>()) != progresses.end()) { + if (base::ranges::adjacent_find(progresses, std::greater<>()) != + progresses.end()) { ADD_FAILURE() << "Progress values should be in order: " << ::testing::PrintToString(progresses); } @@ -1025,8 +1026,8 @@ const std::vector<double>& progresses = delegate->progresses; // All updates should be in order ... - if (std::adjacent_find(progresses.begin(), progresses.end(), - std::greater<>()) != progresses.end()) { + if (base::ranges::adjacent_find(progresses, std::greater<>()) != + progresses.end()) { ADD_FAILURE() << "Progress values should be in order: " << ::testing::PrintToString(progresses); }
diff --git a/content/public/browser/direct_sockets_delegate.h b/content/public/browser/direct_sockets_delegate.h index 3d4424e..ee608cb 100644 --- a/content/public/browser/direct_sockets_delegate.h +++ b/content/public/browser/direct_sockets_delegate.h
@@ -28,9 +28,6 @@ const std::string& address, uint16_t port, blink::mojom::DirectSocketProtocolType) const = 0; - - // If yes, skips post-resolve checks for Direct TCP/UDP sockets. - virtual bool ShouldSkipPostResolveChecks(content::RenderFrameHost*) const = 0; }; } // namespace content
diff --git a/content/public/browser/first_party_sets_handler.h b/content/public/browser/first_party_sets_handler.h index 15a5206..3cdcbe4 100644 --- a/content/public/browser/first_party_sets_handler.h +++ b/content/public/browser/first_party_sets_handler.h
@@ -19,7 +19,6 @@ class FirstPartySetEntry; class FirstPartySetsCacheFilter; class FirstPartySetsContextConfig; -class GlobalFirstPartySets; class SchemefulSite; } @@ -140,10 +139,6 @@ // Resets the state on the instance for testing. virtual void ResetForTesting() = 0; - // Allows tests to override the post-initalization global First-Party Sets. - virtual void SetGlobalSetsForTesting( - net::GlobalFirstPartySets global_sets) = 0; - // Looks up `site` in the global First-Party Sets and `config` to find its // associated FirstPartySetEntry. //
diff --git a/content/public/browser/site_isolation_policy.cc b/content/public/browser/site_isolation_policy.cc index 16f994d..35725be 100644 --- a/content/public/browser/site_isolation_policy.cc +++ b/content/public/browser/site_isolation_policy.cc
@@ -25,6 +25,7 @@ #include "content/public/common/content_client.h" #include "content/public/common/content_features.h" #include "content/public/common/content_switches.h" +#include "third_party/blink/public/common/features.h" #include "url/origin.h" namespace content { @@ -141,7 +142,8 @@ // static bool SiteIsolationPolicy::AreIsolatedSandboxedIframesEnabled() { return !IsSiteIsolationDisabled(SiteIsolationMode::kPartialSiteIsolation) && - base::FeatureList::IsEnabled(features::kIsolateSandboxedIframes); + base::FeatureList::IsEnabled( + blink::features::kIsolateSandboxedIframes); } // static
diff --git a/content/public/common/content_features.cc b/content/public/common/content_features.cc index 5b754837..0ca55f1 100644 --- a/content/public/common/content_features.cc +++ b/content/public/common/content_features.cc
@@ -524,24 +524,6 @@ base::FEATURE_DISABLED_BY_DEFAULT); const char kIsolateOriginsFieldTrialParamName[] = "OriginsList"; -// Allow process isolation of iframes with the 'sandbox' attribute set. Whether -// or not such an iframe will be isolated may depend on options specified with -// the attribute. Note: At present, only iframes with origin-restricted -// sandboxes are isolated. -BASE_FEATURE(kIsolateSandboxedIframes, - "IsolateSandboxedIframes", - base::FEATURE_DISABLED_BY_DEFAULT); -const base::FeatureParam<IsolateSandboxedIframesGrouping>::Option - isolated_sandboxed_iframes_grouping_types[] = { - {IsolateSandboxedIframesGrouping::kPerSite, "per-site"}, - {IsolateSandboxedIframesGrouping::kPerOrigin, "per-origin"}, - {IsolateSandboxedIframesGrouping::kPerDocument, "per-document"}}; -const base::FeatureParam<IsolateSandboxedIframesGrouping> - kIsolateSandboxedIframesGroupingParam{ - &kIsolateSandboxedIframes, "grouping", - IsolateSandboxedIframesGrouping::kPerSite, - &isolated_sandboxed_iframes_grouping_types}; - // Enables the TC39 Array grouping proposal. BASE_FEATURE(kJavaScriptArrayGrouping, "JavaScriptArrayGrouping",
diff --git a/content/public/common/content_features.h b/content/public/common/content_features.h index 1e543fa..e10cdf16 100644 --- a/content/public/common/content_features.h +++ b/content/public/common/content_features.h
@@ -110,22 +110,6 @@ CONTENT_EXPORT BASE_DECLARE_FEATURE(kIsolateFencedFrames); CONTENT_EXPORT BASE_DECLARE_FEATURE(kIsolateOrigins); CONTENT_EXPORT extern const char kIsolateOriginsFieldTrialParamName[]; -CONTENT_EXPORT BASE_DECLARE_FEATURE(kIsolateSandboxedIframes); -enum class IsolateSandboxedIframesGrouping { - // In this grouping, all isolated sandboxed iframes whose URLs share the same - // site in a given BrowsingInstance will share a process. - kPerSite, - // In this grouping, all isolated sandboxed iframes from a given - // BrowsingInstance whose URLs share the same origin will be isolated in an - // origin-keyed process. - kPerOrigin, - // Unlike the other two modes, which group sandboxed frames per-site or - // per-origin, this one doesn't do any grouping at all and uses one process - // per document. - kPerDocument, -}; -CONTENT_EXPORT extern const base::FeatureParam<IsolateSandboxedIframesGrouping> - kIsolateSandboxedIframesGroupingParam; CONTENT_EXPORT BASE_DECLARE_FEATURE(kJavaScriptArrayGrouping); CONTENT_EXPORT BASE_DECLARE_FEATURE(kJavaScriptExperimentalSharedMemory); CONTENT_EXPORT BASE_DECLARE_FEATURE(kLazyFrameLoading);
diff --git a/content/renderer/pepper/ppb_var_deprecated_impl.cc b/content/renderer/pepper/ppb_var_deprecated_impl.cc index c106f9c..5becf078 100644 --- a/content/renderer/pepper/ppb_var_deprecated_impl.cc +++ b/content/renderer/pepper/ppb_var_deprecated_impl.cc
@@ -94,17 +94,14 @@ PepperTryCatchVar try_catch(accessor.instance(), accessor.converter(), exception); + v8::Local<v8::Context> context = try_catch.GetContext(); v8::MicrotasksScope microtasks_scope( - accessor.GetObject()->GetIsolate(), + accessor.GetObject()->GetIsolate(), context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Value> v8_name = try_catch.ToV8(name); if (try_catch.HasException()) return false; - v8::Local<v8::Context> context = try_catch.GetContext(); - if (try_catch.HasException()) - return false; - bool result = false; if (!accessor.GetObject()->Has(context, v8_name).To(&result)) { try_catch.HasException(); @@ -121,17 +118,14 @@ PepperTryCatchVar try_catch(accessor.instance(), accessor.converter(), exception); + v8::Local<v8::Context> context = try_catch.GetContext(); v8::MicrotasksScope microtasks_scope( - accessor.GetObject()->GetIsolate(), + accessor.GetObject()->GetIsolate(), context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Value> v8_name = try_catch.ToV8(name); if (try_catch.HasException()) return false; - v8::Local<v8::Context> context = try_catch.GetContext(); - if (try_catch.HasException()) - return false; - bool has_name = false; if (!accessor.GetObject()->Has(context, v8_name).To(&has_name)) { try_catch.HasException(); @@ -153,16 +147,17 @@ PepperTryCatchVar try_catch(accessor.instance(), accessor.converter(), exception); + v8::Local<v8::Context> context = try_catch.GetContext(); v8::MicrotasksScope microtasks_scope( - accessor.GetObject()->GetIsolate(), + accessor.GetObject()->GetIsolate(), context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Value> v8_name = try_catch.ToV8(name); if (try_catch.HasException()) return PP_MakeUndefined(); v8::Local<v8::Value> result; - ScopedPPVar result_var = try_catch.FromV8Maybe( - accessor.GetObject()->Get(try_catch.GetContext(), v8_name)); + ScopedPPVar result_var = + try_catch.FromV8Maybe(accessor.GetObject()->Get(context, v8_name)); if (try_catch.HasException()) return PP_MakeUndefined(); @@ -179,14 +174,14 @@ PepperTryCatchVar try_catch(accessor.instance(), accessor.converter(), exception); + v8::Local<v8::Context> context = try_catch.GetContext(); v8::MicrotasksScope microtasks_scope( - accessor.GetObject()->GetIsolate(), + accessor.GetObject()->GetIsolate(), context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); *properties = nullptr; *property_count = 0; - v8::Local<v8::Context> context = try_catch.GetContext(); v8::Local<v8::Array> identifiers; if (!accessor.GetObject()->GetPropertyNames(context).ToLocal(&identifiers)) return; @@ -215,8 +210,9 @@ PepperTryCatchVar try_catch(accessor.instance(), accessor.converter(), exception); + v8::Local<v8::Context> context = try_catch.GetContext(); v8::MicrotasksScope microtasks_scope( - accessor.GetObject()->GetIsolate(), + accessor.GetObject()->GetIsolate(), context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Value> v8_name = try_catch.ToV8(name); v8::Local<v8::Value> v8_value = try_catch.ToV8(value); @@ -238,18 +234,15 @@ PepperTryCatchVar try_catch(accessor.instance(), accessor.converter(), exception); + v8::Local<v8::Context> context = try_catch.GetContext(); v8::MicrotasksScope microtasks_scope( - accessor.GetObject()->GetIsolate(), + accessor.GetObject()->GetIsolate(), context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Value> v8_name = try_catch.ToV8(name); if (try_catch.HasException()) return; - v8::Local<v8::Context> context = try_catch.GetContext(); - if (try_catch.HasException()) - return; - if (accessor.GetObject()->Delete(context, v8_name).IsNothing()) { // Ensure exception object is created if V8 has thrown. try_catch.HasException(); @@ -277,8 +270,9 @@ PepperTryCatchVar try_catch(accessor.instance(), accessor.converter(), exception); + v8::Local<v8::Context> context = try_catch.GetContext(); v8::MicrotasksScope microtasks_scope( - accessor.GetObject()->GetIsolate(), + accessor.GetObject()->GetIsolate(), context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Value> v8_method_name = try_catch.ToV8(scoped_name.get()); if (try_catch.HasException()) @@ -290,7 +284,6 @@ } v8::Local<v8::Object> function = accessor.GetObject(); - v8::Local<v8::Context> context = accessor.instance()->GetMainWorldContext(); v8::Local<v8::Object> recv = context->Global(); if (v8_method_name.As<v8::String>()->Length() != 0) { v8::Local<v8::Value> value;
diff --git a/content/renderer/pepper/v8_var_converter.cc b/content/renderer/pepper/v8_var_converter.cc index e02ccd3..cac746dc 100644 --- a/content/renderer/pepper/v8_var_converter.cc +++ b/content/renderer/pepper/v8_var_converter.cc
@@ -330,7 +330,8 @@ v8::Isolate* isolate = context->GetIsolate(); v8::EscapableHandleScope handle_scope(isolate); v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); VarHandleMap visited_ids; ParentVarSet parent_ids; @@ -486,7 +487,8 @@ v8::Isolate* isolate = context->GetIsolate(); v8::HandleScope handle_scope(isolate); v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); HandleVarMap visited_handles; ParentHandleSet parent_handles;
diff --git a/content/renderer/render_frame_impl.cc b/content/renderer/render_frame_impl.cc index d7f19a4e..2e135b98 100644 --- a/content/renderer/render_frame_impl.cc +++ b/content/renderer/render_frame_impl.cc
@@ -4,7 +4,6 @@ #include "content/renderer/render_frame_impl.h" -#include <algorithm> #include <map> #include <memory> #include <string> @@ -34,6 +33,7 @@ #include "base/metrics/histogram_macros.h" #include "base/observer_list.h" #include "base/process/process.h" +#include "base/ranges/algorithm.h" #include "base/run_loop.h" #include "base/strings/strcat.h" #include "base/strings/string_piece.h" @@ -726,8 +726,7 @@ DCHECK(std::is_sorted(params_.digests_of_uris_to_skip.begin(), params_.digests_of_uris_to_skip.end())); // URLs are not duplicated. - DCHECK(std::adjacent_find(params_.digests_of_uris_to_skip.begin(), - params_.digests_of_uris_to_skip.end()) == + DCHECK(base::ranges::adjacent_find(params_.digests_of_uris_to_skip) == params_.digests_of_uris_to_skip.end()); } @@ -3819,9 +3818,6 @@ } void RenderFrameImpl::DidClearWindowObject() { - v8::MicrotasksScope microtasks(blink::MainThreadIsolate(), - v8::MicrotasksScope::kDoNotRunMicrotasks); - if (enabled_bindings_ & BINDINGS_POLICY_WEB_UI) WebUIExtension::Install(frame_); @@ -4339,6 +4335,7 @@ void RenderFrameImpl::DidCreateScriptContext(v8::Local<v8::Context> context, int world_id) { v8::MicrotasksScope microtasks(blink::MainThreadIsolate(), + context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); if (((enabled_bindings_ & BINDINGS_POLICY_MOJO_WEB_UI) || enable_mojo_js_bindings_) &&
diff --git a/content/test/data/attribution_reporting/interop/aggregatable_dedup_key.json b/content/test/data/attribution_reporting/interop/aggregatable_dedup_key.json new file mode 100644 index 0000000..df7579f --- /dev/null +++ b/content/test/data/attribution_reporting/interop/aggregatable_dedup_key.json
@@ -0,0 +1,171 @@ +{ + "description": "2nd trigger with the same deduplication key is not attributed", + "input": { + "sources": [ + { + "timestamp": "1643235573000", + "registration_request": { + "source_origin": "https://source.test", + "attribution_src_url": "https://reporter.test/register-source", + "source_type": "navigation" + }, + "responses": [{ + "url": "https://reporter.test/register-source", + "response": { + "Attribution-Reporting-Register-Source": { + "destination": "https://destination.test", + "source_event_id": "123", + "aggregation_keys": { + "a": "0x159" + } + } + } + }] + } + ], + "triggers": [ + // Should result in an event-level report and an aggregatable report. + { + "timestamp": "1643235574000", + "registration_request": { + "attribution_src_url": "https://reporter.test/register-trigger", + "destination_origin": "https://destination.test" + }, + "responses": [{ + "url": "https://reporter.test/register-trigger", + "response": { + "Attribution-Reporting-Register-Trigger": { + "event_trigger_data": [ + { + "trigger_data": "1", + "deduplication_key": "1" + } + ], + "aggregatable_trigger_data": [ + { + "source_keys": ["a"], + "key_piece": "0x400" + } + ], + "aggregatable_values": { + "a": 123 + }, + "aggregatable_deduplication_key": "1" + } + } + }] + }, + // Should result in an event-level report and the aggregatable report + // will be deduplicated. + { + "timestamp": "1643235575000", + "registration_request": { + "attribution_src_url": "https://reporter.test/register-trigger", + "destination_origin": "https://destination.test" + }, + "responses": [{ + "url": "https://reporter.test/register-trigger", + "response": { + "Attribution-Reporting-Register-Trigger": { + "event_trigger_data": [ + { + "trigger_data": "2" + } + ], + "aggregatable_trigger_data": [ + { + "source_keys": ["a"], + "key_piece": "0x400" + } + ], + "aggregatable_values": { + "a": 456 + }, + "aggregatable_deduplication_key": "1" + } + } + }] + }, + // Should result in an aggregatable report as the deduplication key is + // different. + { + "timestamp": "1643235576000", + "registration_request": { + "attribution_src_url": "https://reporter.test/register-trigger", + "destination_origin": "https://destination.test" + }, + "responses": [{ + "url": "https://reporter.test/register-trigger", + "response": { + "Attribution-Reporting-Register-Trigger": { + "aggregatable_trigger_data": [ + { + "source_keys": ["a"], + "key_piece": "0x400" + } + ], + "aggregatable_values": { + "a": 456 + }, + "aggregatable_deduplication_key": "2" + } + } + }] + } + ] + }, + "output": { + "event_level_results": [ + { + "payload": { + "attribution_destination": "https://destination.test", + "randomized_trigger_rate": 0.0024, + "source_event_id": "123", + "source_type": "navigation", + "trigger_data": "1" + }, + "report_url": "https://reporter.test/.well-known/attribution-reporting/report-event-attribution", + "report_time": "1643408373000" + }, + { + "payload": { + "attribution_destination": "https://destination.test", + "randomized_trigger_rate": 0.0024, + "source_event_id": "123", + "source_type": "navigation", + "trigger_data": "2" + }, + "report_url": "https://reporter.test/.well-known/attribution-reporting/report-event-attribution", + "report_time": "1643408373000" + } + ], + "aggregatable_results": [ + { + "payload": { + "attribution_destination": "https://destination.test", + "histograms": [ + { + "key": "0x559", + "value": 123 + } + ] + }, + "report_url": "https://reporter.test/.well-known/attribution-reporting/report-aggregate-attribution", + "report_time": "1643239174000" + }, + { + "payload": { + "attribution_destination": "https://destination.test", + "histograms": [ + { + "key": "0x559", + "value": 456 + } + ] + }, + "report_url": "https://reporter.test/.well-known/attribution-reporting/report-aggregate-attribution", + "report_time": "1643239176000" + } + ] + } +}
diff --git a/content/test/gpu/gpu_tests/test_expectations/webgl2_conformance_expectations.txt b/content/test/gpu/gpu_tests/test_expectations/webgl2_conformance_expectations.txt index 0b96724..0d1d2cd 100644 --- a/content/test/gpu/gpu_tests/test_expectations/webgl2_conformance_expectations.txt +++ b/content/test/gpu/gpu_tests/test_expectations/webgl2_conformance_expectations.txt
@@ -907,9 +907,21 @@ crbug.com/1284804 [ chromeos chromeos-board-kevin passthrough ] deqp/functional/gles3/transformfeedback/random_separate_triangles.html [ Failure ] crbug.com/1285109 [ chromeos chromeos-board-kevin passthrough ] conformance/canvas/render-after-resize-test.html [ Failure ] crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance/context/context-hidden-alpha.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance/textures/misc/copy-tex-image-2d-formats.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance/ogles/GL/array/array_001_to_006.html [ Failure ] crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance/ogles/GL/functions/functions_033_to_040.html [ Failure ] crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance/ogles/GL/refract/refract_001_to_006.html [ Failure ] crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance/renderbuffers/renderbuffer-initialization.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance/renderbuffers/stencil-renderbuffer-initialization.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance2/renderbuffers/multisampled-depth-renderbuffer-initialization.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance2/rendering/rasterizer-discard-and-implicit-clear.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance2/textures/image_bitmap_from_blob/tex-3d-rgb16f-rgb-float.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance2/textures/image_bitmap_from_blob/tex-2d-rgb8ui-rgb_integer-unsigned_byte.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] conformance2/textures/video/tex-3d-rgba8ui-rgba_integer-unsigned_byte.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] deqp/functional/gles3/fbocolorbuffer/tex2d_03.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] deqp/functional/gles3/fbocolorbuffer/tex3d_00.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] deqp/functional/gles3/framebufferblit/conversion_26.html [ Failure ] +crbug.com/1374285 [ chromeos chromeos-board-kevin passthrough ] deqp/functional/gles3/textureshadow/2d_array_nearest_always.html [ Failure ] crbug.com/1285111 [ chromeos chromeos-board-kevin passthrough ] conformance/textures/misc/copy-tex-image-and-sub-image-2d.html [ Failure ] crbug.com/1285112 [ chromeos chromeos-board-kevin passthrough ] conformance2/rendering/blitframebuffer-multisampled-readbuffer.html [ Failure ] crbug.com/1285114 [ chromeos chromeos-board-kevin passthrough ] conformance2/transform_feedback/transform_feedback.html [ Failure ]
diff --git a/content/web_test/renderer/gc_controller.cc b/content/web_test/renderer/gc_controller.cc index c9b46e2..5743ad7 100644 --- a/content/web_test/renderer/gc_controller.cc +++ b/content/web_test/renderer/gc_controller.cc
@@ -101,7 +101,8 @@ v8::Context::Scope context_scope(context); v8::TryCatch try_catch(isolate); v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); auto result = func->Call(context, context->Global(), 0, nullptr); // Swallow potential exception. std::ignore = result;
diff --git a/content/web_test/renderer/web_frame_test_proxy.cc b/content/web_test/renderer/web_frame_test_proxy.cc index c95fcc33a..0dd00b8 100644 --- a/content/web_test/renderer/web_frame_test_proxy.cc +++ b/content/web_test/renderer/web_frame_test_proxy.cc
@@ -744,8 +744,6 @@ // Avoid installing bindings on the about:blank in between tests. This is // especially problematic for web platform tests that would inject javascript // into the page when installing bindings. - v8::MicrotasksScope microtask_scope(blink::MainThreadIsolate(), - v8::MicrotasksScope::kDoNotRunMicrotasks); if (test_runner()->TestIsRunning()) { blink::WebLocalFrame* frame = GetWebFrame(); // These calls will install the various JS bindings for web tests into the
diff --git a/extensions/browser/api/storage/settings_test_util.cc b/extensions/browser/api/storage/settings_test_util.cc index 61e18d57..15a16b9 100644 --- a/extensions/browser/api/storage/settings_test_util.cc +++ b/extensions/browser/api/storage/settings_test_util.cc
@@ -72,15 +72,14 @@ const std::string& id, Manifest::Type type, const std::set<std::string>& permissions_set) { - base::DictionaryValue manifest; - manifest.SetStringKey("name", std::string("Test extension ") + id); - manifest.SetStringKey("version", "1.0"); - manifest.SetIntKey("manifest_version", 2); + base::Value::Dict manifest; + manifest.Set("name", std::string("Test extension ") + id); + manifest.Set("version", "1.0"); + manifest.Set("manifest_version", 2); - std::unique_ptr<base::ListValue> permissions(new base::ListValue()); - for (auto it = permissions_set.cbegin(); it != permissions_set.cend(); ++it) { - permissions->Append(*it); - } + base::Value::List permissions; + for (const auto& perm : permissions_set) + permissions.Append(perm); manifest.Set("permissions", std::move(permissions)); switch (type) { @@ -88,10 +87,10 @@ break; case Manifest::TYPE_LEGACY_PACKAGED_APP: { - auto app = std::make_unique<base::DictionaryValue>(); - auto app_launch = std::make_unique<base::DictionaryValue>(); - app_launch->SetStringKey("local_path", "fake.html"); - app->Set("launch", std::move(app_launch)); + base::Value::Dict app; + base::Value::Dict app_launch; + app_launch.Set("local_path", "fake.html"); + app.Set("launch", std::move(app_launch)); manifest.Set("app", std::move(app)); break; }
diff --git a/extensions/browser/requirements_checker_unittest.cc b/extensions/browser/requirements_checker_unittest.cc index ddcf5ea..a454de32 100644 --- a/extensions/browser/requirements_checker_unittest.cc +++ b/extensions/browser/requirements_checker_unittest.cc
@@ -47,21 +47,18 @@ class RequirementsCheckerTest : public ExtensionsTest { public: - RequirementsCheckerTest() { - manifest_dict_ = std::make_unique<base::DictionaryValue>(); - } - - ~RequirementsCheckerTest() override {} + RequirementsCheckerTest() = default; + ~RequirementsCheckerTest() override = default; void CreateExtension() { - manifest_dict_->SetStringKey("name", "dummy name"); - manifest_dict_->SetStringKey("version", "1"); - manifest_dict_->SetIntKey("manifest_version", 2); + manifest_dict_.Set("name", "dummy name"); + manifest_dict_.Set("version", "1"); + manifest_dict_.Set("manifest_version", 2); std::string error; extension_ = Extension::Create(base::FilePath(), mojom::ManifestLocation::kUnpacked, - *manifest_dict_, Extension::NO_FLAGS, &error); + manifest_dict_, Extension::NO_FLAGS, &error); ASSERT_TRUE(extension_.get()) << error; } @@ -74,15 +71,14 @@ } void RequireWindowShape() { - manifest_dict_->SetBoolPath("requirements.window.shape", true); + manifest_dict_.SetByDottedPath("requirements.window.shape", true); } void RequireFeature(const char feature[]) { - if (!manifest_dict_->FindKey(kFeaturesKey)) - manifest_dict_->Set(kFeaturesKey, std::make_unique<base::ListValue>()); - base::ListValue* features_list = nullptr; - ASSERT_TRUE(manifest_dict_->GetList(kFeaturesKey, &features_list)); - features_list->Append(feature); + base::Value* features_list = manifest_dict_.Find(kFeaturesKey); + if (!features_list) + features_list = manifest_dict_.Set(kFeaturesKey, base::Value::List()); + features_list->GetList().Append(feature); } std::unique_ptr<RequirementsChecker> checker_; @@ -90,7 +86,7 @@ private: scoped_refptr<Extension> extension_; - std::unique_ptr<base::DictionaryValue> manifest_dict_; + base::Value::Dict manifest_dict_; }; // Tests no requirements.
diff --git a/extensions/common/extension.cc b/extensions/common/extension.cc index 33cd6309..13c985e8 100644 --- a/extensions/common/extension.cc +++ b/extensions/common/extension.cc
@@ -141,12 +141,12 @@ // Computes the |extension_id| from the given parameters. On success, returns // true. On failure, populates |error| and returns false. -bool ComputeExtensionID(const base::DictionaryValue& manifest, +bool ComputeExtensionID(const base::DictAdapterForMigration& manifest, const base::FilePath& path, int creation_flags, std::u16string* error, ExtensionId* extension_id) { - if (const base::Value* public_key = manifest.FindKey(keys::kPublicKey)) { + if (const base::Value* public_key = manifest.Find(keys::kPublicKey)) { std::string public_key_bytes; if (!public_key->is_string() || !Extension::ParsePEMKeyBytes(public_key->GetString(), @@ -222,11 +222,12 @@ } // static -scoped_refptr<Extension> Extension::Create(const base::FilePath& path, - ManifestLocation location, - const base::DictionaryValue& value, - int flags, - std::string* utf8_error) { +scoped_refptr<Extension> Extension::Create( + const base::FilePath& path, + ManifestLocation location, + const base::DictAdapterForMigration& value, + int flags, + std::string* utf8_error) { return Extension::Create(path, location, value, @@ -237,12 +238,13 @@ // TODO(sungguk): Continue removing std::string errors and replacing // with std::u16string. See http://crbug.com/71980. -scoped_refptr<Extension> Extension::Create(const base::FilePath& path, - ManifestLocation location, - const base::DictionaryValue& value, - int flags, - const std::string& explicit_id, - std::string* utf8_error) { +scoped_refptr<Extension> Extension::Create( + const base::FilePath& path, + ManifestLocation location, + const base::DictAdapterForMigration& value, + int flags, + const std::string& explicit_id, + std::string* utf8_error) { base::ElapsedTimer timer; DCHECK(utf8_error); std::u16string error; @@ -262,18 +264,14 @@ } std::unique_ptr<extensions::Manifest> manifest; + auto value_clone = base::DictionaryValue::From( + base::Value::ToUniquePtrValue(base::Value(value.Clone()))); if (flags & FOR_LOGIN_SCREEN) { manifest = Manifest::CreateManifestForLoginScreen( - location, - base::DictionaryValue::From( - base::Value::ToUniquePtrValue(value.Clone())), - std::move(extension_id)); + location, std::move(value_clone), std::move(extension_id)); } else { - manifest = std::make_unique<Manifest>( - location, - base::DictionaryValue::From( - base::Value::ToUniquePtrValue(value.Clone())), - std::move(extension_id)); + manifest = std::make_unique<Manifest>(location, std::move(value_clone), + std::move(extension_id)); } std::vector<InstallWarning> install_warnings;
diff --git a/extensions/common/extension.h b/extensions/common/extension.h index a184292..dde6bf6 100644 --- a/extensions/common/extension.h +++ b/extensions/common/extension.h
@@ -32,6 +32,7 @@ #endif namespace base { +class DictAdapterForMigration; class DictionaryValue; } @@ -155,20 +156,22 @@ Extension(const Extension&) = delete; Extension& operator=(const Extension&) = delete; - static scoped_refptr<Extension> Create(const base::FilePath& path, - mojom::ManifestLocation location, - const base::DictionaryValue& value, - int flags, - std::string* error); + static scoped_refptr<Extension> Create( + const base::FilePath& path, + mojom::ManifestLocation location, + const base::DictAdapterForMigration& value, + int flags, + std::string* error); // In a few special circumstances, we want to create an Extension and give it // an explicit id. Most consumers should just use the other Create() method. - static scoped_refptr<Extension> Create(const base::FilePath& path, - mojom::ManifestLocation location, - const base::DictionaryValue& value, - int flags, - const ExtensionId& explicit_id, - std::string* error); + static scoped_refptr<Extension> Create( + const base::FilePath& path, + mojom::ManifestLocation location, + const base::DictAdapterForMigration& value, + int flags, + const ExtensionId& explicit_id, + std::string* error); // Valid schemes for web extent URLPatterns. static const int kValidWebExtentSchemes;
diff --git a/extensions/renderer/binding_generating_native_handler.cc b/extensions/renderer/binding_generating_native_handler.cc index ce599f4..c9d27a040 100644 --- a/extensions/renderer/binding_generating_native_handler.cc +++ b/extensions/renderer/binding_generating_native_handler.cc
@@ -83,7 +83,8 @@ v8::Local<v8::Value> argv[] = {v8_api_name}; v8::Local<v8::Value> binding_instance_value; v8::MicrotasksScope microtasks_scope( - v8_context->GetIsolate(), v8::MicrotasksScope::kDoNotRunMicrotasks); + v8_context->GetIsolate(), v8_context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); // TODO(devlin): We should not be using v8::Function::Call() directly here. // Instead, we should use JSRunner once it's used outside native bindings. if (!create_binding->Call(v8_context, binding, std::size(argv), argv) @@ -108,7 +109,8 @@ v8::Local<v8::Value> compiled_schema; { v8::MicrotasksScope microtasks_scope( - v8_context->GetIsolate(), v8::MicrotasksScope::kDoNotRunMicrotasks); + v8_context->GetIsolate(), v8_context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); // TODO(devlin): We should not be using v8::Function::Call() directly here. // Instead, we should use JSRunner once it's used outside native bindings. if (!generate->Call(v8_context, binding_instance, 0, nullptr)
diff --git a/extensions/renderer/bindings/api_request_handler.cc b/extensions/renderer/bindings/api_request_handler.cc index 142a44f..a7afe4d 100644 --- a/extensions/renderer/bindings/api_request_handler.cc +++ b/extensions/renderer/bindings/api_request_handler.cc
@@ -275,7 +275,8 @@ v8::Isolate* isolate = context->GetIsolate(); v8::MicrotasksScope microtasks_scope( - isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); + isolate, context->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); if (error.empty()) { v8::Local<v8::Value> result;
diff --git a/extensions/renderer/extension_js_runner.cc b/extensions/renderer/extension_js_runner.cc index d15b259..5e3c4ab 100644 --- a/extensions/renderer/extension_js_runner.cc +++ b/extensions/renderer/extension_js_runner.cc
@@ -46,7 +46,7 @@ v8::Isolate* isolate = context->GetIsolate(); DCHECK(context == isolate->GetCurrentContext()); - v8::MicrotasksScope microtasks(isolate, + v8::MicrotasksScope microtasks(isolate, context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Object> global = context->Global();
diff --git a/extensions/renderer/safe_builtins.cc b/extensions/renderer/safe_builtins.cc index 22e89a5..fae59d8 100644 --- a/extensions/renderer/safe_builtins.cc +++ b/extensions/renderer/safe_builtins.cc
@@ -186,7 +186,9 @@ info[2]->IsObject() && // args info[3]->IsInt32() && // first_arg_index info[4]->IsInt32()); // args_length + v8::Local<v8::Context> context = info.GetIsolate()->GetCurrentContext(); v8::MicrotasksScope microtasks(info.GetIsolate(), + context->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Function> function = info[0].As<v8::Function>(); v8::Local<v8::Object> recv; @@ -207,7 +209,6 @@ int first_arg_index = info[3].As<v8::Int32>()->Value(); int args_length = info[4].As<v8::Int32>()->Value(); - v8::Local<v8::Context> context = info.GetIsolate()->GetCurrentContext(); int argc = args_length - first_arg_index; std::unique_ptr<v8::Local<v8::Value>[]> argv( new v8::Local<v8::Value>[argc]);
diff --git a/extensions/renderer/script_context.cc b/extensions/renderer/script_context.cc index d856f08..dcdd44a 100644 --- a/extensions/renderer/script_context.cc +++ b/extensions/renderer/script_context.cc
@@ -291,7 +291,7 @@ DCHECK(thread_checker_.CalledOnValidThread()); v8::HandleScope handle_scope(isolate()); v8::Context::Scope scope(v8_context()); - v8::MicrotasksScope microtasks(isolate(), + v8::MicrotasksScope microtasks(isolate(), v8_context()->GetMicrotaskQueue(), v8::MicrotasksScope::kDoNotRunMicrotasks); v8::Local<v8::Object> global = v8_context()->Global(); if (web_frame_) { @@ -550,8 +550,8 @@ return v8::Undefined(isolate()); } - v8::MicrotasksScope microtasks( - isolate(), v8::MicrotasksScope::kDoNotRunMicrotasks); + v8::MicrotasksScope microtasks(isolate(), v8_context()->GetMicrotaskQueue(), + v8::MicrotasksScope::kDoNotRunMicrotasks); v8::TryCatch try_catch(isolate()); try_catch.SetCaptureMessage(true); v8::ScriptOrigin origin(isolate(), v8_helpers::ToV8StringUnsafe(
diff --git a/google_apis/BUILD.gn b/google_apis/BUILD.gn index c3ab9da9..6327be5 100644 --- a/google_apis/BUILD.gn +++ b/google_apis/BUILD.gn
@@ -55,11 +55,6 @@ #`use_official_google_api_keys` was explicitly set to true and will fail to # build if the file is missing. use_official_google_api_keys = true - - # TODO(crbug.com/1294931): Remove this override when fixing the issue. - if (is_fuchsia) { - use_official_google_api_keys = false - } } else { # Check if the key file exists. check_internal_result =
diff --git a/ios/chrome/browser/sync/sync_service_factory_unittest.cc b/ios/chrome/browser/sync/sync_service_factory_unittest.cc index 2d3e9184..9fad64c 100644 --- a/ios/chrome/browser/sync/sync_service_factory_unittest.cc +++ b/ios/chrome/browser/sync/sync_service_factory_unittest.cc
@@ -53,7 +53,7 @@ protected: // Returns the collection of default datatypes. syncer::ModelTypeSet DefaultDatatypes() { - static_assert(42 == syncer::GetNumModelTypes(), + static_assert(43 == syncer::GetNumModelTypes(), "When adding a new type, you probably want to add it here as " "well (assuming it is already enabled)."); @@ -78,6 +78,9 @@ datatypes.Put(syncer::PREFERENCES); datatypes.Put(syncer::PRIORITY_PREFERENCES); datatypes.Put(syncer::READING_LIST); + if (base::FeatureList::IsEnabled(syncer::kSyncSegmentationDataType)) { + datatypes.Put(syncer::SEGMENTATION); + } // TODO(crbug.com/919489) Add SECURITY_EVENTS data type once it is enabled. datatypes.Put(syncer::SESSIONS); datatypes.Put(syncer::PROXY_TABS);
diff --git a/ios/chrome/browser/ui/content_suggestions/ntp_home_egtest.mm b/ios/chrome/browser/ui/content_suggestions/ntp_home_egtest.mm index d898aa1..625b83ae 100644 --- a/ios/chrome/browser/ui/content_suggestions/ntp_home_egtest.mm +++ b/ios/chrome/browser/ui/content_suggestions/ntp_home_egtest.mm
@@ -508,7 +508,8 @@ // Tests that the position of the collection view is restored when navigating // back to the NTP. -- (void)testPositionRestored { +// TODO(crbug.com/1364725): Re-enable test after fixing the test failure. +- (void)DISABLED_testPositionRestored { [self addMostVisitedTile]; // Add suggestions to be able to scroll on iPad.
diff --git a/ios/google_internal/frameworks/ChromeInternal.framework.dSYM.ios.zip.sha1 b/ios/google_internal/frameworks/ChromeInternal.framework.dSYM.ios.zip.sha1 index d6d7a37..1594946 100644 --- a/ios/google_internal/frameworks/ChromeInternal.framework.dSYM.ios.zip.sha1 +++ b/ios/google_internal/frameworks/ChromeInternal.framework.dSYM.ios.zip.sha1
@@ -1 +1 @@ -f38a8f003eaa0529f41c39ef89992f8a874b0a5a \ No newline at end of file +41b99b27bfd6780db9527dba15b96d747097604c \ No newline at end of file
diff --git a/ios/google_internal/frameworks/ChromeSSOInternal.framework.dSYM.ios.zip.sha1 b/ios/google_internal/frameworks/ChromeSSOInternal.framework.dSYM.ios.zip.sha1 index e3118dc2..4f16d8c 100644 --- a/ios/google_internal/frameworks/ChromeSSOInternal.framework.dSYM.ios.zip.sha1 +++ b/ios/google_internal/frameworks/ChromeSSOInternal.framework.dSYM.ios.zip.sha1
@@ -1 +1 @@ -846fcf651d96e06dd692197a51338f437522041f \ No newline at end of file +7ee130db21651b87124ac6dbee579f815d1fcd10 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/chrome_internal_dynamic_framework.ios.zip.sha1 b/ios/google_internal/frameworks/chrome_internal_dynamic_framework.ios.zip.sha1 index 34595f3..184c2824 100644 --- a/ios/google_internal/frameworks/chrome_internal_dynamic_framework.ios.zip.sha1 +++ b/ios/google_internal/frameworks/chrome_internal_dynamic_framework.ios.zip.sha1
@@ -1 +1 @@ -5dc796d6d88214d693783958cfd90457d2c33537 \ No newline at end of file +85861226e1845e497d36d4ee7b7c9ad7dbcad8a5 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/chrome_internal_dynamic_framework.iossimulator.zip.sha1 b/ios/google_internal/frameworks/chrome_internal_dynamic_framework.iossimulator.zip.sha1 index 7961104..5ac8e01e 100644 --- a/ios/google_internal/frameworks/chrome_internal_dynamic_framework.iossimulator.zip.sha1 +++ b/ios/google_internal/frameworks/chrome_internal_dynamic_framework.iossimulator.zip.sha1
@@ -1 +1 @@ -1e256610365ffceafca6385c42989ecba29fedf9 \ No newline at end of file +58a43c1c957d5801942af8544a3431466e51ef83 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.ios.zip.sha1 b/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.ios.zip.sha1 index 916d0790..f07765b3 100644 --- a/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.ios.zip.sha1 +++ b/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.ios.zip.sha1
@@ -1 +1 @@ -30f9c067d2c5dd7f11772ad87fa1b30ded294d06 \ No newline at end of file +37c7961be73058cfc484135a53c090819d34f464 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.iossimulator.zip.sha1 b/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.iossimulator.zip.sha1 index 649cb4a..8683eef 100644 --- a/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.iossimulator.zip.sha1 +++ b/ios/google_internal/frameworks/chrome_sso_internal_dynamic_framework.iossimulator.zip.sha1
@@ -1 +1 @@ -b7384d271dd061cc697f0d55d3cea95b80fec4c0 \ No newline at end of file +a95ce3b67de477578cc665a551158c801ed981a2 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.ios.zip.sha1 b/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.ios.zip.sha1 index 21dc00d..e63a7717 100644 --- a/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.ios.zip.sha1 +++ b/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.ios.zip.sha1
@@ -1 +1 @@ -14d12c8e80190e266070271f6b067925f5ee1896 \ No newline at end of file +4d10d859fda92c6aebf3e93978471f3c56c3206b \ No newline at end of file
diff --git a/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.iossimulator.zip.sha1 b/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.iossimulator.zip.sha1 index d74c544d..76a9761 100644 --- a/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.iossimulator.zip.sha1 +++ b/ios/google_internal/frameworks/chrome_test_internal_dynamic_framework.iossimulator.zip.sha1
@@ -1 +1 @@ -69426fb5336094108a04cc3714fae485a452ec61 \ No newline at end of file +465e3f571785d6ca6d5fc02034a118bbb26e822c \ No newline at end of file
diff --git a/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.ios.zip.sha1 b/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.ios.zip.sha1 index 87a5f445..70ec021 100644 --- a/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.ios.zip.sha1 +++ b/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.ios.zip.sha1
@@ -1 +1 @@ -9fb7c1cb99135f5f817ea19eed8b37491748a4dd \ No newline at end of file +f487ba26fd9339143451a268d7819dc3be64cf26 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.iossimulator.zip.sha1 b/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.iossimulator.zip.sha1 index bdfb0c5..f2365e88 100644 --- a/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.iossimulator.zip.sha1 +++ b/ios/google_internal/frameworks/remoting_dogfood_internal_dynamic_framework.iossimulator.zip.sha1
@@ -1 +1 @@ -0ae74e40ec78feca8dba9f62f5b177369e3d10b2 \ No newline at end of file +62ec1a3533e642f50e057cd43ad448429f278f75 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/remoting_internal_dynamic_framework.ios.zip.sha1 b/ios/google_internal/frameworks/remoting_internal_dynamic_framework.ios.zip.sha1 index f743f2a5..718635e 100644 --- a/ios/google_internal/frameworks/remoting_internal_dynamic_framework.ios.zip.sha1 +++ b/ios/google_internal/frameworks/remoting_internal_dynamic_framework.ios.zip.sha1
@@ -1 +1 @@ -2aad958e79bef7e091169afdd3a837fc4115a645 \ No newline at end of file +cd9a9b9980e3eb7c5fed7d4bf0cb0987de95d7cd \ No newline at end of file
diff --git a/ios/google_internal/frameworks/remoting_internal_dynamic_framework.iossimulator.zip.sha1 b/ios/google_internal/frameworks/remoting_internal_dynamic_framework.iossimulator.zip.sha1 index 86782d2c..5b9679c2 100644 --- a/ios/google_internal/frameworks/remoting_internal_dynamic_framework.iossimulator.zip.sha1 +++ b/ios/google_internal/frameworks/remoting_internal_dynamic_framework.iossimulator.zip.sha1
@@ -1 +1 @@ -61b225b35ea1ce3772e2fd82936bc093e60d5f3d \ No newline at end of file +3fe631affa823b5470a33f923e963b547020b5f5 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.ios.zip.sha1 b/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.ios.zip.sha1 index 27e0857..4521cc51 100644 --- a/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.ios.zip.sha1 +++ b/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.ios.zip.sha1
@@ -1 +1 @@ -2931b4faf0b5afc67e39230ccfb314b15d1bcd49 \ No newline at end of file +68a35c8454595a83d879e67c9d602f61b3cffe30 \ No newline at end of file
diff --git a/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.iossimulator.zip.sha1 b/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.iossimulator.zip.sha1 index 0bb59d5..261b0b0 100644 --- a/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.iossimulator.zip.sha1 +++ b/ios/google_internal/frameworks/web_view_shell_internal_dynamic_framework.iossimulator.zip.sha1
@@ -1 +1 @@ -6fe36274fb2afce6699109377148f8d3c628ef98 \ No newline at end of file +4ae6527f1998c39c7cb9448a6eb43f77ff220ff8 \ No newline at end of file
diff --git a/media/media_options.gni b/media/media_options.gni index b75ffaf..4650e50 100644 --- a/media/media_options.gni +++ b/media/media_options.gni
@@ -205,8 +205,8 @@ # Enables the use of library CDMs that implements the interface defined at # media/cdm/api/content_decryption_module.h. If true, the actually library CDM # will be hosted in the mojo CDM service running in the CDM (utility) process. - # Used for all desktop platforms except Fuchsia (crbug.com/1265618). - enable_library_cdms = toolkit_views && !is_fuchsia && !is_castos + # Used for all desktop platforms. + enable_library_cdms = toolkit_views && !is_castos } declare_args() {
diff --git a/net/quic/quic_chromium_client_session.cc b/net/quic/quic_chromium_client_session.cc index a9ff250..07e4288 100644 --- a/net/quic/quic_chromium_client_session.cc +++ b/net/quic/quic_chromium_client_session.cc
@@ -3011,6 +3011,40 @@ return StartProbing(network, peer_address); } +std::unique_ptr<quic::QuicPathValidationContext> +QuicChromiumClientSession::CreateContextForMultiPortPath() { + if (!connection()->connection_migration_use_new_cid()) { + return nullptr; + } + + // Create and configure socket on default network + std::unique_ptr<DatagramClientSocket> probing_socket = + stream_factory_->CreateSocket(net_log_.net_log(), net_log_.source()); + if (stream_factory_->ConfigureSocket( + probing_socket.get(), ToIPEndPoint(peer_address()), default_network_, + session_key_.socket_tag()) != OK) { + return nullptr; + } + + // Create new packet writer and reader on the probing socket. + auto probing_writer = std::make_unique<QuicChromiumPacketWriter>( + probing_socket.get(), task_runner_); + auto probing_reader = std::make_unique<QuicChromiumPacketReader>( + probing_socket.get(), clock_, this, yield_after_packets_, + yield_after_duration_, net_log_); + + probing_reader->StartReading(); + path_validation_writer_delegate_.set_network(default_network_); + path_validation_writer_delegate_.set_peer_address(peer_address()); + probing_writer->set_delegate(&path_validation_writer_delegate_); + IPEndPoint local_address; + probing_socket->GetLocalAddress(&local_address); + return std::make_unique<QuicChromiumPathValidationContext>( + ToQuicSocketAddress(local_address), peer_address(), default_network_, + std::move(probing_socket), std::move(probing_writer), + std::move(probing_reader)); +} + ProbingResult QuicChromiumClientSession::StartProbing( handles::NetworkHandle network, const quic::QuicSocketAddress& peer_address) {
diff --git a/net/quic/quic_chromium_client_session.h b/net/quic/quic_chromium_client_session.h index 54edb82..c594635 100644 --- a/net/quic/quic_chromium_client_session.h +++ b/net/quic/quic_chromium_client_session.h
@@ -695,6 +695,8 @@ void OnPathDegrading() override; void OnForwardProgressMadeAfterPathDegrading() override; void OnKeyUpdate(quic::KeyUpdateReason reason) override; + std::unique_ptr<quic::QuicPathValidationContext> + CreateContextForMultiPortPath() override; // QuicChromiumPacketReader::Visitor methods: bool OnReadError(int result, const DatagramClientSocket* socket) override;
diff --git a/net/quic/quic_stream_factory_test.cc b/net/quic/quic_stream_factory_test.cc index 70c1595..65eaa56 100644 --- a/net/quic/quic_stream_factory_test.cc +++ b/net/quic/quic_stream_factory_test.cc
@@ -4653,6 +4653,134 @@ EXPECT_TRUE(quic_data2.AllWriteDataConsumed()); } +TEST_P(QuicStreamFactoryTest, MultiPortSession) { + if (!version_.HasIetfQuicFrames()) { + // Path validator is only supported in IETF QUIC. + return; + } + SetIetfConnectionMigrationFlagsAndConnectionOptions(); + // Turning on MPQC will implicitly turn on port migration. + quic_params_->connection_options.push_back(quic::kMPQC); + socket_factory_ = std::make_unique<TestPortMigrationSocketFactory>(); + Initialize(); + + ProofVerifyDetailsChromium verify_details = DefaultProofVerifyDetails(); + crypto_client_stream_factory_.AddProofVerifyDetails(&verify_details); + + // Using a testing task runner so that we can control time. + auto task_runner = base::MakeRefCounted<base::TestMockTimeTaskRunner>(); + QuicStreamFactoryPeer::SetTaskRunner(factory_.get(), task_runner.get()); + + MockQuicData quic_data1(version_); + quic_data1.AddWrite(SYNCHRONOUS, ConstructInitialSettingsPacket(1)); + quic_data1.AddWrite( + SYNCHRONOUS, + ConstructGetRequestPacket( + 3, GetNthClientInitiatedBidirectionalStreamId(0), true, true)); + quic_data1.AddRead(ASYNC, ERR_IO_PENDING); // Pause + quic_data1.AddRead( + ASYNC, + ConstructOkResponsePacket( + 2, GetNthClientInitiatedBidirectionalStreamId(0), false, false)); + quic_data1.AddRead(SYNCHRONOUS, ERR_IO_PENDING); + quic_data1.AddWrite( + SYNCHRONOUS, + client_maker_.MakeAckAndDataPacket( + 4, /*include_version=*/false, GetQpackDecoderStreamId(), + /*largest_received=*/2, /*smallest_received=*/1, /*fin=*/false, + StreamCancellationQpackDecoderInstruction(0))); + quic_data1.AddWrite( + SYNCHRONOUS, client_maker_.MakeRstPacket( + 5, false, GetNthClientInitiatedBidirectionalStreamId(0), + quic::QUIC_STREAM_CANCELLED)); + quic_data1.AddSocketDataToFactory(socket_factory_.get()); + + // Set up the second socket data provider that is used for multi-port + MockQuicData quic_data2(version_); + quic::QuicConnectionId cid_on_new_path = + quic::test::TestConnectionId(12345678); + + client_maker_.set_connection_id(cid_on_new_path); + // Connectivity probe to be sent on the new path. + quic_data2.AddWrite(SYNCHRONOUS, + client_maker_.MakeConnectivityProbingPacket(2, true)); + quic_data2.AddRead(ASYNC, ERR_IO_PENDING); // Pause + // Connectivity probe to receive from the server. + quic_data2.AddRead(ASYNC, + server_maker_.MakeConnectivityProbingPacket(1, false)); + + quic_data2.AddRead(SYNCHRONOUS, ERR_IO_PENDING); + quic_data2.AddSocketDataToFactory(socket_factory_.get()); + + // Create request and QuicHttpStream. + QuicStreamRequest request(factory_.get()); + EXPECT_EQ(ERR_IO_PENDING, + request.Request( + scheme_host_port_, version_, privacy_mode_, DEFAULT_PRIORITY, + SocketTag(), NetworkAnonymizationKey(), SecureDnsPolicy::kAllow, + /*use_dns_aliases=*/true, /*require_dns_https_alpn=*/false, + /*cert_verify_flags=*/0, url_, net_log_, &net_error_details_, + failed_on_default_network_callback_, callback_.callback())); + EXPECT_THAT(callback_.WaitForResult(), IsOk()); + std::unique_ptr<HttpStream> stream = CreateStream(&request); + EXPECT_TRUE(stream.get()); + + // Cause QUIC stream to be created. + HttpRequestInfo request_info; + request_info.method = "GET"; + request_info.url = url_; + request_info.traffic_annotation = + MutableNetworkTrafficAnnotationTag(TRAFFIC_ANNOTATION_FOR_TESTS); + stream->RegisterRequest(&request_info); + EXPECT_EQ(OK, stream->InitializeStream(true, DEFAULT_PRIORITY, net_log_, + CompletionOnceCallback())); + + // Ensure that session is alive and active. + QuicChromiumClientSession* session = GetActiveSession(scheme_host_port_); + EXPECT_TRUE(QuicStreamFactoryPeer::IsLiveSession(factory_.get(), session)); + EXPECT_TRUE(HasActiveSession(scheme_host_port_)); + // Manually initialize the connection's self address. In real life, the + // initialization will be done during crypto handshake. + IPEndPoint ip; + session->GetDefaultSocket()->GetLocalAddress(&ip); + quic::test::QuicConnectionPeer::SetSelfAddress(session->connection(), + ToQuicSocketAddress(ip)); + + // This will trigger multi-port path creation. + MaybeMakeNewConnectionIdAvailableToSession(cid_on_new_path, session); + + // Send GET request on stream. + HttpResponseInfo response; + HttpRequestHeaders request_headers; + EXPECT_EQ(OK, stream->SendRequest(request_headers, &response, + callback_.callback())); + + // Resume quic data and a connectivity probe response will be read on the new + // socket. + quic_data2.Resume(); + + // The response is received on the default path. + quic_data1.Resume(); + // Response headers are received over the new port. + EXPECT_EQ(OK, stream->ReadResponseHeaders(callback_.callback())); + EXPECT_EQ(200, response.headers->response_code()); + + EXPECT_TRUE(QuicStreamFactoryPeer::IsLiveSession(factory_.get(), session)); + EXPECT_TRUE(HasActiveSession(scheme_host_port_)); + EXPECT_EQ(1u, session->GetNumActiveStreams()); + + // The periodic probing is handled and tested in quiche. + EXPECT_TRUE(quic::test::QuicConnectionPeer::GetMultiPortProbingAlarm( + session->connection()) + ->IsSet()); + + stream.reset(); + EXPECT_TRUE(quic_data1.AllReadDataConsumed()); + EXPECT_TRUE(quic_data1.AllWriteDataConsumed()); + EXPECT_TRUE(quic_data2.AllReadDataConsumed()); + EXPECT_TRUE(quic_data2.AllWriteDataConsumed()); +} + TEST_P(QuicStreamFactoryTest, MigratePortOnPathDegrading_WithoutNetworkHandle_PathValidator) { if (!version_.HasIetfQuicFrames()) {
diff --git a/net/url_request/url_request.cc b/net/url_request/url_request.cc index 5d754a7e..2446869d 100644 --- a/net/url_request/url_request.cc +++ b/net/url_request/url_request.cc
@@ -12,6 +12,7 @@ #include "base/compiler_specific.h" #include "base/metrics/histogram_macros.h" #include "base/rand_util.h" +#include "base/strings/string_piece.h" #include "base/strings/utf_string_conversions.h" #include "base/synchronization/lock.h" #include "base/threading/thread_task_runner_handle.h" @@ -499,9 +500,9 @@ initiator_ = initiator; } -void URLRequest::set_method(const std::string& method) { +void URLRequest::set_method(base::StringPiece method) { DCHECK(!is_pending_); - method_ = method; + method_ = std::string(method); } #if BUILDFLAG(ENABLE_REPORTING)
diff --git a/net/url_request/url_request.h b/net/url_request/url_request.h index 18db4ce..778a965 100644 --- a/net/url_request/url_request.h +++ b/net/url_request/url_request.h
@@ -14,6 +14,7 @@ #include "base/containers/flat_set.h" #include "base/memory/raw_ptr.h" #include "base/memory/weak_ptr.h" +#include "base/strings/string_piece_forward.h" #include "base/supports_user_data.h" #include "base/threading/thread_checker.h" #include "base/time/time.h" @@ -346,11 +347,12 @@ // This method may only be called before Start(). void set_initiator(const absl::optional<url::Origin>& initiator); - // The request method, as an uppercase string. "GET" is the default value. - // The request method may only be changed before Start() is called and - // should only be assigned an uppercase value. + // The request method. "GET" is the default value. The request method may + // only be changed before Start() is called. Request methods are + // case-sensitive, so standard HTTP methods like GET or POST should be + // specified in uppercase. const std::string& method() const { return method_; } - void set_method(const std::string& method); + void set_method(base::StringPiece method); #if BUILDFLAG(ENABLE_REPORTING) // Reporting upload nesting depth of this request. @@ -953,7 +955,7 @@ bool force_main_frame_for_same_site_cookies_ = false; absl::optional<url::Origin> initiator_; GURL delegate_redirect_url_; - std::string method_; // "GET", "POST", etc. Should be all uppercase. + std::string method_; // "GET", "POST", etc. Case-sensitive. std::string referrer_; ReferrerPolicy referrer_policy_ = ReferrerPolicy::CLEAR_ON_TRANSITION_FROM_SECURE_TO_INSECURE;
diff --git a/net/url_request/url_request_http_job.cc b/net/url_request/url_request_http_job.cc index 0a622f7f..e2a68e8 100644 --- a/net/url_request/url_request_http_job.cc +++ b/net/url_request/url_request_http_job.cc
@@ -572,6 +572,12 @@ } } + if (rv == OK && request_info_.method == "CONNECT") { + // CONNECT has different kinds of targets than other methods (RFC 9110, + // section 9.3.6), which are incompatible with URLRequest. + rv = ERR_METHOD_NOT_SUPPORTED; + } + if (rv == OK) { transaction_->SetConnectedCallback(base::BindRepeating( &URLRequestHttpJob::NotifyConnectedCallback, base::Unretained(this)));
diff --git a/net/url_request/url_request_unittest.cc b/net/url_request/url_request_unittest.cc index 4892eb28..43aa79a 100644 --- a/net/url_request/url_request_unittest.cc +++ b/net/url_request/url_request_unittest.cc
@@ -3988,7 +3988,7 @@ std::unique_ptr<URLRequest> r(default_context().CreateRequest( test_server_.GetURL("/echo"), DEFAULT_PRIORITY, &d, TRAFFIC_ANNOTATION_FOR_TESTS)); - r->set_method(method.c_str()); + r->set_method(method); r->set_upload(CreateSimpleUploadData(uploadBytes.get())); @@ -8451,9 +8451,6 @@ HTTPRedirectMethodTest(url, "PUT", "GET", true); HTTPRedirectMethodTest(url, "HEAD", "HEAD", false); - HTTPRedirectOriginHeaderTest(url, "CONNECT", "GET", std::string()); - HTTPRedirectOriginHeaderTest(https_redirect_url, "CONNECT", "GET", - std::string()); HTTPRedirectOriginHeaderTest(url, "DELETE", "GET", std::string()); HTTPRedirectOriginHeaderTest(https_redirect_url, "DELETE", "GET", std::string()); @@ -13009,6 +13006,18 @@ EXPECT_EQ("/auth-basic", r->auth_challenge_info()->path); } +TEST_F(URLRequestTestHTTP, ConnectNoSupported) { + ASSERT_TRUE(http_test_server()->Start()); + TestDelegate delegate; + std::unique_ptr<URLRequest> r(default_context().CreateRequest( + http_test_server()->GetURL("/"), DEFAULT_PRIORITY, &delegate, + TRAFFIC_ANNOTATION_FOR_TESTS)); + r->set_method("CONNECT"); + r->Start(); + delegate.RunUntilComplete(); + EXPECT_EQ(ERR_METHOD_NOT_SUPPORTED, delegate.request_status()); +} + class URLRequestDnsAliasTest : public TestWithTaskEnvironment { protected: URLRequestDnsAliasTest() {
diff --git a/services/device/generic_sensor/platform_sensor_fusion.cc b/services/device/generic_sensor/platform_sensor_fusion.cc index c8f2560a..d08aa3b 100644 --- a/services/device/generic_sensor/platform_sensor_fusion.cc +++ b/services/device/generic_sensor/platform_sensor_fusion.cc
@@ -4,8 +4,6 @@ #include "services/device/generic_sensor/platform_sensor_fusion.h" -#include <algorithm> - #include "base/bind.h" #include "base/check.h" #include "base/memory/raw_ptr.h" @@ -47,7 +45,7 @@ const auto& types = fusion_algorithm_->source_types(); DCHECK(!types.empty()); // Make sure there are no dups. - DCHECK(std::adjacent_find(types.begin(), types.end()) == types.end()); + DCHECK(base::ranges::adjacent_find(types) == types.end()); DCHECK(result_callback_); DCHECK(reading_buffer_); DCHECK(provider_);
diff --git a/testing/buildbot/chromium.rust.json b/testing/buildbot/chromium.rust.json index d3f3ce15..1795656 100644 --- a/testing/buildbot/chromium.rust.json +++ b/testing/buildbot/chromium.rust.json
@@ -209,6 +209,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "base_unittests", @@ -221,6 +226,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "gnrt_unittests", @@ -233,6 +243,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "rust_gtest_interop_unittests", @@ -246,6 +261,11 @@ "name": "test_cpp_including_rust_unittests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "test_cpp_including_rust_unittests", @@ -259,6 +279,11 @@ "name": "test_serde_json_lenient", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "test_serde_json_lenient", @@ -275,6 +300,11 @@ "name": "autocxx_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "autocxx_tests", @@ -289,6 +319,11 @@ "name": "build_rust_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "build_rust_tests", @@ -303,6 +338,11 @@ "name": "mojo_rust_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "mojo_rust_tests", @@ -326,6 +366,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "base_unittests", @@ -338,6 +383,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "gnrt_unittests", @@ -350,6 +400,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "rust_gtest_interop_unittests", @@ -363,6 +418,11 @@ "name": "test_cpp_including_rust_unittests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "test_cpp_including_rust_unittests", @@ -376,6 +436,11 @@ "name": "test_serde_json_lenient", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "test_serde_json_lenient", @@ -392,6 +457,11 @@ "name": "autocxx_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "autocxx_tests", @@ -406,6 +476,11 @@ "name": "build_rust_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "build_rust_tests", @@ -420,6 +495,11 @@ "name": "mojo_rust_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "mojo_rust_tests", @@ -443,6 +523,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "base_unittests", @@ -455,6 +540,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "gnrt_unittests", @@ -467,6 +557,11 @@ }, "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "rust_gtest_interop_unittests", @@ -480,6 +575,11 @@ "name": "test_cpp_including_rust_unittests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "test_cpp_including_rust_unittests", @@ -493,6 +593,11 @@ "name": "test_serde_json_lenient", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "test_serde_json_lenient", @@ -509,6 +614,11 @@ "name": "autocxx_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "autocxx_tests", @@ -523,6 +633,11 @@ "name": "build_rust_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "build_rust_tests", @@ -537,6 +652,11 @@ "name": "mojo_rust_tests", "swarming": { "can_use_on_swarming_builders": true, + "dimension_sets": [ + { + "os": "Ubuntu-18.04" + } + ], "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com" }, "test": "mojo_rust_tests",
diff --git a/testing/buildbot/waterfalls.pyl b/testing/buildbot/waterfalls.pyl index 4ce3966..a2e4fc2 100644 --- a/testing/buildbot/waterfalls.pyl +++ b/testing/buildbot/waterfalls.pyl
@@ -5572,6 +5572,9 @@ 'rust_build_tests', 'rust_autocxx_tests', ], + 'mixins': [ + 'linux-bionic' + ], 'test_suites': { 'gtest_tests': 'rust_gtests', 'isolated_scripts': 'rust_native_tests', @@ -5585,6 +5588,9 @@ 'rust_build_tests', 'rust_autocxx_tests', ], + 'mixins': [ + 'linux-bionic' + ], 'test_suites': { 'gtest_tests': 'rust_gtests', 'isolated_scripts': 'rust_native_tests', @@ -5598,6 +5604,9 @@ 'rust_build_tests', 'rust_autocxx_tests', ], + 'mixins': [ + 'linux-bionic' + ], 'test_suites': { 'gtest_tests': 'rust_gtests', 'isolated_scripts': 'rust_native_tests',
diff --git a/testing/variations/fieldtrial_testing_config.json b/testing/variations/fieldtrial_testing_config.json index 8a906c7..f7c1742 100644 --- a/testing/variations/fieldtrial_testing_config.json +++ b/testing/variations/fieldtrial_testing_config.json
@@ -6650,7 +6650,7 @@ { "name": "Enabled", "params": { - "nesting": "10" + "nesting": "15" }, "enable_features": [ "MaxUnthrottledTimeoutNestingLevel" @@ -10861,11 +10861,16 @@ { "name": "Enabled", "params": { - "min_gms_core_version_no_dots": "222612000", - "stage": "0" + "api_error_list_version": "1", + "fallback_on_user_affecting_read_operations": "true", + "ignored_api_errors": "11005,11006", + "min_gms_core_version_no_dots": "223012000", + "retriable_api_errors": "7,17,20,22" }, "enable_features": [ - "UnifiedPasswordManagerAndroid" + "UnifiedPasswordManagerAndroid", + "UnifiedPasswordManagerErrorMessages", + "UnifiedPasswordManagerReenrollment" ] } ]
diff --git a/third_party/blink/common/features.cc b/third_party/blink/common/features.cc index 96e7f3f..1dcfe45 100644 --- a/third_party/blink/common/features.cc +++ b/third_party/blink/common/features.cc
@@ -1220,7 +1220,7 @@ "MaxUnthrottledTimeoutNestingLevel", base::FEATURE_DISABLED_BY_DEFAULT); const base::FeatureParam<int> kMaxUnthrottledTimeoutNestingLevelParam{ - &kMaxUnthrottledTimeoutNestingLevel, "nesting", 10}; + &kMaxUnthrottledTimeoutNestingLevel, "nesting", 15}; bool IsMaxUnthrottledTimeoutNestingLevelEnabled() { auto policy = GetUnthrottledNestedTimeoutPolicyOverride(); if (policy != UnthrottledNestedTimeoutPolicyOverride::kNoOverride) @@ -1609,10 +1609,6 @@ "FastPathPaintPropertyUpdates", base::FEATURE_ENABLED_BY_DEFAULT); -BASE_FEATURE(kWildcardSubdomainsInPermissionsPolicy, - "WildcardSubdomainsInPermissionsPolicy", - base::FEATURE_ENABLED_BY_DEFAULT); - BASE_FEATURE(kThreadedBodyLoader, "ThreadedBodyLoader", base::FEATURE_DISABLED_BY_DEFAULT); @@ -1632,5 +1628,23 @@ "WebRtcCombinedNetworkAndWorkerThread", base::FEATURE_DISABLED_BY_DEFAULT); +// Allow process isolation of iframes with the 'sandbox' attribute set. Whether +// or not such an iframe will be isolated may depend on options specified with +// the attribute. Note: At present, only iframes with origin-restricted +// sandboxes are isolated. +BASE_FEATURE(kIsolateSandboxedIframes, + "IsolateSandboxedIframes", + base::FEATURE_DISABLED_BY_DEFAULT); +const base::FeatureParam<IsolateSandboxedIframesGrouping>::Option + isolated_sandboxed_iframes_grouping_types[] = { + {IsolateSandboxedIframesGrouping::kPerSite, "per-site"}, + {IsolateSandboxedIframesGrouping::kPerOrigin, "per-origin"}, + {IsolateSandboxedIframesGrouping::kPerDocument, "per-document"}}; +const base::FeatureParam<IsolateSandboxedIframesGrouping> + kIsolateSandboxedIframesGroupingParam{ + &kIsolateSandboxedIframes, "grouping", + IsolateSandboxedIframesGrouping::kPerSite, + &isolated_sandboxed_iframes_grouping_types}; + } // namespace features } // namespace blink
diff --git a/third_party/blink/common/permissions/permission_utils.cc b/third_party/blink/common/permissions/permission_utils.cc index c8b01c2..c5de113 100644 --- a/third_party/blink/common/permissions/permission_utils.cc +++ b/third_party/blink/common/permissions/permission_utils.cc
@@ -190,7 +190,7 @@ return PermissionType::NFC; case PermissionName::STORAGE_ACCESS: return PermissionType::STORAGE_ACCESS_GRANT; - case PermissionName::WINDOW_PLACEMENT: + case PermissionName::WINDOW_MANAGEMENT: return PermissionType::WINDOW_PLACEMENT; case PermissionName::LOCAL_FONTS: return PermissionType::LOCAL_FONTS;
diff --git a/third_party/blink/common/permissions_policy/origin_with_possible_wildcards.cc b/third_party/blink/common/permissions_policy/origin_with_possible_wildcards.cc index 4244136..7ad2f84e 100644 --- a/third_party/blink/common/permissions_policy/origin_with_possible_wildcards.cc +++ b/third_party/blink/common/permissions_policy/origin_with_possible_wildcards.cc
@@ -4,10 +4,8 @@ #include "third_party/blink/public/common/permissions_policy/origin_with_possible_wildcards.h" -#include "base/feature_list.h" #include "net/base/registry_controlled_domains/registry_controlled_domain.h" #include "services/network/public/cpp/cors/origin_access_entry.h" -#include "third_party/blink/public/common/features.h" #include "url/gurl.h" #include "url/origin.h" @@ -41,9 +39,7 @@ // subdomain wildcard if there is a exactly one `*` and it's after the scheme // and before the rest of the host. Invalid origins return an instance of // OriginWithPossibleWildcards with an opaque origin member. - if (base::FeatureList::IsEnabled( - features::kWildcardSubdomainsInPermissionsPolicy) && - type == NodeType::kHeader && + if (type == NodeType::kHeader && (wildcard_pos = allowlist_entry.find("://*.")) != std::string::npos && allowlist_entry.find('*') == allowlist_entry.rfind('*')) { // We need a copy as erase modifies the original. @@ -90,13 +86,7 @@ bool OriginWithPossibleWildcards::DoesMatchOrigin( const url::Origin& match_origin) const { - // TODO(crbug.com/1345994): Merge logic with IsSubdomainOfHost where possible. if (has_subdomain_wildcard) { - // Only try to match at all if wildcard matching is enabled. - if (!base::FeatureList::IsEnabled( - features::kWildcardSubdomainsInPermissionsPolicy)) { - return false; - } // This function won't match https://*.foo.com with https://foo.com. if (origin == match_origin) { return false;
diff --git a/third_party/blink/common/permissions_policy/origin_with_possible_wildcards_unittest.cc b/third_party/blink/common/permissions_policy/origin_with_possible_wildcards_unittest.cc index 621b2d2..404f0b6 100644 --- a/third_party/blink/common/permissions_policy/origin_with_possible_wildcards_unittest.cc +++ b/third_party/blink/common/permissions_policy/origin_with_possible_wildcards_unittest.cc
@@ -5,34 +5,16 @@ #include "third_party/blink/public/common/permissions_policy/origin_with_possible_wildcards.h" #include "base/test/gtest_util.h" -#include "base/test/scoped_feature_list.h" #include "mojo/public/cpp/test_support/test_utils.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/blink/common/permissions_policy/permissions_policy_mojom_traits.h" -#include "third_party/blink/public/common/features.h" #include "third_party/blink/public/mojom/permissions_policy/permissions_policy.mojom.h" #include "url/gurl.h" #include "url/origin.h" namespace blink { -class OriginWithPossibleWildcardsTest : public testing::TestWithParam<bool> { - public: - void SetUp() override { - scoped_feature_list_.InitWithFeatureState( - features::kWildcardSubdomainsInPermissionsPolicy, - HasWildcardSubdomainsInPermissionsPolicy()); - } - - bool HasWildcardSubdomainsInPermissionsPolicy() { return GetParam(); } - - private: - base::test::ScopedFeatureList scoped_feature_list_; -}; - -INSTANTIATE_TEST_SUITE_P(All, OriginWithPossibleWildcardsTest, testing::Bool()); - -TEST_P(OriginWithPossibleWildcardsTest, DoesMatchOrigin) { +TEST(OriginWithPossibleWildcardsTest, DoesMatchOrigin) { // Tuple of {origin to test, origin in policy, w/ wildcard, result, // description}. const auto& values = { @@ -78,23 +60,20 @@ url::Origin::Create(GURL("https://foo.com")), true, false, "Same origin, w/ wildcard"), std::make_tuple(url::Origin::Create(GURL("https://bar.foo.com")), - url::Origin::Create(GURL("https://foo.com")), true, - HasWildcardSubdomainsInPermissionsPolicy(), + url::Origin::Create(GURL("https://foo.com")), true, true, "Subdomain matches, w/ wildcard"), std::make_tuple(url::Origin::Create(GURL("http://bar.foo.com")), url::Origin::Create(GURL("https://foo.com")), true, false, "Different scheme, w/ wildcard"), std::make_tuple(url::Origin::Create(GURL("https://baz.bar.foo.com")), - url::Origin::Create(GURL("https://foo.com")), true, - HasWildcardSubdomainsInPermissionsPolicy(), + url::Origin::Create(GURL("https://foo.com")), true, true, "Sub-subdomain matches, w/ wildcard"), std::make_tuple(url::Origin::Create(GURL("https://foo.com")), url::Origin::Create(GURL("https://bar.foo.com")), true, false, "Subdomain doesn't match, w/ wildcard"), std::make_tuple(url::Origin::Create(GURL("https://bar.foo.com")), url::Origin::Create(GURL("https://foo.com:443")), true, - HasWildcardSubdomainsInPermissionsPolicy(), - "Ignore default port, w/ wildcard"), + true, "Ignore default port, w/ wildcard"), std::make_tuple(url::Origin(), url::Origin::Create(GURL("https://foo.com")), true, false, "Opaque to origin, w/ wildcard"), @@ -121,7 +100,7 @@ } } -TEST_P(OriginWithPossibleWildcardsTest, Parse) { +TEST(OriginWithPossibleWildcardsTest, Parse) { // Tuple of {serialized value, type, origin, wildcard, description}. const auto& values = { std::make_tuple("https://foo.com", @@ -134,10 +113,7 @@ "Origin without subdomain wildcard in attribute"), std::make_tuple( "https://*.foo.com", OriginWithPossibleWildcards::NodeType::kHeader, - HasWildcardSubdomainsInPermissionsPolicy() ? "https://foo.com" - : "https://%2A.foo.com", - HasWildcardSubdomainsInPermissionsPolicy(), - "Origin with subdomain wildcard in header"), + "https://foo.com", true, "Origin with subdomain wildcard in header"), std::make_tuple("https://*.foo.com", OriginWithPossibleWildcards::NodeType::kAttribute, "https://%2A.foo.com", false, @@ -173,10 +149,7 @@ "Origin with only private tld host wildcard in attribute"), std::make_tuple("https://*.foo.appspot.com", OriginWithPossibleWildcards::NodeType::kHeader, - HasWildcardSubdomainsInPermissionsPolicy() - ? "https://foo.appspot.com" - : "https://%2A.foo.appspot.com", - HasWildcardSubdomainsInPermissionsPolicy(), + "https://foo.appspot.com", true, "Origin with private tld host wildcard in header"), std::make_tuple("https://*.foo.appspot.com", OriginWithPossibleWildcards::NodeType::kAttribute, @@ -184,10 +157,7 @@ "Origin with private tld host wildcard in attribute"), std::make_tuple("https://*.example.test", OriginWithPossibleWildcards::NodeType::kHeader, - HasWildcardSubdomainsInPermissionsPolicy() - ? "https://example.test" - : "https://%2A.example.test", - HasWildcardSubdomainsInPermissionsPolicy(), + "https://example.test", true, "Origin with unknown tld host wildcard in header"), std::make_tuple("https://*.example.test", OriginWithPossibleWildcards::NodeType::kAttribute, @@ -228,7 +198,7 @@ } } -TEST_P(OriginWithPossibleWildcardsTest, Serialize) { +TEST(OriginWithPossibleWildcardsTest, Serialize) { // Tuple of {origin, wildcard, serialized value, description}. const auto& values = { std::make_tuple("https://foo.com", false, "https://foo.com", @@ -249,7 +219,7 @@ } } -TEST_P(OriginWithPossibleWildcardsTest, Constructors) { +TEST(OriginWithPossibleWildcardsTest, Constructors) { OriginWithPossibleWildcards a; OriginWithPossibleWildcards b(url::Origin(), false); OriginWithPossibleWildcards c(b); @@ -261,7 +231,7 @@ EXPECT_EQ(a, b); } -TEST_P(OriginWithPossibleWildcardsTest, Opaque) { +TEST(OriginWithPossibleWildcardsTest, Opaque) { EXPECT_DCHECK_DEATH(OriginWithPossibleWildcards(url::Origin(), true)); OriginWithPossibleWildcards original(url::Origin(), false); original.has_subdomain_wildcard = true;
diff --git a/third_party/blink/public/common/features.h b/third_party/blink/public/common/features.h index 11d35671..e739ed13 100644 --- a/third_party/blink/public/common/features.h +++ b/third_party/blink/public/common/features.h
@@ -850,10 +850,6 @@ // applied directly instead of using the property tree builder. BLINK_COMMON_EXPORT BASE_DECLARE_FEATURE(kFastPathPaintPropertyUpdates); -// If enabled, wildcard subdomains are supported in permissions policies. -BLINK_COMMON_EXPORT BASE_DECLARE_FEATURE( - kWildcardSubdomainsInPermissionsPolicy); - // If enabled, reads and decodes navigation body data off the main thread. BLINK_COMMON_EXPORT BASE_DECLARE_FEATURE(kThreadedBodyLoader); @@ -872,6 +868,24 @@ // Combine WebRTC Network and Worker threads. More info at crbug.com/1373439. BLINK_COMMON_EXPORT BASE_DECLARE_FEATURE(kWebRtcCombinedNetworkAndWorkerThread); +BLINK_COMMON_EXPORT BASE_DECLARE_FEATURE(kIsolateSandboxedIframes); +enum class IsolateSandboxedIframesGrouping { + // In this grouping, all isolated sandboxed iframes whose URLs share the same + // site in a given BrowsingInstance will share a process. + kPerSite, + // In this grouping, all isolated sandboxed iframes from a given + // BrowsingInstance whose URLs share the same origin will be isolated in an + // origin-keyed process. + kPerOrigin, + // Unlike the other two modes, which group sandboxed frames per-site or + // per-origin, this one doesn't do any grouping at all and uses one process + // per document. + kPerDocument, +}; +BLINK_COMMON_EXPORT extern const base::FeatureParam< + IsolateSandboxedIframesGrouping> + kIsolateSandboxedIframesGroupingParam; + } // namespace features } // namespace blink
diff --git a/third_party/blink/public/mojom/permissions/permission.mojom b/third_party/blink/public/mojom/permissions/permission.mojom index cf00630..8721ae5 100644 --- a/third_party/blink/public/mojom/permissions/permission.mojom +++ b/third_party/blink/public/mojom/permissions/permission.mojom
@@ -28,7 +28,7 @@ SYSTEM_WAKE_LOCK, NFC, STORAGE_ACCESS, - WINDOW_PLACEMENT, + WINDOW_MANAGEMENT, LOCAL_FONTS, DISPLAY_CAPTURE, };
diff --git a/third_party/blink/renderer/core/css/properties/computed_style_utils.cc b/third_party/blink/renderer/core/css/properties/computed_style_utils.cc index 74072583..f9a3a06 100644 --- a/third_party/blink/renderer/core/css/properties/computed_style_utils.cc +++ b/third_party/blink/renderer/core/css/properties/computed_style_utils.cc
@@ -1428,28 +1428,58 @@ wtf_size_t auto_repeat_insertion_point = computed_grid_track_list.auto_repeat_insertion_point; - // If the element is a grid container, the resolved value is the used value, - // specifying track sizes in pixels and expanding the repeat() notation. if (is_layout_grid) { const auto* grid = ToInterface<LayoutNGGridInterface>(layout_object); - OrderedNamedLinesCollectorInGridLayout collector( - computed_grid_track_list.ordered_named_grid_lines, - computed_grid_track_list.auto_repeat_ordered_named_grid_lines, - auto_repeat_insertion_point, - grid->AutoRepeatCountForDirection(direction), - auto_repeat_track_sizes.size()); - auto getTrackSize = [&](const LayoutUnit& v) { - return ZoomAdjustedPixelValue(v, style); - }; - // Named grid line indices are relative to the explicit grid, but we are - // including all tracks. So we need to subtract the number of leading - // implicit tracks in order to get the proper line index. - int offset = -base::checked_cast<int>( - grid->ExplicitGridStartForDirection(direction)); - PopulateGridTrackList(list, collector, - grid->TrackSizesForComputedStyle(direction), - getTrackSize, offset); - return list; + if (computed_grid_track_list.IsSubgriddedAxis()) { + // If the track list is subgridded, return the word 'subgrid', followed by + // the specified named grid lines in brackets. Empty brackets are also + // valid. + list->Append( + *MakeGarbageCollected<CSSIdentifierValue>(CSSValueID::kSubgrid)); + + wtf_size_t subgrid_line_names_start = + grid->ExplicitGridStartForDirection(direction); + wtf_size_t subgrid_line_names_end = + grid->ExplicitGridEndForDirection(direction); + for (wtf_size_t i = subgrid_line_names_start; i <= subgrid_line_names_end; + ++i) { + auto iter = computed_grid_track_list.ordered_named_grid_lines.find(i); + + cssvalue::CSSBracketedValueList* value_list = + MakeGarbageCollected<cssvalue::CSSBracketedValueList>(); + + if (iter != computed_grid_track_list.ordered_named_grid_lines.end()) { + for (auto named_grid_line : iter->value) { + value_list->Append(*MakeGarbageCollected<CSSCustomIdentValue>( + named_grid_line.line_name)); + } + } + list->Append(*value_list); + } + return list; + } else { + // If the element is a grid container, the resolved value is the used + // value, specifying track sizes in pixels and expanding the repeat() + // notation. + OrderedNamedLinesCollectorInGridLayout collector( + computed_grid_track_list.ordered_named_grid_lines, + computed_grid_track_list.auto_repeat_ordered_named_grid_lines, + auto_repeat_insertion_point, + grid->AutoRepeatCountForDirection(direction), + auto_repeat_track_sizes.size()); + auto getTrackSize = [&](const LayoutUnit& v) { + return ZoomAdjustedPixelValue(v, style); + }; + // Named grid line indices are relative to the explicit grid, but we are + // including all tracks. So we need to subtract the number of leading + // implicit tracks in order to get the proper line index. + int offset = -base::checked_cast<int>( + grid->ExplicitGridStartForDirection(direction)); + PopulateGridTrackList(list, collector, + grid->TrackSizesForComputedStyle(direction), + getTrackSize, offset); + return list; + } } // Otherwise, the resolved value is the computed value, preserving repeat().
diff --git a/third_party/blink/renderer/core/css/resolver/style_builder_converter.cc b/third_party/blink/renderer/core/css/resolver/style_builder_converter.cc index c28f7f9..e2cd4b4 100644 --- a/third_party/blink/renderer/core/css/resolver/style_builder_converter.cc +++ b/third_party/blink/renderer/core/css/resolver/style_builder_converter.cc
@@ -1015,9 +1015,8 @@ bool is_first_repeat = false) { DCHECK(value.IsGridLineNamesValue()); - // TODO(ansollan): Serialize subgrid's empty lines. for (auto& named_grid_line_value : To<CSSValueList>(value)) { - String named_grid_line = + AtomicString named_grid_line = To<CSSCustomIdentValue>(*named_grid_line_value).Value(); NamedGridLinesMap::AddResult result = named_grid_lines.insert(named_grid_line, Vector<wtf_size_t>());
diff --git a/third_party/blink/renderer/core/css/transition.css b/third_party/blink/renderer/core/css/transition.css index 2a33056..404abc0e 100644 --- a/third_party/blink/renderer/core/css/transition.css +++ b/third_party/blink/renderer/core/css/transition.css
@@ -11,7 +11,12 @@ html::page-transition { position: fixed; - inset: 0; + top: 0; + left: 0; + + /* Size will be set dynamically to correctly account for UI such as + virtual-keyboards */ + pointer-events: none; }
diff --git a/third_party/blink/renderer/core/document_transition/document_transition.h b/third_party/blink/renderer/core/document_transition/document_transition.h index 5e61b0b..f13e7d6 100644 --- a/third_party/blink/renderer/core/document_transition/document_transition.h +++ b/third_party/blink/renderer/core/document_transition/document_transition.h
@@ -135,6 +135,10 @@ bool IsIdle() const { return state_ == State::kIdle; } + bool IsRootTransitioning() const { + return style_tracker_ && style_tracker_->IsRootTransitioning(); + } + // In physical pixels. See comments on equivalent methods in // DocumentTransitionStyleTracker for info. gfx::Rect GetSnapshotViewportRect() const;
diff --git a/third_party/blink/renderer/core/document_transition/document_transition_style_builder.cc b/third_party/blink/renderer/core/document_transition/document_transition_style_builder.cc index 72b8880c..c696f80 100644 --- a/third_party/blink/renderer/core/document_transition/document_transition_style_builder.cc +++ b/third_party/blink/renderer/core/document_transition/document_transition_style_builder.cc
@@ -156,10 +156,23 @@ AddContainerStyles(tag, rule_builder.ReleaseString()); } -void DocumentTransitionStyleBuilder::AddRootStyles(const String& rules) { +void DocumentTransitionStyleBuilder::AddRootStyles( + const gfx::RectF& snapshot_viewport_rect_css) { builder_.Append(kTransitionRootName); builder_.Append("{ "); - builder_.Append(rules); + builder_.AppendFormat( + R"CSS( + width: %.3fpx; + height: %.3fpx; + )CSS", + snapshot_viewport_rect_css.width(), snapshot_viewport_rect_css.height()); + if (!snapshot_viewport_rect_css.OffsetFromOrigin().IsZero()) { + builder_.AppendFormat( + R"CSS( + transform: translate(%.3fpx, %.3fpx); + )CSS", + snapshot_viewport_rect_css.x(), snapshot_viewport_rect_css.y()); + } builder_.Append(" }"); }
diff --git a/third_party/blink/renderer/core/document_transition/document_transition_style_builder.h b/third_party/blink/renderer/core/document_transition/document_transition_style_builder.h index e0367613..6b05019 100644 --- a/third_party/blink/renderer/core/document_transition/document_transition_style_builder.h +++ b/third_party/blink/renderer/core/document_transition/document_transition_style_builder.h
@@ -37,7 +37,7 @@ const ContainerProperties& properties, WritingMode writing_mode); - void AddRootStyles(const String& rules); + void AddRootStyles(const gfx::RectF& snapshot_viewport_rect_css); String Build();
diff --git a/third_party/blink/renderer/core/document_transition/document_transition_style_tracker.cc b/third_party/blink/renderer/core/document_transition/document_transition_style_tracker.cc index 6cc1e8e..44e73d2 100644 --- a/third_party/blink/renderer/core/document_transition/document_transition_style_tracker.cc +++ b/third_party/blink/renderer/core/document_transition/document_transition_style_tracker.cc
@@ -65,27 +65,6 @@ return kAnimationUAStyles; } -absl::optional<String> GetSnapshotViewportOffsetTransform( - const gfx::Vector2d& offset, - float device_pixel_ratio) { - if (!offset.x() && !offset.y()) - return absl::nullopt; - - // Since we're using the offset in style, convert from physical pixels to CSS - // pixels. - gfx::Vector2dF css_offset = - gfx::ScaleVector2d(offset, 1.f / device_pixel_ratio); - - // The root is translated up and left so that the coordinate space for all - // children has its origin at the point that is the top-left when all UI is - // hidden. This requires non-root shared elements to be shifted back down and - // right. - DCHECK_LE(css_offset.x(), 0.f); - DCHECK_LE(css_offset.y(), 0.f); - return String::Format("transform: translate(%.3fpx, %.3fpx);", css_offset.x(), - css_offset.y()); -} - absl::optional<String> ComputeInsetDifference(PhysicalRect reference_rect, const LayoutRect& target_rect, float device_pixel_ratio) { @@ -987,29 +966,31 @@ DCHECK(document.View()); DCHECK(document.GetPage()); DCHECK(document.GetFrame()); - - if (!document.GetFrame()->IsOutermostMainFrame()) - return gfx::Outsets(); - - Page& page = *document.GetPage(); + DCHECK(document.GetLayoutView()); int top = 0; int right = 0; int bottom = 0; int left = 0; - // TODO(bokan): This assumes any shown ratio implies controls are shown. We - // many need to do some synchronization to make this work seamlessly with URL - // bar animations. - BrowserControls& controls = page.GetBrowserControls(); - if (page.GetBrowserControls().TopShownRatio()) { - top += controls.TopHeight() - controls.TopMinHeight(); - bottom += controls.BottomHeight() - controls.BottomMinHeight(); + if (document.GetFrame()->IsOutermostMainFrame()) { + // TODO(bokan): This assumes any shown ratio implies controls are shown. We + // many need to do some synchronization to make this work seamlessly with + // URL bar animations. + BrowserControls& controls = document.GetPage()->GetBrowserControls(); + if (controls.TopShownRatio()) + top += controls.TopHeight() - controls.TopMinHeight(); + if (controls.BottomShownRatio()) + bottom += controls.BottomHeight() - controls.BottomMinHeight(); } // TODO(bokan): Account for virtual-keyboard - // TODO(bokan): Account for scrollbars. + // TODO(bokan): Handle left-hand side vertical scrollbars. + + LocalFrameView& view = *document.View(); + right += view.LayoutViewport()->VerticalScrollbarWidth(); + bottom += view.LayoutViewport()->HorizontalScrollbarHeight(); gfx::Outsets outsets; outsets.set_top(top); @@ -1027,11 +1008,11 @@ LocalFrameView& view = *document_->View(); - // Start with the full FrameView size, i.e. the position: fixed viewport and + // Start with the FrameView size, i.e. the position: fixed viewport, and // expand the viewport by any insetting UI such as the mobile URL bar, - // virtual-keyboard, etc. Note: the FrameView size already includes - // scrollbars. - gfx::Rect snapshot_viewport_rect(view.Size()); + // virtual-keyboard, scrollbars, etc. + gfx::Rect snapshot_viewport_rect( + view.LayoutViewport()->ExcludeScrollbars(view.Size())); snapshot_viewport_rect.Outset(GetFixedToSnapshotViewportOutsets(*document_)); return snapshot_viewport_rect; @@ -1039,8 +1020,14 @@ gfx::Vector2d DocumentTransitionStyleTracker::GetRootSnapshotPaintOffset() const { + DCHECK(document_->GetLayoutView()); + DCHECK(document_->View()); + gfx::Outsets outsets = GetFixedToSnapshotViewportOutsets(*document_); - return gfx::Vector2d(outsets.left(), outsets.top()); + int left = outsets.left(); + int top = outsets.top(); + + return gfx::Vector2d(left, top); } void DocumentTransitionStyleTracker::InvalidateStyle() { @@ -1144,14 +1131,19 @@ ->StyleRef() .EffectiveZoom(); - // Position the root container behind any viewport insetting widgets (such - // as the URL bar) so that it's stable across a transition. - absl::optional<String> snapshot_viewport_offset = - GetSnapshotViewportOffsetTransform( - GetSnapshotViewportRect().OffsetFromOrigin(), device_pixel_ratio); - if (snapshot_viewport_offset) { - builder.AddRootStyles(*snapshot_viewport_offset); - } + // Size and position the root container behind any viewport insetting widgets + // (such as the URL bar) so that it's stable across a transition. This rect + // is called the "snapshot viewport". Since this is applied in style, + // convert from physical pixels to CSS pixels. + gfx::RectF snapshot_viewport_css_pixels = gfx::ScaleRect( + gfx::RectF(GetSnapshotViewportRect()), 1.f / device_pixel_ratio); + + // If adjusted, the root is always translated up and left underneath any UI + // so the direction must always be negative. + DCHECK_LE(snapshot_viewport_css_pixels.x(), 0.f); + DCHECK_LE(snapshot_viewport_css_pixels.y(), 0.f); + + builder.AddRootStyles(snapshot_viewport_css_pixels); for (auto& root_tag : AllRootTags()) { // This is case 3 above.
diff --git a/third_party/blink/renderer/core/frame/local_frame_client.h b/third_party/blink/renderer/core/frame/local_frame_client.h index 83a62e2..4c1364e1 100644 --- a/third_party/blink/renderer/core/frame/local_frame_client.h +++ b/third_party/blink/renderer/core/frame/local_frame_client.h
@@ -293,7 +293,9 @@ HTMLMediaElement&) = 0; virtual void DidCommitDocumentReplacementNavigation(DocumentLoader*) = 0; - virtual void DispatchDidClearWindowObjectInMainWorld() = 0; + virtual void DispatchDidClearWindowObjectInMainWorld( + v8::Isolate* isolate, + v8::MicrotaskQueue* microtask_queue) = 0; virtual void DocumentElementAvailable() = 0; virtual void RunScriptsAtDocumentElementAvailable() = 0; virtual void RunScriptsAtDocumentReady(bool document_is_empty) = 0;
diff --git a/third_party/blink/renderer/core/frame/local_frame_client_impl.cc b/third_party/blink/renderer/core/frame/local_frame_client_impl.cc index 7b726508..ff373fdfa 100644 --- a/third_party/blink/renderer/core/frame/local_frame_client_impl.cc +++ b/third_party/blink/renderer/core/frame/local_frame_client_impl.cc
@@ -191,9 +191,17 @@ } } -void LocalFrameClientImpl::DispatchDidClearWindowObjectInMainWorld() { +void LocalFrameClientImpl::DispatchDidClearWindowObjectInMainWorld( + v8::Isolate* isolate, + v8::MicrotaskQueue* microtask_queue) { if (web_frame_->Client()) { - web_frame_->Client()->DidClearWindowObject(); + // Do not run microtasks while invoking the callback. + { + v8::MicrotasksScope microtasks(isolate, microtask_queue, + + v8::MicrotasksScope::kDoNotRunMicrotasks); + web_frame_->Client()->DidClearWindowObject(); + } Document* document = web_frame_->GetFrame()->GetDocument(); if (document) { const Settings* const settings = web_frame_->GetFrame()->GetSettings();
diff --git a/third_party/blink/renderer/core/frame/local_frame_client_impl.h b/third_party/blink/renderer/core/frame/local_frame_client_impl.h index c974fa2..2fe4b1c 100644 --- a/third_party/blink/renderer/core/frame/local_frame_client_impl.h +++ b/third_party/blink/renderer/core/frame/local_frame_client_impl.h
@@ -72,7 +72,9 @@ // Notifies the WebView delegate that the JS window object has been cleared, // giving it a chance to bind native objects to the window before script // parsing begins. - void DispatchDidClearWindowObjectInMainWorld() override; + void DispatchDidClearWindowObjectInMainWorld( + v8::Isolate* isolate, + v8::MicrotaskQueue* microtask_queue) override; void DocumentElementAvailable() override; void RunScriptsAtDocumentElementAvailable() override; void RunScriptsAtDocumentReady(bool document_is_empty) override;
diff --git a/third_party/blink/renderer/core/html/parser/html_document_parser.cc b/third_party/blink/renderer/core/html/parser/html_document_parser.cc index 58d28da..9afebbe 100644 --- a/third_party/blink/renderer/core/html/parser/html_document_parser.cc +++ b/third_party/blink/renderer/core/html/parser/html_document_parser.cc
@@ -1699,7 +1699,7 @@ // commit. Processing the data now can lead to unexpected states. // TODO(https://crbug.com/1364695): see if this limitation can be removed. if (auto* sink = probe::ToCoreProbeSink(GetDocument())) { - if (sink->HasAgentsGlobal(CoreProbeSink::kInspectorDOMDebuggerAgent)) + if (sink->HasAgentsGlobal(CoreProbeSink::kDevToolsSession)) return false; }
diff --git a/third_party/blink/renderer/core/layout/layout_view.cc b/third_party/blink/renderer/core/layout/layout_view.cc index dc08ce1..7ddf9a9 100644 --- a/third_party/blink/renderer/core/layout/layout_view.cc +++ b/third_party/blink/renderer/core/layout/layout_view.cc
@@ -587,10 +587,12 @@ if (!frame_view_) return PhysicalRect(); + // TODO(bokan): This shouldn't be just for the outermost main frame, we + // should do it for all frames. crbug.com/1311518. if (frame_view_->GetFrame().IsOutermostMainFrame()) { auto* supplement = DocumentTransitionSupplement::FromIfExists(GetDocument()); - if (supplement && !supplement->GetTransition()->IsIdle()) { + if (supplement && supplement->GetTransition()->IsRootTransitioning()) { // If we're capturing a transition snapshot, the root transition needs to // produce the snapshot at a known stable size, excluding all insetting // UI like mobile URL bars and virtual keyboards. @@ -622,7 +624,14 @@ } rect.offset += location; - if (IsScrollContainer()) + + // When capturing the root snapshot for a transition, we paint the background + // color where the scrollbar would be so keep the clip rect the full ViewRect + // size. + auto* supplement = DocumentTransitionSupplement::FromIfExists(GetDocument()); + bool is_in_transition = + supplement && supplement->GetTransition()->IsRootTransitioning(); + if (IsScrollContainer() && !is_in_transition) ExcludeScrollbars(rect, overlay_scrollbar_clip_behavior); return rect;
diff --git a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_data.h b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_data.h index 87cec8f2..7ffc3ed 100644 --- a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_data.h +++ b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_data.h
@@ -22,8 +22,15 @@ // Subgrids need to map named lines from every parent grid. This constructor // should be used exclusively by subgrids to differentiate such scenario. NGGridPlacementData(const ComputedStyle& grid_style, - const NGGridLineResolver& parent_line_resolver) - : line_resolver(grid_style, parent_line_resolver) {} + const NGGridLineResolver& parent_line_resolver, + GridArea subgrid_area) + : line_resolver(grid_style, parent_line_resolver, subgrid_area), + subgridded_column_span_size(subgrid_area.columns.IsTranslatedDefinite() + ? subgrid_area.SpanSize(kForColumns) + : kNotFound), + subgridded_row_span_size(subgrid_area.rows.IsTranslatedDefinite() + ? subgrid_area.SpanSize(kForRows) + : kNotFound) {} // This constructor only copies inputs to the auto-placement algorithm. NGGridPlacementData(const NGGridPlacementData& other)
diff --git a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_layout_algorithm.cc b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_layout_algorithm.cc index cf6580f..3515bab 100644 --- a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_layout_algorithm.cc +++ b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_layout_algorithm.cc
@@ -507,29 +507,30 @@ const auto& style = node.Style(); bool has_nested_subgrid; + auto& sizing_data = sizing_tree->CreateSizingData(node, parent_sizing_data, subgrid_data_in_parent); { + GridArea subgridded_area_in_parent; + if (subgrid_data_in_parent) { + subgridded_area_in_parent = subgrid_data_in_parent->resolved_position; + if (!subgrid_data_in_parent->is_parallel_with_root_grid) { + std::swap(subgridded_area_in_parent.columns, + subgridded_area_in_parent.rows); + } + + if (!subgrid_data_in_parent->has_subgridded_columns) + subgridded_area_in_parent.columns = GridSpan::IndefiniteGridSpan(); + if (!subgrid_data_in_parent->has_subgridded_rows) + subgridded_area_in_parent.rows = GridSpan::IndefiniteGridSpan(); + } + // Initialize this grid's placement data. auto placement_data = - parent_line_resolver ? NGGridPlacementData(style, *parent_line_resolver) + parent_line_resolver ? NGGridPlacementData(style, *parent_line_resolver, + subgridded_area_in_parent) : NGGridPlacementData(style); - if (subgrid_data_in_parent) { - wtf_size_t column_span_size_in_parent = - subgrid_data_in_parent->SpanSize(kForColumns); - wtf_size_t row_span_size_in_parent = - subgrid_data_in_parent->SpanSize(kForRows); - - if (!subgrid_data_in_parent->is_parallel_with_root_grid) - std::swap(column_span_size_in_parent, row_span_size_in_parent); - - if (subgrid_data_in_parent->has_subgridded_columns) - placement_data.subgridded_column_span_size = column_span_size_in_parent; - if (subgrid_data_in_parent->has_subgridded_rows) - placement_data.subgridded_row_span_size = row_span_size_in_parent; - } - // TODO(ethavar): Compute automatic repetitions for subgridded axes as // described in https://drafts.csswg.org/css-grid-2/#auto-repeat. if (!parent_sizing_data) {
diff --git a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.cc b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.cc index fa8a29a..50c0481 100644 --- a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.cc +++ b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.cc
@@ -28,23 +28,73 @@ NGGridLineResolver::NGGridLineResolver( const ComputedStyle& grid_style, - const NGGridLineResolver& parent_line_resolver) - : style_(&grid_style), is_subgrid_line_resolver_(true) { - column_subgrid_merged_grid_line_names_ = - grid_style.GridTemplateColumns().named_grid_lines; - row_subgrid_merged_grid_line_names_ = - grid_style.GridTemplateRows().named_grid_lines; + const NGGridLineResolver& parent_line_resolver, + GridArea subgrid_area) + : style_(&grid_style), + subgrid_column_start_line_(subgrid_area.columns.IsTranslatedDefinite() + ? subgrid_area.StartLine(kForColumns) + : 0), + subgrid_row_start_line_(subgrid_area.rows.IsTranslatedDefinite() + ? subgrid_area.StartLine(kForRows) + : 0), + column_subgrid_merged_grid_line_names_( + grid_style.GridTemplateColumns().named_grid_lines), + row_subgrid_merged_grid_line_names_( + grid_style.GridTemplateRows().named_grid_lines) { + auto MergeNamedGridLinesWithAncestor = [](NamedGridLinesMap& subgrid_map, + const NamedGridLinesMap& parent_map, + GridSpan subgrid_span) -> void { + // Update `subgrid_map` to a merged map from a parent grid or subgrid map + // (`parent_map`). The map is a key-value store with keys as the line name + // and the value as an array of ascending indices. + for (auto& pair : parent_map) { + Vector<wtf_size_t, 16> merged_list; + for (const auto& position : pair.value) { + // Filter out parent named lines are are out of the subgrid range. Also + // offset entries by `subgrid_start_line` before inserting them into the + // merged map so they are all relative to offset 0. These are already in + // ascending order so there's no need to sort. + if ((position >= subgrid_span.StartLine()) && + (position <= + (subgrid_span.StartLine() + subgrid_span.IntegerSpan()))) { + merged_list.push_back(position - subgrid_span.StartLine()); + } + } - // Add the parent grid/subgrid's line numbers to the shared line name set. - // TODO(kschmi): Merge/filter these lists. - for (const auto& pair : - parent_line_resolver.ExplicitNamedLinesMap(kForColumns)) { - column_subgrid_merged_grid_line_names_.insert(pair.key, pair.value); + // If there's a name collision, merge the values and sort. These are from + // the subgrid and not the parent container, so they are already relative + // to index 0 and don't need to be offset. + const auto& existing_entry = subgrid_map.find(pair.key); + if (existing_entry != subgrid_map.end()) { + for (auto& value : existing_entry->value) + merged_list.push_back(value); + std::sort(merged_list.begin(), merged_list.end()); + } + + // Override the existing subgrid's line names map with the new merged list + // for this particular line name entry. If `merged_list` list is empty, + // (this can happen when all entries for a particular line name are out + // of the subgrid range), erase the entry entirely, as + // `NGGridNamedLineCollection` doesn't support named line entries without + // values. + if (merged_list.empty()) + subgrid_map.erase(pair.key); + else + subgrid_map.Set(pair.key, merged_list); + } + }; + + if (subgrid_area.columns.IsTranslatedDefinite()) { + MergeNamedGridLinesWithAncestor( + *column_subgrid_merged_grid_line_names_, + parent_line_resolver.ExplicitNamedLinesMap(kForColumns), + subgrid_area.columns); } - - for (const auto& pair : - parent_line_resolver.ExplicitNamedLinesMap(kForRows)) { - row_subgrid_merged_grid_line_names_.insert(pair.key, pair.value); + if (subgrid_area.rows.IsTranslatedDefinite()) { + MergeNamedGridLinesWithAncestor( + *row_subgrid_merged_grid_line_names_, + parent_line_resolver.ExplicitNamedLinesMap(kForRows), + subgrid_area.rows); } } @@ -134,14 +184,15 @@ int last_line, NGGridNamedLineCollection& lines_collection) const { int start, end; + const int span_position = position.SpanPosition(); if (side == kRowStartSide || side == kColumnStartSide) { - start = LookBackForNamedGridLine(opposite_line - 1, position.SpanPosition(), + start = LookBackForNamedGridLine(opposite_line - 1, span_position, last_line, lines_collection); end = opposite_line; } else { start = opposite_line; - end = LookAheadForNamedGridLine(opposite_line + 1, position.SpanPosition(), - last_line, lines_collection); + end = LookAheadForNamedGridLine(opposite_line + 1, span_position, last_line, + lines_collection); } return GridSpan::UntranslatedDefiniteGridSpan(start, end); @@ -153,7 +204,7 @@ if (subgrid_span_size != kNotFound) return subgrid_span_size; - // TODO(kschmi): Refactor with `is_subgrid_line_resolver_` factored in. + // TODO(kschmi): Refactor so that `subgrid_span_size` isn't necessary. return std::min<wtf_size_t>(std::max(style_->GridTemplateColumns() .track_sizes.NGTrackList() .TrackCountWithoutAutoRepeat() + @@ -168,7 +219,7 @@ if (subgrid_span_size != kNotFound) return subgrid_span_size; - // TODO(kschmi): Refactor with `is_subgrid_line_resolver_` factored in. + // TODO(kschmi): Refactor so that `subgrid_span_size` isn't necessary. return std::min<wtf_size_t>(std::max(style_->GridTemplateRows() .track_sizes.NGTrackList() .TrackCountWithoutAutoRepeat() + @@ -194,7 +245,8 @@ const GridPosition& position, wtf_size_t auto_repeat_tracks_count, GridPositionSide side, - wtf_size_t subgrid_span_size) const { + wtf_size_t subgrid_span_size, + bool is_subgridded_to_parent) const { DCHECK(position.IsSpan()); DCHECK(!position.NamedGridLine().IsNull()); // Negative positions are not allowed per the specification and should have @@ -204,7 +256,6 @@ GridTrackSizingDirection track_direction = DirectionFromSide(side); const auto& implicit_grid_line_names = ImplicitNamedLinesMap(track_direction); const auto& explicit_grid_line_names = ExplicitNamedLinesMap(track_direction); - const auto& computed_grid_track_list = ComputedGridTrackList(track_direction); wtf_size_t last_line = ExplicitGridSizeForSide(side, auto_repeat_tracks_count, @@ -213,7 +264,7 @@ NGGridNamedLineCollection lines_collection( position.NamedGridLine(), track_direction, implicit_grid_line_names, explicit_grid_line_names, computed_grid_track_list, last_line, - auto_repeat_tracks_count); + auto_repeat_tracks_count, is_subgridded_to_parent); return DefiniteGridSpanWithNamedSpanAgainstOpposite( opposite_line, position, side, last_line, lines_collection); } @@ -234,7 +285,7 @@ const NamedGridLinesMap& NGGridLineResolver::ImplicitNamedLinesMap( GridTrackSizingDirection track_direction) const { - // TODO(kschmi): Merge implicit list if `is_subgrid_line_resolver_`. + // TODO(kschmi): Merge implicit list if it's subgridded. return (track_direction == kForColumns) ? style_->ImplicitNamedGridColumnLines() : style_->ImplicitNamedGridRowLines(); @@ -242,20 +293,19 @@ const NamedGridLinesMap& NGGridLineResolver::ExplicitNamedLinesMap( GridTrackSizingDirection track_direction) const { - // Subgrids look at the merged map of the parent's grid line names, while - // standalone grids should look directly at the style object. - if (is_subgrid_line_resolver_) { - return (track_direction == kForColumns) - ? column_subgrid_merged_grid_line_names_ - : row_subgrid_merged_grid_line_names_; - } - return ComputedGridTrackList(track_direction).named_grid_lines; + const auto& subgrid_merged_grid_line_names = + (track_direction == kForColumns) ? column_subgrid_merged_grid_line_names_ + : row_subgrid_merged_grid_line_names_; + + return subgrid_merged_grid_line_names + ? *subgrid_merged_grid_line_names + : ComputedGridTrackList(track_direction).named_grid_lines; } const blink::ComputedGridTrackList& NGGridLineResolver::ComputedGridTrackList( GridTrackSizingDirection track_direction) const { - // TODO(kschmi): Refactor so this isn't necessary when - // `is_subgrid_line_resolver_`. + // TODO(kschmi): Refactor so this isn't necessary and handle auto-repeats + // for subgrids. return (track_direction == kForColumns) ? style_->GridTemplateColumns() : style_->GridTemplateRows(); } @@ -265,7 +315,8 @@ const GridPosition& position, GridPositionSide side, wtf_size_t auto_repeat_tracks_count, - wtf_size_t subgrid_span_size) const { + wtf_size_t subgrid_span_size, + bool is_subgridded_to_parent) const { if (position.IsAuto()) { if (side == kColumnStartSide || side == kRowStartSide) { return GridSpan::UntranslatedDefiniteGridSpan(opposite_line - 1, @@ -283,7 +334,7 @@ // our opposite position. return ResolveNamedGridLinePositionAgainstOppositePosition( opposite_line, position, auto_repeat_tracks_count, side, - subgrid_span_size); + subgrid_span_size, is_subgridded_to_parent); } return DefiniteGridSpanWithSpanAgainstOpposite(opposite_line, position, side); @@ -315,11 +366,12 @@ return SpanSizeFromPositions(initial_position, final_position); } -int NGGridLineResolver::ResolveNamedGridLinePositionFromStyle( +int NGGridLineResolver::ResolveNamedGridLinePosition( const GridPosition& position, GridPositionSide side, wtf_size_t auto_repeat_tracks_count, - wtf_size_t subgrid_span_size) const { + wtf_size_t subgrid_span_size, + bool is_subgridded_to_parent) const { DCHECK(!position.NamedGridLine().IsNull()); wtf_size_t last_line = ExplicitGridSizeForSide(side, auto_repeat_tracks_count, @@ -330,8 +382,8 @@ const auto& track_list = ComputedGridTrackList(track_direction); NGGridNamedLineCollection lines_collection( position.NamedGridLine(), track_direction, implicit_grid_line_names, - explicit_grid_line_names, track_list, last_line, - auto_repeat_tracks_count); + explicit_grid_line_names, track_list, last_line, auto_repeat_tracks_count, + is_subgridded_to_parent); if (position.IsPositive()) { return LookAheadForNamedGridLine(0, abs(position.IntegerPosition()), @@ -342,19 +394,29 @@ last_line, lines_collection); } -int NGGridLineResolver::ResolveGridPositionFromStyle( +int NGGridLineResolver::ResolveGridPosition( const GridPosition& position, GridPositionSide side, wtf_size_t auto_repeat_tracks_count, bool is_subgridded_to_parent, wtf_size_t subgrid_span_size) const { + auto track_direction = DirectionFromSide(side); + + // TODO(kschmi): Remove `subgrid_offset` once `auto_repeat_tracks_count` is + // correct. + const int subgrid_offset = (track_direction == kForColumns) + ? subgrid_column_start_line_.value_or(0) + : subgrid_row_start_line_.value_or(0); switch (position.GetType()) { case kExplicitPosition: { DCHECK(position.IntegerPosition()); + // `ResolveNamedGridLinePosition` already factors in + // `subgrid_offset` via `ExplicitGridSizeForSide`. if (!position.NamedGridLine().IsNull()) { - return ResolveNamedGridLinePositionFromStyle( - position, side, auto_repeat_tracks_count, subgrid_span_size); + return ResolveNamedGridLinePosition( + position, side, auto_repeat_tracks_count, subgrid_span_size, + is_subgridded_to_parent); } // Handle <integer> explicit position. @@ -378,7 +440,6 @@ wtf_size_t last_line = ExplicitGridSizeForSide( side, auto_repeat_tracks_count, subgrid_span_size); - GridTrackSizingDirection track_direction = DirectionFromSide(side); const auto& implicit_grid_line_names = ImplicitNamedLinesMap(track_direction); const auto& explicit_grid_line_names = @@ -390,7 +451,7 @@ implicit_grid_line_names, explicit_grid_line_names, track_list, last_line, auto_repeat_tracks_count); if (implicit_lines.HasNamedLines()) - return implicit_lines.FirstPosition(); + return implicit_lines.FirstPosition() + subgrid_offset; // Otherwise, if there is a named line with the specified name, // contributes the first such line to the grid item's placement. @@ -399,11 +460,11 @@ explicit_grid_line_names, track_list, last_line, auto_repeat_tracks_count, is_subgridded_to_parent); if (explicit_lines.HasNamedLines()) - return explicit_lines.FirstPosition(); + return explicit_lines.FirstPosition() + subgrid_offset; // If none of the above works specs mandate to assume that all the lines // in the implicit grid have this name. - return last_line + 1; + return last_line + subgrid_offset + 1; } case kAutoPosition: case kSpanPosition: @@ -447,31 +508,31 @@ if (initial_should_be_resolved_against_opposite_position) { // Infer the position from the final_position position ('auto / 1' or 'span // 2 / 3' case). - int end_line = ResolveGridPositionFromStyle( + int end_line = ResolveGridPosition( final_position, final_side, auto_repeat_tracks_count, is_subgridded_to_parent, subgrid_span_size); return ResolveGridPositionAgainstOppositePosition( end_line, initial_position, initial_side, auto_repeat_tracks_count, - subgrid_span_size); + subgrid_span_size, is_subgridded_to_parent); } if (final_should_be_resolved_against_opposite_position) { // Infer our position from the initial_position position ('1 / auto' or '3 / // span 2' case). - int start_line = ResolveGridPositionFromStyle( + int start_line = ResolveGridPosition( initial_position, initial_side, auto_repeat_tracks_count, is_subgridded_to_parent, subgrid_span_size); return ResolveGridPositionAgainstOppositePosition( start_line, final_position, final_side, auto_repeat_tracks_count, - subgrid_span_size); + subgrid_span_size, is_subgridded_to_parent); } - int start_line = ResolveGridPositionFromStyle( + int start_line = ResolveGridPosition( initial_position, initial_side, auto_repeat_tracks_count, is_subgridded_to_parent, subgrid_span_size); - int end_line = ResolveGridPositionFromStyle( - final_position, final_side, auto_repeat_tracks_count, - is_subgridded_to_parent, subgrid_span_size); + int end_line = + ResolveGridPosition(final_position, final_side, auto_repeat_tracks_count, + is_subgridded_to_parent, subgrid_span_size); if (end_line < start_line) std::swap(end_line, start_line);
diff --git a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.h b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.h index ac8a4f47..07eb2890 100644 --- a/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.h +++ b/third_party/blink/renderer/core/layout/ng/grid/ng_grid_line_resolver.h
@@ -5,7 +5,9 @@ #ifndef THIRD_PARTY_BLINK_RENDERER_CORE_LAYOUT_NG_GRID_NG_GRID_LINE_RESOLVER_H_ #define THIRD_PARTY_BLINK_RENDERER_CORE_LAYOUT_NG_GRID_NG_GRID_LINE_RESOLVER_H_ +#include "third_party/abseil-cpp/absl/types/optional.h" #include "third_party/blink/renderer/core/style/computed_grid_track_list.h" +#include "third_party/blink/renderer/core/style/grid_area.h" #include "third_party/blink/renderer/core/style/grid_enums.h" #include "third_party/blink/renderer/core/style/named_grid_lines_map.h" #include "third_party/blink/renderer/platform/wtf/allocator/allocator.h" @@ -27,12 +29,13 @@ NGGridLineResolver() = default; explicit NGGridLineResolver(const ComputedStyle& grid_style) - : style_(&grid_style), is_subgrid_line_resolver_(false) {} + : style_(&grid_style) {} // Subgrids need to map named lines from every parent grid. This constructor // should be used exclusively by subgrids to differentiate such scenario. explicit NGGridLineResolver(const ComputedStyle& grid_style, - const NGGridLineResolver& parent_line_resolver); + const NGGridLineResolver& parent_line_resolver, + GridArea subgrid_area); wtf_size_t ExplicitGridColumnCount( wtf_size_t auto_repeat_columns_count, @@ -67,20 +70,22 @@ const GridPosition& position, GridPositionSide side, wtf_size_t auto_repeat_tracks_count, - wtf_size_t subgrid_span_size) const; + wtf_size_t subgrid_span_size, + bool is_subgridded_to_parent) const; GridSpan ResolveNamedGridLinePositionAgainstOppositePosition( int opposite_line, const GridPosition& position, wtf_size_t auto_repeat_tracks_count, GridPositionSide side, - wtf_size_t subgrid_span_size) const; + wtf_size_t subgrid_span_size, + bool is_subgridded_to_parent) const; - int ResolveGridPositionFromStyle(const GridPosition& position, - GridPositionSide side, - wtf_size_t auto_repeat_tracks_count, - bool is_subgridded_to_parent, - wtf_size_t subgrid_span_size) const; + int ResolveGridPosition(const GridPosition& position, + GridPositionSide side, + wtf_size_t auto_repeat_tracks_count, + bool is_subgridded_to_parent, + wtf_size_t subgrid_span_size) const; wtf_size_t ExplicitGridSizeForSide(GridPositionSide side, wtf_size_t auto_repeat_tracks_count, @@ -101,10 +106,11 @@ wtf_size_t SpanSizeFromPositions(const GridPosition& initial_position, const GridPosition& final_position) const; - int ResolveNamedGridLinePositionFromStyle(const GridPosition& position, - GridPositionSide side, - wtf_size_t auto_repeat_tracks_count, - wtf_size_t subgrid_span_size) const; + int ResolveNamedGridLinePosition(const GridPosition& position, + GridPositionSide side, + wtf_size_t auto_repeat_tracks_count, + wtf_size_t subgrid_span_size, + bool is_subgridded_to_parent) const; void InitialAndFinalPositionsFromStyle( const ComputedStyle& grid_item_style, @@ -121,10 +127,13 @@ scoped_refptr<const ComputedStyle> style_; - bool is_subgrid_line_resolver_ : 1; + // TODO(kschmi) remove these, as they will be unnecessary when the rest of + // the line resolution work is completed. + absl::optional<wtf_size_t> subgrid_column_start_line_; + absl::optional<wtf_size_t> subgrid_row_start_line_; - NamedGridLinesMap column_subgrid_merged_grid_line_names_; - NamedGridLinesMap row_subgrid_merged_grid_line_names_; + absl::optional<NamedGridLinesMap> column_subgrid_merged_grid_line_names_; + absl::optional<NamedGridLinesMap> row_subgrid_merged_grid_line_names_; }; } // namespace blink
diff --git a/third_party/blink/renderer/core/loader/empty_clients.h b/third_party/blink/renderer/core/loader/empty_clients.h index ba841a4..f876e56 100644 --- a/third_party/blink/renderer/core/loader/empty_clients.h +++ b/third_party/blink/renderer/core/loader/empty_clients.h
@@ -358,7 +358,9 @@ HTMLMediaElement&) override; void DidCommitDocumentReplacementNavigation(DocumentLoader*) override {} - void DispatchDidClearWindowObjectInMainWorld() override {} + void DispatchDidClearWindowObjectInMainWorld( + v8::Isolate* isolate, + v8::MicrotaskQueue* microtask_queue) override {} void DocumentElementAvailable() override {} void RunScriptsAtDocumentElementAvailable() override {} void RunScriptsAtDocumentReady(bool) override {}
diff --git a/third_party/blink/renderer/core/loader/frame_loader.cc b/third_party/blink/renderer/core/loader/frame_loader.cc index b0c3dfe..bf017fd 100644 --- a/third_party/blink/renderer/core/loader/frame_loader.cc +++ b/third_party/blink/renderer/core/loader/frame_loader.cc
@@ -1729,18 +1729,21 @@ &dispatching_did_clear_window_object_in_main_world_, true); // We just cleared the document, not the entire window object, but for the // embedder that's close enough. - Client()->DispatchDidClearWindowObjectInMainWorld(); + Client()->DispatchDidClearWindowObjectInMainWorld( + window->GetIsolate(), window->GetMicrotaskQueue()); } void FrameLoader::DispatchDidClearWindowObjectInMainWorld() { - if (!frame_->DomWindow()->CanExecuteScripts(kNotAboutToExecuteScript)) + LocalDOMWindow* window = frame_->DomWindow(); + if (!window->CanExecuteScripts(kNotAboutToExecuteScript)) return; if (dispatching_did_clear_window_object_in_main_world_) return; base::AutoReset<bool> in_did_clear_window_object( &dispatching_did_clear_window_object_in_main_world_, true); - Client()->DispatchDidClearWindowObjectInMainWorld(); + Client()->DispatchDidClearWindowObjectInMainWorld( + window->GetIsolate(), window->GetMicrotaskQueue()); } network::mojom::blink::WebSandboxFlags
diff --git a/third_party/blink/renderer/core/paint/paint_layer_scrollable_area.cc b/third_party/blink/renderer/core/paint/paint_layer_scrollable_area.cc index 305a123..8f62d7a 100644 --- a/third_party/blink/renderer/core/paint/paint_layer_scrollable_area.cc +++ b/third_party/blink/renderer/core/paint/paint_layer_scrollable_area.cc
@@ -61,6 +61,8 @@ #include "third_party/blink/renderer/core/animation/scroll_timeline.h" #include "third_party/blink/renderer/core/content_capture/content_capture_manager.h" #include "third_party/blink/renderer/core/css/style_request.h" +#include "third_party/blink/renderer/core/document_transition/document_transition.h" +#include "third_party/blink/renderer/core/document_transition/document_transition_supplement.h" #include "third_party/blink/renderer/core/dom/dom_node_ids.h" #include "third_party/blink/renderer/core/dom/node.h" #include "third_party/blink/renderer/core/dom/shadow_root.h" @@ -702,8 +704,23 @@ gfx::Size PaintLayerScrollableArea::PixelSnappedContentsSize( const PhysicalOffset& paint_offset) const { - return ToPixelSnappedRect(PhysicalRect(paint_offset, overflow_rect_.size)) - .size(); + PhysicalSize size = overflow_rect_.size; + + // If we're capturing a transition snapshot, ensure the content size is + // considered at least as large as the container. Otherwise, the snapshot + // will be clipped by PendingLayer to the content size. + if (IsA<LayoutView>(GetLayoutBox())) { + auto* supplement = DocumentTransitionSupplement::FromIfExists( + GetLayoutBox()->GetDocument()); + if (supplement && supplement->GetTransition()->IsRootTransitioning()) { + PhysicalSize container_size( + supplement->GetTransition()->GetSnapshotViewportRect().size()); + size.width = std::max(container_size.width, size.width); + size.height = std::max(container_size.height, size.height); + } + } + + return ToPixelSnappedRect(PhysicalRect(paint_offset, size)).size(); } void PaintLayerScrollableArea::ContentsResized() {
diff --git a/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc b/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc index 466479f..1fae5b6 100644 --- a/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc +++ b/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc
@@ -2910,7 +2910,8 @@ // origin rather than behind the UI. if (auto* supplement = DocumentTransitionSupplement::FromIfExists(object_.GetDocument())) { - if (object_.IsDocumentElement() && !supplement->GetTransition()->IsIdle()) { + if (object_.IsDocumentElement() && + supplement->GetTransition()->IsRootTransitioning()) { PhysicalOffset offset = PhysicalOffset( supplement->GetTransition()->GetRootSnapshotPaintOffset()); context_.current.paint_offset += offset;
diff --git a/third_party/blink/renderer/core/paint/video_painter_test.cc b/third_party/blink/renderer/core/paint/video_painter_test.cc index ddc26ba..f1c265d 100644 --- a/third_party/blink/renderer/core/paint/video_painter_test.cc +++ b/third_party/blink/renderer/core/paint/video_painter_test.cc
@@ -140,8 +140,10 @@ EXPECT_EQ(gfx::Size(300, 150), layer->bounds()); } -class MockWebMediaPlayer : public EmptyWebMediaPlayer { +class MockWebMediaPlayer : public StubWebMediaPlayer { public: + explicit MockWebMediaPlayer(WebMediaPlayerClient* client) + : StubWebMediaPlayer(client) {} MOCK_CONST_METHOD0(HasAvailableVideoFrame, bool()); MOCK_METHOD3(Paint, void(cc::PaintCanvas*, const gfx::Rect&, cc::PaintFlags&)); @@ -151,14 +153,17 @@ public: WebMediaPlayer* CreateMediaPlayer( const WebMediaPlayerSource&, - WebMediaPlayerClient*, + WebMediaPlayerClient* client, blink::MediaInspectorContext*, WebMediaPlayerEncryptedMediaClient*, WebContentDecryptionModule*, const WebString& sink_id, const cc::LayerTreeSettings& settings, scoped_refptr<base::TaskRunner> compositor_worker_task_runner) override { - return new MockWebMediaPlayer(); + MockWebMediaPlayer* player = new MockWebMediaPlayer(client); + EXPECT_CALL(*player, HasAvailableVideoFrame) + .WillRepeatedly(testing::Return(false)); + return player; } }; @@ -282,6 +287,10 @@ ASSERT_TRUE(PlayVideo()); // Capture using poster. + auto* element = To<HTMLMediaElement>(GetDocument().body()->firstChild()); + MockWebMediaPlayer* player = + static_cast<MockWebMediaPlayer*>(element->GetWebMediaPlayer()); + EXPECT_CALL(*player, Paint(testing::_, testing::_, testing::_)).Times(0); auto record = CapturePaintPreview(/*skip_accelerated_content=*/true); std::vector<std::pair<GURL, SkRect>> links; @@ -294,6 +303,7 @@ EXPECT_EQ(1U, CountImagesOfType(record.get(), cc::ImageType::kGIF)); // Capture using video frame. + EXPECT_CALL(*player, Paint(testing::_, testing::_, testing::_)); record = CapturePaintPreview(/*skip_accelerated_content=*/false); links.clear();
diff --git a/third_party/blink/renderer/core/permissions_policy/permissions_policy_test.cc b/third_party/blink/renderer/core/permissions_policy/permissions_policy_test.cc index b0091d25..c1617dd 100644 --- a/third_party/blink/renderer/core/permissions_policy/permissions_policy_test.cc +++ b/third_party/blink/renderer/core/permissions_policy/permissions_policy_test.cc
@@ -8,9 +8,7 @@ #include <string> #include "base/ranges/algorithm.h" -#include "base/test/scoped_feature_list.h" #include "testing/gtest/include/gtest/gtest.h" -#include "third_party/blink/public/common/features.h" #include "third_party/blink/public/common/permissions_policy/origin_with_possible_wildcards.h" #include "third_party/blink/public/mojom/permissions_policy/permissions_policy.mojom-blink.h" #include "third_party/blink/renderer/core/frame/local_dom_window.h" @@ -140,7 +138,6 @@ const char* permissions_policy_string; const char* self_origin; const char* src_origin; - const bool subdomain_wildcards; // Test expectation. ParsedPolicyForTest expected_parse_result; @@ -222,8 +219,6 @@ } } - base::test::ScopedFeatureList scoped_feature_list_; - public: static const PermissionsPolicyParserTestCase kCases[]; }; @@ -236,7 +231,6 @@ /* permissions_policy_string */ "", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, /* expected_parse_result */ {}, }, { @@ -245,7 +239,6 @@ /* permissions_policy_string */ "geolocation=self", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -262,7 +255,6 @@ /* permissions_policy_string */ "geolocation=(self)", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -279,7 +271,6 @@ /* permissions_policy_string */ "geolocation=*", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -302,7 +293,6 @@ "payment=self", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -338,7 +328,6 @@ "payment=(self \"badorigin\")", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -372,7 +361,6 @@ "geolocation=self,fullscreen=self,payment=self", /* self_origin */ ORIGIN_A, /* src_origin */ nullptr, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -401,7 +389,6 @@ /* permissions_policy_string */ "", /* self_origin */ ORIGIN_A, /* src_origin */ OPAQUE_ORIGIN, - /* subdomain_wildcards */ false, /* expected_parse_result */ {}, }, { @@ -410,7 +397,6 @@ /* permissions_policy_string */ NOT_APPLICABLE, /* self_origin */ ORIGIN_A, /* src_origin */ OPAQUE_ORIGIN, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -427,7 +413,6 @@ /* permissions_policy_string */ NOT_APPLICABLE, /* self_origin */ ORIGIN_A, /* src_origin */ OPAQUE_ORIGIN, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -444,7 +429,6 @@ /* permissions_policy_string */ "geolocation=*", /* self_origin */ ORIGIN_A, /* src_origin */ OPAQUE_ORIGIN, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -462,7 +446,6 @@ "geolocation=(\"" ORIGIN_B "\" \"" ORIGIN_C "\")", /* self_origin */ ORIGIN_A, /* src_origin */ OPAQUE_ORIGIN, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -481,7 +464,6 @@ /* permissions_policy_string */ NOT_APPLICABLE, /* self_origin */ ORIGIN_A, /* src_origin */ OPAQUE_ORIGIN, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -499,7 +481,6 @@ /* permissions_policy_string */ "geolocation=9", /* self_origin */ ORIGIN_A, /* src_origin */ nullptr, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -517,7 +498,6 @@ /* permissions_policy_string */ "geolocation=1.1", /* self_origin */ ORIGIN_A, /* src_origin */ nullptr, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -535,7 +515,6 @@ /* permissions_policy_string */ "geolocation=?0", /* self_origin */ ORIGIN_A, /* src_origin */ nullptr, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -552,7 +531,6 @@ /* permissions_policy_string */ "geolocation=\"\"", /* self_origin */ ORIGIN_A, /* src_origin */ nullptr, - /* subdomain_wildcards */ false, /* expected_parse_result */ { { @@ -564,34 +542,12 @@ }, }, { - /* test_name */ "ProperWildcardIncludedWhileFeatureDisabled", - /* feature_policy_string */ - "fullscreen " ORIGIN_A_SUBDOMAIN_WILDCARD, - /* permissions_policy_string */ - "fullscreen=(\"" ORIGIN_A_SUBDOMAIN_WILDCARD "\")", - /* self_origin */ ORIGIN_A, - /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, - /* expected_parse_result */ - { - { - mojom::blink::PermissionsPolicyFeature::kFullscreen, - /* matches_all_origins */ false, - /* matches_opaque_src */ false, - {{ORIGIN_A_SUBDOMAIN_ESCAPED, - /*has_subdomain_wildcard=*/false}}, - }, - }, - }, - { - /* test_name */ "ProperWildcardIncludedWhileFeatureEnabledForFeatur" - "ePolicy", + /* test_name */ "ProperWildcardIncludedForFeaturePolicy", /* feature_policy_string */ "fullscreen " ORIGIN_A_SUBDOMAIN_WILDCARD, /* permissions_policy_string */ NOT_APPLICABLE, /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ true, /* expected_parse_result */ { { @@ -604,14 +560,12 @@ }, }, { - /* test_name */ "ProperWildcardIncludedWhileFeatureEnabledForPermis" - "sionsPolicy", + /* test_name */ "ProperWildcardIncludedForPermissionsPolicy", /* feature_policy_string */ NOT_APPLICABLE, /* permissions_policy_string */ "fullscreen=(\"" ORIGIN_A_SUBDOMAIN_WILDCARD "\")", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ true, /* expected_parse_result */ { { @@ -624,7 +578,7 @@ }, }, { - /* test_name */ "ImproperWildcardsIncludedWhileFeatureDisabled", + /* test_name */ "ImproperWildcardsIncluded", /* feature_policy_string */ "fullscreen *://example.com https://foo.*.example.com " "https://*.*.example.com https://example.com:*", @@ -633,31 +587,6 @@ "\"https://*.*.example.com\" \"https://example.com:*\")", /* self_origin */ ORIGIN_A, /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ false, - /* expected_parse_result */ - { - { - mojom::blink::PermissionsPolicyFeature::kFullscreen, - /* matches_all_origins */ false, - /* matches_opaque_src */ false, - {{"https://%2A.%2A.example.com", - /*has_subdomain_wildcard=*/false}, - {"https://foo.%2A.example.com", - /*has_subdomain_wildcard=*/false}}, - }, - }, - }, - { - /* test_name */ "ImproperWildcardsIncludedWhileFeatureEnabled", - /* feature_policy_string */ - "fullscreen *://example.com https://foo.*.example.com " - "https://*.*.example.com https://example.com:*", - /* permissions_policy_string */ - "fullscreen=(\"*://example.com\" \"https://foo.*.example.com\" " - "\"https://*.*.example.com\" \"https://example.com:*\")", - /* self_origin */ ORIGIN_A, - /* src_origin */ ORIGIN_B, - /* subdomain_wildcards */ true, /* expected_parse_result */ { { @@ -687,9 +616,6 @@ return; ASSERT_NE(test_case.self_origin, nullptr); - scoped_feature_list_.InitWithFeatureState( - features::kWildcardSubdomainsInPermissionsPolicy, - test_case.subdomain_wildcards); CheckParsedPolicy( ParseFeaturePolicy(test_case.feature_policy_string, test_case.self_origin, test_case.src_origin, logger, test_feature_name_map), @@ -703,9 +629,6 @@ return; ASSERT_NE(test_case.self_origin, nullptr); - scoped_feature_list_.InitWithFeatureState( - features::kWildcardSubdomainsInPermissionsPolicy, - test_case.subdomain_wildcards); CheckParsedPolicy( ParsePermissionsPolicy(test_case.permissions_policy_string, test_case.self_origin, test_case.src_origin,
diff --git a/third_party/blink/renderer/core/permissions_policy/policy_test.cc b/third_party/blink/renderer/core/permissions_policy/policy_test.cc index f9a14993..055de97 100644 --- a/third_party/blink/renderer/core/permissions_policy/policy_test.cc +++ b/third_party/blink/renderer/core/permissions_policy/policy_test.cc
@@ -2,10 +2,8 @@ // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. -#include "base/test/scoped_feature_list.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" -#include "third_party/blink/public/common/features.h" #include "third_party/blink/renderer/core/frame/local_dom_window.h" #include "third_party/blink/renderer/core/frame/local_frame.h" #include "third_party/blink/renderer/core/permissions_policy/dom_feature_policy.h" @@ -26,12 +24,9 @@ using testing::UnorderedElementsAre; -class PolicyTest : public testing::TestWithParam<bool> { +class PolicyTest : public testing::Test { public: void SetUp() override { - scoped_feature_list_.InitWithFeatureState( - features::kWildcardSubdomainsInPermissionsPolicy, - HasWildcardSubdomainsInPermissionsPolicy()); page_holder_ = std::make_unique<DummyPageHolder>(); auto origin = SecurityOrigin::CreateFromString(kSelfOrigin); @@ -53,15 +48,12 @@ DOMFeaturePolicy* GetPolicy() const { return policy_; } - bool HasWildcardSubdomainsInPermissionsPolicy() { return GetParam(); } - PolicyParserMessageBuffer dummy_logger_ = PolicyParserMessageBuffer("", true /* discard_message */); protected: std::unique_ptr<DummyPageHolder> page_holder_; Persistent<DOMFeaturePolicy> policy_; - base::test::ScopedFeatureList scoped_feature_list_; }; class DOMFeaturePolicyTest : public PolicyTest { @@ -73,8 +65,6 @@ } }; -INSTANTIATE_TEST_SUITE_P(All, DOMFeaturePolicyTest, testing::Bool()); - class IFramePolicyTest : public PolicyTest { public: void SetUp() override { @@ -85,9 +75,7 @@ } }; -INSTANTIATE_TEST_SUITE_P(All, IFramePolicyTest, testing::Bool()); - -TEST_P(DOMFeaturePolicyTest, TestAllowsFeature) { +TEST_F(DOMFeaturePolicyTest, TestAllowsFeature) { EXPECT_FALSE(GetPolicy()->allowsFeature(nullptr, "badfeature")); EXPECT_FALSE(GetPolicy()->allowsFeature(nullptr, "midi")); EXPECT_FALSE(GetPolicy()->allowsFeature(nullptr, "midi", kSelfOrigin)); @@ -106,15 +94,14 @@ EXPECT_TRUE(GetPolicy()->allowsFeature(nullptr, "sync-xhr", kOriginA)); EXPECT_TRUE(GetPolicy()->allowsFeature(nullptr, "gyroscope")); EXPECT_FALSE(GetPolicy()->allowsFeature(nullptr, "gyroscope", kOriginA)); - EXPECT_EQ( - HasWildcardSubdomainsInPermissionsPolicy(), + EXPECT_TRUE( GetPolicy()->allowsFeature(nullptr, "gyroscope", kOriginASubdomain)); EXPECT_TRUE(GetPolicy()->allowsFeature(nullptr, "gyroscope", kOriginB)); EXPECT_FALSE( GetPolicy()->allowsFeature(nullptr, "gyroscope", kOriginBSubdomain)); } -TEST_P(DOMFeaturePolicyTest, TestGetAllowList) { +TEST_F(DOMFeaturePolicyTest, TestGetAllowList) { EXPECT_THAT(GetPolicy()->getAllowlistForFeature(nullptr, "camera"), UnorderedElementsAre(kSelfOrigin, kOriginA, kOriginB)); EXPECT_THAT(GetPolicy()->getAllowlistForFeature(nullptr, "payment"), @@ -128,14 +115,12 @@ EXPECT_TRUE(GetPolicy()->getAllowlistForFeature(nullptr, "midi").empty()); EXPECT_THAT(GetPolicy()->getAllowlistForFeature(nullptr, "sync-xhr"), UnorderedElementsAre("*")); - EXPECT_THAT(GetPolicy()->getAllowlistForFeature(nullptr, "gyroscope"), - UnorderedElementsAre(kSelfOrigin, kOriginB, - HasWildcardSubdomainsInPermissionsPolicy() - ? "https://*.example.com" - : "https://%2A.example.com")); + EXPECT_THAT( + GetPolicy()->getAllowlistForFeature(nullptr, "gyroscope"), + UnorderedElementsAre(kSelfOrigin, kOriginB, "https://*.example.com")); } -TEST_P(DOMFeaturePolicyTest, TestAllowedFeatures) { +TEST_F(DOMFeaturePolicyTest, TestAllowedFeatures) { Vector<String> allowed_features = GetPolicy()->allowedFeatures(nullptr); EXPECT_TRUE(allowed_features.Contains("fullscreen")); EXPECT_TRUE(allowed_features.Contains("payment")); @@ -149,7 +134,7 @@ EXPECT_TRUE(allowed_features.Contains("sync-xhr")); } -TEST_P(IFramePolicyTest, TestAllowsFeature) { +TEST_F(IFramePolicyTest, TestAllowsFeature) { EXPECT_FALSE(GetPolicy()->allowsFeature(nullptr, "badfeature")); EXPECT_FALSE(GetPolicy()->allowsFeature(nullptr, "midi")); EXPECT_FALSE(GetPolicy()->allowsFeature(nullptr, "midi", kSelfOrigin)); @@ -176,7 +161,7 @@ GetPolicy()->allowsFeature(nullptr, "gyroscope", kOriginBSubdomain)); } -TEST_P(IFramePolicyTest, TestGetAllowList) { +TEST_F(IFramePolicyTest, TestGetAllowList) { EXPECT_THAT(GetPolicy()->getAllowlistForFeature(nullptr, "camera"), UnorderedElementsAre(kSelfOrigin)); EXPECT_THAT(GetPolicy()->getAllowlistForFeature(nullptr, "payment"), @@ -194,7 +179,7 @@ UnorderedElementsAre(kSelfOrigin)); } -TEST_P(IFramePolicyTest, TestSameOriginAllowedFeatures) { +TEST_F(IFramePolicyTest, TestSameOriginAllowedFeatures) { Vector<String> allowed_features = GetPolicy()->allowedFeatures(nullptr); // These features are allowed in a same origin context, and not restricted by // the parent document's policy. @@ -211,7 +196,7 @@ EXPECT_FALSE(allowed_features.Contains("badfeature")); } -TEST_P(IFramePolicyTest, TestCrossOriginAllowedFeatures) { +TEST_F(IFramePolicyTest, TestCrossOriginAllowedFeatures) { // Update the iframe's policy, given a new origin. GetPolicy()->UpdateContainerPolicy( ParsedPermissionsPolicy(), SecurityOrigin::CreateFromString(kOriginA)); @@ -229,7 +214,7 @@ EXPECT_FALSE(allowed_features.Contains("badfeature")); } -TEST_P(IFramePolicyTest, TestCombinedPolicyOnOriginA) { +TEST_F(IFramePolicyTest, TestCombinedPolicyOnOriginA) { ParsedPermissionsPolicy container_policy = PermissionsPolicyParser::ParseAttribute( "geolocation 'src'; payment 'none'; midi; camera 'src'; gyroscope " @@ -255,7 +240,7 @@ EXPECT_FALSE(allowed_features.Contains("badfeature")); } -TEST_P(IFramePolicyTest, TestCombinedPolicyOnOriginASubdomain) { +TEST_F(IFramePolicyTest, TestCombinedPolicyOnOriginASubdomain) { ParsedPermissionsPolicy container_policy = PermissionsPolicyParser::ParseAttribute( "geolocation 'src'; payment 'none'; midi; camera 'src'; gyroscope " @@ -273,16 +258,15 @@ // These are allowed by the attribute, but still blocked by the parent policy. EXPECT_FALSE(allowed_features.Contains("midi")); EXPECT_FALSE(allowed_features.Contains("camera")); - // These features are allowed if subdomain wildcard matching is on. - EXPECT_EQ(HasWildcardSubdomainsInPermissionsPolicy(), - allowed_features.Contains("gyroscope")); + // These features are allowed via wildcard matching. + EXPECT_TRUE(allowed_features.Contains("gyroscope")); // "sync-xhr" is still implicitly allowed on all origins. EXPECT_TRUE(allowed_features.Contains("sync-xhr")); // This feature does not exist, so should not be advertised as allowed. EXPECT_FALSE(allowed_features.Contains("badfeature")); } -TEST_P(IFramePolicyTest, TestCombinedPolicyOnOriginB) { +TEST_F(IFramePolicyTest, TestCombinedPolicyOnOriginB) { ParsedPermissionsPolicy container_policy = PermissionsPolicyParser::ParseAttribute( "geolocation 'src'; payment 'none'; midi; camera 'src'; gyroscope " @@ -307,7 +291,7 @@ EXPECT_FALSE(allowed_features.Contains("badfeature")); } -TEST_P(IFramePolicyTest, TestCombinedPolicyOnOriginBSubdomain) { +TEST_F(IFramePolicyTest, TestCombinedPolicyOnOriginBSubdomain) { ParsedPermissionsPolicy container_policy = PermissionsPolicyParser::ParseAttribute( "geolocation 'src'; payment 'none'; midi; camera 'src'; gyroscope "
diff --git a/third_party/blink/renderer/core/style/ordered_named_grid_lines.h b/third_party/blink/renderer/core/style/ordered_named_grid_lines.h index 9040ca79..5f169db 100644 --- a/third_party/blink/renderer/core/style/ordered_named_grid_lines.h +++ b/third_party/blink/renderer/core/style/ordered_named_grid_lines.h
@@ -7,13 +7,12 @@ #include "third_party/blink/renderer/platform/wtf/hash_map.h" #include "third_party/blink/renderer/platform/wtf/hash_traits.h" -#include "third_party/blink/renderer/platform/wtf/text/wtf_string.h" #include "third_party/blink/renderer/platform/wtf/vector.h" namespace blink { struct NamedGridLine { - explicit NamedGridLine(const String line_name, + explicit NamedGridLine(const AtomicString& line_name, bool is_in_repeat = false, bool is_first_repeat = false) : line_name(line_name), @@ -25,9 +24,9 @@ is_first_repeat == other.is_first_repeat; } - String line_name; - bool is_in_repeat; - bool is_first_repeat; + AtomicString line_name; + bool is_in_repeat : 1; + bool is_first_repeat : 1; }; using OrderedNamedGridLines =
diff --git a/third_party/blink/renderer/core/workers/threaded_worklet_test.cc b/third_party/blink/renderer/core/workers/threaded_worklet_test.cc index 00eb640c..b51c212 100644 --- a/third_party/blink/renderer/core/workers/threaded_worklet_test.cc +++ b/third_party/blink/renderer/core/workers/threaded_worklet_test.cc
@@ -235,6 +235,7 @@ private: friend class ThreadedWorkletTest; + FRIEND_TEST_ALL_PREFIXES(ThreadedWorkletTest, NestedRunLoopTermination); std::unique_ptr<WorkerThread> CreateWorkerThread() final { return std::make_unique<ThreadedWorkletThreadForTest>(WorkletObjectProxy()); @@ -281,6 +282,16 @@ } Document& GetDocument() { return page_->GetDocument(); } + void WaitForReady(WorkerThread* worker_thread) { + base::WaitableEvent child_waitable; + PostCrossThreadTask( + *worker_thread->GetTaskRunner(TaskType::kInternalTest), FROM_HERE, + CrossThreadBindOnce(&base::WaitableEvent::Signal, + CrossThreadUnretained(&child_waitable))); + + child_waitable.Wait(); + } + private: std::unique_ptr<DummyPageHolder> page_; Persistent<ThreadedWorkletMessagingProxyForTest> messaging_proxy_; @@ -404,4 +415,40 @@ test::EnterRunLoop(); } +TEST_F(ThreadedWorkletTest, NestedRunLoopTermination) { + MessagingProxy()->Start(); + + ThreadedWorkletMessagingProxyForTest* second_messaging_proxy = + MakeGarbageCollected<ThreadedWorkletMessagingProxyForTest>( + GetExecutionContext()); + + // Get a nested event loop where the first one is on the stack + // and the second is still alive. + second_messaging_proxy->Start(); + + // Wait until the workers are setup and ready to accept work before we + // pause them. + WaitForReady(GetWorkerThread()); + WaitForReady(second_messaging_proxy->GetWorkerThread()); + + // Pause the second worker, then the first. + second_messaging_proxy->GetWorkerThread()->Pause(); + GetWorkerThread()->Pause(); + + // Resume then terminate the second worker. + second_messaging_proxy->GetWorkerThread()->Resume(); + second_messaging_proxy->GetWorkerThread()->Terminate(); + second_messaging_proxy = nullptr; + + // Now resume the first worker. + GetWorkerThread()->Resume(); + + // Make sure execution still works without crashing. + PostCrossThreadTask( + *GetWorkerThread()->GetTaskRunner(TaskType::kInternalTest), FROM_HERE, + CrossThreadBindOnce(&ThreadedWorkletThreadForTest::TestTaskRunner, + CrossThreadUnretained(GetWorkerThread()))); + test::EnterRunLoop(); +} + } // namespace blink
diff --git a/third_party/blink/renderer/core/workers/worker_thread.cc b/third_party/blink/renderer/core/workers/worker_thread.cc index b20c2286..8bf05b2 100644 --- a/third_party/blink/renderer/core/workers/worker_thread.cc +++ b/third_party/blink/renderer/core/workers/worker_thread.cc
@@ -30,7 +30,6 @@ #include <memory> #include <utility> -#include "base/auto_reset.h" #include "base/metrics/histogram_functions.h" #include "base/synchronization/lock.h" #include "base/synchronization/waitable_event.h" @@ -598,6 +597,7 @@ const absl::optional<WorkerBackingThreadStartupData>& thread_startup_data, std::unique_ptr<WorkerDevToolsParams> devtools_params) { DCHECK(IsCurrentThread()); + backing_thread_weak_factory_.emplace(this); worker_reporting_proxy_.WillInitializeWorkerContext(); { TRACE_EVENT0("blink.worker", "WorkerThread::InitializeWorkerContext"); @@ -739,11 +739,13 @@ SetThreadState(ThreadState::kReadyToShutdown); } + backing_thread_weak_factory_ = absl::nullopt; if (pause_or_freeze_count_ > 0) { DCHECK(nested_runner_); pause_or_freeze_count_ = 0; nested_runner_->QuitNow(); } + pause_handle_.reset(); if (WorkerThreadDebugger* debugger = WorkerThreadDebugger::From(GetIsolate())) debugger->WorkerThreadDestroyed(this); @@ -888,8 +890,7 @@ if (pause_or_freeze_count_ > 1) return; - std::unique_ptr<scheduler::WorkerScheduler::PauseHandle> pause_handle = - GetScheduler()->Pause(); + pause_handle_ = GetScheduler()->Pause(); { // Since the nested message loop runner needs to be created and destroyed on // the same thread we allocate and destroy a new message loop runner each @@ -897,13 +898,20 @@ // the worker thread such that the resume/terminate can quit this runner. std::unique_ptr<Platform::NestedMessageLoopRunner> nested_runner = Platform::Current()->CreateNestedMessageLoopRunner(); - base::AutoReset<Platform::NestedMessageLoopRunner*> nested_runner_autoreset( - &nested_runner_, nested_runner.get()); + auto weak_this = backing_thread_weak_factory_->GetWeakPtr(); + nested_runner_ = nested_runner.get(); nested_runner->Run(); + + // Careful `this` may be destroyed. + if (!weak_this) { + return; + } + nested_runner_ = nullptr; } GlobalScope()->SetDefersLoadingForResourceFetchers(LoaderFreezeMode::kNone); GlobalScope()->SetIsInBackForwardCache(false); GlobalScope()->SetLifecycleState(mojom::blink::FrameLifecycleState::kRunning); + pause_handle_.reset(); } void WorkerThread::ResumeOnWorkerThread() {
diff --git a/third_party/blink/renderer/core/workers/worker_thread.h b/third_party/blink/renderer/core/workers/worker_thread.h index 7254460e..298c36e 100644 --- a/third_party/blink/renderer/core/workers/worker_thread.h +++ b/third_party/blink/renderer/core/workers/worker_thread.h
@@ -31,6 +31,7 @@ #include "base/gtest_prod_util.h" #include "base/memory/scoped_refptr.h" +#include "base/memory/weak_ptr.h" #include "base/synchronization/lock.h" #include "base/synchronization/waitable_event.h" #include "base/task/single_thread_task_runner.h" @@ -82,7 +83,7 @@ // abstract class. Multiple WorkerThreads may share one WorkerBackingThread for // worklets. // -// WorkerThread start and termination must be initiated on the main thread and +// WorkerThread start and termination must be initiated on the parent thread and // an actual task is executed on the worker thread. // // When termination starts, (debugger) tasks on WorkerThread are handled as @@ -105,7 +106,7 @@ ~WorkerThread() override; // Starts the underlying thread and creates the global scope. Called on the - // main thread. + // parent thread. // Startup data for WorkerBackingThread is absl::nullopt if |this| doesn't own // the underlying WorkerBackingThread. // TODO(nhiroki): We could separate WorkerBackingThread initialization from @@ -117,14 +118,14 @@ std::unique_ptr<WorkerDevToolsParams>); // Posts a task to evaluate a top-level classic script on the worker thread. - // Called on the main thread after Start(). + // Called on the parent thread after Start(). void EvaluateClassicScript(const KURL& script_url, const String& source_code, std::unique_ptr<Vector<uint8_t>> cached_meta_data, const v8_inspector::V8StackTraceId& stack_id); // Posts a task to fetch and run a top-level classic script on the worker - // thread. Called on the main thread after Start(). + // thread. Called on the parent thread after Start(). void FetchAndRunClassicScript( const KURL& script_url, std::unique_ptr<WorkerMainScriptLoadParameters> @@ -136,7 +137,7 @@ const v8_inspector::V8StackTraceId& stack_id); // Posts a task to fetch and run a top-level module script on the worker - // thread. Called on the main thread after Start(). + // thread. Called on the parent thread after Start(). void FetchAndRunModuleScript( const KURL& script_url, std::unique_ptr<WorkerMainScriptLoadParameters> @@ -158,7 +159,7 @@ // Terminates the worker thread. Subclasses of WorkerThread can override this // to do cleanup. The default behavior is to call Terminate() and // synchronously call EnsureScriptExecutionTerminates() to ensure the thread - // is quickly terminated. Called on the main thread. + // is quickly terminated. Called on the parent thread. virtual void TerminateForTesting(); // Thread::TaskObserver. @@ -185,7 +186,7 @@ void DebuggerTaskStarted(); void DebuggerTaskFinished(); - // Callable on both the main thread and the worker thread. + // Callable on both the parent thread and the worker thread. const base::UnguessableToken& GetDevToolsWorkerToken() const { return devtools_worker_token_; } @@ -330,7 +331,7 @@ // already shutting down. Does not terminate if a debugger task is running, // because the debugger task is guaranteed to finish and it heavily uses V8 // API calls which would crash after forcible script termination. Called on - // the main thread. + // the parent thread. void EnsureScriptExecutionTerminates(ExitCode) LOCKS_EXCLUDED(lock_); // These are called in this order during worker thread startup. @@ -417,7 +418,7 @@ // A unique identifier among all WorkerThreads. const int worker_thread_id_; - // Set on the main thread. + // Set on the parent thread. bool requested_to_terminate_ GUARDED_BY(lock_) = false; ThreadState thread_state_ GUARDED_BY(lock_) = ThreadState::kNotStarted; @@ -451,7 +452,7 @@ TaskTypeTraits>; TaskRunnerHashMap worker_task_runners_; - // This lock protects shared states between the main thread and the worker + // This lock protects shared states between the parent thread and the worker // thread. See thread-safety annotations (e.g., GUARDED_BY) in this header // file. base::Lock lock_; @@ -460,6 +461,10 @@ // only on the worker thread. int pause_or_freeze_count_ = 0; + // The `PauseHandle` needs to be destroyed before the scheduler is destroyed + // otherwise we will hit a DCHECK. + std::unique_ptr<scheduler::WorkerScheduler::PauseHandle> pause_handle_; + // A nested message loop for handling pausing. Pointer is not owned. Used only // on the worker thread. Platform::NestedMessageLoopRunner* nested_runner_ = nullptr; @@ -485,6 +490,12 @@ // a pointer to a member in this list. HashSet<std::unique_ptr<InterruptData>> pending_interrupts_ GUARDED_BY(lock_); + // Since the WorkerThread is allocated and deallocated on the parent thread, + // we need a WeakPtrFactory that is allocated and cleared on the backing + // thread. + absl::optional<base::WeakPtrFactory<WorkerThread>> + backing_thread_weak_factory_; + THREAD_CHECKER(parent_thread_checker_); };
diff --git a/third_party/blink/renderer/modules/peerconnection/rtc_peer_connection.idl b/third_party/blink/renderer/modules/peerconnection/rtc_peer_connection.idl index 6a415ec..9468801 100644 --- a/third_party/blink/renderer/modules/peerconnection/rtc_peer_connection.idl +++ b/third_party/blink/renderer/modules/peerconnection/rtc_peer_connection.idl
@@ -86,7 +86,7 @@ readonly attribute boolean? canTrickleIceCandidates; void restartIce(); [CallWith=ScriptState] RTCConfiguration getConfiguration(); - [CallWith=ScriptState, RaisesException] void setConfiguration(RTCConfiguration configuration); + [CallWith=ScriptState, RaisesException] void setConfiguration(optional RTCConfiguration configuration = {}); void close(); attribute EventHandler onnegotiationneeded; attribute EventHandler onicecandidate;
diff --git a/third_party/blink/renderer/modules/peerconnection/testing/mock_peer_connection_interface.h b/third_party/blink/renderer/modules/peerconnection/testing/mock_peer_connection_interface.h index 39140cc..e7b01c1 100644 --- a/third_party/blink/renderer/modules/peerconnection/testing/mock_peer_connection_interface.h +++ b/third_party/blink/renderer/modules/peerconnection/testing/mock_peer_connection_interface.h
@@ -15,194 +15,13 @@ #include "third_party/abseil-cpp/absl/types/optional.h" #include "third_party/webrtc/api/peer_connection_interface.h" #include "third_party/webrtc/api/scoped_refptr.h" +#include "third_party/webrtc/api/test/mock_peerconnectioninterface.h" #include "third_party/webrtc/rtc_base/ref_count.h" namespace blink { class MockPeerConnectionInterface - : public rtc::RefCountedObject<webrtc::PeerConnectionInterface> { - public: - // PeerConnectionInterface - MOCK_METHOD(rtc::scoped_refptr<webrtc::StreamCollectionInterface>, - local_streams, - (), - (override)); - MOCK_METHOD(rtc::scoped_refptr<webrtc::StreamCollectionInterface>, - remote_streams, - (), - (override)); - MOCK_METHOD(bool, AddStream, (webrtc::MediaStreamInterface*), (override)); - MOCK_METHOD(void, RemoveStream, (webrtc::MediaStreamInterface*), (override)); - MOCK_METHOD( - webrtc::RTCErrorOr<rtc::scoped_refptr<webrtc::RtpSenderInterface>>, - AddTrack, - (rtc::scoped_refptr<webrtc::MediaStreamTrackInterface>, - const std::vector<std::string>&), - (override)); - MOCK_METHOD(webrtc::RTCError, - RemoveTrackOrError, - (rtc::scoped_refptr<webrtc::RtpSenderInterface>), - (override)); - MOCK_METHOD( - webrtc::RTCErrorOr<rtc::scoped_refptr<webrtc::RtpTransceiverInterface>>, - AddTransceiver, - (rtc::scoped_refptr<webrtc::MediaStreamTrackInterface>), - (override)); - MOCK_METHOD( - webrtc::RTCErrorOr<rtc::scoped_refptr<webrtc::RtpTransceiverInterface>>, - AddTransceiver, - (rtc::scoped_refptr<webrtc::MediaStreamTrackInterface>, - const webrtc::RtpTransceiverInit&), - (override)); - MOCK_METHOD( - webrtc::RTCErrorOr<rtc::scoped_refptr<webrtc::RtpTransceiverInterface>>, - AddTransceiver, - (cricket::MediaType), - (override)); - MOCK_METHOD( - webrtc::RTCErrorOr<rtc::scoped_refptr<webrtc::RtpTransceiverInterface>>, - AddTransceiver, - (cricket::MediaType, const webrtc::RtpTransceiverInit&), - (override)); - MOCK_METHOD(rtc::scoped_refptr<webrtc::RtpSenderInterface>, - CreateSender, - (const std::string&, const std::string&), - (override)); - MOCK_METHOD(std::vector<rtc::scoped_refptr<webrtc::RtpSenderInterface>>, - GetSenders, - (), - (const override)); - MOCK_METHOD(std::vector<rtc::scoped_refptr<webrtc::RtpReceiverInterface>>, - GetReceivers, - (), - (const override)); - MOCK_METHOD(std::vector<rtc::scoped_refptr<webrtc::RtpTransceiverInterface>>, - GetTransceivers, - (), - (const override)); - MOCK_METHOD(bool, - GetStats, - (webrtc::StatsObserver*, - webrtc::MediaStreamTrackInterface*, - StatsOutputLevel), - (override)); - MOCK_METHOD(void, GetStats, (webrtc::RTCStatsCollectorCallback*), (override)); - MOCK_METHOD(void, - GetStats, - (rtc::scoped_refptr<webrtc::RtpSenderInterface>, - rtc::scoped_refptr<webrtc::RTCStatsCollectorCallback>), - (override)); - MOCK_METHOD(void, - GetStats, - (rtc::scoped_refptr<webrtc::RtpReceiverInterface>, - rtc::scoped_refptr<webrtc::RTCStatsCollectorCallback>), - (override)); - MOCK_METHOD(void, ClearStatsCache, (), (override)); - MOCK_METHOD(rtc::scoped_refptr<webrtc::SctpTransportInterface>, - GetSctpTransport, - (), - (const override)); - MOCK_METHOD(rtc::scoped_refptr<webrtc::DataChannelInterface>, - CreateDataChannel, - (const std::string&, const webrtc::DataChannelInit*), - (override)); - MOCK_METHOD(const webrtc::SessionDescriptionInterface*, - local_description, - (), - (const override)); - MOCK_METHOD(const webrtc::SessionDescriptionInterface*, - remote_description, - (), - (const override)); - MOCK_METHOD(const webrtc::SessionDescriptionInterface*, - current_local_description, - (), - (const override)); - MOCK_METHOD(const webrtc::SessionDescriptionInterface*, - current_remote_description, - (), - (const override)); - MOCK_METHOD(const webrtc::SessionDescriptionInterface*, - pending_local_description, - (), - (const override)); - MOCK_METHOD(const webrtc::SessionDescriptionInterface*, - pending_remote_description, - (), - (const override)); - MOCK_METHOD(void, RestartIce, (), (override)); - MOCK_METHOD(void, - CreateOffer, - (webrtc::CreateSessionDescriptionObserver*, - const RTCOfferAnswerOptions&), - (override)); - MOCK_METHOD(void, - CreateAnswer, - (webrtc::CreateSessionDescriptionObserver*, - const RTCOfferAnswerOptions&), - (override)); - MOCK_METHOD(void, - SetLocalDescription, - (webrtc::SetSessionDescriptionObserver*, - webrtc::SessionDescriptionInterface*), - (override)); - MOCK_METHOD(void, - SetRemoteDescription, - (webrtc::SetSessionDescriptionObserver*, - webrtc::SessionDescriptionInterface*), - (override)); - MOCK_METHOD( - void, - SetRemoteDescription, - (std::unique_ptr<webrtc::SessionDescriptionInterface>, - rtc::scoped_refptr<webrtc::SetRemoteDescriptionObserverInterface>), - (override)); - MOCK_METHOD(PeerConnectionInterface::RTCConfiguration, - GetConfiguration, - (), - (override)); - MOCK_METHOD(webrtc::RTCError, - SetConfiguration, - (const PeerConnectionInterface::RTCConfiguration&), - (override)); - MOCK_METHOD(bool, - AddIceCandidate, - (const webrtc::IceCandidateInterface*), - (override)); - MOCK_METHOD(bool, - RemoveIceCandidates, - (const std::vector<cricket::Candidate>&), - (override)); - MOCK_METHOD(webrtc::RTCError, - SetBitrate, - (const webrtc::BitrateSettings&), - (override)); - MOCK_METHOD(void, SetAudioPlayout, (bool), (override)); - MOCK_METHOD(void, SetAudioRecording, (bool), (override)); - MOCK_METHOD(rtc::scoped_refptr<webrtc::DtlsTransportInterface>, - LookupDtlsTransportByMid, - (const std::string&), - (override)); - MOCK_METHOD(SignalingState, signaling_state, (), (override)); - MOCK_METHOD(IceConnectionState, ice_connection_state, (), (override)); - MOCK_METHOD(IceConnectionState, - standardized_ice_connection_state, - (), - (override)); - MOCK_METHOD(PeerConnectionState, peer_connection_state, (), (override)); - MOCK_METHOD(IceGatheringState, ice_gathering_state, (), (override)); - MOCK_METHOD(absl::optional<bool>, can_trickle_ice_candidates, (), (override)); - MOCK_METHOD(bool, - StartRtcEventLog, - (std::unique_ptr<webrtc::RtcEventLogOutput>, int64_t), - (override)); - MOCK_METHOD(bool, - StartRtcEventLog, - (std::unique_ptr<webrtc::RtcEventLogOutput>), - (override)); - MOCK_METHOD(void, StopRtcEventLog, (), (override)); - MOCK_METHOD(void, Close, (), (override)); -}; + : public rtc::RefCountedObject<webrtc::MockPeerConnectionInterface> {}; static_assert(!std::is_abstract<MockPeerConnectionInterface>::value, "");
diff --git a/third_party/blink/renderer/modules/permissions/permission_utils.cc b/third_party/blink/renderer/modules/permissions/permission_utils.cc index 6e060bc9..81a9cb4 100644 --- a/third_party/blink/renderer/modules/permissions/permission_utils.cc +++ b/third_party/blink/renderer/modules/permissions/permission_utils.cc
@@ -95,7 +95,7 @@ return "nfc"; case PermissionName::STORAGE_ACCESS: return "storage_access"; - case PermissionName::WINDOW_PLACEMENT: + case PermissionName::WINDOW_MANAGEMENT: return "window_placement"; case PermissionName::LOCAL_FONTS: return "local_fonts"; @@ -280,7 +280,7 @@ exception_state.ThrowTypeError("Window Placement is not enabled."); return nullptr; } - return CreatePermissionDescriptor(PermissionName::WINDOW_PLACEMENT); + return CreatePermissionDescriptor(PermissionName::WINDOW_MANAGEMENT); } if (name == "local-fonts") { if (!RuntimeEnabledFeatures::FontAccessEnabled(
diff --git a/third_party/blink/renderer/modules/screen_enumeration/window_screens.cc b/third_party/blink/renderer/modules/screen_enumeration/window_screens.cc index 03a69bc..dbd476bb 100644 --- a/third_party/blink/renderer/modules/screen_enumeration/window_screens.cc +++ b/third_party/blink/renderer/modules/screen_enumeration/window_screens.cc
@@ -71,7 +71,7 @@ } auto permission_descriptor = CreatePermissionDescriptor( - mojom::blink::PermissionName::WINDOW_PLACEMENT); + mojom::blink::PermissionName::WINDOW_MANAGEMENT); auto* resolver = MakeGarbageCollected<ScriptPromiseResolver>(script_state); auto callback = WTF::BindOnce(&WindowScreens::OnPermissionRequestComplete, WrapPersistent(this), WrapPersistent(resolver));
diff --git a/third_party/blink/renderer/modules/webgpu/gpu.cc b/third_party/blink/renderer/modules/webgpu/gpu.cc index 672a82f..ecf1b5a 100644 --- a/third_party/blink/renderer/modules/webgpu/gpu.cc +++ b/third_party/blink/renderer/modules/webgpu/gpu.cc
@@ -41,72 +41,52 @@ #include "third_party/blink/renderer/platform/scheduler/public/main_thread.h" #include "third_party/blink/renderer/platform/scheduler/public/post_cross_thread_task.h" #include "third_party/blink/renderer/platform/weborigin/kurl.h" +#include "third_party/blink/renderer/platform/wtf/cross_thread_copier_base.h" +#include "third_party/blink/renderer/platform/wtf/cross_thread_copier_std.h" #include "third_party/blink/renderer/platform/wtf/cross_thread_functional.h" namespace blink { namespace { -void CreateContextProvider( - const KURL& url, - base::WaitableEvent* waitable_event, - std::unique_ptr<WebGraphicsContext3DProvider>* created_context_provider) { +void CreateContextProviderOnMainThread( + ExecutionContext* execution_context, + scoped_refptr<base::SingleThreadTaskRunner> task_runner, + CrossThreadOnceFunction<void(std::unique_ptr<WebGraphicsContext3DProvider>)> + callback) { DCHECK(IsMainThread()); - *created_context_provider = - Platform::Current()->CreateWebGPUGraphicsContext3DProvider(url); - waitable_event->Signal(); -} - -std::unique_ptr<WebGraphicsContext3DProvider> CreateContextProviderOnMainThread( - const KURL& url) { - scoped_refptr<base::SingleThreadTaskRunner> task_runner = - Thread::MainThread()->GetDeprecatedTaskRunner(); - - base::WaitableEvent waitable_event; - std::unique_ptr<WebGraphicsContext3DProvider> created_context_provider; + const KURL& url = execution_context->Url(); PostCrossThreadTask( *task_runner, FROM_HERE, - CrossThreadBindOnce(&CreateContextProvider, url, - CrossThreadUnretained(&waitable_event), - CrossThreadUnretained(&created_context_provider))); - - waitable_event.Wait(); - return created_context_provider; + CrossThreadBindOnce( + std::move(callback), + Platform::Current()->CreateWebGPUGraphicsContext3DProvider(url))); } -std::unique_ptr<WebGraphicsContext3DProvider> CreateContextProvider( - ExecutionContext& execution_context) { - const KURL& url = execution_context.Url(); - std::unique_ptr<WebGraphicsContext3DProvider> context_provider; +void EnsureDawnControlClientInitialized( + ExecutionContext* execution_context, + base::OnceCallback<void(std::unique_ptr<WebGraphicsContext3DProvider>)> + callback) { if (IsMainThread()) { - context_provider = - Platform::Current()->CreateWebGPUGraphicsContext3DProvider(url); + const KURL& url = execution_context->Url(); + std::move(callback).Run( + Platform::Current()->CreateWebGPUGraphicsContext3DProvider(url)); } else { - context_provider = CreateContextProviderOnMainThread(url); + // Posts a task to the main thread to create context provider + // because the current RendererBlinkPlatformImpl and viz::Gpu + // APIs allow to create it only on the main thread. + // When it is created, posts it back to the current thread + // and call the callback with it. + // TODO(takahiro): Directly create context provider on Workers threads + // if RendererBlinkPlatformImpl and viz::Gpu will start to + // allow the context provider creation on Workers. + PostCrossThreadTask( + *Thread::MainThread()->GetDeprecatedTaskRunner(), FROM_HERE, + CrossThreadBindOnce(&CreateContextProviderOnMainThread, + WrapCrossThreadPersistent(execution_context), + execution_context->GetTaskRunner(TaskType::kWebGPU), + CrossThreadBindOnce(std::move(callback)))); } - - // Note that we check for API blocking *after* creating the context. This is - // because context creation synchronizes against GpuProcessHost lifetime in - // the browser process, and GpuProcessHost destruction is what updates API - // blocking state on a GPU process crash. See https://crbug.com/1215907#c10 - // for more details. - bool blocked = true; - mojo::Remote<mojom::blink::GpuDataManager> gpu_data_manager; - Platform::Current()->GetBrowserInterfaceBroker()->GetInterface( - gpu_data_manager.BindNewPipeAndPassReceiver()); - gpu_data_manager->Are3DAPIsBlockedForUrl(url, &blocked); - if (blocked) { - return nullptr; - } - - // TODO(kainino): we will need a better way of accessing the GPU interface - // from multiple threads than BindToCurrentSequence et al. - if (context_provider && !context_provider->BindToCurrentSequence()) { - // TODO(crbug.com/973017): Collect GPU info and surface context creation - // error. - return nullptr; - } - return context_provider; } [[maybe_unused]] void AddConsoleWarning(ExecutionContext* execution_context, @@ -287,34 +267,95 @@ .Record(context->UkmRecorder()); } -ScriptPromise GPU::requestAdapter(ScriptState* script_state, - const GPURequestAdapterOptions* options) { - auto* resolver = MakeGarbageCollected<ScriptPromiseResolver>(script_state); - ScriptPromise promise = resolver->Promise(); - ExecutionContext* execution_context = ExecutionContext::From(script_state); +std::unique_ptr<WebGraphicsContext3DProvider> CheckContextProvider( + const KURL& url, + std::unique_ptr<WebGraphicsContext3DProvider> context_provider) { + // Note that we check for API blocking *after* creating the context. This is + // because context creation synchronizes against GpuProcessHost lifetime in + // the browser process, and GpuProcessHost destruction is what updates API + // blocking state on a GPU process crash. See https://crbug.com/1215907#c10 + // for more details. + bool blocked = true; + mojo::Remote<mojom::blink::GpuDataManager> gpu_data_manager; + Platform::Current()->GetBrowserInterfaceBroker()->GetInterface( + gpu_data_manager.BindNewPipeAndPassReceiver()); + gpu_data_manager->Are3DAPIsBlockedForUrl(url, &blocked); + if (blocked) { + return nullptr; + } + // TODO(kainino): we will need a better way of accessing the GPU interface + // from multiple threads than BindToCurrentSequence et al. + if (context_provider && !context_provider->BindToCurrentSequence()) { + // TODO(crbug.com/973017): Collect GPU info and surface context creation + // error. + return nullptr; + } + + return context_provider; +} + +void GPU::RequestAdapterImpl(ScriptState* script_state, + const GPURequestAdapterOptions* options, + ScriptPromiseResolver* resolver) { + ExecutionContext* execution_context = ExecutionContext::From(script_state); if (!dawn_control_client_ || dawn_control_client_->IsContextLost()) { + dawn_control_client_initialized_callbacks_.push_back(WTF::BindOnce( + [](GPU* gpu, ScriptState* script_state, + const GPURequestAdapterOptions* options, + ScriptPromiseResolver* resolver) { + if (gpu->dawn_control_client_ && + !gpu->dawn_control_client_->IsContextLost()) { + gpu->RequestAdapterImpl(script_state, options, resolver); + } else { + // Failed to create context provider, won't be able to request + // adapter + // TODO(crbug.com/973017): Collect GPU info and surface context + // creation error. + resolver->Resolve(v8::Null(script_state->GetIsolate())); + } + }, + WrapPersistent(this), WrapPersistent(script_state), + WrapPersistent(options), WrapPersistent(resolver))); + + // Returning since the task to create the control client from a previous + // call to EnsureDawnControlClientInitialized should be already running + if (dawn_control_client_initialized_callbacks_.size() > 1) { + return; + } + // TODO(natlee@microsoft.com): if GPU process is lost, wait for the GPU // process to come back instead of rejecting right away - std::unique_ptr<WebGraphicsContext3DProvider> context_provider = - CreateContextProvider(*execution_context); - if (!context_provider) { - // Failed to create context provider, won't be able to request adapter - // TODO(crbug.com/973017): Collect GPU info and surface context creation - // error. - resolver->Resolve(v8::Null(script_state->GetIsolate())); - return promise; - } else { - context_provider->WebGPUInterface()->SetWebGPUExecutionContextToken( - GetExecutionContextToken(execution_context)); + EnsureDawnControlClientInitialized( + execution_context, + WTF::BindOnce( + [](GPU* gpu, ExecutionContext* execution_context, + std::unique_ptr<WebGraphicsContext3DProvider> context_provider) { + const KURL& url = execution_context->Url(); + context_provider = + CheckContextProvider(url, std::move(context_provider)); + if (context_provider) { + context_provider->WebGPUInterface() + ->SetWebGPUExecutionContextToken( + GetExecutionContextToken(execution_context)); - // Make a new DawnControlClientHolder with the context provider we just - // made and set the lost context callback - dawn_control_client_ = DawnControlClientHolder::Create( - std::move(context_provider), - execution_context->GetTaskRunner(TaskType::kWebGPU)); - } + // Make a new DawnControlClientHolder with the context provider + // we just made and set the lost context callback + gpu->dawn_control_client_ = DawnControlClientHolder::Create( + std::move(context_provider), + execution_context->GetTaskRunner(TaskType::kWebGPU)); + } + + WTF::Vector<base::OnceCallback<void()>> callbacks = + std::move(gpu->dawn_control_client_initialized_callbacks_); + for (auto& callback : callbacks) { + std::move(callback).Run(); + } + }, + WrapPersistent(this), WrapPersistent(execution_context))); + + return; } DCHECK_NE(dawn_control_client_, nullptr); @@ -332,7 +373,13 @@ *execution_context->GetAgent()->event_loop()); UseCounter::Count(execution_context, WebFeature::kWebGPU); +} +ScriptPromise GPU::requestAdapter(ScriptState* script_state, + const GPURequestAdapterOptions* options) { + auto* resolver = MakeGarbageCollected<ScriptPromiseResolver>(script_state); + ScriptPromise promise = resolver->Promise(); + RequestAdapterImpl(script_state, options, resolver); return promise; }
diff --git a/third_party/blink/renderer/modules/webgpu/gpu.h b/third_party/blink/renderer/modules/webgpu/gpu.h index 827d8dc..c699aa0 100644 --- a/third_party/blink/renderer/modules/webgpu/gpu.h +++ b/third_party/blink/renderer/modules/webgpu/gpu.h
@@ -8,6 +8,7 @@ #include <dawn/webgpu.h> #include "base/memory/scoped_refptr.h" +#include "third_party/blink/public/platform/web_graphics_context_3d_provider.h" #include "third_party/blink/renderer/bindings/core/v8/script_promise.h" #include "third_party/blink/renderer/core/execution_context/execution_context.h" #include "third_party/blink/renderer/core/execution_context/execution_context_lifecycle_observer.h" @@ -105,7 +106,21 @@ const GPURequestAdapterOptions* options, GPUAdapter* adapter) const; + void OnContextProviderCreated( + const KURL& url, + ScriptState* script_state, + const GPURequestAdapterOptions* options, + ScriptPromiseResolver* resolver, + ExecutionContext* execution_context, + std::unique_ptr<WebGraphicsContext3DProvider> context_provider); + + void RequestAdapterImpl(ScriptState* script_state, + const GPURequestAdapterOptions* options, + ScriptPromiseResolver* resolver); + scoped_refptr<DawnControlClientHolder> dawn_control_client_; + WTF::Vector<base::OnceCallback<void()>> + dawn_control_client_initialized_callbacks_; HeapHashSet<WeakMember<GPUBuffer>> mappable_buffers_; // Mappable buffers remove themselves from this set on destruction. // It is boxed in a scoped_refptr so GPUBuffer can access it in its
diff --git a/third_party/blink/renderer/platform/fonts/shaping/shape_result_bloberizer.cc b/third_party/blink/renderer/platform/fonts/shaping/shape_result_bloberizer.cc index 32df292..69a153a9 100644 --- a/third_party/blink/renderer/platform/fonts/shaping/shape_result_bloberizer.cc +++ b/third_party/blink/renderer/platform/fonts/shaping/shape_result_bloberizer.cc
@@ -7,6 +7,7 @@ #include <hb.h> #include "base/logging.h" +#include "base/ranges/algorithm.h" #include "third_party/blink/renderer/platform/fonts/font.h" #include "third_party/blink/renderer/platform/fonts/shaping/caching_word_shaper.h" #include "third_party/blink/renderer/platform/fonts/shaping/shape_result.h" @@ -411,9 +412,8 @@ void Finish(unsigned from, unsigned to) { std::sort(cluster_starts_.begin(), cluster_starts_.end()); - DCHECK_EQ( - std::adjacent_find(cluster_starts_.begin(), cluster_starts_.end()), - cluster_starts_.end()); + DCHECK_EQ(base::ranges::adjacent_find(cluster_starts_), + cluster_starts_.end()); DVLOG(4) << " Cluster starts: " << base::make_span(cluster_starts_); if (!cluster_starts_.empty()) { // 'from' may point inside a cluster; the least seen index may be larger.
diff --git a/third_party/blink/renderer/platform/graphics/video_frame_submitter.cc b/third_party/blink/renderer/platform/graphics/video_frame_submitter.cc index 5ebd1f1..b6da23bd 100644 --- a/third_party/blink/renderer/platform/graphics/video_frame_submitter.cc +++ b/third_party/blink/renderer/platform/graphics/video_frame_submitter.cc
@@ -347,7 +347,7 @@ continue; auto& feedback = timing_details.find(frame_token)->value.presentation_feedback; -#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) +#if BUILDFLAG(IS_LINUX) // TODO: On Linux failure flag is unreliable, and perfectly rendered frames // are reported as failures all the time. bool presentation_failure = false;
diff --git a/third_party/blink/web_tests/TestExpectations b/third_party/blink/web_tests/TestExpectations index de5e6e97..ee01a0c 100644 --- a/third_party/blink/web_tests/TestExpectations +++ b/third_party/blink/web_tests/TestExpectations
@@ -4025,10 +4025,6 @@ crbug.com/618969 external/wpt/css/css-grid/subgrid/grid-template-computed-nogrid.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/independent-formatting-context.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/item-percentage-height-001.html [ Failure ] -crbug.com/618969 external/wpt/css/css-grid/subgrid/line-names-001.html [ Failure ] -crbug.com/618969 external/wpt/css/css-grid/subgrid/line-names-002.html [ Failure ] -crbug.com/618969 external/wpt/css/css-grid/subgrid/line-names-003.html [ Failure ] -crbug.com/618969 external/wpt/css/css-grid/subgrid/line-names-004.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/line-names-005.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/line-names-006.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/line-names-007.html [ Failure ] @@ -4037,6 +4033,8 @@ crbug.com/618969 external/wpt/css/css-grid/subgrid/orthogonal-writing-mode-002.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/orthogonal-writing-mode-003.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/orthogonal-writing-mode-004.html [ Failure ] +crbug.com/618969 external/wpt/css/css-grid/subgrid/parent-repeat-auto-fit-001.html [ Failure ] +crbug.com/618969 external/wpt/css/css-grid/subgrid/parent-repeat-auto-fit-002.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/repeat-auto-fill-001.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/repeat-auto-fill-002.html [ Failure ] crbug.com/618969 external/wpt/css/css-grid/subgrid/repeat-auto-fill-003.html [ Failure ] @@ -6665,6 +6663,7 @@ crbug.com/1304956 storage/indexeddb/dont-wedge.html [ Failure Pass ] # Sheriff 2022-03-15 +crbug.com/626703 external/wpt/scroll-to-text-fragment/force-load-at-top.html [ Pass Timeout ] crbug.com/1269534 fast/spatial-navigation/snav-zero-margin-content.html [ Failure Pass ] crbug.com/1295980 virtual/fenced-frame-mparch/wpt_internal/fenced_frame/user-activation.https.html [ Failure Pass Timeout ] crbug.com/1295980 virtual/fenced-frame-shadow-dom/wpt_internal/fenced_frame/user-activation.https.html [ Failure Pass Timeout ]
diff --git a/third_party/blink/web_tests/VirtualTestSuites b/third_party/blink/web_tests/VirtualTestSuites index f120845..267642c 100644 --- a/third_party/blink/web_tests/VirtualTestSuites +++ b/third_party/blink/web_tests/VirtualTestSuites
@@ -698,12 +698,6 @@ "args": ["--disable-features=PrefersReducedMotionClientHintHeader"] }, { - "prefix": "disable-wildcard-subdomains-in-permissions-policy", - "platforms": ["Linux", "Mac", "Win"], - "bases": ["external/wpt/permissions-policy/"], - "args": ["--disable-features=WildcardSubdomainsInPermissionsPolicy"] - }, - { "prefix": "storage-access-api", "platforms": ["Linux", "Mac", "Win"], "bases": [ "external/wpt/storage-access-api" ],
diff --git a/third_party/blink/web_tests/external/wpt/cookies/samesite/get_all_cookies-default-samesite.html b/third_party/blink/web_tests/external/wpt/cookies/samesite/get_all_cookies-default-samesite.html new file mode 100644 index 0000000..22727e39 --- /dev/null +++ b/third_party/blink/web_tests/external/wpt/cookies/samesite/get_all_cookies-default-samesite.html
@@ -0,0 +1,21 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>TestDriver get_all_cookies method w/ default SameSite</title> +<script src="/resources/testharness.js"></script> +<script src="/resources/testharnessreport.js"></script> +<script src="/resources/testdriver.js"></script> +<script src="/resources/testdriver-vendor.js"></script> +<script> +promise_test(async t => { + document.cookie = "test0=0"; + const cookies = await test_driver.get_all_cookies(); + assert_equals(cookies.length, 1); + let cookieMap = new Map(); + for (const cookie of cookies) { + cookieMap.set(cookie["name"], cookie); + } + assert_equals(cookieMap.get("test0")["name"], "test0"); + assert_equals(cookieMap.get("test0")["value"], "0"); + assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); +}, "Get all cookies w/ default SameSite"); +</script>
diff --git a/third_party/blink/web_tests/external/wpt/cookies/samesite/get_named_cookie-default-samesite.html b/third_party/blink/web_tests/external/wpt/cookies/samesite/get_named_cookie-default-samesite.html new file mode 100644 index 0000000..405473c --- /dev/null +++ b/third_party/blink/web_tests/external/wpt/cookies/samesite/get_named_cookie-default-samesite.html
@@ -0,0 +1,16 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>TestDriver get_named_cookie method w/ default SameSite</title> +<script src="/resources/testharness.js"></script> +<script src="/resources/testharnessreport.js"></script> +<script src="/resources/testdriver.js"></script> +<script src="/resources/testdriver-vendor.js"></script> +<script> +promise_test(async t => { + document.cookie = "test0=0"; + const cookie = await test_driver.get_named_cookie("test0"); + assert_equals(cookie["name"], "test0"); + assert_equals(cookie["value"], "0"); + assert_equals(cookie["sameSite"], "Lax"); +}, "Get Named cookie w/ default SameSite"); +</script>
diff --git a/third_party/blink/web_tests/external/wpt/css/css-grid/subgrid/line-names-009-ref.html b/third_party/blink/web_tests/external/wpt/css/css-grid/subgrid/line-names-009-ref.html new file mode 100644 index 0000000..103a359 --- /dev/null +++ b/third_party/blink/web_tests/external/wpt/css/css-grid/subgrid/line-names-009-ref.html
@@ -0,0 +1,75 @@ +<!DOCTYPE HTML> +<!-- + Any copyright is dedicated to the Public Domain. + http://creativecommons.org/publicdomain/zero/1.0/ +--> +<html><head> + <meta charset="utf-8"> + <title>CSS Grid Test: subgrid line names are not inherited to a non-subgridded axis</title> + <link rel="author" title="Kurt Catti-Schmidt" href="mailto:kschmi@microsoft.com"> + <link rel="help" href="https://drafts.csswg.org/css-grid-2"> + <style> + html,body { + color:black; background-color:white; font:12px/1 monospace; padding:0; margin:0; + } + + div > div { background: grey; grid-column:2 / span 2; } + + i { + grid-row: 1; + counter-increment: i; + } + i::before { content: counter(i, decimal); } + + x { background: silver; } + </style> +</head> +<body> + +<div style="display:grid;"> + <i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][d][e]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:1 / 4;"> + <x style="grid-column:c">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][d][e]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c / 4;"> + <x style="grid-column:span 1">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][c][c]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c 2 / 4;"> + <x style="grid-column:4">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][c][c]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c 2 / 4;"> + <x style="grid-column:span 1">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][c][c]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c-1 / c 3;"> + <x style="grid-column:span 1">x</x> + </div> + </div> +</div> + +</body> +</html>
diff --git a/third_party/blink/web_tests/external/wpt/css/css-grid/subgrid/line-names-009.html b/third_party/blink/web_tests/external/wpt/css/css-grid/subgrid/line-names-009.html new file mode 100644 index 0000000..bd8a776 --- /dev/null +++ b/third_party/blink/web_tests/external/wpt/css/css-grid/subgrid/line-names-009.html
@@ -0,0 +1,76 @@ +<!DOCTYPE HTML> +<!-- + Any copyright is dedicated to the Public Domain. + http://creativecommons.org/publicdomain/zero/1.0/ +--> +<html><head> + <meta charset="utf-8"> + <title>CSS Grid Test: subgrid line names are not inherited to a non-subgridded axis</title> + <link rel="author" title="Kurt Catti-Schmidt" href="mailto:kschmi@microsoft.com"> + <link rel="help" href="https://drafts.csswg.org/css-grid-2"> + <link rel="match" href="line-names-009-ref.html"> + <style> + html,body { + color:black; background-color:white; font:12px/1 monospace; padding:0; margin:0; + } + + div > div { background: grey; grid-column:2 / span 2; } + + i { + grid-row: 1; + counter-increment: i; + } + i::before { content: counter(i, decimal); } + + x { background: silver; } + </style> +</head> +<body> + +<div style="display:grid;"> + <i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][d][e]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:1 / 4;"> + <x style="grid-column:c">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][d][e]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c / 4;"> + <x style="grid-column:span c">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][c][c]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c 2 / 4;"> + <x style="grid-column:c 2">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][c][c]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c 2 / 4;"> + <x style="grid-column:span c 2">x</x> + </div> + </div> +</div> + +<div style="display:grid;"> +<i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i> + <div style="display:grid; grid:auto/subgrid [a][b][c][c][c]; grid-column:2 / 9;"> + <div style="display:grid; grid:subgrid/auto; grid-column:c-1 / c 3;"> + <x style="grid-column:span c">x</x> + </div> + </div> +</div> + +</body> +</html>
diff --git a/third_party/blink/web_tests/external/wpt/event-timing/duration-with-target-low.html b/third_party/blink/web_tests/external/wpt/event-timing/duration-with-target-low.html new file mode 100644 index 0000000..11a8c3a --- /dev/null +++ b/third_party/blink/web_tests/external/wpt/event-timing/duration-with-target-low.html
@@ -0,0 +1,16 @@ +<!DOCTYPE html> +<html> +<meta charset=utf-8 /> +<title>Event Timing: PerformanceObserver with a durationThreshold way smaller than processingDelay</title> +<script src=/resources/testharness.js></script> +<script src=/resources/testharnessreport.js></script> +<script src=/resources/testdriver.js></script> +<script src=/resources/testdriver-vendor.js></script> +<script src=resources/event-timing-test-utils.js></script> +<div id='myDiv'>Click me</div> +<script> +promise_test(async t => { + return testDurationWithDurationThreshold(t, 'myDiv', numEntries=5, durThreshold=300, processingDelay=0); +}, "PerformanceObserver with durationThreshold of 300 and processingDelay of 0 doesn't see any entries in the observer"); +</script> +</html>
diff --git a/third_party/blink/web_tests/external/wpt/event-timing/large-duration-threshold.html b/third_party/blink/web_tests/external/wpt/event-timing/large-duration-threshold.html index 5665d42..4eed8d97 100644 --- a/third_party/blink/web_tests/external/wpt/event-timing/large-duration-threshold.html +++ b/third_party/blink/web_tests/external/wpt/event-timing/large-duration-threshold.html
@@ -10,7 +10,7 @@ <div id='myDiv'>Click me</div> <script> promise_test(async t => { - return testDuration(t, 'myDiv', numEntries=2, dur=125, fastDur=70, slowDur=140); + return testDuration(t, 'myDiv', numEntries=2, dur=125, slowDur=140); }, "PerformanceObserver observes events according to its durationThreshold"); </script> </html>
diff --git a/third_party/blink/web_tests/external/wpt/event-timing/medium-duration-threshold.html b/third_party/blink/web_tests/external/wpt/event-timing/medium-duration-threshold.html index a827f22..a2f79cf6 100644 --- a/third_party/blink/web_tests/external/wpt/event-timing/medium-duration-threshold.html +++ b/third_party/blink/web_tests/external/wpt/event-timing/medium-duration-threshold.html
@@ -10,7 +10,7 @@ <div id='myDiv'>Click me</div> <script> promise_test(async t => { - return testDuration(t, 'myDiv', numEntries=3, dur=50, fastDur=20, slowDur=70); + return testDuration(t, 'myDiv', numEntries=3, dur=50, slowDur=70); }, "PerformanceObserver observes events according to its durationThreshold"); </script> </html>
diff --git a/third_party/blink/web_tests/external/wpt/event-timing/min-duration-threshold.html b/third_party/blink/web_tests/external/wpt/event-timing/min-duration-threshold.html index 443b38a2..b7382f9 100644 --- a/third_party/blink/web_tests/external/wpt/event-timing/min-duration-threshold.html +++ b/third_party/blink/web_tests/external/wpt/event-timing/min-duration-threshold.html
@@ -10,7 +10,7 @@ <div id='myDiv'>Click me</div> <script> promise_test(async t => { - return testDuration(t, 'myDiv', numEntries=5, dur=0, fastDur=0, slowDur=20); + return testDuration(t, 'myDiv', numEntries=5, dur=0, slowDur=20); }, "PerformanceObserver with durationThreshold of 0 sees events of duration >= 16"); </script> </html>
diff --git a/third_party/blink/web_tests/external/wpt/event-timing/resources/event-timing-test-utils.js b/third_party/blink/web_tests/external/wpt/event-timing/resources/event-timing-test-utils.js index ab3f110c..fc4d1ae 100644 --- a/third_party/blink/web_tests/external/wpt/event-timing/resources/event-timing-test-utils.js +++ b/third_party/blink/web_tests/external/wpt/event-timing/resources/event-timing-test-utils.js
@@ -5,10 +5,12 @@ const element = document.getElementById(id); const clickHandler = () => { mainThreadBusy(delay); - if (callback) + if (callback) { callback(); + } element.removeEventListener("pointerdown", clickHandler); }; + element.addEventListener("pointerdown", clickHandler); await test_driver.click(element); } @@ -81,17 +83,15 @@ // attempt to check that the duration is appropriately checked by: // * Asserting that entries received have a duration which is the smallest multiple of 8 // that is greater than or equal to |dur|. - // * Issuing |numEntries| entries that are fast, of duration |slowDur|. - // * Issuing |numEntries| entries that are slow, of duration |fastDur|. - // * Asserting that at least |numEntries| entries are received (at least the slow ones). + // * Issuing |numEntries| entries that has duration greater than |slowDur|. + // * Asserting that exactly |numEntries| entries are received. // Parameters: // |t| - the test harness. // |dur| - the durationThreshold for the PerformanceObserver. // |id| - the ID of the element to be clicked. - // |numEntries| - the number of slow and number of fast entries. + // |numEntries| - the number of entries. // |slowDur| - the min duration of a slow entry. - // |fastDur| - the min duration of a fast entry. -async function testDuration(t, id, numEntries, dur, fastDur, slowDur) { +async function testDuration(t, id, numEntries, dur, slowDur) { assert_implements(window.PerformanceEventTiming, 'Event Timing is not supported.'); const observerPromise = new Promise(async resolve => { let minDuration = Math.ceil(dur / 8) * 8; @@ -106,29 +106,62 @@ }); }); numEntriesReceived += pointerDowns.length; - // Note that we may receive more entries if the 'fast' click events turn out slower - // than expected. - if (numEntriesReceived >= numEntries) + // All the entries should be received since the slowDur is higher + // than the duration threshold. + if (numEntriesReceived === numEntries) resolve(); }).observe({type: "event", durationThreshold: dur}); }); const clicksPromise = new Promise(async resolve => { for (let index = 0; index < numEntries; index++) { - // Add some fast click events. + // Add some click events that has at least slowDur for duration. await clickOnElementAndDelay(id, slowDur); - // Add some slow click events. - if (fastDur > 0) { - await clickOnElementAndDelay(id, fastDur); - } else { - // We can just directly call test_driver when |fastDur| is 0. - await test_driver.click(document.getElementById(id)); - } } resolve(); }); return Promise.all([observerPromise, clicksPromise]); } + // Add a PerformanceObserver and observe with a durationThreshold of |durThreshold|. This test will + // attempt to check that the duration is appropriately checked by: + // * Asserting that entries received have a duration which is the smallest multiple of 8 + // that is greater than or equal to |durThreshold|. + // * Issuing |numEntries| entries that have at least |processingDelay| as duration. + // * Asserting that the entries we receive has duration greater than or equals to the + // duration threshold we setup + // Parameters: + // |t| - the test harness. + // |id| - the ID of the element to be clicked. + // |durThreshold| - the durationThreshold for the PerformanceObserver. + // |numEntries| - the number of slow and number of fast entries. + // |processingDelay| - the event duration we add on each event. + async function testDurationWithDurationThreshold(t, id, numEntries, durThreshold, processingDelay) { + assert_implements(window.PerformanceEventTiming, 'Event Timing is not supported.'); + const observerPromise = new Promise(async resolve => { + let minDuration = Math.ceil(durThreshold / 8) * 8; + // Exposed events must always have a minimum duration of 16. + minDuration = Math.max(minDuration, 16); + new PerformanceObserver(t.step_func(list => { + const pointerDowns = list.getEntriesByName('pointerdown'); + pointerDowns.forEach(p => { + assert_greater_than_equal(p.duration, minDuration, + "The entry's duration should be greater than or equal to " + minDuration + " ms."); + }); + resolve(); + })).observe({type: "event", durationThreshold: durThreshold}); + }); + for (let index = 0; index < numEntries; index++) { + // These clicks are expected to be ignored, unless the test has some extra delays. + // In that case, the test will verify the event duration to ensure the event duration is + // greater than the duration threshold + await clickOnElementAndDelay(id, processingDelay); + } + // Send click with event duration equals to or greater than |durThreshold|, so the + // observer promise can be resolved + await clickOnElementAndDelay(id, durThreshold); + return observerPromise; + } + // Apply events that trigger an event of the given |eventType| to be dispatched to the // |target|. Some of these assume that the target is not on the top left corner of the // screen, which means that (0, 0) of the viewport is outside of the |target|.
diff --git a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.html b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.html index 8545d06..a145e52 100644 --- a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.html +++ b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.html
@@ -31,7 +31,8 @@ assert_equals(cookieMap.get("test0")["secure"], false); assert_equals(cookieMap.get("test0")["httpOnly"], false); assert_equals(cookieMap.get("test0")["expiry"], undefined); - assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test1 [Expires in 10 days] assert_equals(cookieMap.get("test1")["name"], "test1"); @@ -41,7 +42,8 @@ assert_equals(cookieMap.get("test1")["secure"], false); assert_equals(cookieMap.get("test1")["httpOnly"], false); assert_equals(cookieMap.get("test1")["expiry"], Math.floor(kTenDaysFromNow.getTime()/1000)); - assert_equals(cookieMap.get("test1")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test2 [Path /] assert_equals(cookieMap.get("test2")["name"], "test2"); @@ -51,7 +53,8 @@ assert_equals(cookieMap.get("test2")["secure"], false); assert_equals(cookieMap.get("test2")["httpOnly"], false); assert_equals(cookieMap.get("test2")["expiry"], undefined); - assert_equals(cookieMap.get("test2")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test3 [HttpOnly] assert_equals(cookieMap.get("test3")["name"], "test3"); @@ -61,7 +64,8 @@ assert_equals(cookieMap.get("test3")["secure"], false); assert_equals(cookieMap.get("test3")["httpOnly"], true); assert_equals(cookieMap.get("test3")["expiry"], undefined); - assert_equals(cookieMap.get("test3")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test4 [Secure] Omitted
diff --git a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.https.html b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.https.html index 08ab8cc3..e3402780 100644 --- a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.https.html +++ b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_all_cookies.sub.https.html
@@ -31,7 +31,8 @@ assert_equals(cookieMap.get("test0")["secure"], false); assert_equals(cookieMap.get("test0")["httpOnly"], false); assert_equals(cookieMap.get("test0")["expiry"], undefined); - assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test1 [Expires in 10 days] assert_equals(cookieMap.get("test1")["name"], "test1"); @@ -41,7 +42,8 @@ assert_equals(cookieMap.get("test1")["secure"], false); assert_equals(cookieMap.get("test1")["httpOnly"], false); assert_equals(cookieMap.get("test1")["expiry"], Math.floor(kTenDaysFromNow.getTime()/1000)); - assert_equals(cookieMap.get("test1")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test2 [Path /] assert_equals(cookieMap.get("test2")["name"], "test2"); @@ -51,7 +53,8 @@ assert_equals(cookieMap.get("test2")["secure"], false); assert_equals(cookieMap.get("test2")["httpOnly"], false); assert_equals(cookieMap.get("test2")["expiry"], undefined); - assert_equals(cookieMap.get("test2")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test3 [HttpOnly] assert_equals(cookieMap.get("test3")["name"], "test3"); @@ -61,7 +64,8 @@ assert_equals(cookieMap.get("test3")["secure"], false); assert_equals(cookieMap.get("test3")["httpOnly"], true); assert_equals(cookieMap.get("test3")["expiry"], undefined); - assert_equals(cookieMap.get("test3")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test4 [Secure] assert_equals(cookieMap.get("test4")["name"], "test4"); @@ -71,7 +75,8 @@ assert_equals(cookieMap.get("test4")["secure"], true); assert_equals(cookieMap.get("test4")["httpOnly"], false); assert_equals(cookieMap.get("test4")["expiry"], undefined); - assert_equals(cookieMap.get("test4")["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookieMap.get("test0")["sameSite"], "Lax"); // test5 [SameSite Strict] assert_equals(cookieMap.get("test5")["name"], "test5");
diff --git a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.html b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.html index c3a8eed..3cbc784 100644 --- a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.html +++ b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.html
@@ -26,7 +26,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], false); assert_equals(cookie["expiry"], undefined); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test1 [Expires in 10 days] cookie = await test_driver.get_named_cookie("test1"); @@ -37,7 +38,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], false); assert_equals(cookie["expiry"], Math.floor(kTenDaysFromNow.getTime()/1000)); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test2 [Path /] cookie = await test_driver.get_named_cookie("test2"); @@ -48,7 +50,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], false); assert_equals(cookie["expiry"], undefined); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test3 [HttpOnly] cookie = await test_driver.get_named_cookie("test3"); @@ -59,7 +62,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], true); assert_equals(cookie["expiry"], undefined); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test4 [Secure] Omitted try {
diff --git a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.https.html b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.https.html index d316d4e..d0d8612a 100644 --- a/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.https.html +++ b/third_party/blink/web_tests/external/wpt/infrastructure/testdriver/get_named_cookie.sub.https.html
@@ -26,7 +26,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], false); assert_equals(cookie["expiry"], undefined); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test1 [Expires in 10 days] cookie = await test_driver.get_named_cookie("test1"); @@ -37,7 +38,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], false); assert_equals(cookie["expiry"], Math.floor(kTenDaysFromNow.getTime()/1000)); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test2 [Path /] cookie = await test_driver.get_named_cookie("test2"); @@ -48,7 +50,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], false); assert_equals(cookie["expiry"], undefined); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test3 [HttpOnly] cookie = await test_driver.get_named_cookie("test3"); @@ -59,7 +62,8 @@ assert_equals(cookie["secure"], false); assert_equals(cookie["httpOnly"], true); assert_equals(cookie["expiry"], undefined); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test4 [Secure] cookie = await test_driver.get_named_cookie("test4"); @@ -70,7 +74,8 @@ assert_equals(cookie["secure"], true); assert_equals(cookie["httpOnly"], false); assert_equals(cookie["expiry"], undefined); - assert_equals(cookie["sameSite"], "Lax"); + // The 'default lax' behavior varies by browser. + // assert_equals(cookie["sameSite"], "Lax"); // test5 [SameSite Strict] cookie = await test_driver.get_named_cookie("test5");
diff --git a/third_party/blink/web_tests/external/wpt/webrtc/idlharness.https.window-expected.txt b/third_party/blink/web_tests/external/wpt/webrtc/idlharness.https.window-expected.txt index c9ca12c2..19ec66a 100644 --- a/third_party/blink/web_tests/external/wpt/webrtc/idlharness.https.window-expected.txt +++ b/third_party/blink/web_tests/external/wpt/webrtc/idlharness.https.window-expected.txt
@@ -1,5 +1,5 @@ This is a testharness.js-based test. -Found 499 tests; 483 PASS, 16 FAIL, 0 TIMEOUT, 0 NOTRUN. +Found 499 tests; 484 PASS, 15 FAIL, 0 TIMEOUT, 0 NOTRUN. PASS idl_test setup PASS idl_test validation PASS Test driver for asyncInitCertificate @@ -41,7 +41,7 @@ PASS RTCPeerConnection interface: attribute canTrickleIceCandidates PASS RTCPeerConnection interface: operation restartIce() PASS RTCPeerConnection interface: operation getConfiguration() -FAIL RTCPeerConnection interface: operation setConfiguration(optional RTCConfiguration) assert_equals: property has wrong .length expected 0 but got 1 +PASS RTCPeerConnection interface: operation setConfiguration(optional RTCConfiguration) PASS RTCPeerConnection interface: operation close() PASS RTCPeerConnection interface: attribute onnegotiationneeded PASS RTCPeerConnection interface: attribute onicecandidate
diff --git a/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/README.md b/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/README.md deleted file mode 100644 index 31860a83..0000000 --- a/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/README.md +++ /dev/null
@@ -1 +0,0 @@ -Tests with WildcardSubdomainsInPermissionsPolicy enabled.
diff --git a/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/external/wpt/permissions-policy/README.txt b/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/external/wpt/permissions-policy/README.txt deleted file mode 100644 index 31860a83..0000000 --- a/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/external/wpt/permissions-policy/README.txt +++ /dev/null
@@ -1 +0,0 @@ -Tests with WildcardSubdomainsInPermissionsPolicy enabled.
diff --git a/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/external/wpt/permissions-policy/permissions-policy-header-policy-allowed-for-wildcard.https.sub-expected.txt b/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/external/wpt/permissions-policy/permissions-policy-header-policy-allowed-for-wildcard.https.sub-expected.txt deleted file mode 100644 index 23dbddc..0000000 --- a/third_party/blink/web_tests/virtual/disable-wildcard-subdomains-in-permissions-policy/external/wpt/permissions-policy/permissions-policy-header-policy-allowed-for-wildcard.https.sub-expected.txt +++ /dev/null
@@ -1,12 +0,0 @@ -This is a testharness.js-based test. -FAIL Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test allowlist is [self wildcard_origin]. assert_array_equals: expected property 0 to be "https://*.web-platform.test:8444" but got "https://%2A.web-platform.test:8444" (expected array ["https://*.web-platform.test:8444", "https://web-platform.test:8444"] got ["https://%2A.web-platform.test:8444", "https://web-platform.test:8444"]) -PASS Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is allowed on same-origin subframe -PASS Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is allowed on same-origin subframe even with allow attribute -PASS Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is disallowed on cross-origin subframe -FAIL Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is allowed on cross-origin subframe allow attribute assert_true: fullscreen expected true got false -PASS Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is disallowed on another cross-origin subframe -FAIL Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is allowed on another cross-origin subframe allow attribute assert_true: fullscreen expected true got false -PASS Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is disallowed on cross-origin subframe with wildcard allow attribute -PASS Permissions-Policy: fullscreen=("https://*.web-platform.test:8444") -- test fullscreen is disallowed on another cross-origin subframe with wildcard allow attribute -Harness: the test ran to completion. -
diff --git a/third_party/blink/web_tests/virtual/document-transition-wide-gamut/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt b/third_party/blink/web_tests/virtual/document-transition-wide-gamut/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt index 95e3f8f..06613e3 100644 --- a/third_party/blink/web_tests/virtual/document-transition-wide-gamut/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt +++ b/third_party/blink/web_tests/virtual/document-transition-wide-gamut/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt
@@ -101,28 +101,56 @@ value : 0px } [2] : { - name : right - value : 0px - } - [3] : { - name : bottom - value : 0px - } - [4] : { name : left value : 0px } - [5] : { + [3] : { name : pointer-events value : none } ] shorthandEntries : [ + ] + } + supports : [ + ] + } + } + [1] : { + matchingSelectors : [ + [0] : 0 + ] + rule : { + containerQueries : [ + ] + layers : [ + ] + media : [ + ] + origin : user-agent + scopes : [ + ] + selectorList : { + selectors : [ [0] : { - name : inset - value : 0px + text : html::page-transition } ] + text : html::page-transition + } + style : { + cssProperties : [ + [0] : { + name : width + value : 800px + } + [1] : { + name : height + value : 600px + } + ] + shorthandEntries : [ + ] } supports : [ ] @@ -1672,28 +1700,56 @@ value : 0px } [2] : { - name : right - value : 0px - } - [3] : { - name : bottom - value : 0px - } - [4] : { name : left value : 0px } - [5] : { + [3] : { name : pointer-events value : none } ] shorthandEntries : [ + ] + } + supports : [ + ] + } + } + [1] : { + matchingSelectors : [ + [0] : 0 + ] + rule : { + containerQueries : [ + ] + layers : [ + ] + media : [ + ] + origin : user-agent + scopes : [ + ] + selectorList : { + selectors : [ [0] : { - name : inset - value : 0px + text : html::page-transition } ] + text : html::page-transition + } + style : { + cssProperties : [ + [0] : { + name : width + value : 800px + } + [1] : { + name : height + value : 600px + } + ] + shorthandEntries : [ + ] } supports : [ ]
diff --git a/third_party/blink/web_tests/virtual/document-transition/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt b/third_party/blink/web_tests/virtual/document-transition/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt index 95e3f8f..06613e3 100644 --- a/third_party/blink/web_tests/virtual/document-transition/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt +++ b/third_party/blink/web_tests/virtual/document-transition/inspector-protocol/css/css-get-styles-for-document-transition-expected.txt
@@ -101,28 +101,56 @@ value : 0px } [2] : { - name : right - value : 0px - } - [3] : { - name : bottom - value : 0px - } - [4] : { name : left value : 0px } - [5] : { + [3] : { name : pointer-events value : none } ] shorthandEntries : [ + ] + } + supports : [ + ] + } + } + [1] : { + matchingSelectors : [ + [0] : 0 + ] + rule : { + containerQueries : [ + ] + layers : [ + ] + media : [ + ] + origin : user-agent + scopes : [ + ] + selectorList : { + selectors : [ [0] : { - name : inset - value : 0px + text : html::page-transition } ] + text : html::page-transition + } + style : { + cssProperties : [ + [0] : { + name : width + value : 800px + } + [1] : { + name : height + value : 600px + } + ] + shorthandEntries : [ + ] } supports : [ ] @@ -1672,28 +1700,56 @@ value : 0px } [2] : { - name : right - value : 0px - } - [3] : { - name : bottom - value : 0px - } - [4] : { name : left value : 0px } - [5] : { + [3] : { name : pointer-events value : none } ] shorthandEntries : [ + ] + } + supports : [ + ] + } + } + [1] : { + matchingSelectors : [ + [0] : 0 + ] + rule : { + containerQueries : [ + ] + layers : [ + ] + media : [ + ] + origin : user-agent + scopes : [ + ] + selectorList : { + selectors : [ [0] : { - name : inset - value : 0px + text : html::page-transition } ] + text : html::page-transition + } + style : { + cssProperties : [ + [0] : { + name : width + value : 800px + } + [1] : { + name : height + value : 600px + } + ] + shorthandEntries : [ + ] } supports : [ ]
diff --git a/third_party/blink/web_tests/wpt_internal/document-transition/content-with-clip-root.html b/third_party/blink/web_tests/wpt_internal/document-transition/content-with-clip-root.html index fa57a3b..1aa2d060 100644 --- a/third_party/blink/web_tests/wpt_internal/document-transition/content-with-clip-root.html +++ b/third_party/blink/web_tests/wpt_internal/document-transition/content-with-clip-root.html
@@ -41,7 +41,6 @@ } html::page-transition-container(root) { - position: fixed; top: -50vh; bottom: 50vh; }
diff --git a/third_party/blink/web_tests/wpt_internal/document-transition/new-content-has-scrollbars-ref.html b/third_party/blink/web_tests/wpt_internal/document-transition/new-content-has-scrollbars-ref.html new file mode 100644 index 0000000..b84efaa --- /dev/null +++ b/third_party/blink/web_tests/wpt_internal/document-transition/new-content-has-scrollbars-ref.html
@@ -0,0 +1,27 @@ +<!DOCTYPE html> +<title>Shared transitions: incoming viewport has scrollbars (ref)</title> +<link rel="help" href="https://github.com/WICG/shared-element-transitions"> +<link rel="help" href="https://github.com/w3c/csswg-drafts/issues/7859"> +<link rel="author" href="mailto:bokan@chromium.org"> +<style> + html, body { + width: 100%; + height: 100%; + background-color: lightpink; + } + body { + margin: 50px; + } + div { + background-image: + linear-gradient(45deg, #000 25%, transparent 25%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, #000 25%, lightgreen 25%); + background-size: 200px 200px; + background-position: 0 0, 0 0, -100px -100px, 100px 100px; + width: 200%; + height: 200%; + } +</style> +<div></div>
diff --git a/third_party/blink/web_tests/wpt_internal/document-transition/new-content-has-scrollbars.html b/third_party/blink/web_tests/wpt_internal/document-transition/new-content-has-scrollbars.html new file mode 100644 index 0000000..eb112b2d --- /dev/null +++ b/third_party/blink/web_tests/wpt_internal/document-transition/new-content-has-scrollbars.html
@@ -0,0 +1,61 @@ +<!DOCTYPE html> +<meta name="timeout" content="long"> +<html class=reftest-wait> +<title>Shared transitions: incoming viewport has scrollbars</title> +<link rel="help" href="https://github.com/WICG/shared-element-transitions"> +<link rel="help" href="https://github.com/w3c/csswg-drafts/issues/7859"> +<link rel="author" href="mailto:bokan@chromium.org"> +<link rel="match" href="new-content-has-scrollbars-ref.html"> +<meta name=fuzzy content="new-content-has-scrollbars-ref.html:0-40;0-30000"> +<script src="/common/reftest-wait.js"></script> +<style> + html, body { + width: 100%; + height: 100%; + background-color: lightpink; + } + html { + overflow: hidden; + } + body { + /* Margin to make sure background color is correctly drawn into the snapshot. */ + margin: 50px; + } + div { + /* Draw a checkerboard pattern to make sure the snapshot is captured at the + * full size, rather than scaled. */ + background-image: + linear-gradient(45deg, #000 25%, transparent 25%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, #000 25%, lightgreen 25%); + background-size: 200px 200px; + background-position: 0 0, 0 0, -100px -100px, 100px 100px; + width: 200%; + height: 200%; + } + + /* We're verifying what we capture, so just display the old contents for 5 minutes. */ + html::page-transition-incoming-image(root) { animation: unset; opacity: 0; } + html::page-transition-outgoing-image(root) { + animation-duration: 3s; + animation-timing-function: steps(1, end); + opacity: 1; + } +</style> + +<div></div> +<script> +// Ensure a root snapshot captured in the absence of scrollbars is displayed at +// full size when rendered in the incoming viewport which is inset by +// scrollbars. The content must not be scaled-to-fit. +async function runTest() { + let t = document.createDocumentTransition(); + t.start(() => { + document.documentElement.style.overflow = "unset"; + requestAnimationFrame(() => requestAnimationFrame(takeScreenshot)); + }); +} +onload = () => requestAnimationFrame(() => requestAnimationFrame(runTest)); +</script> +
diff --git a/third_party/blink/web_tests/wpt_internal/document-transition/old-content-has-scrollbars-ref.html b/third_party/blink/web_tests/wpt_internal/document-transition/old-content-has-scrollbars-ref.html new file mode 100644 index 0000000..4c7d9760 --- /dev/null +++ b/third_party/blink/web_tests/wpt_internal/document-transition/old-content-has-scrollbars-ref.html
@@ -0,0 +1,30 @@ +<!DOCTYPE html> +<title>Shared transitions: outgoing viewport has scrollbars (ref)</title> +<link rel="help" href="https://github.com/WICG/shared-element-transitions"> +<link rel="help" href="https://github.com/w3c/csswg-drafts/issues/7859"> +<link rel="author" href="mailto:bokan@chromium.org"> +<style> + html, body { + width: 100%; + height: 100%; + background-color: lightpink; + } + html { + overflow: hidden; + } + body { + margin: 50px; + } + div { + background-image: + linear-gradient(45deg, #000 25%, transparent 25%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, #000 25%, lightgreen 25%); + background-size: 200px 200px; + background-position: 0 0, 0 0, -100px -100px, 100px 100px; + width: 200%; + height: 200%; + } +</style> +<div></div>
diff --git a/third_party/blink/web_tests/wpt_internal/document-transition/old-content-has-scrollbars.html b/third_party/blink/web_tests/wpt_internal/document-transition/old-content-has-scrollbars.html new file mode 100644 index 0000000..b5314f8d --- /dev/null +++ b/third_party/blink/web_tests/wpt_internal/document-transition/old-content-has-scrollbars.html
@@ -0,0 +1,59 @@ +<!DOCTYPE html> +<meta name="timeout" content="long"> +<html class=reftest-wait> +<title>Shared transitions: outgoing viewport has scrollbars</title> +<link rel="help" href="https://github.com/WICG/shared-element-transitions"> +<link rel="help" href="https://github.com/w3c/csswg-drafts/issues/7859"> +<link rel="author" href="mailto:bokan@chromium.org"> +<link rel="match" href="old-content-has-scrollbars-ref.html"> +<meta name=fuzzy content="old-content-has-scrollbars-ref.html:0-40;0-30000"> +<script src="/common/reftest-wait.js"></script> +<style> + html, body { + width: 100%; + height: 100%; + background-color: lightpink; + } + body { + /* Margin to make sure background color is correctly drawn into the snapshot. */ + margin: 50px; + } + div { + /* Draw a checkerboard pattern to make sure the snapshot is captured at the + * full size, rather than scaled. */ + background-image: + linear-gradient(45deg, #000 25%, transparent 25%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, transparent 75%, #000 75%), + linear-gradient(45deg, #000 25%, lightgreen 25%); + background-size: 200px 200px; + background-position: 0 0, 0 0, -100px -100px, 100px 100px; + width: 200%; + height: 200%; + } + + /* We're verifying what we capture, so just display the old contents for 5 minutes. */ + html::page-transition-incoming-image(root) { animation: unset; opacity: 0; } + html::page-transition-outgoing-image(root) { + animation-duration: 300s; + animation-timing-function: steps(1, end); + opacity: 1; + } +</style> + +<div></div> +<script> +// Ensure a root snapshot captured on a page with scrollbars uses the full +// viewport size (including scrollbars). Areas obscured by scrollbars should be +// filled with background and content; the snapshot should not be scaled-to-fit. +async function runTest() { + let t = document.createDocumentTransition(); + t.start(() => { + document.querySelector('div').remove(); + document.documentElement.style.overflow = "hidden"; + requestAnimationFrame(() => requestAnimationFrame(takeScreenshot)); + }); +} +onload = () => requestAnimationFrame(() => requestAnimationFrame(runTest)); +</script> +
diff --git a/third_party/closure_compiler/externs/accessibility_private.js b/third_party/closure_compiler/externs/accessibility_private.js index 5237e93..c8731eb 100644 --- a/third_party/closure_compiler/externs/accessibility_private.js +++ b/third_party/closure_compiler/externs/accessibility_private.js
@@ -265,6 +265,7 @@ ENHANCED_NETWORK_VOICES: 'enhancedNetworkVoices', GOOGLE_TTS_LANGUAGE_PACKS: 'googleTtsLanguagePacks', DICTATION_PUMPKIN_PARSING: 'dictationPumpkinParsing', + DICTATION_MORE_COMMANDS: 'dictationMoreCommands', SELECT_TO_SPEAK_VOICE_SWITCHING: 'selectToSpeakVoiceSwitching', };
diff --git a/third_party/d3/OWNERS b/third_party/d3/OWNERS index 81967977..1bef084 100644 --- a/third_party/d3/OWNERS +++ b/third_party/d3/OWNERS
@@ -1,2 +1 @@ -chrisha@chromium.org zentaro@chromium.org
diff --git a/third_party/dom_distiller_js/BUILD.gn b/third_party/dom_distiller_js/BUILD.gn index 7096af7..81b46bc 100644 --- a/third_party/dom_distiller_js/BUILD.gn +++ b/third_party/dom_distiller_js/BUILD.gn
@@ -1,4 +1,4 @@ -# Copyright 2014 The Chromium Authors. All rights reserved. +# Copyright 2014 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file.
diff --git a/third_party/dom_distiller_js/LICENSE b/third_party/dom_distiller_js/LICENSE index 085efa7..3a8aeb9 100644 --- a/third_party/dom_distiller_js/LICENSE +++ b/third_party/dom_distiller_js/LICENSE
@@ -1,4 +1,4 @@ -Copyright 2014 The Chromium Authors. All rights reserved. +Copyright 2014 The Chromium Authors Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are
diff --git a/third_party/dom_distiller_js/protoc_plugins/json_values_converter.py b/third_party/dom_distiller_js/protoc_plugins/json_values_converter.py index a65f08f..6c044001 100755 --- a/third_party/dom_distiller_js/protoc_plugins/json_values_converter.py +++ b/third_party/dom_distiller_js/protoc_plugins/json_values_converter.py
@@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# Copyright 2016 The Chromium Authors. All rights reserved. +# Copyright 2016 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file.
diff --git a/third_party/dom_distiller_js/protoc_plugins/json_values_converter_tests.py b/third_party/dom_distiller_js/protoc_plugins/json_values_converter_tests.py index 0318983..54ae94fe 100755 --- a/third_party/dom_distiller_js/protoc_plugins/json_values_converter_tests.py +++ b/third_party/dom_distiller_js/protoc_plugins/json_values_converter_tests.py
@@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# Copyright (c) 2016 The Chromium Authors. All rights reserved. +# Copyright (c) 2016 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file.
diff --git a/third_party/dom_distiller_js/protoc_plugins/util/plugin_protos.py b/third_party/dom_distiller_js/protoc_plugins/util/plugin_protos.py index 352591c..2c03e6835 100644 --- a/third_party/dom_distiller_js/protoc_plugins/util/plugin_protos.py +++ b/third_party/dom_distiller_js/protoc_plugins/util/plugin_protos.py
@@ -1,4 +1,4 @@ -# Copyright 2016 The Chromium Authors. All rights reserved. +# Copyright 2016 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file.
diff --git a/third_party/dom_distiller_js/protoc_plugins/util/types.py b/third_party/dom_distiller_js/protoc_plugins/util/types.py index b82b8500..f33e01b 100644 --- a/third_party/dom_distiller_js/protoc_plugins/util/types.py +++ b/third_party/dom_distiller_js/protoc_plugins/util/types.py
@@ -1,4 +1,4 @@ -# Copyright 2016 The Chromium Authors. All rights reserved. +# Copyright 2016 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file.
diff --git a/third_party/dom_distiller_js/protoc_plugins/util/writer.py b/third_party/dom_distiller_js/protoc_plugins/util/writer.py index a56e9cb..9ee673a 100644 --- a/third_party/dom_distiller_js/protoc_plugins/util/writer.py +++ b/third_party/dom_distiller_js/protoc_plugins/util/writer.py
@@ -1,4 +1,4 @@ -# Copyright 2016 The Chromium Authors. All rights reserved. +# Copyright 2016 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file.
diff --git a/third_party/dom_distiller_js/test_sample.proto b/third_party/dom_distiller_js/test_sample.proto index bceb5c7..30167fe 100644 --- a/third_party/dom_distiller_js/test_sample.proto +++ b/third_party/dom_distiller_js/test_sample.proto
@@ -1,4 +1,4 @@ -// Copyright 2016 The Chromium Authors. All rights reserved. +// Copyright 2016 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file.
diff --git a/third_party/dom_distiller_js/update_domdistiller_js.sh b/third_party/dom_distiller_js/update_domdistiller_js.sh index 4cd40433..50417d1 100755 --- a/third_party/dom_distiller_js/update_domdistiller_js.sh +++ b/third_party/dom_distiller_js/update_domdistiller_js.sh
@@ -1,6 +1,6 @@ #!/bin/bash # -# Copyright 2014 The Chromium Authors. All rights reserved. +# Copyright 2014 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. #
diff --git a/third_party/grpc/BUILD.gn b/third_party/grpc/BUILD.gn index 1fd7a2e..986d256 100644 --- a/third_party/grpc/BUILD.gn +++ b/third_party/grpc/BUILD.gn
@@ -54,6 +54,10 @@ "-Wno-shadow", "-Wno-unreachable-code-return", "-Wno-unreachable-code-break", + + # TODO(crbug.com/1372394): Remove after integrating + # https://github.com/grpc/grpc/pull/31306 + "-Wno-thread-safety-analysis", ] if (is_android) {
diff --git a/third_party/grpc/template/BUILD.chromium.gn.template b/third_party/grpc/template/BUILD.chromium.gn.template index 991c94c..b30a7db 100644 --- a/third_party/grpc/template/BUILD.chromium.gn.template +++ b/third_party/grpc/template/BUILD.chromium.gn.template
@@ -15,11 +15,12 @@ # TODO(169395837): Somehow gRPC symbols cannot be found on Android. # Keep using static linking for now. - # In windows and mac use static linking. + # In windows, mac and iOS use static linking. # Use static linking on Chrome OS as a workaround for the symbol lookup # error(crbug/1241330) due to a gRPC version mismatch between what Chrome # uses and what CrOS provides. - grpc_use_static_linking = is_android || is_win || is_chromeos || is_mac + grpc_use_static_linking = + is_android || is_win || is_chromeos || is_mac || is_ios } if (is_android) { @@ -58,6 +59,10 @@ "-Wno-shadow", "-Wno-unreachable-code-return", "-Wno-unreachable-code-break", + + # TODO(crbug.com/1372394): Remove after integrating + # https://github.com/grpc/grpc/pull/31306 + "-Wno-thread-safety-analysis", ] if (is_android) {
diff --git a/tools/android/test_health/README.md b/tools/android/test_health/README.md index b10e50d9..9267322 100644 --- a/tools/android/test_health/README.md +++ b/tools/android/test_health/README.md
@@ -11,7 +11,7 @@ The `get_test_health.py` script extracts Java test health from a Git repository. The script defaults to the Chromium repository containing this script itself. The test health data includes a listing of tests that are disabled (annotated as -`@DisabledTest`), conditionally-disabled (`@DisableIf`) or flaky (`@FlakyTest`). +`@DisabledTest`) or conditionally-disabled (`@DisableIf`). The script exports the data in newline-delimited JSON ([JSON Lines](http://jsonlines.org)) format so that it can be easily [ingested into BigQuery][bq-load-gcs-json]. @@ -43,8 +43,8 @@ ### Java Test Utilities The `java_test_utils` module contains utility functions to extract counts of -test cases annotated with `@DisabledTest`, `@DisableIf` and `@FlakyTest`, as -well as the Java package name, from the source code of Java test files. +test cases annotated with `@DisabledTest` and `@DisableIf`, as well as the +Java package name, from the source code of Java test files. ### Other Modules
diff --git a/tools/android/test_health/java_test_utils.py b/tools/android/test_health/java_test_utils.py index 1082bd7..a1cd9f138 100644 --- a/tools/android/test_health/java_test_utils.py +++ b/tools/android/test_health/java_test_utils.py
@@ -32,7 +32,6 @@ _DISABLED_TEST_ANNOTATION = 'DisabledTest' _DISABLE_IF_TEST_ANNOTATION = 'DisableIf' -_FLAKY_TEST_ANNOTATION = 'FlakyTest' _DISABLE_IF_TEST_PATTERN = re.compile(_DISABLE_IF_TEST_ANNOTATION + r'\.\w+') @@ -47,7 +46,7 @@ disable_if_tests_count: int """The number of test cases annotated with @DisableIf.""" flaky_tests_count: int - """The number of test cases annotated with @FlakyTest.""" + """The number of test cases annotated with the removed @FlakyTest, now always 0.""" def get_java_test_health(test_path: pathlib.Path) -> JavaTestHealth: @@ -108,7 +107,7 @@ java_package=_get_java_package_name(java_ast), disabled_tests_count=annotation_counter[_DISABLED_TEST_ANNOTATION], disable_if_tests_count=annotation_counter[_DISABLE_IF_TEST_ANNOTATION], - flaky_tests_count=annotation_counter[_FLAKY_TEST_ANNOTATION]) + flaky_tests_count=0) def _get_java_package_name(java_ast: CompilationUnit) -> Optional[str]: @@ -137,8 +136,6 @@ counter[_DISABLED_TEST_ANNOTATION] += 1 elif re.fullmatch(_DISABLE_IF_TEST_PATTERN, annotation.name): counter[_DISABLE_IF_TEST_ANNOTATION] += 1 - elif annotation.name == _FLAKY_TEST_ANNOTATION: - counter[_FLAKY_TEST_ANNOTATION] += 1 return counter
diff --git a/tools/android/test_health/java_test_utils_unittest.py b/tools/android/test_health/java_test_utils_unittest.py index b290b62..1cdce9e 100755 --- a/tools/android/test_health/java_test_utils_unittest.py +++ b/tools/android/test_health/java_test_utils_unittest.py
@@ -33,13 +33,11 @@ 'SampleDisabledTest.java') _DISABLE_IF_TEST_PATH = (_TEST_FILES_PATH / 'disabled_tests' / 'SampleDisableIfTest.java') -_FLAKY_TEST_PATH = _TEST_FILES_PATH / 'flaky_tests' / 'SampleFlakyTest.java' _BASE_JAVA_PACKAGE = 'org.chromium.chrome.browser.test_health' _JAVA_PACKAGE_HEALTHY_TESTS = _BASE_JAVA_PACKAGE + '.healthy_tests' _JAVA_PACKAGE_UNHEALTHY_TESTS = _BASE_JAVA_PACKAGE + '.unhealthy_tests' _JAVA_PACKAGE_DISABLED_TESTS = _BASE_JAVA_PACKAGE + '.disabled_tests' -_JAVA_PACKAGE_FLAKY_TESTS = _BASE_JAVA_PACKAGE + '.flaky_tests' class TestJavaTestHealthStats(unittest.TestCase): @@ -70,7 +68,7 @@ test_health.java_package) self.assertEqual(1, test_health.disabled_tests_count) self.assertEqual(1, test_health.disable_if_tests_count) - self.assertEqual(1, test_health.flaky_tests_count) + self.assertEqual(0, test_health.flaky_tests_count) def test_get_java_test_health_stats_disabled_tests(self): test_health = java_test_utils.get_java_test_health(_DISABLED_TEST_PATH) @@ -91,14 +89,6 @@ self.assertEqual(2, test_health.disable_if_tests_count) self.assertEqual(0, test_health.flaky_tests_count) - def test_get_java_test_health_stats_flaky_tests(self): - test_health = java_test_utils.get_java_test_health(_FLAKY_TEST_PATH) - - self.assertEqual(_JAVA_PACKAGE_FLAKY_TESTS, test_health.java_package) - self.assertEqual(0, test_health.disabled_tests_count) - self.assertEqual(0, test_health.disable_if_tests_count) - self.assertEqual(2, test_health.flaky_tests_count) - def test_get_java_test_health_invalid_test_syntax(self): expected_filename = str( _INVALID_SYNTAX_TEST_PATH.relative_to(_CHROMIUM_SRC_PATH))
diff --git a/tools/android/test_health/test_health_extractor.py b/tools/android/test_health/test_health_extractor.py index c1f5251..aac763e 100644 --- a/tools/android/test_health/test_health_extractor.py +++ b/tools/android/test_health/test_health_extractor.py
@@ -65,7 +65,7 @@ ) -> List[TestHealthInfo]: """Gets test health information and stats for a Git repository. - This function checks for Java tests annotated as disabled or flaky but could + This function checks for Java tests annotated as disabled but could be extended to check other metrics or languages in the future. Args:
diff --git a/tools/android/test_health/test_health_extractor_unittest.py b/tools/android/test_health/test_health_extractor_unittest.py index ab05ae3d2..5318ddb 100755 --- a/tools/android/test_health/test_health_extractor_unittest.py +++ b/tools/android/test_health/test_health_extractor_unittest.py
@@ -31,7 +31,7 @@ _UNHEALTHY_TESTS_PATH / 'InvalidSyntaxTest.java').relative_to(_TEST_FILES_PATH) -_IGNORED_DIRS = ('disabled_tests', 'flaky_tests', 'unhealthy_tests') +_IGNORED_DIRS = ('disabled_tests', 'unhealthy_tests') _BASE_JAVA_PACKAGE = 'org.chromium.chrome.browser.test_health' _JAVA_PACKAGE_HEALTHY_TESTS = _BASE_JAVA_PACKAGE + '.healthy_tests' @@ -74,12 +74,12 @@ java_package=_JAVA_PACKAGE_UNHEALTHY_TESTS, disabled_tests_count=1, disable_if_tests_count=1, - flaky_tests_count=1), + flaky_tests_count=0), git_repo_info=_CHROMIUM_REPO_INFO) test_health_infos = test_health_extractor.get_repo_test_health( test_dir=_TEST_FILES_PATH, - ignored_dirs=('disabled_tests', 'flaky_tests', 'healthy_tests'), + ignored_dirs=('disabled_tests', 'healthy_tests'), ignored_files={str(_INVALID_SYNTAX_TEST_PATH)}) self.assertEqual(expected_test_health_info, test_health_infos[0]) @@ -101,8 +101,7 @@ with self.assertLogs(level=logging.WARNING) as logging_cm: test_health_infos = test_health_extractor.get_repo_test_health( test_dir=_TEST_FILES_PATH, - ignored_dirs=('disabled_tests', 'flaky_tests', - 'healthy_tests'), + ignored_dirs=('disabled_tests', 'healthy_tests'), ignored_files={ str((_UNHEALTHY_TESTS_PATH / 'SampleTest.java').relative_to(_TEST_FILES_PATH))
diff --git a/tools/android/test_health/testdata/javatests/org/chromium/chrome/browser/test_health/flaky_tests/SampleFlakyTest.java b/tools/android/test_health/testdata/javatests/org/chromium/chrome/browser/test_health/flaky_tests/SampleFlakyTest.java deleted file mode 100644 index 327844b..0000000 --- a/tools/android/test_health/testdata/javatests/org/chromium/chrome/browser/test_health/flaky_tests/SampleFlakyTest.java +++ /dev/null
@@ -1,41 +0,0 @@ -// Copyright 2021 The Chromium Authors -// Use of this source code is governed by a BSD-style license that can be -// found in the LICENSE file. - -package org.chromium.chrome.browser.test_health.flaky_tests; - -import androidx.test.filters.SmallTest; - -import org.junit.Assert; -import org.junit.Test; -import org.junit.runner.RunWith; - -import org.chromium.base.test.BaseJUnit4ClassRunner; - -import java.util.Random; - -/** A sample Java test with flaky test cases. */ -@SmallTest -@RunWith(BaseJUnit4ClassRunner.class) -public class SampleFlakyTest { - @FlakyTest(message = "Flaky since value is sometimes false.") - @Test - public void testFalseIsTrue() { - Random random = new Random(); - - boolean value = random.nextBoolean(); - - Assert.assertTrue(value); - } - - @FlakyTest - @Test - public void testTrueIsFalse() { - // Flaky since value is sometimes true. - Random random = new Random(); - - boolean value = random.nextBoolean(); - - Assert.assertFalse(value); - } -}
diff --git a/tools/android/test_health/testdata/javatests/org/chromium/chrome/browser/test_health/unhealthy_tests/SampleTest.java b/tools/android/test_health/testdata/javatests/org/chromium/chrome/browser/test_health/unhealthy_tests/SampleTest.java index 3d60625..cc7f969 100644 --- a/tools/android/test_health/testdata/javatests/org/chromium/chrome/browser/test_health/unhealthy_tests/SampleTest.java +++ b/tools/android/test_health/testdata/javatests/org/chromium/chrome/browser/test_health/unhealthy_tests/SampleTest.java
@@ -13,9 +13,6 @@ import org.chromium.base.test.BaseJUnit4ClassRunner; import org.chromium.base.test.util.DisableIf; import org.chromium.base.test.util.DisabledTest; -import org.chromium.base.test.util.FlakyTest; - -import java.util.Random; /** A sample Java test. */ @SmallTest @@ -42,14 +39,4 @@ public void testDisableIfTest() { Assert.assertTrue(false); } - - @FlakyTest - @Test - public void testFlakyTest() { - Random random = new Random(); - - boolean value = random.nextBoolean(); - - Assert.assertFalse(value); - } }
diff --git a/tools/clang/scripts/build.py b/tools/clang/scripts/build.py index 4b44cd26..764d487 100755 --- a/tools/clang/scripts/build.py +++ b/tools/clang/scripts/build.py
@@ -1019,15 +1019,20 @@ # results on # https://chromium-review.googlesource.com/c/chromium/src/+/3702739/4 # Maybe it should work for builtins too? - ('armv7-unknown-linux-gnueabihf', - compiler_rt_cmake_flags(sanitizers=True, profile=True) + [ - 'CMAKE_SYSROOT=%s' % sysroot_arm, - ])) - runtimes_triples_args.append( - ('aarch64-unknown-linux-gnu', - compiler_rt_cmake_flags(sanitizers=True, profile=True) + [ - 'CMAKE_SYSROOT=%s' % sysroot_arm64, - ])) + ( + 'armv7-unknown-linux-gnueabihf', + compiler_rt_cmake_flags(sanitizers=True, profile=True) + [ + 'CMAKE_SYSROOT=%s' % sysroot_arm, + # Can't run tests on x86 host. + 'LLVM_INCLUDE_TESTS=OFF', + ])) + runtimes_triples_args.append(( + 'aarch64-unknown-linux-gnu', + compiler_rt_cmake_flags(sanitizers=True, profile=True) + [ + 'CMAKE_SYSROOT=%s' % sysroot_arm64, + # Can't run tests on x86 host. + 'LLVM_INCLUDE_TESTS=OFF', + ])) elif sys.platform == 'win32': runtimes_triples_args.append( ('i386-pc-windows-msvc',
diff --git a/tools/gritsettings/resource_ids.spec b/tools/gritsettings/resource_ids.spec index 01f27f9..8b0842c 100644 --- a/tools/gritsettings/resource_ids.spec +++ b/tools/gritsettings/resource_ids.spec
@@ -474,6 +474,10 @@ "META": {"sizes": {"includes": [10]}}, "includes": [2665], }, + "<(SHARED_INTERMEDIATE_DIR)/chrome/browser/resources/lens/resources.grd": { + "META": {"sizes": {"includes": [10]}}, + "includes": [2675], + }, # END chrome/ WebUI resources section # START chrome/ miscellaneous section.
diff --git a/tools/metrics/actions/actions.xml b/tools/metrics/actions/actions.xml index 12903c68..d093ce1 100644 --- a/tools/metrics/actions/actions.xml +++ b/tools/metrics/actions/actions.xml
@@ -3619,6 +3619,9 @@ </action> <action name="AppList_ShowPlayStoreQueryResults"> + <obsolete> + Deprecated in M108. App List search no longer shows tile results. + </obsolete> <owner>jennyz@chromium.org</owner> <owner>newcomer@chromium.org</owner> <description> @@ -3660,6 +3663,7 @@ </action> <action name="AppList_ZeroStateOpenInstalledApp"> + <obsolete>Deprecated in M108. Launcher zero state has been removed.</obsolete> <owner>jennyz@chromium.org</owner> <owner>newcomer@chromium.org</owner> <description>User opens a suggested installed app in zero state.</description>
diff --git a/tools/metrics/histograms/enums.xml b/tools/metrics/histograms/enums.xml index ebc6183e..f8f36a345 100644 --- a/tools/metrics/histograms/enums.xml +++ b/tools/metrics/histograms/enums.xml
@@ -21688,6 +21688,7 @@ <int value="24" label="SMART_SELECT_BTWN_INCL"/> <int value="25" label="NAV_NEXT_SENT"/> <int value="26" label="NAV_PREV_SENT"/> + <int value="27" label="DELETE_ALL_TEXT"/> </enum> <enum name="CrosDictationToggleDictationMethod"> @@ -56726,6 +56727,7 @@ <int value="-2112547309" label="ClickToCallReceiver:enabled"/> <int value="-2111273398" label="SendTabToSelfManageDevicesLink:enabled"/> <int value="-2111196530" label="EnableAssistantLauncherUI:disabled"/> + <int value="-2108866112" label="FeedHeaderStickToTop:enabled"/> <int value="-2108564200" label="AutofillUpstream:disabled"/> <int value="-2106960993" label="HarfBuzzPDFSubsetter:disabled"/> <int value="-2106763275" label="ArcRtVcpuQuadCore:enabled"/> @@ -56820,6 +56822,7 @@ <int value="-2054612904" label="BuiltInModuleInfra:enabled"/> <int value="-2053860791" label="XGEOVisibleNetworks:enabled"/> <int value="-2053062676" label="EnableNeuralPalmAdaptiveHold:enabled"/> + <int value="-2050829390" label="LensRegionSearchStaticPage:enabled"/> <int value="-2050039250" label="JourneysOnDeviceClusteringContentClustering:enabled"/> <int value="-2048927838" label="AutoplayWhitelistSettings:enabled"/> @@ -56943,6 +56946,7 @@ <int value="-1991935790" label="ShareCrowLaunchTab:enabled"/> <int value="-1990678808" label="SafetyCheckUnusedSitePermissions:disabled"/> <int value="-1990614981" label="StoragePressureUI:disabled"/> + <int value="-1990583427" label="FeedHeaderStickToTop:disabled"/> <int value="-1990238241" label="ChromeOSHWVBREncoding:enabled"/> <int value="-1989747818" label="TabStripKeyboardFocus:disabled"/> <int value="-1989624484" label="StartSurfaceRefactor:disabled"/> @@ -61223,6 +61227,7 @@ <int value="564355877" label="OopRasterizationDDL:disabled"/> <int value="564452531" label="SystemEmojiPickerExtension:enabled"/> <int value="564522013" label="Av1Decoder:disabled"/> + <int value="565254510" label="LensRegionSearchStaticPage:disabled"/> <int value="565406673" label="EnableVirtualKeyboardMdUi:enabled"/> <int value="567368307" label="enable-experimental-canvas-features"/> <int value="570469494" label="LoginDetection:disabled"/> @@ -95575,6 +95580,7 @@ <int value="52" label="Printers Authorization Servers"/> <int value="53" label="Contact info"/> <int value="54" label="Autofill Wallet Usage"/> + <int value="55" label="Segmentation"/> </enum> <enum name="SyncModelTypeStoreInitResult"> @@ -103357,6 +103363,27 @@ <int value="3" label="Transferred WebContents"/> </enum> +<enum name="WebFeedChangeReason"> + <int value="0" label="Unspecified">Unknown change reason</int> + <int value="1" label="Web page menu"> + The user tapped the Follow/Unfollow option from the three dot menu while on + a web page. + </int> + <int value="2" label="Web page accelerator"> + The user tapped the Follow accelerator while visiting a site. + </int> + <int value="3" label="Management page"> + The user tapped to Follow/Unfollow on the Follow Management page. + </int> + <int value="4" label="In feed recommendation"> + The user tapped to Follow/Unfollow a recommendation embedded in feed + content. + </int> + <int value="5" label="Back of card unfollow"> + The user tapped 'Unfollow' on the back of card menu. + </int> +</enum> + <enum name="WebFeedPageInformationRequestReason"> <int value="0" label="User Requested Follow"> The user requested to Follow the current web page.
diff --git a/tools/metrics/histograms/metadata/apps/histograms.xml b/tools/metrics/histograms/metadata/apps/histograms.xml index beb5ac0..95f1efc3 100644 --- a/tools/metrics/histograms/metadata/apps/histograms.xml +++ b/tools/metrics/histograms/metadata/apps/histograms.xml
@@ -1499,6 +1499,9 @@ <histogram name="Apps.AppListPlayStoreAppLaunchedIndex" units="indices" expires_after="2021-12-31"> + <obsolete> + Deprecated in M108. Tiled app results are no longer shown. + </obsolete> <owner>jennyz@chromium.org</owner> <owner>newcomer@chromium.org</owner> <summary> @@ -1515,10 +1518,14 @@ <histogram name="Apps.AppListPlayStoreSearchAppsDisplayed" units="apss" expires_after="2021-12-31"> + <obsolete> + Deprecated in M108. Tiled app results are no longer shown. + </obsolete> <owner>jennyz@chromium.org</owner> <owner>newcomer@chromium.org</owner> <summary> - The number of the play store apps displayed to user for a query. + Deprecated in Oct 2022. The number of the play store apps displayed to user + as tile items for a query. </summary> </histogram>
diff --git a/tools/metrics/histograms/metadata/arc/histograms.xml b/tools/metrics/histograms/metadata/arc/histograms.xml index 1a0ba11..eb68c155 100644 --- a/tools/metrics/histograms/metadata/arc/histograms.xml +++ b/tools/metrics/histograms/metadata/arc/histograms.xml
@@ -1658,6 +1658,9 @@ <histogram name="Arc.PlayStoreSearch.DefaultResultClickLatency" units="ms" expires_after="2021-12-31"> + <obsolete> + Deprecated in M108. Tiled app results are no longer shown. + </obsolete> <owner>jennyz@chromium.org</owner> <owner>newcomer@chromium.org</owner> <summary>
diff --git a/tools/metrics/histograms/metadata/content/histograms.xml b/tools/metrics/histograms/metadata/content/histograms.xml index 427bda19..596259c 100644 --- a/tools/metrics/histograms/metadata/content/histograms.xml +++ b/tools/metrics/histograms/metadata/content/histograms.xml
@@ -1473,6 +1473,19 @@ </token> </histogram> +<histogram name="ContentSuggestions.Feed.WebFeed.NewFollow.ChangeReason" + enum="WebFeedChangeReason" expires_after="2023-04-01"> + <owner>harringtond@chromium.org</owner> + <owner>feed@chromium.org</owner> + <summary> + Change reason for the follow. + + Reported upon successfully following a web feed. Reported as 'true' if the + web feed was recommended, even if the user did not see or interact with + recommendation UI. + </summary> +</histogram> + <histogram name="ContentSuggestions.Feed.WebFeed.NewFollow.IsRecommended" enum="Boolean" expires_after="2023-03-01"> <owner>harringtond@chromium.org</owner> @@ -1480,9 +1493,7 @@ <summary> Whether the followed web feed was recommended by the server. - Reported upon successfully following a web feed. Reported as 'true' if the - web feed was recommended, even if the user did not see or interact with - recommendation UI. + Reported upon successfully following a web feed. </summary> </histogram>
diff --git a/tools/metrics/histograms/metadata/sync/histograms.xml b/tools/metrics/histograms/metadata/sync/histograms.xml index 073c8c66..dea969ba 100644 --- a/tools/metrics/histograms/metadata/sync/histograms.xml +++ b/tools/metrics/histograms/metadata/sync/histograms.xml
@@ -54,6 +54,7 @@ <variant name=".READING_LIST" summary="READING_LIST"/> <variant name=".SEARCH_ENGINE" summary="SEARCH_ENGINE"/> <variant name=".SECURITY_EVENT" summary="SECURITY_EVENT"/> + <variant name=".SEGMENTATION" summary="SEGMENTATION"/> <variant name=".SEND_TAB_TO_SELF" summary="SEND_TAB_TO_SELF"/> <variant name=".SESSION" summary="SESSION"/> <variant name=".SHARING_MESSAGE" summary="SHARING_MESSAGE"/>
diff --git a/tools/metrics/ukm/ukm.xml b/tools/metrics/ukm/ukm.xml index 7c97b6eb..2950587 100644 --- a/tools/metrics/ukm/ukm.xml +++ b/tools/metrics/ukm/ukm.xml
@@ -16542,6 +16542,11 @@ PageTimelineState UKM events. One PageTimelineState is recorded for each browser tab when a slice is collected. </summary> + <metric name="BatterySaverMode" enum="Boolean"> + <summary> + True if Batter Saver Mode is enabled at the time this event is recorded. + </summary> + </metric> <metric name="ChangedFaviconOrTitleInBackground" enum="Boolean"> <summary> True if this page has been observed to change its favicon or title when in @@ -16553,6 +16558,43 @@ The state the tab was in when the slice was recorded. </summary> </metric> + <metric name="HasNotificationPermission" enum="Boolean"> + <summary> + True if this page has been granted the notification permission. + </summary> + </metric> + <metric name="HighEfficiencyMode" enum="Boolean"> + <summary> + True if High Efficiency Mode is enabled at the time this event is + recorded. + </summary> + </metric> + <metric name="IsActiveTab" enum="Boolean"> + <summary> + True if this page is the active tab in its tab strip. + </summary> + </metric> + <metric name="IsCapturingMedia" enum="Boolean"> + <summary> + True if this page is capturing media (audio/video/screen). + </summary> + </metric> + <metric name="IsConnectedToDevice" enum="Boolean"> + <summary> + True if this page is connected to a BT or USB device. + </summary> + </metric> + <metric name="IsPlayingAudio" enum="Boolean"> + <summary> + True if this page is playing audio. + </summary> + </metric> + <metric name="ResidentSetSize"> + <summary> + The total resident set size footprint of this page at the moment the + sample was recorded, in bytes. + </summary> + </metric> <metric name="SliceId"> <summary> A strictly increasing ID used to order slices in relation to each other.
diff --git a/tools/perf/benchmarks/loading_metrics_category.py b/tools/perf/benchmarks/loading_metrics_category.py index b507ecf..25d9db7e 100644 --- a/tools/perf/benchmarks/loading_metrics_category.py +++ b/tools/perf/benchmarks/loading_metrics_category.py
@@ -26,5 +26,6 @@ # properly compute time-to-interactive. cat_filter.AddDisabledByDefault('disabled-by-default-network') - tbm_options.AddTimelineBasedMetric('loadingMetric') + tbm_options.ExtendTimelineBasedMetric( + ['loadingMetric', 'coreWebVitalsMetric']) return tbm_options
diff --git a/tools/perf/core/perfetto_binary_roller/binary_deps.json b/tools/perf/core/perfetto_binary_roller/binary_deps.json index 3ef368e..dc3fa53 100644 --- a/tools/perf/core/perfetto_binary_roller/binary_deps.json +++ b/tools/perf/core/perfetto_binary_roller/binary_deps.json
@@ -6,15 +6,15 @@ }, "win": { "hash": "cd36010f4aaa8261c0ead7927e3b5b0c51b23680", - "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/win/08c1f851b8a083fbf2e052f42f190db9b8915e0b/trace_processor_shell.exe" + "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/win/7bd83a8f9af8604e932cd717fdd56e586c54d3a4/trace_processor_shell.exe" }, "linux_arm": { "hash": "58893933be305d3bfe0a72ebebcacde2ac3ca893", "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/linux_arm/49b4b5dcbc312d8d2c3751cf29238b8efeb4e494/trace_processor_shell" }, "mac": { - "hash": "3c5d52cc52064a87d37f0e22efbb8d0ea38856ff", - "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/mac/fabd415c7415ae91f871fc08683cd3bdcc47830e/trace_processor_shell" + "hash": "b3d9c24d59f3e90d4cc0351d61bd90f45f8ac025", + "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/mac/7bd83a8f9af8604e932cd717fdd56e586c54d3a4/trace_processor_shell" }, "mac_arm64": { "hash": "92318bea34f5c9beec69d2d826a9a92ec9d3cdcd", @@ -22,7 +22,7 @@ }, "linux": { "hash": "9ec936285839034b556c7e91136aeda7c8a59150", - "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/linux/08c1f851b8a083fbf2e052f42f190db9b8915e0b/trace_processor_shell" + "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/linux/7bd83a8f9af8604e932cd717fdd56e586c54d3a4/trace_processor_shell" } }, "power_profile.sql": {
diff --git a/ui/compositor/total_animation_throughput_reporter.cc b/ui/compositor/total_animation_throughput_reporter.cc index ffd295b..53f0608 100644 --- a/ui/compositor/total_animation_throughput_reporter.cc +++ b/ui/compositor/total_animation_throughput_reporter.cc
@@ -10,19 +10,6 @@ namespace ui { -TotalAnimationThroughputReporter::ScopedThroughputReporterBlocker:: - ScopedThroughputReporterBlocker( - base::WeakPtr<TotalAnimationThroughputReporter> reporter) - : reporter_(std::move(reporter)) { - reporter_->scoped_blocker_count_++; -} - -TotalAnimationThroughputReporter::ScopedThroughputReporterBlocker:: - ~ScopedThroughputReporterBlocker() { - if (reporter_) - reporter_->scoped_blocker_count_--; -} - TotalAnimationThroughputReporter::TotalAnimationThroughputReporter( ui::Compositor* compositor, ReportOnceCallback once_callback, @@ -58,9 +45,6 @@ void TotalAnimationThroughputReporter::OnFirstNonAnimatedFrameStarted( ui::Compositor* compositor) { - if (IsBlocked()) - return; - throughput_tracker_->Stop(); throughput_tracker_.reset(); // Stop observing if no need to report multiple times. @@ -80,19 +64,6 @@ delete this; } -base::WeakPtr<ui::TotalAnimationThroughputReporter> -TotalAnimationThroughputReporter::GetWeakPtr() { - return ptr_factory_.GetWeakPtr(); -} - -std::unique_ptr< - TotalAnimationThroughputReporter::ScopedThroughputReporterBlocker> -TotalAnimationThroughputReporter::NewScopedBlocker() { - return std::make_unique< - ui::TotalAnimationThroughputReporter::ScopedThroughputReporterBlocker>( - ptr_factory_.GetWeakPtr()); -} - TotalAnimationThroughputReporter::TotalAnimationThroughputReporter( ui::Compositor* compositor, ReportRepeatingCallback repeating_callback, @@ -123,8 +94,4 @@ report_repeating_callback_.Run(data); } -bool TotalAnimationThroughputReporter::IsBlocked() const { - return scoped_blocker_count_; -} - } // namespace ui
diff --git a/ui/compositor/total_animation_throughput_reporter.h b/ui/compositor/total_animation_throughput_reporter.h index 866676e..bddc5e4 100644 --- a/ui/compositor/total_animation_throughput_reporter.h +++ b/ui/compositor/total_animation_throughput_reporter.h
@@ -14,10 +14,6 @@ #include "ui/compositor/compositor_observer.h" #include "ui/compositor/throughput_tracker.h" -namespace ash { -class LoginUnlockThroughputRecorderTestBase; -} - namespace ui { // Reports cc::FrameSequenceMetrics::ThroughputData between the first animation @@ -36,31 +32,9 @@ // report callback is invoked on the next begin frame if there is enough data. // Since this observes multiple animations, aborting one of animations will // not cancel the tracking, and the data will be reported as normal. -// -// The reporter will not fire if ScopedThroughputReporterBlocker is active. -// This allows to measure throughput from the very first animation (when -// reporter was created) till the specific expected animation ends even if -// there were delays between the animations. class COMPOSITOR_EXPORT TotalAnimationThroughputReporter : public CompositorObserver { public: - // This allows to temporarily ignore OnFirstNonAnimatedFrameStarted event - // until an interesting event happens. - class COMPOSITOR_EXPORT ScopedThroughputReporterBlocker { - public: - explicit ScopedThroughputReporterBlocker( - base::WeakPtr<TotalAnimationThroughputReporter> reporter); - ScopedThroughputReporterBlocker(const ScopedThroughputReporterBlocker&) = - delete; - ~ScopedThroughputReporterBlocker(); - - ScopedThroughputReporterBlocker& operator=( - const ScopedThroughputReporterBlocker&) = delete; - - private: - base::WeakPtr<TotalAnimationThroughputReporter> reporter_; - }; - using ReportOnceCallback = base::OnceCallback<void( const cc::FrameSequenceMetrics::CustomReportData& data)>; using ReportRepeatingCallback = base::RepeatingCallback<void( @@ -90,18 +64,9 @@ void OnFirstNonAnimatedFrameStarted(Compositor* compositor) override; void OnCompositingShuttingDown(Compositor* compositor) override; - base::WeakPtr<ui::TotalAnimationThroughputReporter> GetWeakPtr(); - bool IsMeasuringForTesting() const { return bool{throughput_tracker_}; } - // The returned scope will delay the animation report until the next - // |OnFirstNonAnimatedFrameStarted| received after it is destructed. See - // |ScopedThroughputReporterBlocker| above. - std::unique_ptr<ScopedThroughputReporterBlocker> NewScopedBlocker(); - private: - friend class ash::LoginUnlockThroughputRecorderTestBase; - TotalAnimationThroughputReporter(Compositor* compositor, ReportRepeatingCallback repeating_callback, ReportOnceCallback once_callback, @@ -109,18 +74,12 @@ void Report(const cc::FrameSequenceMetrics::CustomReportData& data); - // Returns true if there is an active ScopedThroughputReporterBlocker. - bool IsBlocked() const; - raw_ptr<Compositor> compositor_; ReportRepeatingCallback report_repeating_callback_; ReportOnceCallback report_once_callback_; bool should_delete_ = false; absl::optional<ThroughputTracker> throughput_tracker_; - // Number of active ScopedThroughputReporterBlocker objects. - int scoped_blocker_count_ = 0; - base::WeakPtrFactory<TotalAnimationThroughputReporter> ptr_factory_{this}; };
diff --git a/ui/compositor/total_animation_throughput_reporter_unittest.cc b/ui/compositor/total_animation_throughput_reporter_unittest.cc index b4c02dc..7f8d369 100644 --- a/ui/compositor/total_animation_throughput_reporter_unittest.cc +++ b/ui/compositor/total_animation_throughput_reporter_unittest.cc
@@ -48,58 +48,6 @@ layer.SetOpacity(opacity); } -class TestCompositorMonitor : public ui::CompositorObserver { - public: - explicit TestCompositorMonitor(ui::Compositor* compositor) - : compositor_(compositor) { - compositor->AddObserver(this); - } - - ~TestCompositorMonitor() override { compositor_->RemoveObserver(this); } - - // ui::CompositorObserver - void OnFirstAnimationStarted(Compositor* compositor) override { - animatins_running_ = true; - } - - void OnFirstNonAnimatedFrameStarted(Compositor* compositor) override { - DCHECK_EQ(compositor_, compositor); - if (animatins_running_) { - waiting_for_did_present_compositor_frame_ = true; - } - animatins_running_ = false; - } - - void OnDidPresentCompositorFrame( - uint32_t frame_token, - const gfx::PresentationFeedback& feedback) override { - if (waiting_for_did_present_compositor_frame_) { - waiting_for_did_present_compositor_frame_ = false; - if (animatins_running_) - return; - - if (run_loop_) - run_loop_->Quit(); - } - } - - void WaitForAllAnimationsEnd() { - if (!animatins_running_ && !waiting_for_did_present_compositor_frame_) - return; - - run_loop_ = std::make_unique<base::RunLoop>( - base::RunLoop::Type::kNestableTasksAllowed); - run_loop_->Run(); - run_loop_.reset(); - } - - private: - const base::raw_ptr<ui::Compositor> compositor_; - bool animatins_running_ = false; - bool waiting_for_did_present_compositor_frame_ = false; - std::unique_ptr<base::RunLoop> run_loop_; -}; - } // namespace using TotalAnimationThroughputReporterTest = @@ -113,21 +61,7 @@ ThroughputReportChecker checker(this); TotalAnimationThroughputReporter reporter(compositor(), checker.repeating_callback()); - auto scoped_blocker = reporter.NewScopedBlocker(); SetLayerOpacity(layer, 1.0f, base::Milliseconds(48)); - Advance(base::Milliseconds(200)); - - // No report should happen while scoped_blocker exists. - EXPECT_FALSE(checker.reported()); - - scoped_blocker.reset(); - // No animation should be running yet, nothing to report. - EXPECT_FALSE(checker.reported()); - Advance(base::Milliseconds(200)); - EXPECT_FALSE(checker.reported()); - - // Animation of opacity goes to 0.5. - SetLayerOpacity(layer, 0.5f, base::Milliseconds(48)); Advance(base::Milliseconds(32)); EXPECT_FALSE(checker.reported()); EXPECT_TRUE(checker.WaitUntilReported()); @@ -192,6 +126,7 @@ ThroughputReportChecker checker(this); TotalAnimationThroughputReporter reporter(compositor(), checker.repeating_callback()); + SetLayerOpacity(layer, 1.0f, base::Milliseconds(48)); { LayerAnimator* animator = layer.GetAnimator(); @@ -354,7 +289,6 @@ // Make sure the once reporter is called only once. TEST_F(TotalAnimationThroughputReporterTest, OnceReporter) { - TestCompositorMonitor compositor_monitor(compositor()); Layer layer; layer.SetOpacity(0.5f); root_layer()->Add(&layer); @@ -366,7 +300,6 @@ ThroughputReportChecker checker(this); TotalAnimationThroughputReporter reporter( compositor(), checker.once_callback(), /*should_delete=*/false); - auto scoped_blocker = reporter.NewScopedBlocker(); // Make sure the TotalAnimationThroughputReporter removes itself // from compositor as observer. @@ -374,22 +307,6 @@ // Report data for animation of opacity goes to 1. SetLayerOpacity(layer, 1.0f, base::Milliseconds(48)); - Advance(base::Milliseconds(100)); - - // No report should happen while scoped_blocker exists. - EXPECT_FALSE(checker.reported()); - - // Make sure there are no animations running. - compositor_monitor.WaitForAllAnimationsEnd(); - - scoped_blocker.reset(); - // No animation should be running yet, nothing to report. - EXPECT_FALSE(checker.reported()); - Advance(base::Milliseconds(100)); - EXPECT_FALSE(checker.reported()); - - // Animation of opacity goes to 0.5. - SetLayerOpacity(layer, 0.7f, base::Milliseconds(48)); EXPECT_TRUE(checker.WaitUntilReported()); // Report data for animation of opacity goes to 0.5. @@ -417,7 +334,6 @@ raw_ptr<bool> deleted_; }; - TestCompositorMonitor compositor_monitor(compositor()); Layer layer; layer.SetOpacity(0.5f); root_layer()->Add(&layer); @@ -430,40 +346,21 @@ base::RunLoop run_loop; bool deleted = false; - TotalAnimationThroughputReporter* reporter = new DeleteTestReporter( + new DeleteTestReporter( compositor(), base::BindLambdaForTesting( [&](const cc::FrameSequenceMetrics::CustomReportData&) { run_loop.Quit(); }), &deleted); - auto scoped_blocker = reporter->NewScopedBlocker(); // Report data for animation of opacity goes to 1. SetLayerOpacity(layer, 1.0f, base::Milliseconds(48)); - Advance(base::Milliseconds(100)); - - // No report should happen while scoped_blocker exists. - EXPECT_FALSE(deleted); - - // Make sure there are no animations running. - compositor_monitor.WaitForAllAnimationsEnd(); - - scoped_blocker.reset(); - // No animation should be running yet, nothing to report. - EXPECT_FALSE(deleted); - Advance(base::Milliseconds(100)); - EXPECT_FALSE(deleted); - - // Animation of opacity goes to 0.5. - layer.SetOpacity(0.7f); - EXPECT_FALSE(deleted); - Advance(base::Milliseconds(100)); + run_loop.Run(); EXPECT_TRUE(deleted); } TEST_F(TotalAnimationThroughputReporterTest, ThreadCheck) { - TestCompositorMonitor compositor_monitor(compositor()); Layer layer; layer.SetOpacity(0.5f); root_layer()->Add(&layer); @@ -486,26 +383,9 @@ TotalAnimationThroughputReporter reporter(c, std::move(callback), /*should_delete=*/false); - auto scoped_blocker = reporter.NewScopedBlocker(); // Report data for animation of opacity goes to 1. layer.SetOpacity(1.0f); - Advance(base::Milliseconds(100)); - - // No report should happen while scoped_blocker exists. - EXPECT_FALSE(checker.reported()); - - // Make sure there are no animations running. - compositor_monitor.WaitForAllAnimationsEnd(); - - scoped_blocker.reset(); - // No animation should be running yet, nothing to report. - EXPECT_FALSE(checker.reported()); - Advance(base::Milliseconds(100)); - EXPECT_FALSE(checker.reported()); - - // Animation of opacity goes to 0.5. - layer.SetOpacity(0.7f); EXPECT_TRUE(checker.WaitUntilReported()); }
diff --git a/ui/views/bubble/bubble_dialog_model_host.cc b/ui/views/bubble/bubble_dialog_model_host.cc index 907ad7640..8766c097 100644 --- a/ui/views/bubble/bubble_dialog_model_host.cc +++ b/ui/views/bubble/bubble_dialog_model_host.cc
@@ -17,7 +17,6 @@ #include "ui/base/models/dialog_model_field.h" #include "ui/base/ui_base_types.h" #include "ui/gfx/geometry/insets.h" -#include "ui/views/accessibility/accessibility_paint_checks.h" #include "ui/views/accessibility/view_accessibility.h" #include "ui/views/border.h" #include "ui/views/bubble/bubble_dialog_delegate_view.h" @@ -797,12 +796,6 @@ // TODO(pbos): Support updates to the existing model. auto textfield = std::make_unique<Textfield>(); - // TODO(crbug.com/1218186): Remove this, this is in place temporarily to be - // able to submit accessibility checks, but this focusable View needs to - // add a name so that the screen reader knows what to announce. The - // placeholder name may need to be pushed into DialogModel, unless we can tie - // this to the label. Maybe SetAssociatedField on Textfield is sufficient? - textfield->SetProperty(kSkipAccessibilityPaintChecks, true); textfield->SetAccessibleName( model_field->accessible_name(GetPassKey()).empty() ? model_field->label(GetPassKey())
diff --git a/ui/views/controls/combobox/combobox.cc b/ui/views/controls/combobox/combobox.cc index 1be7478..8ece700 100644 --- a/ui/views/controls/combobox/combobox.cc +++ b/ui/views/controls/combobox/combobox.cc
@@ -432,9 +432,6 @@ } if (new_index.has_value()) { - if (menu_selection_at_callback_) - menu_selection_at_callback_.Run(new_index.value()); - SetSelectedIndex(new_index); OnPerformAction(); }
diff --git a/ui/webui/resources/BUILD.gn b/ui/webui/resources/BUILD.gn index eeb848b..e287da1 100644 --- a/ui/webui/resources/BUILD.gn +++ b/ui/webui/resources/BUILD.gn
@@ -160,7 +160,6 @@ "js/cr.m.js", "js/cr/ui.js", "js/load_time_data.m.js", - "js/icon.js", "js/util.js", ] @@ -185,6 +184,7 @@ "js/focus_grid.ts", "js/focus_row.ts", "js/focus_outline_manager.ts", + "js/icon.ts", "js/keyboard_shortcut_list.ts", "js/plural_string_proxy.ts", "js/static_types.ts",
diff --git a/ui/webui/resources/js/BUILD.gn b/ui/webui/resources/js/BUILD.gn index acbc5ae..ccb1561 100644 --- a/ui/webui/resources/js/BUILD.gn +++ b/ui/webui/resources/js/BUILD.gn
@@ -48,6 +48,7 @@ "focus_grid.ts", "focus_outline_manager.ts", "focus_row.ts", + "icon.ts", "keyboard_shortcut_list.ts", "plural_string_proxy.ts", "search_highlight_utils.ts", @@ -78,7 +79,6 @@ "cr.m.js", "cr/event_target.js", "cr/ui.js", - "icon.js", "load_time_data.m.js", "load_time_data_deprecated.js", "parse_html_subset.js", @@ -138,7 +138,6 @@ deps = [ ":assert", ":cr.m", - ":icon", ":load_time_data.m", ":parse_html_subset", ":promise_resolver", @@ -157,10 +156,6 @@ externs_list = [ "$externs_path/chrome_send.js" ] } -js_library("icon") { - deps = [ ":cr.m" ] -} - js_library("load_time_data.m") { }
diff --git a/ui/webui/resources/js/icon.js b/ui/webui/resources/js/icon.ts similarity index 76% rename from ui/webui/resources/js/icon.js rename to ui/webui/resources/js/icon.ts index 5cd55f0..a19ce7b 100644 --- a/ui/webui/resources/js/icon.js +++ b/ui/webui/resources/js/icon.ts
@@ -5,10 +5,9 @@ import {isAndroid, isIOS} from './cr.m.js'; /** - * @return {!Array<number>} The scale factors supported by this platform for - * webui resources. + * @return The scale factors supported by this platform for webui resources. */ -function getSupportedScaleFactors() { +function getSupportedScaleFactors(): number[] { const supportedScaleFactors = []; if (!isIOS) { // This matches the code in ResourceBundle::InitSharedInstance() that @@ -33,10 +32,10 @@ /** * Generates a CSS url string. - * @param {string} s The URL to generate the CSS url for. - * @return {string} The CSS url string. + * @param s The URL to generate the CSS url for. + * @return The CSS url string. */ -export function getUrlForCss(s) { +export function getUrlForCss(s: string): string { // http://www.w3.org/TR/css3-values/#uris // Parentheses, commas, whitespace characters, single quotes (') and double // quotes (") appearing in a URI must be escaped with a backslash @@ -46,10 +45,8 @@ /** * A URL for the filetype icon for |filePath|. OS and theme dependent. - * @param {string} filePath - * @return {string} */ -export function getFileIconUrl(filePath) { +export function getFileIconUrl(filePath: string): string { const url = new URL('chrome://fileicon/'); url.searchParams.set('path', filePath); url.searchParams.set('scale', window.devicePixelRatio + 'x'); @@ -62,11 +59,11 @@ * The scale-factor-specific url is generated by replacing the first instance * of 'scalefactor' in |path| with the numeric scale factor. * - * @param {string} path The URL to generate an image set for. + * @param path The URL to generate an image set for. * 'scalefactor' should be a substring of |path|. - * @return {string} The CSS -webkit-image-set. + * @return The CSS -webkit-image-set. */ -function getImageSet(path) { +function getImageSet(path: string): string { const supportedScaleFactors = getSupportedScaleFactors(); const replaceStartIndex = path.indexOf('SCALEFACTOR'); @@ -93,10 +90,10 @@ * Returns the URL of the image, or an image set of URLs for the provided * path. Resources in chrome://theme have multiple supported scale factors. * - * @param {string} path The path of the image. - * @return {string} The url, or an image set of URLs. + * @param path The path of the image. + * @return The url, or an image set of URLs. */ -export function getImage(path) { +export function getImage(path: string): string { const chromeThemePath = 'chrome://theme'; const isChromeThemeUrl = (path.slice(0, chromeThemePath.length) === chromeThemePath); @@ -104,7 +101,7 @@ getUrlForCss(path); } -function getBaseFaviconUrl() { +function getBaseFaviconUrl(): URL { const faviconUrl = new URL('chrome://favicon2/'); faviconUrl.searchParams.set('size', '16'); faviconUrl.searchParams.set('scaleFactor', 'SCALEFACTORx'); @@ -114,10 +111,10 @@ /** * Creates a CSS -webkit-image-set for a favicon. * - * @param {string} url URL of the favicon - * @return {string} -webkit-image-set for the favicon + * @param url URL of the favicon + * @return -webkit-image-set for the favicon */ -export function getFavicon(url) { +export function getFavicon(url: string): string { const faviconUrl = getBaseFaviconUrl(); faviconUrl.searchParams.set('iconUrl', url); return getImageSet(faviconUrl.toString()); @@ -126,25 +123,26 @@ /** * Creates a CSS -webkit-image-set for a favicon request based on a page URL. * - * @param {string} url URL of the original page - * @param {boolean} isSyncedUrlForHistoryUi Should be set to true only if the + * @param url URL of the original page + * @param isSyncedUrlForHistoryUi Should be set to true only if the * caller is an UI aimed at displaying user history, and the requested url * is known to be present in Chrome sync data. - * @param {string=} remoteIconUrlForUma In case the entry is contained in sync + * @param remoteIconUrlForUma In case the entry is contained in sync * data, we can pass the associated icon url. - * @param {number=} size The favicon size. - * @param {boolean=} forceLightMode Flag to force the service to show the light + * @param size The favicon size. + * @param forceLightMode Flag to force the service to show the light * mode version of the default favicon. * - * @return {string} -webkit-image-set for the favicon. + * @return -webkit-image-set for the favicon. */ export function getFaviconForPageURL( - url, isSyncedUrlForHistoryUi, remoteIconUrlForUma = '', size = 16, - forceLightMode = false) { + url: string, isSyncedUrlForHistoryUi: boolean, + remoteIconUrlForUma: string = '', size: number = 16, + forceLightMode: boolean = false): string { // Note: URL param keys used below must match those in the description of // chrome://favicon2 format in components/favicon_base/favicon_url_parser.h. const faviconUrl = getBaseFaviconUrl(); - faviconUrl.searchParams.set('size', size); + faviconUrl.searchParams.set('size', size.toString()); faviconUrl.searchParams.set('pageUrl', url); // TODO(dbeam): use the presence of 'allowGoogleServerFallback' to // indicate true, otherwise false. @@ -154,7 +152,7 @@ faviconUrl.searchParams.set('iconUrl', remoteIconUrlForUma); } if (forceLightMode) { - faviconUrl.searchParams.set('forceLightMode', true); + faviconUrl.searchParams.set('forceLightMode', 'true'); } return getImageSet(faviconUrl.toString());