#include "chrome/browser/device_notifications/device_status_icon_unittest.h"
#include <optional>
#include <string>
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "build/build_config.h"
#include "build/chromeos_buildflags.h"
#include "chrome/app/chrome_command_ids.h"
#include "chrome/browser/profiles/profile_attributes_storage.h"
#include "chrome/browser/status_icons/status_icon.h"
#include "chrome/browser/status_icons/status_icon_menu_model.h"
#include "chrome/browser/status_icons/status_tray.h"
#include "chrome/test/base/browser_with_test_window_test.h"
#include "chrome/test/base/testing_browser_process.h"
#include "chrome/test/base/testing_profile_manager.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#if BUILDFLAG(ENABLE_EXTENSIONS)
#include "extensions/browser/extension_registry.h"
#include "extensions/common/constants.h"
#include "extensions/common/extension.h"
#endif
namespace {
std::u16string GetExpectedOriginConnectionCountLabel(Profile* profile,
const url::Origin& origin,
const std::string& name,
int connection_count) {
#if BUILDFLAG(ENABLE_EXTENSIONS)
if (origin.scheme() == extensions::kExtensionScheme) {
if (connection_count == 0) {
return base::UTF8ToUTF16(base::StringPrintf(
"Extension \"%s\" was accessing devices", name.c_str()));
}
return base::UTF8ToUTF16(base::StringPrintf(
"Extension \"%s\" is accessing %d %s", name.c_str(), connection_count,
(connection_count <= 1 ? "device" : "devices")));
}
#endif
NOTREACHED();
}
class MockStatusIcon : public StatusIcon {
public:
explicit MockStatusIcon(const std::u16string& tool_tip)
: tool_tip_(tool_tip) {}
void SetImage(const gfx::ImageSkia& image) override {}
void SetToolTip(const std::u16string& tool_tip) override {
tool_tip_ = tool_tip;
}
void DisplayBalloon(const gfx::ImageSkia& icon,
const std::u16string& title,
const std::u16string& contents,
const message_center::NotifierId& notifier_id) override {}
void UpdatePlatformContextMenu(StatusIconMenuModel* menu) override {
menu_item_ = menu;
}
const std::u16string& tool_tip() const { return tool_tip_; }
StatusIconMenuModel* menu_item() const { return menu_item_; }
private:
raw_ptr<StatusIconMenuModel> menu_item_ = nullptr;
std::u16string tool_tip_;
};
class MockStatusTray : public StatusTray {
public:
std::unique_ptr<StatusIcon> CreatePlatformStatusIcon(
StatusIconType type,
const gfx::ImageSkia& image,
const std::u16string& tool_tip) override {
return std::make_unique<MockStatusIcon>(tool_tip);
}
const StatusIcons& GetStatusIconsForTest() const { return status_icons(); }
};
}
DeviceStatusIconTestBase::DeviceStatusIconTestBase(
std::u16string about_device_label,
std::u16string device_content_settings_label)
: about_device_label_(std::move(about_device_label)),
device_content_settings_label_(std::move(device_content_settings_label)) {
}
void DeviceStatusIconTestBase::SetUp() {
DeviceSystemTrayIconTestBase::SetUp();
TestingBrowserProcess::GetGlobal()->SetStatusTray(
std::make_unique<MockStatusTray>());
}
void DeviceStatusIconTestBase::TearDown() {
DeviceSystemTrayIconTestBase::TearDown();
TestingBrowserProcess::GetGlobal()->SetStatusTray(nullptr);
}
void DeviceStatusIconTestBase::CheckIcon(
const std::vector<DeviceSystemTrayIconTestBase::ProfileItem>&
profile_connection_counts) {
const auto* status_tray = static_cast<MockStatusTray*>(
TestingBrowserProcess::GetGlobal()->status_tray());
ASSERT_TRUE(status_tray);
ASSERT_EQ(status_tray->GetStatusIconsForTest().size(), 1u);
const auto* status_icon = static_cast<MockStatusIcon*>(
status_tray->GetStatusIconsForTest().back().icon.get());
auto sorted_profile_connection_counts = profile_connection_counts;
std::ranges::sort(sorted_profile_connection_counts);
size_t total_connection_count = 0;
size_t total_origin_count = 0;
auto* menu_item = status_icon->menu_item();
int menu_idx = 1;
int expected_command_id = IDC_DEVICE_SYSTEM_TRAY_ICON_FIRST;
CheckClickableMenuItem(menu_item, menu_idx++, about_device_label_,
expected_command_id++, false);
for (const auto& [profile, origin_items] : sorted_profile_connection_counts) {
total_origin_count += origin_items.size();
auto sorted_origin_items = origin_items;
std::ranges::sort(sorted_origin_items);
auto* connection_tracker = GetDeviceConnectionTracker(profile,
false);
ASSERT_TRUE(connection_tracker);
CheckSeparatorMenuItem(menu_item, menu_idx++);
CheckMenuItemLabel(menu_item, menu_idx++,
base::UTF8ToUTF16(profile->GetProfileUserName()));
EXPECT_CALL(*GetMockDeviceConnectionTracker(connection_tracker),
ShowContentSettingsExceptions());
CheckClickableMenuItem(menu_item, menu_idx++,
device_content_settings_label_,
expected_command_id++, true);
for (const auto& [origin, connection_count, name] : sorted_origin_items) {
EXPECT_CALL(*GetMockDeviceConnectionTracker(connection_tracker),
ShowSiteSettings(origin));
CheckClickableMenuItem(menu_item, menu_idx++,
GetExpectedOriginConnectionCountLabel(
profile, origin, name, connection_count),
expected_command_id++, true);
total_connection_count += connection_count;
}
}
CheckMenuItemLabel(
menu_item, 0,
GetExpectedTitle(total_origin_count,
override_title_total_connection_count_.value_or(
total_connection_count)));
EXPECT_EQ(status_icon->tool_tip(),
GetExpectedTitle(total_origin_count,
override_title_total_connection_count_.value_or(
total_connection_count)));
EXPECT_LE(expected_command_id, IDC_DEVICE_SYSTEM_TRAY_ICON_LAST + 1);
}
void DeviceStatusIconTestBase::CheckIconHidden() {
const auto* status_tray = static_cast<MockStatusTray*>(
TestingBrowserProcess::GetGlobal()->status_tray());
ASSERT_TRUE(status_tray);
EXPECT_TRUE(status_tray->GetStatusIconsForTest().empty());
}
#if BUILDFLAG(ENABLE_EXTENSIONS)
void DeviceStatusIconTestBase::TestNumCommandIdOverLimitExtensionOrigin() {
size_t num_profiles = 19;
std::vector<DeviceSystemTrayIconTestBase::ProfileItem>
profile_connection_counts;
for (size_t idx = 0; idx < num_profiles; idx++) {
std::string profile_name = base::StringPrintf("user%zu", idx);
auto* profile = CreateTestingProfile(profile_name);
auto extension = CreateExtensionWithName("Test Extension");
AddExtensionToProfile(profile, extension.get());
auto* connection_tracker =
GetDeviceConnectionTracker(profile, true);
connection_tracker->IncrementConnectionCount(extension->origin());
profile_connection_counts.push_back(
{profile, {{extension->origin(), 1, extension->name()}}});
}
CheckIcon(profile_connection_counts);
{
std::string profile_name = base::StringPrintf("user%zu", num_profiles);
auto* profile = CreateTestingProfile(profile_name);
auto extension = CreateExtensionWithName("Test Extension");
AddExtensionToProfile(profile, extension.get());
auto* connection_tracker = GetDeviceConnectionTracker(profile,
true);
connection_tracker->IncrementConnectionCount(extension->origin());
profile_connection_counts.push_back(
{profile, {{extension->origin(), 1, extension->name()}}});
std::ranges::sort(profile_connection_counts);
profile_connection_counts.back().second.clear();
override_title_total_connection_count_ = 20;
CheckIcon(profile_connection_counts);
}
}
void DeviceStatusIconTestBase::TestProfileUserNameExtensionOrigin() {
std::vector<DeviceSystemTrayIconTestBase::ProfileItem>
profile_connection_counts;
std::vector<std::tuple<Profile*, std::string, std::string>> profiles;
for (size_t idx = 0; idx < 2; idx++) {
std::string profile_name = base::StringPrintf("user%zu", idx);
std::string new_profile_name = base::StringPrintf("user%zu-newname", idx);
auto* profile = CreateTestingProfile(profile_name);
auto extension = CreateExtensionWithName("Test Extension");
AddExtensionToProfile(profile, extension.get());
auto* connection_tracker = GetDeviceConnectionTracker(profile,
true);
connection_tracker->IncrementConnectionCount(extension->origin());
profile_connection_counts.push_back(
{profile, {{extension->origin(), 1, extension->name()}}});
profiles.emplace_back(profile, profile_name, new_profile_name);
}
CheckIcon(profile_connection_counts);
const auto* status_tray = static_cast<MockStatusTray*>(
TestingBrowserProcess::GetGlobal()->status_tray());
ASSERT_TRUE(status_tray);
ASSERT_EQ(status_tray->GetStatusIconsForTest().size(), 1u);
const auto* status_icon = static_cast<MockStatusIcon*>(
status_tray->GetStatusIconsForTest().back().icon.get());
std::ranges::sort(profiles);
int profile_position1 = 3;
int profile_position2 = 7;
{
auto* menu_item = status_icon->menu_item();
CheckMenuItemLabel(
menu_item, profile_position1,
base::UTF8ToUTF16(std::get<0>(profiles[0])->GetProfileUserName()));
CheckMenuItemLabel(
menu_item, profile_position2,
base::UTF8ToUTF16(std::get<0>(profiles[1])->GetProfileUserName()));
}
{
profile_manager()
->profile_attributes_storage()
->GetProfileAttributesWithPath(std::get<0>(profiles[0])->GetPath())
->SetLocalProfileName(base::UTF8ToUTF16(std::get<2>(profiles[0])),
false);
auto* menu_item = status_icon->menu_item();
CheckMenuItemLabel(menu_item, profile_position1,
base::UTF8ToUTF16(std::get<2>(profiles[0])));
CheckMenuItemLabel(menu_item, profile_position2,
base::UTF8ToUTF16(std::get<1>(profiles[1])));
}
{
profile_manager()
->profile_attributes_storage()
->GetProfileAttributesWithPath(std::get<0>(profiles[1])->GetPath())
->SetLocalProfileName(base::UTF8ToUTF16(std::get<2>(profiles[1])),
false);
auto* menu_item = status_icon->menu_item();
CheckMenuItemLabel(menu_item, profile_position1,
base::UTF8ToUTF16(std::get<2>(profiles[0])));
CheckMenuItemLabel(menu_item, profile_position2,
base::UTF8ToUTF16(std::get<2>(profiles[1])));
}
}
#endif
void DeviceStatusIconTestBase::CheckSeparatorMenuItem(
StatusIconMenuModel* menu_item,
size_t menu_idx) {
ASSERT_LT(menu_idx, menu_item->GetItemCount());
EXPECT_EQ(menu_item->GetSeparatorTypeAt(menu_idx), ui::NORMAL_SEPARATOR);
}
void DeviceStatusIconTestBase::CheckMenuItemLabel(
StatusIconMenuModel* menu_item,
size_t menu_idx,
std::u16string label) {
ASSERT_LT(menu_idx, menu_item->GetItemCount());
EXPECT_EQ(menu_item->GetLabelAt(menu_idx), label);
}
void DeviceStatusIconTestBase::CheckClickableMenuItem(
StatusIconMenuModel* menu_item,
size_t menu_idx,
std::u16string label,
int command_id,
bool click) {
CheckMenuItemLabel(menu_item, menu_idx, label);
ASSERT_LT(menu_idx, menu_item->GetItemCount());
EXPECT_EQ(menu_item->GetCommandIdAt(menu_idx), command_id);
if (click) {
menu_item->ActivatedAt(menu_idx);
}
}