| 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 "chrome/browser/ui/simple_message_box.h" | 5 #include "chrome/browser/ui/simple_message_box.h" |
| 6 | 6 |
| 7 #include "base/basictypes.h" | 7 #include "base/basictypes.h" |
| 8 #include "base/compiler_specific.h" | 8 #include "base/compiler_specific.h" |
| 9 #include "base/memory/ref_counted.h" | 9 #include "base/memory/ref_counted.h" |
| 10 #include "base/message_loop/message_loop.h" | 10 #include "base/message_loop/message_loop.h" |
| 11 #include "base/message_loop/message_pump_dispatcher.h" | 11 #include "base/message_loop/message_pump_dispatcher.h" |
| 12 #include "base/run_loop.h" | 12 #include "base/run_loop.h" |
| 13 #include "chrome/browser/browser_process.h" | 13 #include "chrome/browser/browser_process.h" |
| 14 #include "chrome/browser/ui/views/constrained_window_views.h" | 14 #include "chrome/browser/ui/views/constrained_window_views.h" |
| 15 #include "grit/generated_resources.h" | 15 #include "grit/generated_resources.h" |
| 16 #include "ui/aura/client/dispatcher_client.h" | 16 #include "ui/aura/client/dispatcher_client.h" |
| 17 #include "ui/aura/env.h" | 17 #include "ui/aura/env.h" |
| 18 #include "ui/aura/window_event_dispatcher.h" | 18 #include "ui/aura/window_event_dispatcher.h" |
| 19 #include "ui/base/l10n/l10n_util.h" | 19 #include "ui/base/l10n/l10n_util.h" |
| 20 #include "ui/gfx/native_widget_types.h" | 20 #include "ui/gfx/native_widget_types.h" |
| 21 #include "ui/views/controls/message_box_view.h" | 21 #include "ui/views/controls/message_box_view.h" |
| 22 #include "ui/views/corewm/window_util.h" |
| 22 #include "ui/views/widget/widget.h" | 23 #include "ui/views/widget/widget.h" |
| 23 #include "ui/views/window/dialog_delegate.h" | 24 #include "ui/views/window/dialog_delegate.h" |
| 24 | 25 |
| 25 #if defined(OS_WIN) | 26 #if defined(OS_WIN) |
| 26 #include "ui/base/win/message_box_win.h" | 27 #include "ui/base/win/message_box_win.h" |
| 27 #include "ui/views/win/hwnd_util.h" | 28 #include "ui/views/win/hwnd_util.h" |
| 28 #endif | 29 #endif |
| 29 | 30 |
| 30 namespace chrome { | 31 namespace chrome { |
| 31 | 32 |
| 32 namespace { | 33 namespace { |
| 33 | 34 |
| 34 // Multiple SimpleMessageBoxViews can show up at the same time. Each of these | 35 // Multiple SimpleMessageBoxViews can show up at the same time. Each of these |
| 35 // start a nested message-loop. However, these SimpleMessageBoxViews can be | 36 // start a nested message-loop. However, these SimpleMessageBoxViews can be |
| 36 // deleted in any order. This creates problems if a box in an inner-loop gets | 37 // deleted in any order. This creates problems if a box in an inner-loop gets |
| 37 // destroyed before a box in an outer-loop. So to avoid this, ref-counting is | 38 // destroyed before a box in an outer-loop. So to avoid this, ref-counting is |
| 38 // used so that the SimpleMessageBoxViews gets deleted at the right time. | 39 // used so that the SimpleMessageBoxViews gets deleted at the right time. |
| 39 class SimpleMessageBoxViews : public views::DialogDelegate, | 40 class SimpleMessageBoxViews : public views::DialogDelegate, |
| 40 public base::MessagePumpDispatcher, | |
| 41 public base::RefCounted<SimpleMessageBoxViews> { | 41 public base::RefCounted<SimpleMessageBoxViews> { |
| 42 public: | 42 public: |
| 43 SimpleMessageBoxViews(const base::string16& title, | 43 SimpleMessageBoxViews(const base::string16& title, |
| 44 const base::string16& message, | 44 const base::string16& message, |
| 45 MessageBoxType type, | 45 MessageBoxType type, |
| 46 const base::string16& yes_text, | 46 const base::string16& yes_text, |
| 47 const base::string16& no_text); | 47 const base::string16& no_text); |
| 48 | 48 |
| 49 MessageBoxResult result() const { return result_; } | 49 MessageBoxResult result() const { return result_; } |
| 50 | 50 |
| 51 // Overridden from views::DialogDelegate: | 51 // Overridden from views::DialogDelegate: |
| 52 virtual int GetDialogButtons() const OVERRIDE; | 52 virtual int GetDialogButtons() const OVERRIDE; |
| 53 virtual base::string16 GetDialogButtonLabel( | 53 virtual base::string16 GetDialogButtonLabel( |
| 54 ui::DialogButton button) const OVERRIDE; | 54 ui::DialogButton button) const OVERRIDE; |
| 55 virtual bool Cancel() OVERRIDE; | 55 virtual bool Cancel() OVERRIDE; |
| 56 virtual bool Accept() OVERRIDE; | 56 virtual bool Accept() OVERRIDE; |
| 57 | 57 |
| 58 // Overridden from views::WidgetDelegate: | 58 // Overridden from views::WidgetDelegate: |
| 59 virtual base::string16 GetWindowTitle() const OVERRIDE; | 59 virtual base::string16 GetWindowTitle() const OVERRIDE; |
| 60 virtual void DeleteDelegate() OVERRIDE; | 60 virtual void DeleteDelegate() OVERRIDE; |
| 61 virtual ui::ModalType GetModalType() const OVERRIDE; | 61 virtual ui::ModalType GetModalType() const OVERRIDE; |
| 62 virtual views::View* GetContentsView() OVERRIDE; | 62 virtual views::View* GetContentsView() OVERRIDE; |
| 63 virtual views::Widget* GetWidget() OVERRIDE; | 63 virtual views::Widget* GetWidget() OVERRIDE; |
| 64 virtual const views::Widget* GetWidget() const OVERRIDE; | 64 virtual const views::Widget* GetWidget() const OVERRIDE; |
| 65 | 65 |
| 66 // Overridden from MessagePumpDispatcher: | |
| 67 virtual uint32_t Dispatch(const base::NativeEvent& event) OVERRIDE; | |
| 68 | |
| 69 private: | 66 private: |
| 70 friend class base::RefCounted<SimpleMessageBoxViews>; | 67 friend class base::RefCounted<SimpleMessageBoxViews>; |
| 71 virtual ~SimpleMessageBoxViews(); | 68 virtual ~SimpleMessageBoxViews(); |
| 72 | 69 |
| 70 // This terminates the nested message-loop. |
| 71 void Done(); |
| 72 |
| 73 const base::string16 window_title_; | 73 const base::string16 window_title_; |
| 74 const MessageBoxType type_; | 74 const MessageBoxType type_; |
| 75 base::string16 yes_text_; | 75 base::string16 yes_text_; |
| 76 base::string16 no_text_; | 76 base::string16 no_text_; |
| 77 MessageBoxResult result_; | 77 MessageBoxResult result_; |
| 78 views::MessageBoxView* message_box_view_; | 78 views::MessageBoxView* message_box_view_; |
| 79 | 79 |
| 80 // Set to false as soon as the user clicks a dialog button; this tells the | |
| 81 // dispatcher we're done. | |
| 82 bool should_show_dialog_; | |
| 83 | |
| 84 DISALLOW_COPY_AND_ASSIGN(SimpleMessageBoxViews); | 80 DISALLOW_COPY_AND_ASSIGN(SimpleMessageBoxViews); |
| 85 }; | 81 }; |
| 86 | 82 |
| 87 //////////////////////////////////////////////////////////////////////////////// | 83 //////////////////////////////////////////////////////////////////////////////// |
| 88 // SimpleMessageBoxViews, public: | 84 // SimpleMessageBoxViews, public: |
| 89 | 85 |
| 90 SimpleMessageBoxViews::SimpleMessageBoxViews(const base::string16& title, | 86 SimpleMessageBoxViews::SimpleMessageBoxViews(const base::string16& title, |
| 91 const base::string16& message, | 87 const base::string16& message, |
| 92 MessageBoxType type, | 88 MessageBoxType type, |
| 93 const base::string16& yes_text, | 89 const base::string16& yes_text, |
| 94 const base::string16& no_text) | 90 const base::string16& no_text) |
| 95 : window_title_(title), | 91 : window_title_(title), |
| 96 type_(type), | 92 type_(type), |
| 97 yes_text_(yes_text), | 93 yes_text_(yes_text), |
| 98 no_text_(no_text), | 94 no_text_(no_text), |
| 99 result_(MESSAGE_BOX_RESULT_NO), | 95 result_(MESSAGE_BOX_RESULT_NO), |
| 100 message_box_view_(new views::MessageBoxView( | 96 message_box_view_(new views::MessageBoxView( |
| 101 views::MessageBoxView::InitParams(message))), | 97 views::MessageBoxView::InitParams(message))) { |
| 102 should_show_dialog_(true) { | |
| 103 AddRef(); | 98 AddRef(); |
| 104 | 99 |
| 105 if (yes_text_.empty()) { | 100 if (yes_text_.empty()) { |
| 106 if (type_ == MESSAGE_BOX_TYPE_QUESTION) | 101 if (type_ == MESSAGE_BOX_TYPE_QUESTION) |
| 107 yes_text_ = | 102 yes_text_ = |
| 108 l10n_util::GetStringUTF16(IDS_CONFIRM_MESSAGEBOX_YES_BUTTON_LABEL); | 103 l10n_util::GetStringUTF16(IDS_CONFIRM_MESSAGEBOX_YES_BUTTON_LABEL); |
| 109 else if (type_ == MESSAGE_BOX_TYPE_OK_CANCEL) | 104 else if (type_ == MESSAGE_BOX_TYPE_OK_CANCEL) |
| 110 yes_text_ = l10n_util::GetStringUTF16(IDS_OK); | 105 yes_text_ = l10n_util::GetStringUTF16(IDS_OK); |
| 111 else | 106 else |
| 112 yes_text_ = l10n_util::GetStringUTF16(IDS_OK); | 107 yes_text_ = l10n_util::GetStringUTF16(IDS_OK); |
| (...skipping 18 matching lines...) Expand all Loading... |
| 131 } | 126 } |
| 132 | 127 |
| 133 base::string16 SimpleMessageBoxViews::GetDialogButtonLabel( | 128 base::string16 SimpleMessageBoxViews::GetDialogButtonLabel( |
| 134 ui::DialogButton button) const { | 129 ui::DialogButton button) const { |
| 135 if (button == ui::DIALOG_BUTTON_CANCEL) | 130 if (button == ui::DIALOG_BUTTON_CANCEL) |
| 136 return no_text_; | 131 return no_text_; |
| 137 return yes_text_; | 132 return yes_text_; |
| 138 } | 133 } |
| 139 | 134 |
| 140 bool SimpleMessageBoxViews::Cancel() { | 135 bool SimpleMessageBoxViews::Cancel() { |
| 141 should_show_dialog_= false; | |
| 142 result_ = MESSAGE_BOX_RESULT_NO; | 136 result_ = MESSAGE_BOX_RESULT_NO; |
| 137 Done(); |
| 143 return true; | 138 return true; |
| 144 } | 139 } |
| 145 | 140 |
| 146 bool SimpleMessageBoxViews::Accept() { | 141 bool SimpleMessageBoxViews::Accept() { |
| 147 should_show_dialog_ = false; | |
| 148 result_ = MESSAGE_BOX_RESULT_YES; | 142 result_ = MESSAGE_BOX_RESULT_YES; |
| 143 Done(); |
| 149 return true; | 144 return true; |
| 150 } | 145 } |
| 151 | 146 |
| 152 base::string16 SimpleMessageBoxViews::GetWindowTitle() const { | 147 base::string16 SimpleMessageBoxViews::GetWindowTitle() const { |
| 153 return window_title_; | 148 return window_title_; |
| 154 } | 149 } |
| 155 | 150 |
| 156 void SimpleMessageBoxViews::DeleteDelegate() { | 151 void SimpleMessageBoxViews::DeleteDelegate() { |
| 157 Release(); | 152 Release(); |
| 158 } | 153 } |
| 159 | 154 |
| 160 ui::ModalType SimpleMessageBoxViews::GetModalType() const { | 155 ui::ModalType SimpleMessageBoxViews::GetModalType() const { |
| 161 return ui::MODAL_TYPE_WINDOW; | 156 return ui::MODAL_TYPE_WINDOW; |
| 162 } | 157 } |
| 163 | 158 |
| 164 views::View* SimpleMessageBoxViews::GetContentsView() { | 159 views::View* SimpleMessageBoxViews::GetContentsView() { |
| 165 return message_box_view_; | 160 return message_box_view_; |
| 166 } | 161 } |
| 167 | 162 |
| 168 views::Widget* SimpleMessageBoxViews::GetWidget() { | 163 views::Widget* SimpleMessageBoxViews::GetWidget() { |
| 169 return message_box_view_->GetWidget(); | 164 return message_box_view_->GetWidget(); |
| 170 } | 165 } |
| 171 | 166 |
| 172 const views::Widget* SimpleMessageBoxViews::GetWidget() const { | 167 const views::Widget* SimpleMessageBoxViews::GetWidget() const { |
| 173 return message_box_view_->GetWidget(); | 168 return message_box_view_->GetWidget(); |
| 174 } | 169 } |
| 175 | 170 |
| 176 uint32_t SimpleMessageBoxViews::Dispatch(const base::NativeEvent& event) { | |
| 177 uint32_t action = POST_DISPATCH_PERFORM_DEFAULT; | |
| 178 if (!should_show_dialog_) | |
| 179 action |= POST_DISPATCH_QUIT_LOOP; | |
| 180 return action; | |
| 181 } | |
| 182 | |
| 183 //////////////////////////////////////////////////////////////////////////////// | 171 //////////////////////////////////////////////////////////////////////////////// |
| 184 // SimpleMessageBoxViews, private: | 172 // SimpleMessageBoxViews, private: |
| 185 | 173 |
| 186 SimpleMessageBoxViews::~SimpleMessageBoxViews() { | 174 SimpleMessageBoxViews::~SimpleMessageBoxViews() { |
| 187 } | 175 } |
| 188 | 176 |
| 189 #if defined(OS_WIN) | 177 #if defined(OS_WIN) |
| 190 UINT GetMessageBoxFlagsFromType(MessageBoxType type) { | 178 UINT GetMessageBoxFlagsFromType(MessageBoxType type) { |
| 191 UINT flags = MB_SETFOREGROUND; | 179 UINT flags = MB_SETFOREGROUND; |
| 192 switch (type) { | 180 switch (type) { |
| 193 case MESSAGE_BOX_TYPE_INFORMATION: | 181 case MESSAGE_BOX_TYPE_INFORMATION: |
| 194 return flags | MB_OK | MB_ICONINFORMATION; | 182 return flags | MB_OK | MB_ICONINFORMATION; |
| 195 case MESSAGE_BOX_TYPE_WARNING: | 183 case MESSAGE_BOX_TYPE_WARNING: |
| 196 return flags | MB_OK | MB_ICONWARNING; | 184 return flags | MB_OK | MB_ICONWARNING; |
| 197 case MESSAGE_BOX_TYPE_QUESTION: | 185 case MESSAGE_BOX_TYPE_QUESTION: |
| 198 return flags | MB_YESNO | MB_ICONQUESTION; | 186 return flags | MB_YESNO | MB_ICONQUESTION; |
| 199 case MESSAGE_BOX_TYPE_OK_CANCEL: | 187 case MESSAGE_BOX_TYPE_OK_CANCEL: |
| 200 return flags | MB_OKCANCEL | MB_ICONWARNING; | 188 return flags | MB_OKCANCEL | MB_ICONWARNING; |
| 201 } | 189 } |
| 202 NOTREACHED(); | 190 NOTREACHED(); |
| 203 return flags | MB_OK | MB_ICONWARNING; | 191 return flags | MB_OK | MB_ICONWARNING; |
| 204 } | 192 } |
| 205 #endif | 193 #endif |
| 206 | 194 |
| 195 void SimpleMessageBoxViews::Done() { |
| 196 // The nested-loop is started from the parent's dispatcher client (if there is |
| 197 // one), otherwise it's started from the widget's own window's dispatcher |
| 198 // client. Make sure the termination of the nested loop happens from the |
| 199 // correct dispatcher client. |
| 200 aura::Window* window = GetWidget()->GetNativeView(); |
| 201 aura::Window* parent = views::corewm::GetTransientParent(window); |
| 202 aura::client::DispatcherClient* client = NULL; |
| 203 if (parent) |
| 204 client = aura::client::GetDispatcherClient(parent->GetRootWindow()); |
| 205 if (!client) |
| 206 client = aura::client::GetDispatcherClient(window->GetRootWindow()); |
| 207 client->QuitNestedMessageLoop(); |
| 208 } |
| 209 |
| 207 MessageBoxResult ShowMessageBoxImpl(gfx::NativeWindow parent, | 210 MessageBoxResult ShowMessageBoxImpl(gfx::NativeWindow parent, |
| 208 const base::string16& title, | 211 const base::string16& title, |
| 209 const base::string16& message, | 212 const base::string16& message, |
| 210 MessageBoxType type, | 213 MessageBoxType type, |
| 211 const base::string16& yes_text, | 214 const base::string16& yes_text, |
| 212 const base::string16& no_text) { | 215 const base::string16& no_text) { |
| 213 #if defined(OS_WIN) | 216 #if defined(OS_WIN) |
| 214 // GPU-based dialogs can't be used early on; fallback to a Windows MessageBox. | 217 // GPU-based dialogs can't be used early on; fallback to a Windows MessageBox. |
| 215 if (!ui::ContextFactory::GetInstance()) { | 218 if (!ui::ContextFactory::GetInstance()) { |
| 216 int result = ui::MessageBox(views::HWNDForNativeWindow(parent), message, | 219 int result = ui::MessageBox(views::HWNDForNativeWindow(parent), message, |
| (...skipping 10 matching lines...) Expand all Loading... |
| 227 aura::Window* anchor = parent; | 230 aura::Window* anchor = parent; |
| 228 aura::client::DispatcherClient* client = anchor ? | 231 aura::client::DispatcherClient* client = anchor ? |
| 229 aura::client::GetDispatcherClient(anchor->GetRootWindow()) : NULL; | 232 aura::client::GetDispatcherClient(anchor->GetRootWindow()) : NULL; |
| 230 if (!client) { | 233 if (!client) { |
| 231 // Use the widget's window itself so that the message loop | 234 // Use the widget's window itself so that the message loop |
| 232 // exists when the dialog is closed by some other means than | 235 // exists when the dialog is closed by some other means than |
| 233 // |Cancel| or |Accept|. | 236 // |Cancel| or |Accept|. |
| 234 anchor = dialog->GetWidget()->GetNativeWindow(); | 237 anchor = dialog->GetWidget()->GetNativeWindow(); |
| 235 client = aura::client::GetDispatcherClient(anchor->GetRootWindow()); | 238 client = aura::client::GetDispatcherClient(anchor->GetRootWindow()); |
| 236 } | 239 } |
| 237 client->RunWithDispatcher(dialog.get(), anchor); | 240 client->RunWithDispatcher(NULL, anchor); |
| 238 return dialog->result(); | 241 return dialog->result(); |
| 239 } | 242 } |
| 240 | 243 |
| 241 } // namespace | 244 } // namespace |
| 242 | 245 |
| 243 MessageBoxResult ShowMessageBox(gfx::NativeWindow parent, | 246 MessageBoxResult ShowMessageBox(gfx::NativeWindow parent, |
| 244 const base::string16& title, | 247 const base::string16& title, |
| 245 const base::string16& message, | 248 const base::string16& message, |
| 246 MessageBoxType type) { | 249 MessageBoxType type) { |
| 247 return ShowMessageBoxImpl( | 250 return ShowMessageBoxImpl( |
| 248 parent, title, message, type, base::string16(), base::string16()); | 251 parent, title, message, type, base::string16(), base::string16()); |
| 249 } | 252 } |
| 250 | 253 |
| 251 MessageBoxResult ShowMessageBoxWithButtonText(gfx::NativeWindow parent, | 254 MessageBoxResult ShowMessageBoxWithButtonText(gfx::NativeWindow parent, |
| 252 const base::string16& title, | 255 const base::string16& title, |
| 253 const base::string16& message, | 256 const base::string16& message, |
| 254 const base::string16& yes_text, | 257 const base::string16& yes_text, |
| 255 const base::string16& no_text) { | 258 const base::string16& no_text) { |
| 256 return ShowMessageBoxImpl( | 259 return ShowMessageBoxImpl( |
| 257 parent, title, message, MESSAGE_BOX_TYPE_QUESTION, yes_text, no_text); | 260 parent, title, message, MESSAGE_BOX_TYPE_QUESTION, yes_text, no_text); |
| 258 } | 261 } |
| 259 | 262 |
| 260 } // namespace chrome | 263 } // namespace chrome |
| OLD | NEW |