blob: e4469b8fda3a33aeacddfbe0fc3e53786ba7f290 [file] [log] [blame]
// Copyright 2023 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/tabs/tab_strip_model.h"
#include "ash/constants/web_app_id_constants.h"
#include "base/json/json_reader.h"
#include "base/scoped_observation.h"
#include "base/test/metrics/histogram_tester.h"
#include "base/test/mock_callback.h"
#include "base/test/scoped_feature_list.h"
#include "base/values.h"
#include "chrome/browser/policy/policy_test_utils.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_list.h"
#include "chrome/browser/ui/browser_tabstrip.h"
#include "chrome/browser/ui/tabs/organization/tab_organization_service.h"
#include "chrome/browser/ui/tabs/organization/tab_organization_service_factory.h"
#include "chrome/browser/ui/tabs/organization/tab_organization_session.h"
#include "chrome/browser/ui/tabs/tab_strip_model_observer.h"
#include "chrome/browser/ui/ui_features.h"
#include "chrome/browser/ui/web_applications/test/web_app_browsertest_util.h"
#include "chrome/browser/web_applications/test/os_integration_test_override_impl.h"
#include "chrome/browser/web_applications/test/prevent_close_test_base.h"
#include "chrome/browser/web_applications/test/web_app_install_test_utils.h"
#include "chrome/browser/web_applications/web_app_install_info.h"
#include "chrome/common/chrome_features.h"
#include "components/policy/core/browser/browser_policy_connector.h"
#include "components/policy/core/browser/browser_policy_connector_base.h"
#include "components/policy/core/common/mock_configuration_policy_provider.h"
#include "components/policy/core/common/policy_map.h"
#include "components/policy/policy_constants.h"
#include "components/saved_tab_groups/public/features.h"
#include "components/tab_groups/tab_group_id.h"
#include "components/tabs/public/split_tab_visual_data.h"
#include "components/tabs/public/tab_interface.h"
#include "components/webapps/common/web_app_id.h"
#include "content/public/browser/web_contents.h"
#include "content/public/test/browser_test.h"
#include "ui/base/ui_base_features.h"
#include "ui/base/window_open_disposition.h"
#include "url/gurl.h"
using testing::_;
namespace {
constexpr char kCalculatorAppUrl[] = "https://calculator.apps.chrome/";
constexpr char kPreventCloseEnabledForCalculator[] = R"([
{
"manifest_id": "https://calculator.apps.chrome/",
"run_on_os_login": "run_windowed",
"prevent_close_after_run_on_os_login": true
}
])";
constexpr char kCalculatorForceInstalled[] = R"([
{
"url": "https://calculator.apps.chrome/",
"default_launch_container": "window"
}
])";
#if BUILDFLAG(IS_CHROMEOS)
constexpr bool kShouldPreventClose = true;
#else
constexpr bool kShouldPreventClose = false;
#endif // BUILDFLAG(IS_CHROMEOS)
} // namespace
class TabStripModelPreventCloseTest : public PreventCloseTestBase,
public BrowserListObserver,
public TabStripModelObserver {
public:
TabStripModelPreventCloseTest() { BrowserList::AddObserver(this); }
explicit TabStripModelPreventCloseTest(const PreventCloseTestBase&) = delete;
TabStripModelPreventCloseTest& operator=(
const TabStripModelPreventCloseTest&) = delete;
~TabStripModelPreventCloseTest() override {
BrowserList::RemoveObserver(this);
}
// BrowserListObserver:
void OnBrowserRemoved(Browser* browser) override { observer_.Reset(); }
// TabStripModelObserver:
MOCK_METHOD(void,
TabCloseCancelled,
(const content::WebContents* contents),
(override));
protected:
web_app::OsIntegrationTestOverrideBlockingRegistration faked_os_integration_;
base::ScopedObservation<TabStripModel, TabStripModelPreventCloseTest>
observer_{this};
};
IN_PROC_BROWSER_TEST_F(TabStripModelPreventCloseTest,
PreventCloseEnforedByPolicy) {
InstallPWA(GURL(kCalculatorAppUrl), ash::kCalculatorAppId);
SetPoliciesAndWaitUntilInstalled(ash::kCalculatorAppId,
kPreventCloseEnabledForCalculator,
kCalculatorForceInstalled);
Browser* const browser =
LaunchPWA(ash::kCalculatorAppId, /*launch_in_window=*/true);
ASSERT_TRUE(browser);
observer_.Observe(browser->tab_strip_model());
TabStripModel* const tab_strip_model = browser->tab_strip_model();
EXPECT_EQ(1, tab_strip_model->count());
EXPECT_EQ(!kShouldPreventClose, tab_strip_model->IsTabClosable(
tab_strip_model->GetActiveWebContents()));
EXPECT_CALL(*this, TabCloseCancelled(_)).Times(kShouldPreventClose ? 1 : 0);
tab_strip_model->CloseAllTabs();
EXPECT_EQ(kShouldPreventClose ? 1 : 0, tab_strip_model->count());
if (kShouldPreventClose) {
ClearWebAppSettings();
EXPECT_TRUE(tab_strip_model->IsTabClosable(
tab_strip_model->GetActiveWebContents()));
tab_strip_model->CloseAllTabs();
EXPECT_EQ(0, tab_strip_model->count());
}
}
// TODO(b/321593065): enable this flaky test.
#if BUILDFLAG(IS_CHROMEOS)
#define MAYBE_PreventCloseEnforcedByPolicyTabbedAppShallBeClosable \
DISABLED_PreventCloseEnforcedByPolicyTabbedAppShallBeClosable
#else
#define MAYBE_PreventCloseEnforcedByPolicyTabbedAppShallBeClosable \
PreventCloseEnforcedByPolicyTabbedAppShallBeClosable
#endif
IN_PROC_BROWSER_TEST_F(
TabStripModelPreventCloseTest,
MAYBE_PreventCloseEnforcedByPolicyTabbedAppShallBeClosable) {
InstallPWA(GURL(kCalculatorAppUrl), ash::kCalculatorAppId);
SetPoliciesAndWaitUntilInstalled(ash::kCalculatorAppId,
kPreventCloseEnabledForCalculator,
kCalculatorForceInstalled);
Browser* const browser =
LaunchPWA(ash::kCalculatorAppId, /*launch_in_window=*/false);
ASSERT_TRUE(browser);
observer_.Observe(browser->tab_strip_model());
TabStripModel* const tab_strip_model = browser->tab_strip_model();
EXPECT_NE(0, tab_strip_model->count());
EXPECT_TRUE(
tab_strip_model->IsTabClosable(tab_strip_model->GetActiveWebContents()));
EXPECT_CALL(*this, TabCloseCancelled(_)).Times(0);
tab_strip_model->CloseAllTabs();
EXPECT_EQ(0, tab_strip_model->count());
}
class TabStripModelBrowserTest : public InProcessBrowserTest,
public TabStripModelObserver {
public:
TabStripModelBrowserTest() {
feature_list_.InitWithFeatures(
{features::kTabOrganization, features::kSideBySide}, {});
}
void TearDownOnMainThread() override { observer_.Reset(); }
MOCK_METHOD(void,
OnTabGroupAdded,
(const tab_groups::TabGroupId& group_id),
(override));
MOCK_METHOD(void,
OnTabGroupWillBeRemoved,
(const tab_groups::TabGroupId& group_id),
(override));
void AddTabs(int num_tabs) {
for (int i = 0; i < num_tabs; i++) {
chrome::AddTabAt(browser(), GURL(url::kAboutBlankURL), 1, false);
}
}
base::test::ScopedFeatureList feature_list_;
base::ScopedObservation<TabStripModel, TabStripModelBrowserTest> observer_{
this};
};
IN_PROC_BROWSER_TEST_F(TabStripModelBrowserTest, OnTabGroupAdded) {
EXPECT_EQ(1, browser()->tab_strip_model()->count());
// We should already have a tab. Add it to a group and see if
// TabStripModelObserver::OnTabGroupAdded is called.
EXPECT_CALL(*this, OnTabGroupAdded(_)).Times(1);
observer_.Observe(browser()->tab_strip_model());
browser()->tab_strip_model()->AddToNewGroup({0});
}
IN_PROC_BROWSER_TEST_F(TabStripModelBrowserTest, OnTabGroupWillBeRemoved) {
EXPECT_EQ(1, browser()->tab_strip_model()->count());
tab_groups::TabGroupId group_id =
browser()->tab_strip_model()->AddToNewGroup({0});
// Close the group and see if TabStripModelObserver::OnTabGroupWillBeRemoved
// is called.
EXPECT_CALL(*this, OnTabGroupWillBeRemoved(group_id)).Times(1);
observer_.Observe(browser()->tab_strip_model());
browser()->tab_strip_model()->CloseAllTabsInGroup(group_id);
}
IN_PROC_BROWSER_TEST_F(TabStripModelBrowserTest, CommandOrganizeTabs) {
base::HistogramTester histogram_tester;
TabStripModel* const tab_strip_model = browser()->tab_strip_model();
EXPECT_EQ(1, tab_strip_model->count());
EXPECT_TRUE(tab_strip_model->IsContextMenuCommandEnabled(
0, TabStripModel::CommandOrganizeTabs));
// Execute CommandOrganizeTabs once. Expect a request to have been started.
tab_strip_model->ExecuteContextMenuCommand(
0, TabStripModel::CommandOrganizeTabs);
TabOrganizationService* const service =
TabOrganizationServiceFactory::GetForProfile(browser()->profile());
const TabOrganizationSession* const session =
service->GetSessionForBrowser(browser());
EXPECT_NE(session, nullptr);
EXPECT_EQ(session->request()->state(),
TabOrganizationRequest::State::NOT_STARTED);
histogram_tester.ExpectUniqueSample("Tab.Organization.AllEntrypoints.Clicked",
true, 1);
histogram_tester.ExpectUniqueSample("Tab.Organization.TabContextMenu.Clicked",
true, 1);
}
IN_PROC_BROWSER_TEST_F(TabStripModelBrowserTest,
DetachWebContentsAtForInsertion) {
class WebContentsRemovedObserver : public TabStripModelObserver {
public:
WebContentsRemovedObserver() = default;
WebContentsRemovedObserver(const WebContentsRemovedObserver&) = delete;
WebContentsRemovedObserver& operator=(const WebContentsRemovedObserver&) =
delete;
~WebContentsRemovedObserver() override = default;
// TabStripModelObserver:
void OnTabStripModelChanged(
TabStripModel* tab_strip_model,
const TabStripModelChange& change,
const TabStripSelectionChange& selection) override {
if (change.type() == TabStripModelChange::kRemoved) {
const TabStripModelChange::RemovedTab& removed_tab =
change.GetRemove()->contents[0];
remove_reason_ = removed_tab.remove_reason;
tab_detach_reason_ = removed_tab.tab_detach_reason;
}
}
std::optional<TabStripModelChange::RemoveReason> remove_reason() const {
return remove_reason_;
}
std::optional<tabs::TabInterface::DetachReason> tab_detach_reason() const {
return tab_detach_reason_;
}
private:
std::optional<TabStripModelChange::RemoveReason> remove_reason_;
std::optional<tabs::TabInterface::DetachReason> tab_detach_reason_;
};
// Start with a browser window with 2 tabs.
TabStripModel* tab_strip_model = browser()->tab_strip_model();
chrome::AddTabAt(browser(), GURL(url::kAboutBlankURL), 1, true);
tabs::TabInterface* const initial_tab = tab_strip_model->GetTabAtIndex(1);
EXPECT_EQ(2, tab_strip_model->count());
base::MockCallback<tabs::TabInterface::WillDetach> tab_detached_callback;
base::CallbackListSubscription tab_subscription =
initial_tab->RegisterWillDetach(tab_detached_callback.Get());
WebContentsRemovedObserver removed_observer;
tab_strip_model->AddObserver(&removed_observer);
// Extract the new WebContents for re-insertion.
EXPECT_CALL(tab_detached_callback,
Run(tab_strip_model->GetTabAtIndex(1),
tabs::TabInterface::DetachReason::kDelete));
std::unique_ptr<content::WebContents> extracted_contents =
tab_strip_model->DetachWebContentsAtForInsertion(1);
EXPECT_EQ(TabStripModelChange::RemoveReason::kInsertedIntoOtherTabStrip,
removed_observer.remove_reason());
EXPECT_EQ(tabs::TabInterface::DetachReason::kDelete,
removed_observer.tab_detach_reason());
tab_strip_model->AppendWebContents(std::move(extracted_contents), true);
}
// Tests IsContextMenuCommandEnabled and ExecuteContextMenuCommand with
// CommandTogglePinned.
IN_PROC_BROWSER_TEST_F(TabStripModelBrowserTest, CommandAddToSplit) {
TabStripModel* const tab_strip_model = browser()->tab_strip_model();
AddTabs(4);
EXPECT_EQ(tab_strip_model->count(), 5);
tab_strip_model->SetTabPinned(0, true);
tab_strip_model->SetTabPinned(1, true);
// Add tab at index 4 to a group.
tab_strip_model->AddToNewGroup({4});
tab_strip_model->ActivateTabAt(3);
EXPECT_TRUE(tab_strip_model->IsContextMenuCommandEnabled(
0, TabStripModel::CommandAddToSplit));
EXPECT_TRUE(tab_strip_model->IsContextMenuCommandEnabled(
1, TabStripModel::CommandAddToSplit));
EXPECT_TRUE(tab_strip_model->IsContextMenuCommandEnabled(
2, TabStripModel::CommandAddToSplit));
EXPECT_TRUE(tab_strip_model->IsContextMenuCommandEnabled(
3, TabStripModel::CommandAddToSplit));
EXPECT_TRUE(tab_strip_model->IsContextMenuCommandEnabled(
4, TabStripModel::CommandAddToSplit));
tab_strip_model->ExecuteContextMenuCommand(0,
TabStripModel::CommandAddToSplit);
// The first tab should become unpinned and adjacent to the active tab.
EXPECT_TRUE(tab_strip_model->GetTabAtIndex(0)->IsPinned());
EXPECT_FALSE(tab_strip_model->GetTabAtIndex(1)->IsPinned());
EXPECT_FALSE(tab_strip_model->GetTabAtIndex(1)->IsSplit());
EXPECT_TRUE(tab_strip_model->GetTabAtIndex(2)->IsSplit());
EXPECT_TRUE(tab_strip_model->GetTabAtIndex(3)->IsSplit());
EXPECT_TRUE(tab_strip_model->GetTabAtIndex(4)->GetGroup().has_value());
EXPECT_FALSE(tab_strip_model->GetTabAtIndex(4)->IsSplit());
}
IN_PROC_BROWSER_TEST_F(TabStripModelBrowserTest, CommandSwapWithActiveSplit) {
TabStripModel* const tab_strip_model = browser()->tab_strip_model();
AddTabs(3);
tab_strip_model->ActivateTabAt(0);
tab_strip_model->AddToNewSplit({1}, split_tabs::SplitTabLayout::kVertical);
EXPECT_TRUE(tab_strip_model->IsContextMenuCommandEnabled(
2, TabStripModel::CommandSwapWithActiveSplit));
tabs::TabInterface* tab_outside_split = tab_strip_model->GetTabAtIndex(2);
tab_strip_model->ExecuteContextMenuCommand(
2, TabStripModel::CommandSwapWithActiveSplit);
EXPECT_TRUE(tab_strip_model->GetTabAtIndex(0)->IsSplit());
EXPECT_TRUE(tab_strip_model->GetTabAtIndex(1)->IsSplit());
EXPECT_EQ(tab_outside_split, tab_strip_model->GetTabAtIndex(0));
}