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.
5 #include "content/browser/android/web_contents_observer_android.h"
11 #include "base/android/jni_android.h"
12 #include "base/android/jni_string.h"
13 #include "base/android/scoped_java_ref.h"
14 #include "content/browser/renderer_host/render_widget_host_impl.h"
15 #include "content/browser/web_contents/web_contents_impl.h"
16 #include "content/public/browser/navigation_details.h"
17 #include "content/public/browser/navigation_entry.h"
18 #include "jni/WebContentsObserverAndroid_jni.h"
20 using base::android::AttachCurrentThread;
21 using base::android::ScopedJavaLocalRef;
22 using base::android::ConvertUTF8ToJavaString;
23 using base::android::ConvertUTF16ToJavaString;
27 // TODO(dcheng): File a bug. This class incorrectly passes just a frame ID,
28 // which is not sufficient to identify a frame (since frame IDs are scoped per
29 // render process, and so may collide).
30 WebContentsObserverAndroid::WebContentsObserverAndroid(
33 WebContents* web_contents)
34 : WebContentsObserver(web_contents),
35 weak_java_observer_(env, obj){
38 WebContentsObserverAndroid::~WebContentsObserverAndroid() {
41 jlong Init(JNIEnv* env, jobject obj, jobject java_web_contents) {
42 WebContents* web_contents =
43 WebContents::FromJavaWebContents(java_web_contents);
46 WebContentsObserverAndroid* native_observer = new WebContentsObserverAndroid(
47 env, obj, web_contents);
48 return reinterpret_cast<intptr_t>(native_observer);
51 void WebContentsObserverAndroid::Destroy(JNIEnv* env, jobject obj) {
55 void WebContentsObserverAndroid::WebContentsDestroyed() {
56 JNIEnv* env = AttachCurrentThread();
57 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
61 // The java side will destroy |this|
62 Java_WebContentsObserverAndroid_detachFromWebContents(env, obj.obj());
66 void WebContentsObserverAndroid::RenderProcessGone(
67 base::TerminationStatus termination_status) {
68 JNIEnv* env = AttachCurrentThread();
69 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
72 jboolean was_oom_protected =
73 termination_status == base::TERMINATION_STATUS_OOM_PROTECTED;
74 Java_WebContentsObserverAndroid_renderProcessGone(
75 env, obj.obj(), was_oom_protected);
78 void WebContentsObserverAndroid::DidStartLoading(
79 RenderViewHost* render_view_host) {
80 JNIEnv* env = AttachCurrentThread();
81 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
84 ScopedJavaLocalRef<jstring> jstring_url(ConvertUTF8ToJavaString(
85 env, web_contents()->GetVisibleURL().spec()));
86 Java_WebContentsObserverAndroid_didStartLoading(
87 env, obj.obj(), jstring_url.obj());
90 void WebContentsObserverAndroid::DidStopLoading(
91 RenderViewHost* render_view_host) {
92 JNIEnv* env = AttachCurrentThread();
93 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
96 ScopedJavaLocalRef<jstring> jstring_url(ConvertUTF8ToJavaString(
97 env, web_contents()->GetLastCommittedURL().spec()));
98 Java_WebContentsObserverAndroid_didStopLoading(
99 env, obj.obj(), jstring_url.obj());
102 void WebContentsObserverAndroid::DidFailProvisionalLoad(
103 RenderFrameHost* render_frame_host,
104 const GURL& validated_url,
106 const base::string16& error_description) {
107 DidFailLoadInternal(true,
108 !render_frame_host->GetParent(),
114 void WebContentsObserverAndroid::DidFailLoad(
115 RenderFrameHost* render_frame_host,
116 const GURL& validated_url,
118 const base::string16& error_description) {
119 DidFailLoadInternal(false,
120 !render_frame_host->GetParent(),
126 void WebContentsObserverAndroid::DidNavigateMainFrame(
127 const LoadCommittedDetails& details,
128 const FrameNavigateParams& params) {
129 JNIEnv* env = AttachCurrentThread();
130 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
133 ScopedJavaLocalRef<jstring> jstring_url(
134 ConvertUTF8ToJavaString(env, params.url.spec()));
135 ScopedJavaLocalRef<jstring> jstring_base_url(
136 ConvertUTF8ToJavaString(env, params.base_url.spec()));
138 // See http://crbug.com/251330 for why it's determined this way.
139 url::Replacements<char> replacements;
140 replacements.ClearRef();
141 bool urls_same_ignoring_fragment =
142 params.url.ReplaceComponents(replacements) ==
143 details.previous_url.ReplaceComponents(replacements);
145 // is_fragment_navigation is indicative of the intent of this variable.
146 // However, there isn't sufficient information here to determine whether this
147 // is actually a fragment navigation, or a history API navigation to a URL
148 // that would also be valid for a fragment navigation.
149 bool is_fragment_navigation = urls_same_ignoring_fragment &&
150 (details.type == NAVIGATION_TYPE_IN_PAGE || details.is_in_page);
151 Java_WebContentsObserverAndroid_didNavigateMainFrame(
152 env, obj.obj(), jstring_url.obj(), jstring_base_url.obj(),
153 details.is_navigation_to_different_page(), is_fragment_navigation,
154 details.http_status_code);
157 void WebContentsObserverAndroid::DidNavigateAnyFrame(
158 const LoadCommittedDetails& details,
159 const FrameNavigateParams& params) {
160 JNIEnv* env = AttachCurrentThread();
161 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
164 ScopedJavaLocalRef<jstring> jstring_url(
165 ConvertUTF8ToJavaString(env, params.url.spec()));
166 ScopedJavaLocalRef<jstring> jstring_base_url(
167 ConvertUTF8ToJavaString(env, params.base_url.spec()));
168 jboolean jboolean_is_reload = ui::PageTransitionCoreTypeIs(
169 params.transition, ui::PAGE_TRANSITION_RELOAD);
171 Java_WebContentsObserverAndroid_didNavigateAnyFrame(
172 env, obj.obj(), jstring_url.obj(), jstring_base_url.obj(),
176 void WebContentsObserverAndroid::DidStartProvisionalLoadForFrame(
177 RenderFrameHost* render_frame_host,
178 const GURL& validated_url,
180 bool is_iframe_srcdoc) {
181 JNIEnv* env = AttachCurrentThread();
182 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
185 ScopedJavaLocalRef<jstring> jstring_url(
186 ConvertUTF8ToJavaString(env, validated_url.spec()));
187 // TODO(dcheng): Does Java really need the parent frame ID? It doesn't appear
188 // to be used at all, and it just adds complexity here.
189 Java_WebContentsObserverAndroid_didStartProvisionalLoadForFrame(
192 render_frame_host->GetRoutingID(),
193 render_frame_host->GetParent()
194 ? render_frame_host->GetParent()->GetRoutingID()
196 !render_frame_host->GetParent(),
202 void WebContentsObserverAndroid::DidCommitProvisionalLoadForFrame(
203 RenderFrameHost* render_frame_host,
205 ui::PageTransition transition_type) {
206 JNIEnv* env = AttachCurrentThread();
207 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
210 ScopedJavaLocalRef<jstring> jstring_url(
211 ConvertUTF8ToJavaString(env, url.spec()));
212 Java_WebContentsObserverAndroid_didCommitProvisionalLoadForFrame(
215 render_frame_host->GetRoutingID(),
216 !render_frame_host->GetParent(),
221 void WebContentsObserverAndroid::DidFinishLoad(
222 RenderFrameHost* render_frame_host,
223 const GURL& validated_url) {
224 JNIEnv* env = AttachCurrentThread();
225 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
229 std::string url_string = validated_url.spec();
230 NavigationEntry* entry =
231 web_contents()->GetController().GetLastCommittedEntry();
232 // Note that GetBaseURLForDataURL is only used by the Android WebView.
233 if (entry && !entry->GetBaseURLForDataURL().is_empty())
234 url_string = entry->GetBaseURLForDataURL().possibly_invalid_spec();
236 ScopedJavaLocalRef<jstring> jstring_url(
237 ConvertUTF8ToJavaString(env, url_string));
238 Java_WebContentsObserverAndroid_didFinishLoad(
241 render_frame_host->GetRoutingID(),
243 !render_frame_host->GetParent());
246 void WebContentsObserverAndroid::DocumentLoadedInFrame(
247 RenderFrameHost* render_frame_host) {
248 JNIEnv* env = AttachCurrentThread();
249 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
252 Java_WebContentsObserverAndroid_documentLoadedInFrame(
253 env, obj.obj(), render_frame_host->GetRoutingID());
256 void WebContentsObserverAndroid::NavigationEntryCommitted(
257 const LoadCommittedDetails& load_details) {
258 JNIEnv* env = AttachCurrentThread();
259 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
262 Java_WebContentsObserverAndroid_navigationEntryCommitted(env, obj.obj());
265 void WebContentsObserverAndroid::DidAttachInterstitialPage() {
266 JNIEnv* env = AttachCurrentThread();
267 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
270 Java_WebContentsObserverAndroid_didAttachInterstitialPage(env, obj.obj());
273 void WebContentsObserverAndroid::DidDetachInterstitialPage() {
274 JNIEnv* env = AttachCurrentThread();
275 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
278 Java_WebContentsObserverAndroid_didDetachInterstitialPage(env, obj.obj());
281 void WebContentsObserverAndroid::DidChangeThemeColor(SkColor color) {
282 JNIEnv* env = AttachCurrentThread();
283 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
286 Java_WebContentsObserverAndroid_didChangeThemeColor(env, obj.obj(), color);
289 void WebContentsObserverAndroid::DidFailLoadInternal(
290 bool is_provisional_load,
293 const base::string16& description,
295 JNIEnv* env = AttachCurrentThread();
296 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
299 ScopedJavaLocalRef<jstring> jstring_error_description(
300 ConvertUTF16ToJavaString(env, description));
301 ScopedJavaLocalRef<jstring> jstring_url(
302 ConvertUTF8ToJavaString(env, url.spec()));
304 Java_WebContentsObserverAndroid_didFailLoad(
309 jstring_error_description.obj(), jstring_url.obj());
312 void WebContentsObserverAndroid::DidFirstVisuallyNonEmptyPaint() {
313 JNIEnv* env = AttachCurrentThread();
314 ScopedJavaLocalRef<jobject> obj(weak_java_observer_.get(env));
317 Java_WebContentsObserverAndroid_didFirstVisuallyNonEmptyPaint(
321 bool RegisterWebContentsObserverAndroid(JNIEnv* env) {
322 return RegisterNativesImpl(env);
324 } // namespace content