Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(164)

Side by Side Diff: android_webview/native/aw_web_contents_delegate.cc

Issue 2863233002: [WebView] Move files from native to browser (Closed)
Patch Set: Created 3 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
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
3 // found in the LICENSE file.
4
5 #include "android_webview/native/aw_web_contents_delegate.h"
6
7 #include "android_webview/browser/aw_javascript_dialog_manager.h"
8 #include "android_webview/browser/find_helper.h"
9 #include "android_webview/native/aw_contents.h"
10 #include "android_webview/native/aw_contents_io_thread_client_impl.h"
11 #include "android_webview/native/permission/media_access_permission_request.h"
12 #include "android_webview/native/permission/permission_request_handler.h"
13 #include "base/android/jni_array.h"
14 #include "base/android/jni_string.h"
15 #include "base/android/scoped_java_ref.h"
16 #include "base/lazy_instance.h"
17 #include "base/location.h"
18 #include "base/memory/ptr_util.h"
19 #include "base/single_thread_task_runner.h"
20 #include "base/strings/string_util.h"
21 #include "base/strings/utf_string_conversions.h"
22 #include "base/threading/thread_task_runner_handle.h"
23 #include "content/public/browser/render_frame_host.h"
24 #include "content/public/browser/render_process_host.h"
25 #include "content/public/browser/render_view_host.h"
26 #include "content/public/browser/render_widget_host.h"
27 #include "content/public/browser/web_contents.h"
28 #include "content/public/common/file_chooser_file_info.h"
29 #include "content/public/common/file_chooser_params.h"
30 #include "content/public/common/media_stream_request.h"
31 #include "jni/AwWebContentsDelegate_jni.h"
32 #include "net/base/escape.h"
33
34 using base::android::AttachCurrentThread;
35 using base::android::ConvertUTF16ToJavaString;
36 using base::android::ConvertUTF8ToJavaString;
37 using base::android::JavaParamRef;
38 using base::android::ScopedJavaLocalRef;
39 using content::FileChooserParams;
40 using content::WebContents;
41
42 namespace android_webview {
43
44 namespace {
45
46 // WARNING: these constants are exposed in the public interface Java side, so
47 // must remain in sync with what clients are expecting.
48 const int kFileChooserModeOpenMultiple = 1 << 0;
49 const int kFileChooserModeOpenFolder = 1 << 1;
50
51 base::LazyInstance<AwJavaScriptDialogManager>::Leaky
52 g_javascript_dialog_manager = LAZY_INSTANCE_INITIALIZER;
53 }
54
55 AwWebContentsDelegate::AwWebContentsDelegate(
56 JNIEnv* env,
57 jobject obj)
58 : WebContentsDelegateAndroid(env, obj),
59 is_fullscreen_(false) {
60 }
61
62 AwWebContentsDelegate::~AwWebContentsDelegate() {
63 }
64
65 content::JavaScriptDialogManager*
66 AwWebContentsDelegate::GetJavaScriptDialogManager(WebContents* source) {
67 return g_javascript_dialog_manager.Pointer();
68 }
69
70 void AwWebContentsDelegate::FindReply(WebContents* web_contents,
71 int request_id,
72 int number_of_matches,
73 const gfx::Rect& selection_rect,
74 int active_match_ordinal,
75 bool final_update) {
76 AwContents* aw_contents = AwContents::FromWebContents(web_contents);
77 if (!aw_contents)
78 return;
79
80 aw_contents->GetFindHelper()->HandleFindReply(request_id,
81 number_of_matches,
82 active_match_ordinal,
83 final_update);
84 }
85
86 void AwWebContentsDelegate::CanDownload(
87 const GURL& url,
88 const std::string& request_method,
89 const base::Callback<void(bool)>& callback) {
90 // Android webview intercepts download in its resource dispatcher host
91 // delegate, so should not reach here.
92 NOTREACHED();
93 callback.Run(false);
94 }
95
96 void AwWebContentsDelegate::RunFileChooser(
97 content::RenderFrameHost* render_frame_host,
98 const FileChooserParams& params) {
99 JNIEnv* env = AttachCurrentThread();
100 ScopedJavaLocalRef<jobject> java_delegate = GetJavaDelegate(env);
101 if (!java_delegate.obj())
102 return;
103
104 int mode_flags = 0;
105 if (params.mode == FileChooserParams::OpenMultiple) {
106 mode_flags |= kFileChooserModeOpenMultiple;
107 } else if (params.mode == FileChooserParams::UploadFolder) {
108 // Folder implies multiple in Chrome.
109 mode_flags |= kFileChooserModeOpenMultiple | kFileChooserModeOpenFolder;
110 } else if (params.mode == FileChooserParams::Save) {
111 // Save not supported, so cancel it.
112 render_frame_host->FilesSelectedInChooser(
113 std::vector<content::FileChooserFileInfo>(), params.mode);
114 return;
115 } else {
116 DCHECK_EQ(FileChooserParams::Open, params.mode);
117 }
118 Java_AwWebContentsDelegate_runFileChooser(
119 env, java_delegate, render_frame_host->GetProcess()->GetID(),
120 render_frame_host->GetRoutingID(), mode_flags,
121 ConvertUTF16ToJavaString(
122 env, base::JoinString(params.accept_types, base::ASCIIToUTF16(","))),
123 params.title.empty() ? nullptr
124 : ConvertUTF16ToJavaString(env, params.title),
125 params.default_file_name.empty()
126 ? nullptr
127 : ConvertUTF8ToJavaString(env, params.default_file_name.value()),
128 params.capture);
129 }
130
131 void AwWebContentsDelegate::AddNewContents(WebContents* source,
132 WebContents* new_contents,
133 WindowOpenDisposition disposition,
134 const gfx::Rect& initial_rect,
135 bool user_gesture,
136 bool* was_blocked) {
137 JNIEnv* env = AttachCurrentThread();
138
139 bool is_dialog = disposition == WindowOpenDisposition::NEW_POPUP;
140 ScopedJavaLocalRef<jobject> java_delegate = GetJavaDelegate(env);
141 bool create_popup = false;
142
143 if (java_delegate.obj()) {
144 create_popup = Java_AwWebContentsDelegate_addNewContents(
145 env, java_delegate, is_dialog, user_gesture);
146 }
147
148 if (create_popup) {
149 // The embedder would like to display the popup and we will receive
150 // a callback from them later with an AwContents to use to display
151 // it. The source AwContents takes ownership of the new WebContents
152 // until then, and when the callback is made we will swap the WebContents
153 // out into the new AwContents.
154 AwContents::FromWebContents(source)->SetPendingWebContentsForPopup(
155 base::WrapUnique(new_contents));
156 // Hide the WebContents for the pop up now, we will show it again
157 // when the user calls us back with an AwContents to use to show it.
158 new_contents->WasHidden();
159 } else {
160 // The embedder has forgone their chance to display this popup
161 // window, so we're done with the WebContents now. We use
162 // DeleteSoon as WebContentsImpl may call methods on |new_contents|
163 // after this method returns.
164 base::ThreadTaskRunnerHandle::Get()->DeleteSoon(FROM_HERE, new_contents);
165 }
166
167 if (was_blocked) {
168 *was_blocked = !create_popup;
169 }
170 }
171
172 void AwWebContentsDelegate::NavigationStateChanged(
173 content::WebContents* source,
174 content::InvalidateTypes changed_flags) {
175 JNIEnv* env = AttachCurrentThread();
176
177 ScopedJavaLocalRef<jobject> java_delegate = GetJavaDelegate(env);
178 if (java_delegate.obj()) {
179 Java_AwWebContentsDelegate_navigationStateChanged(env, java_delegate,
180 changed_flags);
181 }
182 }
183
184 // Notifies the delegate about the creation of a new WebContents. This
185 // typically happens when popups are created.
186 void AwWebContentsDelegate::WebContentsCreated(
187 WebContents* source_contents,
188 int opener_render_process_id,
189 int opener_render_frame_id,
190 const std::string& frame_name,
191 const GURL& target_url,
192 content::WebContents* new_contents) {
193 AwContentsIoThreadClientImpl::RegisterPendingContents(new_contents);
194 }
195
196 void AwWebContentsDelegate::CloseContents(WebContents* source) {
197 JNIEnv* env = AttachCurrentThread();
198
199 ScopedJavaLocalRef<jobject> java_delegate = GetJavaDelegate(env);
200 if (java_delegate.obj()) {
201 Java_AwWebContentsDelegate_closeContents(env, java_delegate);
202 }
203 }
204
205 void AwWebContentsDelegate::ActivateContents(WebContents* contents) {
206 JNIEnv* env = AttachCurrentThread();
207
208 ScopedJavaLocalRef<jobject> java_delegate = GetJavaDelegate(env);
209 if (java_delegate.obj()) {
210 Java_AwWebContentsDelegate_activateContents(env, java_delegate);
211 }
212 }
213
214 void AwWebContentsDelegate::LoadingStateChanged(WebContents* source,
215 bool to_different_document) {
216 // Page title may have changed, need to inform the embedder.
217 // |source| may be null if loading has started.
218 JNIEnv* env = AttachCurrentThread();
219
220 ScopedJavaLocalRef<jobject> java_delegate = GetJavaDelegate(env);
221 if (java_delegate.obj()) {
222 Java_AwWebContentsDelegate_loadingStateChanged(env, java_delegate);
223 }
224 }
225
226 bool AwWebContentsDelegate::ShouldResumeRequestsForCreatedWindow() {
227 // Always return false here since we need to defer loading the created window
228 // until after we have attached a new delegate to the new webcontents (which
229 // happens asynchronously).
230 return false;
231 }
232
233 void AwWebContentsDelegate::RequestMediaAccessPermission(
234 WebContents* web_contents,
235 const content::MediaStreamRequest& request,
236 const content::MediaResponseCallback& callback) {
237 AwContents* aw_contents = AwContents::FromWebContents(web_contents);
238 if (!aw_contents) {
239 callback.Run(content::MediaStreamDevices(),
240 content::MEDIA_DEVICE_FAILED_DUE_TO_SHUTDOWN,
241 std::unique_ptr<content::MediaStreamUI>());
242 return;
243 }
244 aw_contents->GetPermissionRequestHandler()->SendRequest(
245 std::unique_ptr<AwPermissionRequestDelegate>(
246 new MediaAccessPermissionRequest(request, callback)));
247 }
248
249 void AwWebContentsDelegate::EnterFullscreenModeForTab(
250 content::WebContents* web_contents, const GURL& origin) {
251 WebContentsDelegateAndroid::EnterFullscreenModeForTab(web_contents, origin);
252 is_fullscreen_ = true;
253 web_contents->GetRenderViewHost()->GetWidget()->WasResized();
254 }
255
256 void AwWebContentsDelegate::ExitFullscreenModeForTab(
257 content::WebContents* web_contents) {
258 WebContentsDelegateAndroid::ExitFullscreenModeForTab(web_contents);
259 is_fullscreen_ = false;
260 web_contents->GetRenderViewHost()->GetWidget()->WasResized();
261 }
262
263 bool AwWebContentsDelegate::IsFullscreenForTabOrPending(
264 const content::WebContents* web_contents) const {
265 return is_fullscreen_;
266 }
267
268 static void FilesSelectedInChooser(
269 JNIEnv* env,
270 const JavaParamRef<jclass>& clazz,
271 jint process_id,
272 jint render_id,
273 jint mode_flags,
274 const JavaParamRef<jobjectArray>& file_paths,
275 const JavaParamRef<jobjectArray>& display_names) {
276 content::RenderFrameHost* rfh =
277 content::RenderFrameHost::FromID(process_id, render_id);
278 if (!rfh)
279 return;
280
281 std::vector<std::string> file_path_str;
282 std::vector<std::string> display_name_str;
283 // Note file_paths maybe NULL, but this will just yield a zero-length vector.
284 base::android::AppendJavaStringArrayToStringVector(env, file_paths,
285 &file_path_str);
286 base::android::AppendJavaStringArrayToStringVector(env, display_names,
287 &display_name_str);
288 std::vector<content::FileChooserFileInfo> files;
289 files.reserve(file_path_str.size());
290 for (size_t i = 0; i < file_path_str.size(); ++i) {
291 GURL url(file_path_str[i]);
292 if (!url.is_valid())
293 continue;
294 base::FilePath path(url.SchemeIsFile() ?
295 net::UnescapeURLComponent(url.path(),
296 net::UnescapeRule::SPACES |
297 net::UnescapeRule::URL_SPECIAL_CHARS_EXCEPT_PATH_SEPARATORS) :
298 file_path_str[i]);
299 content::FileChooserFileInfo file_info;
300 file_info.file_path = path;
301 if (!display_name_str[i].empty())
302 file_info.display_name = display_name_str[i];
303 files.push_back(file_info);
304 }
305 FileChooserParams::Mode mode;
306 if (mode_flags & kFileChooserModeOpenFolder) {
307 mode = FileChooserParams::UploadFolder;
308 } else if (mode_flags & kFileChooserModeOpenMultiple) {
309 mode = FileChooserParams::OpenMultiple;
310 } else {
311 mode = FileChooserParams::Open;
312 }
313 DVLOG(0) << "File Chooser result: mode = " << mode
314 << ", file paths = " << base::JoinString(file_path_str, ":");
315 rfh->FilesSelectedInChooser(files, mode);
316 }
317
318 bool RegisterAwWebContentsDelegate(JNIEnv* env) {
319 return RegisterNativesImpl(env);
320 }
321
322 } // namespace android_webview
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698