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
|
// Copyright (c) 2010 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/js_modal_dialog.h"
#include "chrome/browser/extensions/extension_host.h"
#include "chrome/browser/tab_contents/tab_contents.h"
#include "chrome/common/notification_service.h"
#include "chrome/common/notification_type.h"
#include "ipc/ipc_message.h"
JavaScriptAppModalDialog::JavaScriptAppModalDialog(
JavaScriptMessageBoxClient* client,
const std::wstring& title,
int dialog_flags,
const std::wstring& message_text,
const std::wstring& default_prompt_text,
bool display_suppress_checkbox,
bool is_before_unload_dialog,
IPC::Message* reply_msg)
: AppModalDialog(client->AsTabContents(), title),
client_(client),
extension_host_(client->AsExtensionHost()),
dialog_flags_(dialog_flags),
message_text_(message_text),
default_prompt_text_(default_prompt_text),
display_suppress_checkbox_(display_suppress_checkbox),
is_before_unload_dialog_(is_before_unload_dialog),
reply_msg_(reply_msg) {
DCHECK((tab_contents_ != NULL) != (extension_host_ != NULL));
InitNotifications();
}
JavaScriptAppModalDialog::~JavaScriptAppModalDialog() {
}
void JavaScriptAppModalDialog::Observe(NotificationType type,
const NotificationSource& source,
const NotificationDetails& details) {
if (skip_this_dialog_)
return;
if (NotificationType::EXTENSION_HOST_DESTROYED == type &&
Details<ExtensionHost>(extension_host_) != details)
return;
// If we reach here, we know the notification is relevant to us, either
// because we're only observing applicable sources or because we passed the
// check above. Both of those indicate that we should ignore this dialog.
// Also clear the client, since it's now invalid.
skip_this_dialog_ = true;
client_ = NULL;
CloseModalDialog();
}
void JavaScriptAppModalDialog::InitNotifications() {
// Make sure we get relevant navigation notifications so we know when our
// parent contents will disappear or navigate to a different page.
if (tab_contents_) {
registrar_.Add(this, NotificationType::NAV_ENTRY_COMMITTED,
Source<NavigationController>(&tab_contents_->controller()));
registrar_.Add(this, NotificationType::TAB_CONTENTS_DESTROYED,
Source<TabContents>(tab_contents_));
} else if (extension_host_) {
// EXTENSION_HOST_DESTROYED uses the Profile as its source, but we care
// about the ExtensionHost (which is passed in the details).
registrar_.Add(this, NotificationType::EXTENSION_HOST_DESTROYED,
NotificationService::AllSources());
} else {
NOTREACHED();
}
}
void JavaScriptAppModalDialog::OnCancel() {
// We need to do this before WM_DESTROY (WindowClosing()) as any parent frame
// will receive its activation messages before this dialog receives
// WM_DESTROY. The parent frame would then try to activate any modal dialogs
// that were still open in the ModalDialogQueue, which would send activation
// back to this one. The framework should be improved to handle this, so this
// is a temporary workaround.
CompleteDialog();
if (!skip_this_dialog_) {
client_->OnMessageBoxClosed(reply_msg_, false, std::wstring());
}
Cleanup();
}
void JavaScriptAppModalDialog::OnAccept(const std::wstring& prompt_text,
bool suppress_js_messages) {
CompleteDialog();
if (!skip_this_dialog_) {
client_->OnMessageBoxClosed(reply_msg_, true, prompt_text);
if (suppress_js_messages)
client_->SetSuppressMessageBoxes(true);
}
Cleanup();
}
void JavaScriptAppModalDialog::OnClose() {
Cleanup();
}
void JavaScriptAppModalDialog::Cleanup() {
if (skip_this_dialog_) {
// We can't use the client_, because we might be in the process of
// destroying it.
if (tab_contents_)
tab_contents_->OnMessageBoxClosed(reply_msg_, false, L"");
// The extension_host_ will always be a dirty pointer on OS X because the alert
// window will cause the extension popup to close since it is resigning its key
// state, destroying the host. http://crbug.com/29355
#if !defined(OS_MACOSX)
else if (extension_host_)
extension_host_->OnMessageBoxClosed(reply_msg_, false, L"");
else
NOTREACHED();
#endif
}
AppModalDialog::Cleanup();
}
|