1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
ash / system / power / power_status_unittest.cc [blame]
// 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/system/power/power_status.h"
#include <memory>
#include "ash/resources/vector_icons/vector_icons.h"
#include "ash/test/ash_test_base.h"
#include "base/memory/raw_ptr.h"
#include "base/run_loop.h"
#include "base/test/scoped_feature_list.h"
#include "chromeos/dbus/power/fake_power_manager_client.h"
#include "third_party/cros_system_api/dbus/service_constants.h"
#include "ui/chromeos/styles/cros_tokens_color_mappings.h"
#include "ui/gfx/image/image.h"
#include "ui/gfx/image/image_unittest_util.h"
#include "ui/views/widget/widget.h"
using power_manager::PowerSupplyProperties;
namespace ash {
namespace {
class TestObserver : public PowerStatus::Observer {
public:
TestObserver() : power_changed_count_(0) {}
TestObserver(const TestObserver&) = delete;
TestObserver& operator=(const TestObserver&) = delete;
~TestObserver() override = default;
int power_changed_count() const { return power_changed_count_; }
// PowerStatus::Observer overrides:
void OnPowerStatusChanged() override { ++power_changed_count_; }
private:
int power_changed_count_;
};
} // namespace
class PowerStatusTest : public AshTestBase {
public:
PowerStatusTest() = default;
PowerStatusTest(const PowerStatusTest&) = delete;
PowerStatusTest& operator=(const PowerStatusTest&) = delete;
~PowerStatusTest() override = default;
void SetUp() override {
chromeos::PowerManagerClient::InitializeFake();
AshTestBase::SetUp();
power_status_ = PowerStatus::Get();
test_observer_ = std::make_unique<TestObserver>();
power_status_->AddObserver(test_observer_.get());
}
void TearDown() override {
power_status_->RemoveObserver(test_observer_.get());
test_observer_.reset();
AshTestBase::TearDown();
chromeos::PowerManagerClient::Shutdown();
}
protected:
raw_ptr<PowerStatus, DanglingUntriaged> power_status_ =
nullptr; // Not owned.
std::unique_ptr<TestObserver> test_observer_;
};
TEST_F(PowerStatusTest, InitializeAndUpdate) {
// Test that the initial power supply state should be acquired after
// PowerStatus is instantiated. This depends on
// PowerManagerClientStubImpl, which responds to power status update
// requests, pretends there is a battery present, and generates some valid
// power supply status data.
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1, test_observer_->power_changed_count());
// Test RequestUpdate, test_obsever_ should be notified for power suuply
// status change.
power_status_->RequestStatusUpdate();
base::RunLoop().RunUntilIdle();
EXPECT_EQ(2, test_observer_->power_changed_count());
}
TEST_F(PowerStatusTest, GenerateBatteryImageInfo) {
PowerSupplyProperties prop;
prop.set_external_power(PowerSupplyProperties::AC);
prop.set_battery_state(PowerSupplyProperties::CHARGING);
prop.set_battery_percent(98.0);
power_status_->SetProtoForTesting(prop);
const PowerStatus* power_status = power_status_;
auto get_battery_info = [&power_status]() {
return power_status->GenerateBatteryImageInfo(SK_ColorWHITE);
};
const PowerStatus::BatteryImageInfo info_charging_98 = get_battery_info();
// 99% should use the same icon as 98%.
prop.set_battery_percent(99.0);
power_status_->SetProtoForTesting(prop);
EXPECT_TRUE(info_charging_98.ApproximatelyEqual(get_battery_info()));
// A different icon should be used when the battery is full.
prop.set_battery_state(PowerSupplyProperties::FULL);
prop.set_battery_percent(100.0);
power_status_->SetProtoForTesting(prop);
EXPECT_FALSE(info_charging_98.ApproximatelyEqual(get_battery_info()));
// A much-lower battery level should use a different icon.
prop.set_battery_state(PowerSupplyProperties::CHARGING);
prop.set_battery_percent(20.0);
power_status_->SetProtoForTesting(prop);
EXPECT_FALSE(info_charging_98.ApproximatelyEqual(get_battery_info()));
// Ditto for 98%, but on USB instead of AC.
prop.set_external_power(PowerSupplyProperties::USB);
prop.set_battery_percent(98.0);
power_status_->SetProtoForTesting(prop);
EXPECT_FALSE(info_charging_98.ApproximatelyEqual(get_battery_info()));
}
// Tests that the |icon_badge| member of BatteryImageInfo is set correctly
// with various power supply property values.
TEST_F(PowerStatusTest, BatteryImageInfoIconBadge) {
PowerSupplyProperties prop;
// A charging battery connected to AC power should have a bolt badge.
prop.set_external_power(PowerSupplyProperties::AC);
prop.set_battery_state(PowerSupplyProperties::CHARGING);
prop.set_battery_percent(98.0);
power_status_->SetProtoForTesting(prop);
const PowerStatus* power_status = power_status_;
const auto get_battery_icon = [&power_status]() {
return power_status->GenerateBatteryImageInfo(SK_ColorWHITE).icon_badge;
};
const gfx::VectorIcon* bolt_icon = get_battery_icon();
EXPECT_TRUE(bolt_icon);
// A discharging battery connected to AC should also have a bolt badge.
prop.set_battery_state(PowerSupplyProperties::DISCHARGING);
power_status_->SetProtoForTesting(prop);
EXPECT_EQ(bolt_icon, get_battery_icon());
// A charging battery connected to USB power should have an
// unreliable badge.
prop.set_external_power(PowerSupplyProperties::USB);
prop.set_battery_state(PowerSupplyProperties::CHARGING);
power_status_->SetProtoForTesting(prop);
const gfx::VectorIcon* unreliable_icon = get_battery_icon();
EXPECT_NE(unreliable_icon, bolt_icon);
EXPECT_TRUE(unreliable_icon);
// A discharging battery connected to USB power should also have an
// unreliable badge.
prop.set_battery_state(PowerSupplyProperties::DISCHARGING);
power_status_->SetProtoForTesting(prop);
EXPECT_EQ(unreliable_icon, get_battery_icon());
// Show the right icon when no battery is present.
prop.set_external_power(PowerSupplyProperties::DISCONNECTED);
prop.set_battery_state(PowerSupplyProperties::NOT_PRESENT);
power_status_->SetProtoForTesting(prop);
const gfx::VectorIcon* x_icon = get_battery_icon();
EXPECT_TRUE(x_icon);
EXPECT_NE(bolt_icon, x_icon);
EXPECT_NE(unreliable_icon, x_icon);
// Do not show a badge when the battery is discharging.
prop.set_battery_state(PowerSupplyProperties::DISCHARGING);
power_status_->SetProtoForTesting(prop);
EXPECT_FALSE(get_battery_icon());
// Show the right icon for a discharging battery when it falls below
// a charge level of PowerStatus::kCriticalBatteryChargePercentage.
prop.set_battery_percent(PowerStatus::kCriticalBatteryChargePercentage);
power_status_->SetProtoForTesting(prop);
EXPECT_FALSE(get_battery_icon());
prop.set_battery_percent(PowerStatus::kCriticalBatteryChargePercentage - 1);
power_status_->SetProtoForTesting(prop);
const gfx::VectorIcon* alert_icon = get_battery_icon();
EXPECT_TRUE(alert_icon);
EXPECT_NE(bolt_icon, alert_icon);
EXPECT_NE(unreliable_icon, alert_icon);
EXPECT_NE(x_icon, alert_icon);
}
// Tests that the battery image changes appropriately with various power supply
// property values.
TEST_F(PowerStatusTest, BatteryImageInfoChargeLevel) {
const std::unique_ptr<views::Widget> test_widget =
CreateTestWidget(views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET);
PowerSupplyProperties prop;
// No charge level is drawn when the battery is not present.
prop.set_external_power(PowerSupplyProperties::DISCONNECTED);
prop.set_battery_state(PowerSupplyProperties::NOT_PRESENT);
power_status_->SetProtoForTesting(prop);
const PowerStatus* power_status = power_status_;
const auto get_battery_charge_percent = [&power_status]() {
return power_status->GenerateBatteryImageInfo(SK_ColorWHITE).charge_percent;
};
EXPECT_EQ(0, get_battery_charge_percent());
auto get_battery_image = [&power_status, &test_widget]() {
const ui::ColorProvider* color_provider = test_widget->GetColorProvider();
PowerStatus::BatteryImageInfo info =
power_status->GenerateBatteryImageInfo(SK_ColorGREEN);
return gfx::Image(power_status->GetBatteryImage(info, 16, color_provider));
};
prop.set_external_power(PowerSupplyProperties::AC);
prop.set_battery_state(PowerSupplyProperties::CHARGING);
prop.set_battery_percent(0.0);
EXPECT_EQ(0, get_battery_charge_percent());
gfx::Image empty_image = get_battery_image();
// 10% and 20% look different.
prop.set_battery_percent(10.0);
power_status_->SetProtoForTesting(prop);
EXPECT_EQ(10.0, get_battery_charge_percent());
gfx::Image image_10 = get_battery_image();
EXPECT_FALSE(gfx::test::AreImagesEqual(empty_image, image_10));
prop.set_battery_percent(20.0);
power_status_->SetProtoForTesting(prop);
EXPECT_EQ(20.0, get_battery_charge_percent());
gfx::Image image_20 = get_battery_image();
EXPECT_FALSE(gfx::test::AreImagesEqual(image_10, image_20));
// 99% and 100% look different.
prop.set_battery_percent(99.0);
power_status_->SetProtoForTesting(prop);
EXPECT_EQ(99, get_battery_charge_percent());
gfx::Image image_99 = get_battery_image();
prop.set_battery_percent(100.0);
power_status_->SetProtoForTesting(prop);
EXPECT_EQ(100, get_battery_charge_percent());
gfx::Image image_100 = get_battery_image();
EXPECT_FALSE(gfx::test::AreImagesEqual(image_99, image_100));
}
// Tests that positive time-to-full and time-to-empty estimates are honored.
TEST_F(PowerStatusTest, PositiveBatteryTimeEstimates) {
constexpr auto kTime = base::Seconds(120);
PowerSupplyProperties prop;
prop.set_external_power(PowerSupplyProperties::AC);
prop.set_battery_state(PowerSupplyProperties::CHARGING);
prop.set_battery_time_to_full_sec(kTime.InSeconds());
power_status_->SetProtoForTesting(prop);
std::optional<base::TimeDelta> time = power_status_->GetBatteryTimeToFull();
ASSERT_TRUE(time);
EXPECT_EQ(kTime, *time);
prop.Clear();
prop.set_external_power(PowerSupplyProperties::DISCONNECTED);
prop.set_battery_state(PowerSupplyProperties::DISCHARGING);
prop.set_battery_time_to_empty_sec(kTime.InSeconds());
power_status_->SetProtoForTesting(prop);
time = power_status_->GetBatteryTimeToEmpty();
ASSERT_TRUE(time);
EXPECT_EQ(kTime, *time);
}
// Tests that missing time-to-full and time-to-empty estimates (which powerd
// sends when no battery is present) and negative ones (which powerd sends when
// the battery current is close to zero) are disregarded:
// https://crbug.com/930358
TEST_F(PowerStatusTest, MissingBatteryTimeEstimates) {
// No battery.
PowerSupplyProperties prop;
prop.set_external_power(PowerSupplyProperties::AC);
prop.set_battery_state(PowerSupplyProperties::NOT_PRESENT);
power_status_->SetProtoForTesting(prop);
std::optional<base::TimeDelta> time = power_status_->GetBatteryTimeToFull();
EXPECT_FALSE(time) << *time << " returned despite missing battery";
time = power_status_->GetBatteryTimeToEmpty();
EXPECT_FALSE(time) << *time << " returned despite missing battery";
// Battery is charging, but negative estimate provided.
prop.set_battery_state(PowerSupplyProperties::CHARGING);
prop.set_battery_time_to_full_sec(-1);
power_status_->SetProtoForTesting(prop);
time = power_status_->GetBatteryTimeToFull();
EXPECT_FALSE(time) << *time << " returned despite negative estimate";
// Battery is discharging, but negative estimate provided.
prop.Clear();
prop.set_external_power(PowerSupplyProperties::DISCONNECTED);
prop.set_battery_state(PowerSupplyProperties::DISCHARGING);
prop.set_battery_time_to_empty_sec(-1);
power_status_->SetProtoForTesting(prop);
time = power_status_->GetBatteryTimeToEmpty();
EXPECT_FALSE(time) << *time << " returned despite negative estimate";
}
TEST_F(PowerStatusTest, PreferredMinimumExternalPower) {
PowerSupplyProperties prop;
prop.set_external_power(PowerSupplyProperties::USB);
prop.set_battery_state(PowerSupplyProperties::NOT_PRESENT);
prop.set_preferred_minimum_external_power(23.45);
power_status_->SetProtoForTesting(prop);
EXPECT_EQ(23.45, power_status_->GetPreferredMinimumPower());
}
TEST_F(PowerStatusTest, BatteryImageColorResolution) {
// Set up Color Resolution Testing
PowerSupplyProperties prop;
prop.set_external_power(PowerSupplyProperties::DISCONNECTED);
prop.set_battery_state(PowerSupplyProperties::DISCHARGING);
power_status_->SetProtoForTesting(prop);
const std::unique_ptr<views::Widget> test_widget =
CreateTestWidget(views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET);
const ui::ColorProvider* color_provider =
test_widget->GetRootView()->GetColorProvider();
BatteryColors resolved_colors = PowerStatus::BatteryImageInfo::ResolveColors(
power_status_->GenerateBatteryImageInfo(SK_ColorRED, SK_ColorBLUE),
color_provider);
// Ensure preferences are honored.
EXPECT_EQ(resolved_colors.foreground_color, SK_ColorRED);
EXPECT_EQ(resolved_colors.badge_color, SK_ColorBLUE);
// Test Badge Color Resolution
prop.set_external_power(PowerSupplyProperties::AC);
prop.set_battery_state(PowerSupplyProperties::CHARGING);
prop.set_battery_percent(51);
power_status_->SetProtoForTesting(prop);
// Test badge color resolution (>50%).
resolved_colors = PowerStatus::BatteryImageInfo::ResolveColors(
power_status_->GenerateBatteryImageInfo(SK_ColorRED), color_provider);
EXPECT_EQ(resolved_colors.badge_color, SK_ColorRED);
// Test badge color with <= 50%.
prop.set_battery_percent(50);
power_status_->SetProtoForTesting(prop);
resolved_colors = PowerStatus::BatteryImageInfo::ResolveColors(
power_status_->GenerateBatteryImageInfo(SK_ColorRED), color_provider);
EXPECT_EQ(resolved_colors.badge_color, SK_ColorRED);
// Test alert color resolution.
resolved_colors = PowerStatus::BatteryImageInfo::ResolveColors(
power_status_->GenerateBatteryImageInfo(SK_ColorRED), color_provider);
EXPECT_EQ(resolved_colors.alert_color,
test_widget->GetRootView()->GetColorProvider()->GetColor(
cros_tokens::kColorAlert));
}
} // namespace ash