// 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/system/network/network_icon.h"
#include <memory>
#include <set>
#include "ash/public/cpp/network_config_service.h"
#include "ash/public/cpp/style/dark_light_mode_controller.h"
#include "ash/shell.h"
#include "ash/strings/grit/ash_strings.h"
#include "ash/style/color_util.h"
#include "ash/system/network/active_network_icon.h"
#include "ash/system/network/tray_network_state_model.h"
#include "ash/test/ash_test_base.h"
#include "base/logging.h"
#include "base/run_loop.h"
#include "base/test/scoped_feature_list.h"
#include "chromeos/ash/components/network/network_state_handler.h"
#include "chromeos/ash/components/network/network_state_test_helper.h"
#include "chromeos/ash/components/network/tether_constants.h"
#include "chromeos/ash/services/network_config/public/cpp/cros_network_config_test_helper.h"
#include "chromeos/constants/chromeos_features.h"
#include "third_party/cros_system_api/dbus/shill/dbus-constants.h"
#include "ui/base/l10n/l10n_util.h"
#include "ui/gfx/image/image_unittest_util.h"
#include "ui/views/controls/image_view.h"
// This tests both the helper functions in network_icon, and ActiveNetworkIcon
// which is a primary consumer of the helper functions.
namespace ash::network_icon {
using chromeos::network_config::mojom::ConnectionStateType;
using chromeos::network_config::mojom::NetworkStateProperties;
using chromeos::network_config::mojom::NetworkStatePropertiesPtr;
using chromeos::network_config::mojom::NetworkType;
class NetworkIconTest : public AshTestBase,
public testing::WithParamInterface<bool> {
public:
NetworkIconTest() = default;
NetworkIconTest(const NetworkIconTest&) = delete;
NetworkIconTest& operator=(const NetworkIconTest&) = delete;
~NetworkIconTest() override = default;
void SetUp() override {
if (IsJellyrollEnabled()) {
feature_list_.InitWithFeatures(
/*enabled_features=*/{chromeos::features::kJellyroll,
chromeos::features::kJelly},
/*disabled_features=*/{});
} else {
feature_list_.InitWithFeatures(
/*enabled_features=*/{},
/*disabled_features=*/{chromeos::features::kJellyroll,
chromeos::features::kJelly});
}
AshTestBase::SetUp();
SetUpDefaultNetworkState();
network_state_model_ = std::make_unique<TrayNetworkStateModel>();
active_network_icon_ =
std::make_unique<ActiveNetworkIcon>(network_state_model_.get());
}
void TearDown() override {
active_network_icon_.reset();
PurgeNetworkIconCache(std::set<std::string>());
AshTestBase::TearDown();
}
const ui::ColorProvider* GetColorProvider() {
// TODO(b/279177422): Replace with a stable ColorProvider
return ColorUtil::GetColorProviderSourceForWindow(
Shell::GetPrimaryRootWindow())
->GetColorProvider();
}
bool IsJellyrollEnabled() const { return GetParam(); }
std::string ConfigureService(const std::string& shill_json_string) {
return helper().ConfigureService(shill_json_string);
}
void SetServiceProperty(const std::string& service_path,
const std::string& key,
const base::Value& value) {
helper().SetServiceProperty(service_path, key, value);
}
void SetUpDefaultNetworkState() {
// NetworkStateTestHelper default has a wifi device only and no services.
helper().device_test()->AddDevice("/device/stub_cellular_device",
shill::kTypeCellular,
"stub_cellular_device");
base::RunLoop().RunUntilIdle();
wifi1_path_ = ConfigureService(
R"({"GUID": "wifi1_guid", "Type": "wifi", "State": "idle"})");
ASSERT_FALSE(wifi1_path_.empty());
wifi2_path_ = ConfigureService(
R"({"GUID": "wifi2_guid", "Type": "wifi", "State": "idle"})");
ASSERT_FALSE(wifi2_path_.empty());
cellular_path_ = ConfigureService(
R"({"GUID": "cellular_guid", "Type": "cellular", "Technology": "LTE",
"State": "idle"})");
ASSERT_FALSE(cellular_path_.empty());
}
NetworkStatePropertiesPtr CreateStandaloneNetworkProperties(
const std::string& id,
NetworkType type,
ConnectionStateType connection_state,
int signal_strength) {
return network_config_helper_.CreateStandaloneNetworkProperties(
id, type, connection_state, signal_strength);
}
gfx::Image GetImageForNonVirtualNetwork(const NetworkStateProperties* network,
bool badge_vpn) {
return gfx::Image(network_icon::GetImageForNonVirtualNetwork(
GetColorProvider(), network, icon_type_, badge_vpn));
}
gfx::Image ImageForNetwork(const NetworkStateProperties* network) {
return GetImageForNonVirtualNetwork(network, false /* show_vpn_badge */);
}
gfx::ImageSkia GetDefaultNetworkImage(IconType icon_type, bool* animating) {
return active_network_icon_->GetImage(GetColorProvider(),
ActiveNetworkIcon::Type::kSingle,
icon_type, animating);
}
// The icon for a Tether network should be the same as one for a cellular
// network. The icon for a Tether network should be different from one for a
// Wi-Fi network. The icon for a cellular network should be different from one
// for a Wi-Fi network. The icon for a Tether network should be the same as
// one for a Wi-Fi network with an associated Tether guid.
void GetAndCompareImagesByNetworkType(
const NetworkStateProperties* wifi_network,
const NetworkStateProperties* cellular_network,
const NetworkStateProperties* tether_network) {
ASSERT_EQ(wifi_network->type, NetworkType::kWiFi);
gfx::Image wifi_image = ImageForNetwork(wifi_network);
ASSERT_EQ(cellular_network->type, NetworkType::kCellular);
gfx::Image cellular_image = ImageForNetwork(cellular_network);
ASSERT_EQ(tether_network->type, NetworkType::kTether);
gfx::Image tether_image = ImageForNetwork(tether_network);
EXPECT_FALSE(gfx::test::AreImagesEqual(tether_image, wifi_image));
EXPECT_FALSE(gfx::test::AreImagesEqual(cellular_image, wifi_image));
EXPECT_TRUE(gfx::test::AreImagesEqual(tether_image, cellular_image));
}
void SetCellularUnavailable() {
helper().manager_test()->RemoveTechnology(shill::kTypeCellular);
base::RunLoop().RunUntilIdle();
ASSERT_EQ(NetworkStateHandler::TechnologyState::TECHNOLOGY_UNAVAILABLE,
helper().network_state_handler()->GetTechnologyState(
NetworkTypePattern::Cellular()));
}
NetworkStateTestHelper& helper() {
return network_config_helper_.network_state_helper();
}
const std::string& wifi1_path() const { return wifi1_path_; }
const std::string& wifi2_path() const { return wifi2_path_; }
const std::string& cellular_path() const { return cellular_path_; }
IconType icon_type_ = ICON_TYPE_TRAY_REGULAR;
private:
base::test::ScopedFeatureList feature_list_;
network_config::CrosNetworkConfigTestHelper network_config_helper_;
std::unique_ptr<TrayNetworkStateModel> network_state_model_;
std::unique_ptr<ActiveNetworkIcon> active_network_icon_;
// Preconfigured service paths:
std::string wifi1_path_;
std::string wifi2_path_;
std::string cellular_path_;
};
INSTANTIATE_TEST_SUITE_P(Jellyroll, NetworkIconTest, testing::Bool());
// This tests that the correct icons are being generated for the correct
// networks by pairwise comparison of three different network types, verifying
// that the Tether and cellular icon are the same, Tether and Wi-Fi icons are
// different, and cellular and Wi-Fi icons are different. Additionally, it
// verifies that the Tether network and Wi-Fi network with associated Tether
// guid are treated the same for purposes of icon display
TEST_P(NetworkIconTest, CompareImagesByNetworkType_NotVisible) {
NetworkStatePropertiesPtr wifi_network = CreateStandaloneNetworkProperties(
"wifi", NetworkType::kWiFi, ConnectionStateType::kNotConnected, 50);
NetworkStatePropertiesPtr cellular_network =
CreateStandaloneNetworkProperties("cellular", NetworkType::kCellular,
ConnectionStateType::kNotConnected, 50);
NetworkStatePropertiesPtr tether_network = CreateStandaloneNetworkProperties(
"tether", NetworkType::kTether, ConnectionStateType::kNotConnected, 50);
GetAndCompareImagesByNetworkType(wifi_network.get(), cellular_network.get(),
tether_network.get());
}
TEST_P(NetworkIconTest, CompareImagesByNetworkType_Connecting) {
NetworkStatePropertiesPtr wifi_network = CreateStandaloneNetworkProperties(
"wifi", NetworkType::kWiFi, ConnectionStateType::kConnecting, 50);
NetworkStatePropertiesPtr cellular_network =
CreateStandaloneNetworkProperties("cellular", NetworkType::kCellular,
ConnectionStateType::kConnecting, 50);
NetworkStatePropertiesPtr tether_network = CreateStandaloneNetworkProperties(
"tether", NetworkType::kTether, ConnectionStateType::kConnecting, 50);
GetAndCompareImagesByNetworkType(wifi_network.get(), cellular_network.get(),
tether_network.get());
}
TEST_P(NetworkIconTest, CompareImagesByNetworkType_Connected) {
NetworkStatePropertiesPtr wifi_network = CreateStandaloneNetworkProperties(
"wifi", NetworkType::kWiFi, ConnectionStateType::kOnline, 50);
NetworkStatePropertiesPtr cellular_network =
CreateStandaloneNetworkProperties("cellular", NetworkType::kCellular,
ConnectionStateType::kOnline, 50);
NetworkStatePropertiesPtr tether_network = CreateStandaloneNetworkProperties(
"tether", NetworkType::kTether, ConnectionStateType::kOnline, 50);
GetAndCompareImagesByNetworkType(wifi_network.get(), cellular_network.get(),
tether_network.get());
}
TEST_P(NetworkIconTest, NetworkSignalStrength) {
// Signal strength is divided into four categories: none, weak, medium and
// strong. They are meant to match the number of sections in the wifi icon.
// The wifi icon currently has four levels; signals [0, 100] are mapped to [1,
// 4]. There are only three signal strengths so icons that were mapped to 2
// are also considered weak.
EXPECT_EQ(SignalStrength::NONE, GetSignalStrength(0));
EXPECT_EQ(SignalStrength::WEAK, GetSignalStrength(50));
EXPECT_EQ(SignalStrength::MEDIUM, GetSignalStrength(51));
EXPECT_EQ(SignalStrength::MEDIUM, GetSignalStrength(75));
EXPECT_EQ(SignalStrength::STRONG, GetSignalStrength(76));
EXPECT_EQ(SignalStrength::STRONG, GetSignalStrength(100));
}
TEST_P(NetworkIconTest, DefaultImageWifiConnected) {
// Set the Wifi service as connected.
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kWiFi,
ConnectionStateType::kOnline, 45);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
TEST_P(NetworkIconTest, DefaultImageWifiConnecting) {
// Set the Wifi service as connected.
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateAssociation));
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_TRUE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kWiFi,
ConnectionStateType::kConnecting, 45);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
TEST_P(NetworkIconTest, ConnectingIconChangesInDarkMode) {
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateAssociation));
DarkLightModeController::Get()->SetDarkModeEnabledForTest(true);
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_TRUE(animating);
DarkLightModeController::Get()->SetDarkModeEnabledForTest(false);
gfx::ImageSkia light_mode_image =
GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(light_mode_image.isNull());
EXPECT_TRUE(animating);
EXPECT_FALSE(gfx::test::AreImagesEqual(gfx::Image(default_image),
gfx::Image(light_mode_image)));
}
// Tests that the default network image is a cellular network icon when cellular
// network is the default network, even if a wifi network is connected.
// Generally, shill will prefer wifi over cellular networks when both are
// connected, but that is not always the case. For example, if the connected
// wifi service has no Internet connectivity, cellular service will be selected
// as default.
TEST_P(NetworkIconTest, DefaultImageCellularDefaultWithWifiConnected) {
// Set both wifi and cellular networks in a connected state, but with wifi not
// online - this should prompt fake shill manager implementation to prefer
// cellular network over wifi.
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateReady));
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kCellular,
ConnectionStateType::kOnline, 65);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
// Tests the use case where the default network starts reconnecting while
// another network is connected.
TEST_P(NetworkIconTest, DefaultImageReconnectingWifiWithCellularConnected) {
// First connect both wifi and cellular network (with wifi as default).
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
// Start reconnecting wifi network.
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateAssociation));
// Verify that the default network is connecting icon for the initial default
// network (even though the default network as reported by shill actually
// changed).
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_TRUE(animating);
NetworkStatePropertiesPtr reference_network_1 =
CreateStandaloneNetworkProperties("reference1", NetworkType::kWiFi,
ConnectionStateType::kConnecting, 45);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network_1.get())));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateReady));
// At this point the wifi network is not connecting, but is not yet default -
// the default network icon should have image for the cellular network (which
// is the default one at this point).
// TODO(tbarzic): This creates a unoptimal behavior in the UI where cellular
// icon could flash between wifi connecting and connected icons - this
// should be fixed, for example by not showing connecting icon when
// reconnecting a network if another network is connected.
NetworkStatePropertiesPtr reference_network_2 =
CreateStandaloneNetworkProperties("reference2", NetworkType::kCellular,
ConnectionStateType::kOnline, 65);
default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network_2.get())));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
// The wifi network is online, and thus default - the default network icon
// should display the wifi network's associated image again.
NetworkStatePropertiesPtr reference_network_3 =
CreateStandaloneNetworkProperties("reference3", NetworkType::kWiFi,
ConnectionStateType::kOnline, 45);
default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network_3.get())));
}
TEST_P(NetworkIconTest, DefaultImageDisconnectWifiWithCellularConnected) {
// First connect both wifi and cellular network (with wifi as default).
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
// Disconnect wifi network, and verify the default icon changes to cellular.
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateIdle));
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kCellular,
ConnectionStateType::kOnline, 65);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
// Tests that the default network image remains the same if non-default network
// reconnects.
TEST_P(NetworkIconTest, DefaultImageWhileNonDefaultNetworkReconnecting) {
// First connect both wifi and cellular network (with wifi as default).
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
// Start reconnecting the non-default, cellular network.
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateAssociation));
// Currently, a connecting icon is used as default network icon even if
// another network connected and used as default.
// TODO(tbarzic): Consider changing network icon logic to use a connected
// network icon if a network is connected while a network is reconnecting.
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_TRUE(animating);
NetworkStatePropertiesPtr reference_network_1 =
CreateStandaloneNetworkProperties("reference1", NetworkType::kCellular,
ConnectionStateType::kConnecting, 65);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network_1.get())));
// Move the cellular network to connected, but not yet online state - the
// default network image changes back to the default network.
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateReady));
default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_network_2 =
CreateStandaloneNetworkProperties("reference2", NetworkType::kWiFi,
ConnectionStateType::kOnline, 45);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network_2.get())));
// Move the cellular network to online state - the default network image
// should remain the same.
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network_2.get())));
}
// Tests that the default network image shows a cellular network icon if
// cellular network is connected while wifi is connecting.
TEST_P(NetworkIconTest, DefaultImageConnectingToWifiWhileCellularConnected) {
// Connect cellular network, and set the wifi as connecting.
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateAssociation));
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kCellular,
ConnectionStateType::kOnline, 65);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
// Test that a connecting cellular icon is displayed when activating a cellular
// network (if other networks are not connected).
TEST_P(NetworkIconTest, DefaultNetworkImageActivatingCellularNetwork) {
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kActivationStateProperty,
base::Value(shill::kActivationStateActivating));
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kCellular,
ConnectionStateType::kNotConnected, 65);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
// Tests that default network image is a wifi network image if wifi network is
// connected during cellular network activation.
TEST_P(NetworkIconTest,
DefaultNetworkImageActivatingCellularNetworkWithConnectedWifi) {
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kActivationStateProperty,
base::Value(shill::kActivationStateActivating));
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kWiFi,
ConnectionStateType::kNotConnected, 45);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
// Tests VPN badging for the default network.
TEST_P(NetworkIconTest, DefaultNetworkVpnBadge) {
// Set up initial state with Ethernet and WiFi connected.
std::string ethernet_path = ConfigureService(
R"({"GUID": "ethernet_guid", "Type": "ethernet", "State": "online"})");
ASSERT_FALSE(ethernet_path.empty());
// Wifi1 is set up by default but not connected. Also set its strength.
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateOnline));
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(45));
// Gets the benchmark ethernet images.
NetworkStatePropertiesPtr reference_eth = CreateStandaloneNetworkProperties(
"reference_eth", NetworkType::kEthernet, ConnectionStateType::kOnline, 0);
gfx::Image reference_eth_unbadged = GetImageForNonVirtualNetwork(
reference_eth.get(), false /* show_vpn_badge */);
gfx::Image reference_eth_badged = GetImageForNonVirtualNetwork(
reference_eth.get(), true /* show_vpn_badge */);
// With Ethernet and WiFi connected, the default icon should be the Ethernet
// icon.
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
EXPECT_TRUE(gfx::test::AreImagesEqual(gfx::Image(default_image),
reference_eth_unbadged));
// Add a connected VPN.
std::string vpn_path = ConfigureService(
R"({"GUID": "vpn_guid", "Type": "vpn", "State": "online"})");
ASSERT_FALSE(vpn_path.empty());
// When a VPN is connected, the default icon should be Ethernet with a badge.
default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
EXPECT_FALSE(gfx::test::AreImagesEqual(gfx::Image(default_image),
reference_eth_unbadged));
EXPECT_TRUE(gfx::test::AreImagesEqual(gfx::Image(default_image),
reference_eth_badged));
// Disconnect Ethernet. The default icon should become WiFi with a badge.
SetServiceProperty(ethernet_path, shill::kStateProperty,
base::Value(shill::kStateIdle));
default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_FALSE(animating);
NetworkStatePropertiesPtr reference_wifi = CreateStandaloneNetworkProperties(
"reference_wifi", NetworkType::kWiFi, ConnectionStateType::kOnline, 45);
gfx::Image reference_wifi_badged = GetImageForNonVirtualNetwork(
reference_wifi.get(), true /* show_vpn_badge */);
EXPECT_TRUE(gfx::test::AreImagesEqual(gfx::Image(default_image),
reference_wifi_badged));
// Set the VPN to connecting; the default icon should be animating.
SetServiceProperty(vpn_path, shill::kStateProperty,
base::Value(shill::kStateAssociation));
default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_TRUE(animating);
}
// Tests that wifi image is shown when connecting to wifi network with vpn.
TEST_P(NetworkIconTest, DefaultNetworkImageVpnAndWifi) {
SetServiceProperty(wifi1_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(wifi1_path(), shill::kStateProperty,
base::Value(shill::kStateAssociation));
std::string vpn_path = ConfigureService(
R"({"GUID": "vpn_guid", "Type": "vpn", "State": "online"})");
ASSERT_FALSE(vpn_path.empty());
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_TRUE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kWiFi,
ConnectionStateType::kConnecting, 65);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
// Tests that cellular image is shown when connecting to cellular network with
// VPN.
TEST_P(NetworkIconTest, DefaultNetworkImageVpnAndCellular) {
SetServiceProperty(cellular_path(), shill::kSignalStrengthProperty,
base::Value(65));
SetServiceProperty(cellular_path(), shill::kStateProperty,
base::Value(shill::kStateAssociation));
std::string vpn_path = ConfigureService(
R"({"GUID": "vpn_guid", "Type": "vpn", "State": "online"})");
ASSERT_FALSE(vpn_path.empty());
bool animating = false;
gfx::ImageSkia default_image = GetDefaultNetworkImage(icon_type_, &animating);
ASSERT_FALSE(default_image.isNull());
EXPECT_TRUE(animating);
NetworkStatePropertiesPtr reference_network =
CreateStandaloneNetworkProperties("reference", NetworkType::kCellular,
ConnectionStateType::kConnecting, 65);
EXPECT_TRUE(gfx::test::AreImagesEqual(
gfx::Image(default_image), ImageForNetwork(reference_network.get())));
}
// Tests the case of getting the WiFi Enabled state icon when there is
// no color provider, in which case the window background color is used.
TEST_P(NetworkIconTest, GetImageModelForWiFiEnabledState) {
views::ImageView* image_view =
new views::ImageView(GetImageModelForWiFiEnabledState(true));
std::unique_ptr<views::Widget> widget = CreateFramelessTestWidget();
widget->SetFullscreen(true);
widget->SetContentsView(image_view);
ui::NativeTheme* native_theme = widget->GetNativeTheme();
native_theme->set_use_dark_colors(true);
native_theme->NotifyOnNativeThemeUpdated();
gfx::Image dark_mode_image = gfx::Image(image_view->GetImage());
// Change the color scheme.
native_theme->set_use_dark_colors(false);
native_theme->NotifyOnNativeThemeUpdated();
gfx::Image light_mode_image = gfx::Image(image_view->GetImage());
EXPECT_FALSE(gfx::test::AreImagesEqual(dark_mode_image, light_mode_image));
}
} // namespace ash::network_icon