| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ash/system/chromeos/power/tray_power.h" | 5 #include "ash/system/chromeos/power/tray_power.h" |
| 6 | 6 |
| 7 #include "ash/ash_switches.h" | 7 #include "ash/ash_switches.h" |
| 8 #include "ash/shell.h" | |
| 9 #include "ash/shell_delegate.h" | |
| 10 #include "ash/system/chromeos/power/power_status_view.h" | 8 #include "ash/system/chromeos/power/power_status_view.h" |
| 11 #include "ash/system/date/date_view.h" | 9 #include "ash/system/date/date_view.h" |
| 12 #include "ash/system/tray/tray_constants.h" | 10 #include "ash/system/tray/tray_constants.h" |
| 13 #include "ash/system/tray/tray_notification_view.h" | 11 #include "ash/system/tray/tray_notification_view.h" |
| 14 #include "ash/system/tray/tray_utils.h" | 12 #include "ash/system/tray/tray_utils.h" |
| 15 #include "base/command_line.h" | 13 #include "base/command_line.h" |
| 16 #include "base/strings/string_number_conversions.h" | |
| 17 #include "base/strings/stringprintf.h" | |
| 18 #include "base/strings/utf_string_conversions.h" | |
| 19 #include "chromeos/dbus/power_supply_status.h" | |
| 20 #include "grit/ash_resources.h" | 14 #include "grit/ash_resources.h" |
| 21 #include "grit/ash_strings.h" | 15 #include "grit/ash_strings.h" |
| 22 #include "third_party/icu/public/i18n/unicode/fieldpos.h" | 16 #include "third_party/icu/public/i18n/unicode/fieldpos.h" |
| 23 #include "third_party/icu/public/i18n/unicode/fmtable.h" | 17 #include "third_party/icu/public/i18n/unicode/fmtable.h" |
| 24 #include "third_party/skia/include/core/SkRect.h" | |
| 25 #include "ui/base/accessibility/accessible_view_state.h" | 18 #include "ui/base/accessibility/accessible_view_state.h" |
| 26 #include "ui/base/l10n/l10n_util.h" | |
| 27 #include "ui/base/resource/resource_bundle.h" | 19 #include "ui/base/resource/resource_bundle.h" |
| 28 #include "ui/gfx/image/image.h" | |
| 29 #include "ui/gfx/image/image_skia.h" | |
| 30 #include "ui/gfx/image/image_skia_operations.h" | |
| 31 #include "ui/gfx/size.h" | |
| 32 #include "ui/message_center/message_center.h" | 20 #include "ui/message_center/message_center.h" |
| 33 #include "ui/message_center/notification.h" | 21 #include "ui/message_center/notification.h" |
| 34 #include "ui/views/controls/button/button.h" | 22 #include "ui/views/controls/button/button.h" |
| 35 #include "ui/views/controls/image_view.h" | 23 #include "ui/views/controls/image_view.h" |
| 36 #include "ui/views/controls/label.h" | 24 #include "ui/views/controls/label.h" |
| 37 #include "ui/views/layout/box_layout.h" | 25 #include "ui/views/layout/box_layout.h" |
| 38 #include "ui/views/layout/fill_layout.h" | 26 #include "ui/views/layout/fill_layout.h" |
| 39 #include "ui/views/layout/grid_layout.h" | 27 #include "ui/views/layout/grid_layout.h" |
| 40 #include "ui/views/view.h" | 28 #include "ui/views/view.h" |
| 41 #include "ui/views/widget/widget.h" | 29 #include "ui/views/widget/widget.h" |
| 42 | 30 |
| 43 using chromeos::PowerSupplyStatus; | |
| 44 using message_center::MessageCenter; | 31 using message_center::MessageCenter; |
| 45 using message_center::Notification; | 32 using message_center::Notification; |
| 46 | 33 |
| 47 namespace ash { | 34 namespace ash { |
| 48 namespace internal { | 35 namespace internal { |
| 49 | 36 |
| 50 namespace { | 37 namespace { |
| 51 // Width and height of battery images. | |
| 52 const int kBatteryImageHeight = 25; | |
| 53 const int kBatteryImageWidth = 25; | |
| 54 // Number of different power states. | |
| 55 const int kNumPowerImages = 15; | |
| 56 // Top/bottom padding of the text items. | 38 // Top/bottom padding of the text items. |
| 57 const int kPaddingVertical = 10; | 39 const int kPaddingVertical = 10; |
| 58 // Specify min width of status label for layout. | 40 // Specify min width of status label for layout. |
| 59 const int kLabelMinWidth = 120; | 41 const int kLabelMinWidth = 120; |
| 60 // Notification times. | 42 // Notification times. |
| 61 const int kCriticalSeconds = 5 * 60; | 43 const int kCriticalSeconds = 5 * 60; |
| 62 const int kLowPowerSeconds = 15 * 60; | 44 const int kLowPowerSeconds = 15 * 60; |
| 63 const int kNoWarningSeconds = 30 * 60; | 45 const int kNoWarningSeconds = 30 * 60; |
| 64 // Minimum battery percentage rendered in UI. | |
| 65 const int kMinBatteryPercent = 1; | |
| 66 // Notification in battery percentage. | 46 // Notification in battery percentage. |
| 67 const double kCriticalPercentage = 5.0; | 47 const double kCriticalPercentage = 5.0; |
| 68 const double kLowPowerPercentage = 10.0; | 48 const double kLowPowerPercentage = 10.0; |
| 69 const double kNoWarningPercentage = 15.0; | 49 const double kNoWarningPercentage = 15.0; |
| 70 | 50 |
| 71 base::string16 GetBatteryTimeAccessibilityString(int hour, int min) { | |
| 72 DCHECK(hour || min); | |
| 73 if (hour && !min) { | |
| 74 return Shell::GetInstance()->delegate()->GetTimeDurationLongString( | |
| 75 base::TimeDelta::FromHours(hour)); | |
| 76 } | |
| 77 if (min && !hour) { | |
| 78 return Shell::GetInstance()->delegate()->GetTimeDurationLongString( | |
| 79 base::TimeDelta::FromMinutes(min)); | |
| 80 } | |
| 81 return l10n_util::GetStringFUTF16( | |
| 82 IDS_ASH_STATUS_TRAY_BATTERY_TIME_ACCESSIBLE, | |
| 83 Shell::GetInstance()->delegate()->GetTimeDurationLongString( | |
| 84 base::TimeDelta::FromHours(hour)), | |
| 85 Shell::GetInstance()->delegate()->GetTimeDurationLongString( | |
| 86 base::TimeDelta::FromMinutes(min))); | |
| 87 } | |
| 88 | |
| 89 } // namespace | 51 } // namespace |
| 90 | 52 |
| 91 namespace tray { | 53 namespace tray { |
| 92 | 54 |
| 93 // This view is used only for the tray. | 55 // This view is used only for the tray. |
| 94 class PowerTrayView : public views::ImageView { | 56 class PowerTrayView : public views::ImageView { |
| 95 public: | 57 public: |
| 96 PowerTrayView() | 58 PowerTrayView() { |
| 97 : battery_icon_index_(-1), | |
| 98 battery_icon_offset_(0), | |
| 99 battery_charging_unreliable_(false) { | |
| 100 UpdateImage(); | 59 UpdateImage(); |
| 101 } | 60 } |
| 102 | 61 |
| 103 virtual ~PowerTrayView() { | 62 virtual ~PowerTrayView() { |
| 104 } | 63 } |
| 105 | 64 |
| 106 // Overriden from views::View. | 65 // Overriden from views::View. |
| 107 virtual void GetAccessibleState(ui::AccessibleViewState* state) OVERRIDE { | 66 virtual void GetAccessibleState(ui::AccessibleViewState* state) OVERRIDE { |
| 108 state->name = accessible_name_; | 67 state->name = accessible_name_; |
| 109 state->role = ui::AccessibilityTypes::ROLE_PUSHBUTTON; | 68 state->role = ui::AccessibilityTypes::ROLE_PUSHBUTTON; |
| 110 } | 69 } |
| 111 | 70 |
| 112 void UpdatePowerStatus(const PowerSupplyStatus& status, | 71 void UpdateStatus(bool battery_alert) { |
| 113 bool battery_alert) { | |
| 114 supply_status_ = status; | |
| 115 // Sanitize. | |
| 116 if (supply_status_.battery_is_full) | |
| 117 supply_status_.battery_percentage = 100.0; | |
| 118 | |
| 119 UpdateImage(); | 72 UpdateImage(); |
| 120 SetVisible(status.battery_is_present); | 73 SetVisible(PowerStatus::Get()->IsBatteryPresent()); |
| 121 | 74 |
| 122 if (battery_alert) { | 75 if (battery_alert) { |
| 123 accessible_name_ = TrayPower::GetAccessibleNameString(status); | 76 accessible_name_ = PowerStatus::Get()->GetAccessibleNameString(); |
| 124 NotifyAccessibilityEvent(ui::AccessibilityTypes::EVENT_ALERT, true); | 77 NotifyAccessibilityEvent(ui::AccessibilityTypes::EVENT_ALERT, true); |
| 125 } | 78 } |
| 126 } | 79 } |
| 127 | 80 |
| 128 private: | 81 private: |
| 129 void UpdateImage() { | 82 void UpdateImage() { |
| 130 int index = TrayPower::GetBatteryImageIndex(supply_status_); | 83 SetImage(PowerStatus::Get()->GetBatteryImage(PowerStatus::ICON_LIGHT)); |
| 131 int offset = TrayPower::GetBatteryImageOffset(supply_status_); | |
| 132 bool charging_unreliable = | |
| 133 TrayPower::IsBatteryChargingUnreliable(supply_status_); | |
| 134 if (battery_icon_index_ != index || | |
| 135 battery_icon_offset_ != offset || | |
| 136 battery_charging_unreliable_ != charging_unreliable) { | |
| 137 battery_icon_index_ = index; | |
| 138 battery_icon_offset_ = offset; | |
| 139 battery_charging_unreliable_ = charging_unreliable; | |
| 140 if (battery_icon_index_ != -1) | |
| 141 SetImage(TrayPower::GetBatteryImage(battery_icon_index_, | |
| 142 battery_icon_offset_, | |
| 143 battery_charging_unreliable_, | |
| 144 ICON_LIGHT)); | |
| 145 } | |
| 146 } | 84 } |
| 147 | 85 |
| 148 PowerSupplyStatus supply_status_; | |
| 149 base::string16 accessible_name_; | 86 base::string16 accessible_name_; |
| 150 | 87 |
| 151 // Index of the current icon in the icon array image, or -1 if unknown. | |
| 152 int battery_icon_index_; | |
| 153 int battery_icon_offset_; | |
| 154 bool battery_charging_unreliable_; | |
| 155 | |
| 156 DISALLOW_COPY_AND_ASSIGN(PowerTrayView); | 88 DISALLOW_COPY_AND_ASSIGN(PowerTrayView); |
| 157 }; | 89 }; |
| 158 | 90 |
| 159 class PowerNotificationView : public TrayNotificationView { | 91 class PowerNotificationView : public TrayNotificationView { |
| 160 public: | 92 public: |
| 161 explicit PowerNotificationView(TrayPower* owner) | 93 explicit PowerNotificationView(TrayPower* owner) |
| 162 : TrayNotificationView(owner, 0), | 94 : TrayNotificationView(owner, 0) { |
| 163 battery_icon_index_(-1), | |
| 164 battery_icon_offset_(0), | |
| 165 battery_charging_unreliable_(false) { | |
| 166 power_status_view_ = | 95 power_status_view_ = |
| 167 new PowerStatusView(PowerStatusView::VIEW_NOTIFICATION, true); | 96 new PowerStatusView(PowerStatusView::VIEW_NOTIFICATION, true); |
| 168 InitView(power_status_view_); | 97 InitView(power_status_view_); |
| 169 } | 98 } |
| 170 | 99 |
| 171 void UpdatePowerStatus(const PowerSupplyStatus& status) { | 100 void UpdateStatus() { |
| 172 int index = TrayPower::GetBatteryImageIndex(status); | 101 SetIconImage(PowerStatus::Get()->GetBatteryImage(PowerStatus::ICON_DARK)); |
| 173 int offset = TrayPower::GetBatteryImageOffset(status); | |
| 174 bool charging_unreliable = TrayPower::IsBatteryChargingUnreliable(status); | |
| 175 if (battery_icon_index_ != index || | |
| 176 battery_icon_offset_ != offset || | |
| 177 battery_charging_unreliable_ != charging_unreliable) { | |
| 178 battery_icon_index_ = index; | |
| 179 battery_icon_offset_ = offset; | |
| 180 battery_charging_unreliable_ = charging_unreliable; | |
| 181 if (battery_icon_index_ != -1) { | |
| 182 SetIconImage(TrayPower::GetBatteryImage( | |
| 183 battery_icon_index_, | |
| 184 battery_icon_offset_, | |
| 185 battery_charging_unreliable_, | |
| 186 ICON_DARK)); | |
| 187 } | |
| 188 } | |
| 189 power_status_view_->UpdatePowerStatus(status); | |
| 190 } | 102 } |
| 191 | 103 |
| 192 private: | 104 private: |
| 193 PowerStatusView* power_status_view_; | 105 PowerStatusView* power_status_view_; |
| 194 | 106 |
| 195 // Index of the current icon in the icon array image, or -1 if unknown. | |
| 196 int battery_icon_index_; | |
| 197 int battery_icon_offset_; | |
| 198 bool battery_charging_unreliable_; | |
| 199 | |
| 200 DISALLOW_COPY_AND_ASSIGN(PowerNotificationView); | 107 DISALLOW_COPY_AND_ASSIGN(PowerNotificationView); |
| 201 }; | 108 }; |
| 202 | 109 |
| 203 } // namespace tray | 110 } // namespace tray |
| 204 | 111 |
| 205 using tray::PowerNotificationView; | 112 using tray::PowerNotificationView; |
| 206 | 113 |
| 207 TrayPower::TrayPower(SystemTray* system_tray, MessageCenter* message_center) | 114 TrayPower::TrayPower(SystemTray* system_tray, MessageCenter* message_center) |
| 208 : SystemTrayItem(system_tray), | 115 : SystemTrayItem(system_tray), |
| 209 message_center_(message_center), | 116 message_center_(message_center), |
| 210 power_tray_(NULL), | 117 power_tray_(NULL), |
| 211 notification_view_(NULL), | 118 notification_view_(NULL), |
| 212 notification_state_(NOTIFICATION_NONE) { | 119 notification_state_(NOTIFICATION_NONE), |
| 120 usb_charger_was_connected_(false) { |
| 213 PowerStatus::Get()->AddObserver(this); | 121 PowerStatus::Get()->AddObserver(this); |
| 214 } | 122 } |
| 215 | 123 |
| 216 TrayPower::~TrayPower() { | 124 TrayPower::~TrayPower() { |
| 217 PowerStatus::Get()->RemoveObserver(this); | 125 PowerStatus::Get()->RemoveObserver(this); |
| 218 } | 126 } |
| 219 | 127 |
| 220 // static | |
| 221 bool TrayPower::IsBatteryChargingUnreliable( | |
| 222 const chromeos::PowerSupplyStatus& supply_status) { | |
| 223 // Sometimes devices can get into a state where the battery is almost fully | |
| 224 // charged and the power subsystem reports "neither charging nor discharging" | |
| 225 // despite the battery not at 100%. For now, only report unreliable charging | |
| 226 // on USB. | |
| 227 // TODO(derat): Update this when the power manager code is refactored for M29. | |
| 228 return supply_status.battery_state == PowerSupplyStatus::CONNECTED_TO_USB; | |
| 229 } | |
| 230 | |
| 231 // static | |
| 232 int TrayPower::GetBatteryImageIndex( | |
| 233 const chromeos::PowerSupplyStatus& supply_status) { | |
| 234 int image_index = 0; | |
| 235 if (supply_status.battery_percentage >= 100) { | |
| 236 image_index = kNumPowerImages - 1; | |
| 237 } else if (!supply_status.battery_is_present) { | |
| 238 image_index = kNumPowerImages; | |
| 239 } else { | |
| 240 image_index = static_cast<int>(supply_status.battery_percentage / | |
| 241 100.0 * (kNumPowerImages - 1)); | |
| 242 image_index = std::max(std::min(image_index, kNumPowerImages - 2), 0); | |
| 243 } | |
| 244 return image_index; | |
| 245 } | |
| 246 | |
| 247 // static | |
| 248 int TrayPower::GetBatteryImageOffset( | |
| 249 const chromeos::PowerSupplyStatus& supply_status) { | |
| 250 if (IsBatteryChargingUnreliable(supply_status) || | |
| 251 !supply_status.line_power_on) | |
| 252 return 0; | |
| 253 return 1; | |
| 254 } | |
| 255 | |
| 256 // static | |
| 257 gfx::ImageSkia TrayPower::GetBatteryImage(int image_index, | |
| 258 int image_offset, | |
| 259 bool charging_unreliable, | |
| 260 IconSet icon_set) { | |
| 261 gfx::Image all; | |
| 262 if (charging_unreliable) { | |
| 263 all = ui::ResourceBundle::GetSharedInstance().GetImageNamed( | |
| 264 icon_set == ICON_DARK ? | |
| 265 IDR_AURA_UBER_TRAY_POWER_SMALL_CHARGING_UNRELIABLE_DARK : | |
| 266 IDR_AURA_UBER_TRAY_POWER_SMALL_CHARGING_UNRELIABLE); | |
| 267 } else { | |
| 268 all = ui::ResourceBundle::GetSharedInstance().GetImageNamed( | |
| 269 icon_set == ICON_DARK ? | |
| 270 IDR_AURA_UBER_TRAY_POWER_SMALL_DARK : IDR_AURA_UBER_TRAY_POWER_SMALL); | |
| 271 } | |
| 272 gfx::Rect region( | |
| 273 image_offset * kBatteryImageWidth, | |
| 274 image_index * kBatteryImageHeight, | |
| 275 kBatteryImageWidth, kBatteryImageHeight); | |
| 276 return gfx::ImageSkiaOperations::ExtractSubset(*all.ToImageSkia(), region); | |
| 277 } | |
| 278 | |
| 279 // static | |
| 280 base::string16 TrayPower::GetAccessibleNameString( | |
| 281 const chromeos::PowerSupplyStatus& supply_status) { | |
| 282 ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); | |
| 283 if (supply_status.line_power_on && supply_status.battery_is_full) { | |
| 284 return rb.GetLocalizedString( | |
| 285 IDS_ASH_STATUS_TRAY_BATTERY_FULL_CHARGE_ACCESSIBLE); | |
| 286 } | |
| 287 bool charging_unreliable = | |
| 288 IsBatteryChargingUnreliable(supply_status); | |
| 289 if (supply_status.battery_percentage < 0.0f) { | |
| 290 if (charging_unreliable) { | |
| 291 return rb.GetLocalizedString( | |
| 292 IDS_ASH_STATUS_TRAY_BATTERY_CHARGING_UNRELIABLE_ACCESSIBLE); | |
| 293 } | |
| 294 return rb.GetLocalizedString( | |
| 295 IDS_ASH_STATUS_TRAY_BATTERY_CALCULATING_ACCESSIBLE); | |
| 296 } | |
| 297 base::string16 battery_percentage_accessbile = l10n_util::GetStringFUTF16( | |
| 298 supply_status.line_power_on ? | |
| 299 IDS_ASH_STATUS_TRAY_BATTERY_PERCENT_CHARGING_ACCESSIBLE: | |
| 300 IDS_ASH_STATUS_TRAY_BATTERY_PERCENT_ACCESSIBLE , | |
| 301 base::IntToString16(GetRoundedBatteryPercentage( | |
| 302 supply_status.battery_percentage))); | |
| 303 base::string16 battery_time_accessible = base::string16(); | |
| 304 if (charging_unreliable) { | |
| 305 battery_time_accessible = rb.GetLocalizedString( | |
| 306 IDS_ASH_STATUS_TRAY_BATTERY_CHARGING_UNRELIABLE_ACCESSIBLE); | |
| 307 } else { | |
| 308 if (supply_status.is_calculating_battery_time) { | |
| 309 battery_time_accessible = rb.GetLocalizedString( | |
| 310 IDS_ASH_STATUS_TRAY_BATTERY_CALCULATING_ACCESSIBLE); | |
| 311 } else { | |
| 312 base::TimeDelta time = base::TimeDelta::FromSeconds( | |
| 313 supply_status.line_power_on ? | |
| 314 supply_status.battery_seconds_to_full : | |
| 315 supply_status.battery_seconds_to_empty); | |
| 316 int hour = time.InHours(); | |
| 317 int min = (time - base::TimeDelta::FromHours(hour)).InMinutes(); | |
| 318 if (hour || min) { | |
| 319 base::string16 minute = min < 10 ? | |
| 320 ASCIIToUTF16("0") + base::IntToString16(min) : | |
| 321 base::IntToString16(min); | |
| 322 battery_time_accessible = | |
| 323 l10n_util::GetStringFUTF16( | |
| 324 supply_status.line_power_on ? | |
| 325 IDS_ASH_STATUS_TRAY_BATTERY_TIME_UNTIL_FULL_ACCESSIBLE : | |
| 326 IDS_ASH_STATUS_TRAY_BATTERY_TIME_LEFT_ACCESSIBLE, | |
| 327 GetBatteryTimeAccessibilityString(hour, min)); | |
| 328 } | |
| 329 } | |
| 330 } | |
| 331 return battery_time_accessible.empty() ? | |
| 332 battery_percentage_accessbile : | |
| 333 battery_percentage_accessbile + ASCIIToUTF16(". ") | |
| 334 + battery_time_accessible; | |
| 335 } | |
| 336 | |
| 337 // static | |
| 338 int TrayPower::GetRoundedBatteryPercentage(double battery_percentage) { | |
| 339 DCHECK(battery_percentage >= 0.0); | |
| 340 return std::max(kMinBatteryPercent, | |
| 341 static_cast<int>(battery_percentage + 0.5)); | |
| 342 } | |
| 343 | |
| 344 views::View* TrayPower::CreateTrayView(user::LoginStatus status) { | 128 views::View* TrayPower::CreateTrayView(user::LoginStatus status) { |
| 345 // There may not be enough information when this is created about whether | 129 // There may not be enough information when this is created about whether |
| 346 // there is a battery or not. So always create this, and adjust visibility as | 130 // there is a battery or not. So always create this, and adjust visibility as |
| 347 // necessary. | 131 // necessary. |
| 348 PowerSupplyStatus power_status = PowerStatus::Get()->GetPowerSupplyStatus(); | |
| 349 CHECK(power_tray_ == NULL); | 132 CHECK(power_tray_ == NULL); |
| 350 power_tray_ = new tray::PowerTrayView(); | 133 power_tray_ = new tray::PowerTrayView(); |
| 351 power_tray_->UpdatePowerStatus(power_status, false); | 134 power_tray_->UpdateStatus(false); |
| 352 return power_tray_; | 135 return power_tray_; |
| 353 } | 136 } |
| 354 | 137 |
| 355 views::View* TrayPower::CreateDefaultView(user::LoginStatus status) { | 138 views::View* TrayPower::CreateDefaultView(user::LoginStatus status) { |
| 356 // Make sure icon status is up-to-date. (Also triggers stub activation). | 139 // Make sure icon status is up-to-date. (Also triggers stub activation). |
| 357 RequestStatusUpdate(); | 140 PowerStatus::Get()->RequestStatusUpdate(); |
| 358 return NULL; | 141 return NULL; |
| 359 } | 142 } |
| 360 | 143 |
| 361 views::View* TrayPower::CreateNotificationView(user::LoginStatus status) { | 144 views::View* TrayPower::CreateNotificationView(user::LoginStatus status) { |
| 362 CHECK(notification_view_ == NULL); | 145 CHECK(notification_view_ == NULL); |
| 363 PowerSupplyStatus power_status = PowerStatus::Get()->GetPowerSupplyStatus(); | 146 if (!PowerStatus::Get()->IsBatteryPresent()) |
| 364 if (!power_status.battery_is_present) | |
| 365 return NULL; | 147 return NULL; |
| 366 | 148 |
| 367 notification_view_ = new PowerNotificationView(this); | 149 notification_view_ = new PowerNotificationView(this); |
| 368 notification_view_->UpdatePowerStatus(power_status); | 150 notification_view_->UpdateStatus(); |
| 369 | 151 |
| 370 return notification_view_; | 152 return notification_view_; |
| 371 } | 153 } |
| 372 | 154 |
| 373 void TrayPower::DestroyTrayView() { | 155 void TrayPower::DestroyTrayView() { |
| 374 power_tray_ = NULL; | 156 power_tray_ = NULL; |
| 375 } | 157 } |
| 376 | 158 |
| 377 void TrayPower::DestroyDefaultView() { | 159 void TrayPower::DestroyDefaultView() { |
| 378 } | 160 } |
| 379 | 161 |
| 380 void TrayPower::DestroyNotificationView() { | 162 void TrayPower::DestroyNotificationView() { |
| 381 notification_view_ = NULL; | 163 notification_view_ = NULL; |
| 382 } | 164 } |
| 383 | 165 |
| 384 void TrayPower::UpdateAfterLoginStatusChange(user::LoginStatus status) { | 166 void TrayPower::UpdateAfterLoginStatusChange(user::LoginStatus status) { |
| 385 } | 167 } |
| 386 | 168 |
| 387 void TrayPower::UpdateAfterShelfAlignmentChange(ShelfAlignment alignment) { | 169 void TrayPower::UpdateAfterShelfAlignmentChange(ShelfAlignment alignment) { |
| 388 SetTrayImageItemBorder(power_tray_, alignment); | 170 SetTrayImageItemBorder(power_tray_, alignment); |
| 389 } | 171 } |
| 390 | 172 |
| 391 void TrayPower::OnPowerStatusChanged( | 173 void TrayPower::OnPowerStatusChanged() { |
| 392 const chromeos::PowerSupplyStatus& status) { | 174 bool battery_alert = UpdateNotificationState(); |
| 393 bool battery_alert = UpdateNotificationState(status); | |
| 394 if (power_tray_) | 175 if (power_tray_) |
| 395 power_tray_->UpdatePowerStatus(status, battery_alert); | 176 power_tray_->UpdateStatus(battery_alert); |
| 396 if (notification_view_) | 177 if (notification_view_) |
| 397 notification_view_->UpdatePowerStatus(status); | 178 notification_view_->UpdateStatus(); |
| 398 | 179 |
| 399 // Factory testing may place the battery into unusual states. | 180 // Factory testing may place the battery into unusual states. |
| 400 if (CommandLine::ForCurrentProcess()->HasSwitch( | 181 if (CommandLine::ForCurrentProcess()->HasSwitch( |
| 401 ash::switches::kAshHideNotificationsForFactory)) | 182 ash::switches::kAshHideNotificationsForFactory)) |
| 402 return; | 183 return; |
| 403 | 184 |
| 404 if (ash::switches::UseUsbChargerNotification()) | 185 if (ash::switches::UseUsbChargerNotification()) |
| 405 MaybeShowUsbChargerNotification(last_power_supply_status_, status); | 186 MaybeShowUsbChargerNotification(); |
| 406 | 187 |
| 407 if (battery_alert) | 188 if (battery_alert) |
| 408 ShowNotificationView(); | 189 ShowNotificationView(); |
| 409 else if (notification_state_ == NOTIFICATION_NONE) | 190 else if (notification_state_ == NOTIFICATION_NONE) |
| 410 HideNotificationView(); | 191 HideNotificationView(); |
| 411 | 192 |
| 412 last_power_supply_status_ = status; | 193 usb_charger_was_connected_ = PowerStatus::Get()->IsUsbChargerConnected(); |
| 413 } | 194 } |
| 414 | 195 |
| 415 void TrayPower::RequestStatusUpdate() const { | 196 bool TrayPower::MaybeShowUsbChargerNotification() { |
| 416 PowerStatus::Get()->RequestStatusUpdate(); | |
| 417 } | |
| 418 | |
| 419 bool TrayPower::MaybeShowUsbChargerNotification( | |
| 420 const PowerSupplyStatus& old_status, | |
| 421 const PowerSupplyStatus& new_status) { | |
| 422 ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); | 197 ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); |
| 423 const char kNotificationId[] = "usb-charger"; | 198 const char kNotificationId[] = "usb-charger"; |
| 199 bool usb_charger_is_connected = PowerStatus::Get()->IsUsbChargerConnected(); |
| 200 |
| 424 // Check for a USB charger being connected. | 201 // Check for a USB charger being connected. |
| 425 if (new_status.battery_state == PowerSupplyStatus::CONNECTED_TO_USB && | 202 if (usb_charger_is_connected && !usb_charger_was_connected_) { |
| 426 old_status.battery_state != PowerSupplyStatus::CONNECTED_TO_USB) { | |
| 427 scoped_ptr<Notification> notification(new Notification( | 203 scoped_ptr<Notification> notification(new Notification( |
| 428 message_center::NOTIFICATION_TYPE_SIMPLE, | 204 message_center::NOTIFICATION_TYPE_SIMPLE, |
| 429 kNotificationId, | 205 kNotificationId, |
| 430 rb.GetLocalizedString(IDS_ASH_STATUS_TRAY_LOW_POWER_CHARGER_TITLE), | 206 rb.GetLocalizedString(IDS_ASH_STATUS_TRAY_LOW_POWER_CHARGER_TITLE), |
| 431 rb.GetLocalizedString(IDS_ASH_STATUS_TRAY_LOW_POWER_CHARGER_MESSAGE), | 207 rb.GetLocalizedString(IDS_ASH_STATUS_TRAY_LOW_POWER_CHARGER_MESSAGE), |
| 432 rb.GetImageNamed(IDR_AURA_NOTIFICATION_LOW_POWER_CHARGER), | 208 rb.GetImageNamed(IDR_AURA_NOTIFICATION_LOW_POWER_CHARGER), |
| 433 base::string16(), | 209 base::string16(), |
| 434 std::string(), | 210 std::string(), |
| 435 message_center::RichNotificationData(), | 211 message_center::RichNotificationData(), |
| 436 NULL)); | 212 NULL)); |
| 437 message_center_->AddNotification(notification.Pass()); | 213 message_center_->AddNotification(notification.Pass()); |
| 438 return true; | 214 return true; |
| 439 } | 215 } |
| 440 | 216 |
| 441 // Check for unplug of a USB charger while the USB charger notification is | 217 // Check for unplug of a USB charger while the USB charger notification is |
| 442 // showing. | 218 // showing. |
| 443 if (new_status.battery_state != PowerSupplyStatus::CONNECTED_TO_USB && | 219 if (!usb_charger_is_connected && usb_charger_was_connected_) { |
| 444 old_status.battery_state == PowerSupplyStatus::CONNECTED_TO_USB) { | |
| 445 message_center_->RemoveNotification(kNotificationId, false); | 220 message_center_->RemoveNotification(kNotificationId, false); |
| 446 return true; | 221 return true; |
| 447 } | 222 } |
| 448 return false; | 223 return false; |
| 449 } | 224 } |
| 450 | 225 |
| 451 bool TrayPower::UpdateNotificationState( | 226 bool TrayPower::UpdateNotificationState() { |
| 452 const chromeos::PowerSupplyStatus& status) { | 227 const PowerStatus& status = *PowerStatus::Get(); |
| 453 if (!status.battery_is_present || | 228 if (!status.IsBatteryPresent() || |
| 454 status.is_calculating_battery_time || | 229 status.IsBatteryTimeBeingCalculated() || |
| 455 status.battery_state == PowerSupplyStatus::CHARGING || | 230 status.IsMainsChargerConnected()) { |
| 456 status.battery_state == | |
| 457 PowerSupplyStatus::NEITHER_CHARGING_NOR_DISCHARGING) { | |
| 458 notification_state_ = NOTIFICATION_NONE; | 231 notification_state_ = NOTIFICATION_NONE; |
| 459 return false; | 232 return false; |
| 460 } | 233 } |
| 461 | 234 |
| 462 if (TrayPower::IsBatteryChargingUnreliable(status)) { | 235 return status.IsUsbChargerConnected() ? |
| 463 return UpdateNotificationStateForRemainingPercentage( | 236 UpdateNotificationStateForRemainingPercentage() : |
| 464 status.battery_percentage); | 237 UpdateNotificationStateForRemainingTime(); |
| 465 } else { | |
| 466 return UpdateNotificationStateForRemainingTime( | |
| 467 status.battery_seconds_to_empty); | |
| 468 } | |
| 469 } | 238 } |
| 470 | 239 |
| 471 bool TrayPower::UpdateNotificationStateForRemainingTime(int remaining_seconds) { | 240 bool TrayPower::UpdateNotificationStateForRemainingTime() { |
| 241 const int remaining_seconds = |
| 242 PowerStatus::Get()->GetBatteryTimeToEmpty().InSeconds(); |
| 243 |
| 472 if (remaining_seconds >= kNoWarningSeconds) { | 244 if (remaining_seconds >= kNoWarningSeconds) { |
| 473 notification_state_ = NOTIFICATION_NONE; | 245 notification_state_ = NOTIFICATION_NONE; |
| 474 return false; | 246 return false; |
| 475 } | 247 } |
| 476 | 248 |
| 477 switch (notification_state_) { | 249 switch (notification_state_) { |
| 478 case NOTIFICATION_NONE: | 250 case NOTIFICATION_NONE: |
| 479 if (remaining_seconds <= kCriticalSeconds) { | 251 if (remaining_seconds <= kCriticalSeconds) { |
| 480 notification_state_ = NOTIFICATION_CRITICAL; | 252 notification_state_ = NOTIFICATION_CRITICAL; |
| 481 return true; | 253 return true; |
| 482 } | 254 } |
| 483 if (remaining_seconds <= kLowPowerSeconds) { | 255 if (remaining_seconds <= kLowPowerSeconds) { |
| 484 notification_state_ = NOTIFICATION_LOW_POWER; | 256 notification_state_ = NOTIFICATION_LOW_POWER; |
| 485 return true; | 257 return true; |
| 486 } | 258 } |
| 487 return false; | 259 return false; |
| 488 case NOTIFICATION_LOW_POWER: | 260 case NOTIFICATION_LOW_POWER: |
| 489 if (remaining_seconds <= kCriticalSeconds) { | 261 if (remaining_seconds <= kCriticalSeconds) { |
| 490 notification_state_ = NOTIFICATION_CRITICAL; | 262 notification_state_ = NOTIFICATION_CRITICAL; |
| 491 return true; | 263 return true; |
| 492 } | 264 } |
| 493 return false; | 265 return false; |
| 494 case NOTIFICATION_CRITICAL: | 266 case NOTIFICATION_CRITICAL: |
| 495 return false; | 267 return false; |
| 496 } | 268 } |
| 497 NOTREACHED(); | 269 NOTREACHED(); |
| 498 return false; | 270 return false; |
| 499 } | 271 } |
| 500 | 272 |
| 501 bool TrayPower::UpdateNotificationStateForRemainingPercentage( | 273 bool TrayPower::UpdateNotificationStateForRemainingPercentage() { |
| 502 double remaining_percentage) { | 274 const double remaining_percentage = PowerStatus::Get()->GetBatteryPercent(); |
| 275 |
| 503 if (remaining_percentage > kNoWarningPercentage) { | 276 if (remaining_percentage > kNoWarningPercentage) { |
| 504 notification_state_ = NOTIFICATION_NONE; | 277 notification_state_ = NOTIFICATION_NONE; |
| 505 return false; | 278 return false; |
| 506 } | 279 } |
| 507 | 280 |
| 508 switch (notification_state_) { | 281 switch (notification_state_) { |
| 509 case NOTIFICATION_NONE: | 282 case NOTIFICATION_NONE: |
| 510 if (remaining_percentage <= kCriticalPercentage) { | 283 if (remaining_percentage <= kCriticalPercentage) { |
| 511 notification_state_ = NOTIFICATION_CRITICAL; | 284 notification_state_ = NOTIFICATION_CRITICAL; |
| 512 return true; | 285 return true; |
| (...skipping 11 matching lines...) Expand all Loading... |
| 524 return false; | 297 return false; |
| 525 case NOTIFICATION_CRITICAL: | 298 case NOTIFICATION_CRITICAL: |
| 526 return false; | 299 return false; |
| 527 } | 300 } |
| 528 NOTREACHED(); | 301 NOTREACHED(); |
| 529 return false; | 302 return false; |
| 530 } | 303 } |
| 531 | 304 |
| 532 } // namespace internal | 305 } // namespace internal |
| 533 } // namespace ash | 306 } // namespace ash |
| OLD | NEW |