blob: a765d610fec89b826d06f220c872d7decebfa538 [file] [log] [blame]
[email protected]6ef71d72013-08-10 18:13:441// Copyright 2013 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "ash/display/resolution_notification_controller.h"
6
7#include "ash/display/display_controller.h"
8#include "ash/display/display_manager.h"
9#include "ash/shell.h"
10#include "base/strings/utf_string_conversions.h"
11#include "grit/ash_resources.h"
12#include "grit/ash_strings.h"
13#include "ui/base/l10n/l10n_util.h"
14#include "ui/base/l10n/time_format.h"
15#include "ui/base/resource/resource_bundle.h"
16#include "ui/gfx/display.h"
17#include "ui/gfx/screen.h"
18#include "ui/message_center/message_center.h"
19#include "ui/message_center/notification.h"
20#include "ui/message_center/notification_delegate.h"
21
22using message_center::Notification;
23
24namespace ash {
25namespace internal {
26namespace {
27
28bool g_use_timer = true;
29
30class ResolutionChangeNotificationDelegate
31 : public message_center::NotificationDelegate {
32 public:
33 ResolutionChangeNotificationDelegate(
34 ResolutionNotificationController* controller,
35 bool has_timeout);
36
37 protected:
38 virtual ~ResolutionChangeNotificationDelegate();
39
40 private:
41 // message_center::NotificationDelegate overrides:
42 virtual void Display() OVERRIDE;
43 virtual void Error() OVERRIDE;
44 virtual void Close(bool by_user) OVERRIDE;
45 virtual void Click() OVERRIDE;
46 virtual bool HasClickedListener() OVERRIDE;
47 virtual void ButtonClick(int button_index) OVERRIDE;
48
49 ResolutionNotificationController* controller_;
50 bool has_timeout_;
51
52 DISALLOW_COPY_AND_ASSIGN(ResolutionChangeNotificationDelegate);
53};
54
55ResolutionChangeNotificationDelegate::ResolutionChangeNotificationDelegate(
56 ResolutionNotificationController* controller,
57 bool has_timeout)
58 : controller_(controller),
59 has_timeout_(has_timeout) {
60 DCHECK(controller_);
61}
62
63ResolutionChangeNotificationDelegate::~ResolutionChangeNotificationDelegate() {
64}
65
66void ResolutionChangeNotificationDelegate::Display() {
67}
68
69void ResolutionChangeNotificationDelegate::Error() {
70}
71
72void ResolutionChangeNotificationDelegate::Close(bool by_user) {
73 if (by_user)
74 controller_->AcceptResolutionChange();
75}
76
77void ResolutionChangeNotificationDelegate::Click() {
78 controller_->AcceptResolutionChange();
79}
80
81bool ResolutionChangeNotificationDelegate::HasClickedListener() {
82 return true;
83}
84
85void ResolutionChangeNotificationDelegate::ButtonClick(int button_index) {
86 // If there's the timeout, the first button is "Accept". Otherwise the
87 // button click should be "Revert".
88 if (has_timeout_ && button_index == 0)
89 controller_->AcceptResolutionChange();
90 else
91 controller_->RevertResolutionChange();
92}
93
94} // namespace
95
96// static
97const int ResolutionNotificationController::kTimeoutInSec = 15;
98
99// static
100const char ResolutionNotificationController::kNotificationId[] =
101 "chrome://settings/display/resolution";
102
103struct ResolutionNotificationController::ResolutionChangeInfo {
104 ResolutionChangeInfo(int64 display_id,
105 const gfx::Size& old_resolution,
106 const gfx::Size& new_resolution,
107 const base::Closure& accept_callback);
108 ~ResolutionChangeInfo();
109
110 // The id of the display where the resolution change happens.
111 int64 display_id;
112
113 // The resolution before the change.
114 gfx::Size old_resolution;
115
116 // The new resolution after the change.
117 gfx::Size new_resolution;
118
119 // The callback when accept is chosen.
120 base::Closure accept_callback;
121
122 // The remaining timeout in seconds. 0 if the change does not time out.
123 uint8 timeout_count;
124
125 // The timer to invoke OnTimerTick() every second. This cannot be
126 // OneShotTimer since the message contains text "automatically closed in xx
127 // seconds..." which has to be updated every second.
128 base::RepeatingTimer<ResolutionNotificationController> timer;
129
130 private:
131 DISALLOW_COPY_AND_ASSIGN(ResolutionChangeInfo);
132};
133
134ResolutionNotificationController::ResolutionChangeInfo::ResolutionChangeInfo(
135 int64 display_id,
136 const gfx::Size& old_resolution,
137 const gfx::Size& new_resolution,
138 const base::Closure& accept_callback)
139 : display_id(display_id),
140 old_resolution(old_resolution),
141 new_resolution(new_resolution),
142 accept_callback(accept_callback),
143 timeout_count(0) {
144 DisplayManager* display_manager = Shell::GetInstance()->display_manager();
145 if (!display_manager->HasInternalDisplay() &&
146 display_manager->num_connected_displays() == 1u) {
147 timeout_count = kTimeoutInSec;
148 }
149}
150
151ResolutionNotificationController::ResolutionChangeInfo::
152 ~ResolutionChangeInfo() {
153}
154
155ResolutionNotificationController::ResolutionNotificationController() {
156 Shell::GetInstance()->display_controller()->AddObserver(this);
157 Shell::GetScreen()->AddObserver(this);
158}
159
160ResolutionNotificationController::~ResolutionNotificationController() {
161 Shell::GetInstance()->display_controller()->RemoveObserver(this);
162 Shell::GetScreen()->RemoveObserver(this);
163}
164
165void ResolutionNotificationController::SetDisplayResolutionAndNotify(
166 int64 display_id,
167 const gfx::Size& old_resolution,
168 const gfx::Size& new_resolution,
169 const base::Closure& accept_callback) {
170 // If multiple resolution changes are invoked for the same display,
171 // the original resolution for the first resolution change has to be used
172 // instead of the specified |old_resolution|.
173 gfx::Size original_resolution;
174 if (change_info_ && change_info_->display_id == display_id) {
175 DCHECK(change_info_->new_resolution == old_resolution);
176 original_resolution = change_info_->old_resolution;
177 }
178
179 change_info_.reset(new ResolutionChangeInfo(
180 display_id, old_resolution, new_resolution, accept_callback));
181 if (!original_resolution.IsEmpty())
182 change_info_->old_resolution = original_resolution;
183
184 // SetDisplayResolution() causes OnConfigurationChanged() and the notification
185 // will be shown at that point.
186 Shell::GetInstance()->display_manager()->SetDisplayResolution(
187 display_id, new_resolution);
188}
189
190bool ResolutionNotificationController::DoesNotificationTimeout() {
191 return change_info_ && change_info_->timeout_count > 0;
192}
193
194void ResolutionNotificationController::CreateOrUpdateNotification() {
195 message_center::MessageCenter* message_center =
196 message_center::MessageCenter::Get();
197 if (!change_info_) {
198 message_center->RemoveNotification(kNotificationId, false /* by_user */);
199 return;
200 }
201
202 base::string16 timeout_message;
203 message_center::RichNotificationData data;
204 if (change_info_->timeout_count > 0) {
205 data.buttons.push_back(message_center::ButtonInfo(
206 l10n_util::GetStringUTF16(IDS_ASH_DISPLAY_RESOLUTION_CHANGE_ACCEPT)));
207 timeout_message = l10n_util::GetStringFUTF16(
208 IDS_ASH_DISPLAY_RESOLUTION_TIMEOUT,
209 ui::TimeFormat::TimeDurationLong(
210 base::TimeDelta::FromSeconds(change_info_->timeout_count)));
211 }
212 data.buttons.push_back(message_center::ButtonInfo(
213 l10n_util::GetStringUTF16(IDS_ASH_DISPLAY_RESOLUTION_CHANGE_REVERT)));
214
215 ui::ResourceBundle& bundle = ui::ResourceBundle::GetSharedInstance();
216 scoped_ptr<Notification> notification(new Notification(
217 message_center::NOTIFICATION_TYPE_SIMPLE,
218 kNotificationId,
219 l10n_util::GetStringFUTF16(
220 IDS_ASH_STATUS_TRAY_DISPLAY_RESOLUTION_CHANGED,
221 UTF8ToUTF16(Shell::GetInstance()->display_manager()->
222 GetDisplayNameForId(change_info_->display_id)),
223 UTF8ToUTF16(change_info_->new_resolution.ToString())),
224 timeout_message,
225 bundle.GetImageNamed(IDR_AURA_UBER_TRAY_DISPLAY),
226 base::string16() /* display_source */,
227 std::string() /* extension_id */,
228 data,
229 new ResolutionChangeNotificationDelegate(
230 this, change_info_->timeout_count > 0)));
231 notification->SetSystemPriority();
232 message_center->AddNotification(notification.Pass());
233}
234
235void ResolutionNotificationController::OnTimerTick() {
236 if (!change_info_)
237 return;
238
239 --change_info_->timeout_count;
240 if (change_info_->timeout_count == 0)
241 RevertResolutionChange();
242 else
243 CreateOrUpdateNotification();
244}
245
246void ResolutionNotificationController::AcceptResolutionChange() {
247 message_center::MessageCenter::Get()->RemoveNotification(
248 kNotificationId, false /* by_user */);
249 base::Closure callback = change_info_->accept_callback;
250 change_info_.reset();
251 callback.Run();
252}
253
254void ResolutionNotificationController::RevertResolutionChange() {
255 message_center::MessageCenter::Get()->RemoveNotification(
256 kNotificationId, false /* by_user */);
257 int64 display_id = change_info_->display_id;
258 gfx::Size old_resolution = change_info_->old_resolution;
259 change_info_.reset();
260 Shell::GetInstance()->display_manager()->SetDisplayResolution(
261 display_id, old_resolution);
262}
263
264void ResolutionNotificationController::OnDisplayBoundsChanged(
265 const gfx::Display& display) {
266}
267
268void ResolutionNotificationController::OnDisplayAdded(
269 const gfx::Display& new_display) {
270}
271
272void ResolutionNotificationController::OnDisplayRemoved(
273 const gfx::Display& old_display) {
274 if (change_info_ && change_info_->display_id == old_display.id())
275 RevertResolutionChange();
276}
277
278void ResolutionNotificationController::OnDisplayConfigurationChanged() {
279 if (!change_info_)
280 return;
281
282 CreateOrUpdateNotification();
283 if (g_use_timer && change_info_->timeout_count > 0) {
284 change_info_->timer.Start(FROM_HERE,
285 base::TimeDelta::FromSeconds(1),
286 this,
287 &ResolutionNotificationController::OnTimerTick);
288 }
289}
290
291void ResolutionNotificationController::SuppressTimerForTest() {
292 g_use_timer = false;
293}
294
295} // namespace internal
296} // namespace ash