// Copyright (c) 2012 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 "android_webview/native/aw_contents.h" #include "android_webview/browser/aw_browser_main_parts.h" #include "android_webview/browser/net_disk_cache_remover.h" #include "android_webview/browser/renderer_host/aw_render_view_host_ext.h" #include "android_webview/browser/renderer_host/aw_resource_dispatcher_host_delegate.h" #include "android_webview/common/aw_hit_test_data.h" #include "android_webview/native/aw_browser_dependency_factory.h" #include "android_webview/native/aw_contents_io_thread_client_impl.h" #include "android_webview/native/aw_web_contents_delegate.h" #include "android_webview/native/state_serializer.h" #include "base/android/jni_android.h" #include "base/android/jni_array.h" #include "base/android/jni_string.h" #include "base/bind.h" #include "base/callback.h" #include "base/message_loop.h" #include "base/pickle.h" #include "base/supports_user_data.h" #include "cc/layer.h" #include "content/components/navigation_interception/intercept_navigation_delegate.h" #include "content/public/browser/android/content_view_core.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/cert_store.h" #include "content/public/browser/navigation_entry.h" #include "content/public/browser/render_process_host.h" #include "content/public/browser/web_contents.h" #include "content/public/common/ssl_status.h" #include "jni/AwContents_jni.h" #include "net/base/x509_certificate.h" #include "ui/gfx/transform.h" using base::android::AttachCurrentThread; using base::android::ConvertJavaStringToUTF16; using base::android::ConvertJavaStringToUTF8; using base::android::ConvertUTF16ToJavaString; using base::android::ConvertUTF8ToJavaString; using base::android::JavaRef; using base::android::ScopedJavaGlobalRef; using base::android::ScopedJavaLocalRef; using content::BrowserThread; using content::ContentViewCore; using content::InterceptNavigationDelegate; using content::WebContents; extern "C" { static AwDrawGLFunction DrawGLFunction; static void DrawGLFunction(int view_context, AwDrawGLInfo* draw_info, void* spare) { // |view_context| is the value that was returned from the java // AwContents.onPrepareDrawGL; this cast must match the code there. reinterpret_cast(view_context)->DrawGL( draw_info); } } namespace android_webview { namespace { const void* kAwContentsUserDataKey = &kAwContentsUserDataKey; class AwContentsUserData : public base::SupportsUserData::Data { public: AwContentsUserData(AwContents* ptr) : contents_(ptr) {} static AwContents* GetContents(WebContents* web_contents) { if (!web_contents) return NULL; AwContentsUserData* data = reinterpret_cast( web_contents->GetUserData(kAwContentsUserDataKey)); return data ? data->contents_ : NULL; } private: AwContents* contents_; }; // Work-around for http://crbug.com/161864. TODO(joth): Remove this class when // that bug is closed. class NullCompositor : public content::Compositor { public: NullCompositor() {} virtual ~NullCompositor() {} // Compositor virtual void SetRootLayer(scoped_refptr root) OVERRIDE {} virtual void SetWindowBounds(const gfx::Size& size) OVERRIDE {} virtual void SetVisible(bool visible) OVERRIDE {} virtual void SetWindowSurface(ANativeWindow* window) OVERRIDE {} virtual bool CompositeAndReadback(void *pixels, const gfx::Rect& rect) OVERRIDE { return false; } virtual void Composite() {} virtual WebKit::WebGLId GenerateTexture(gfx::JavaBitmap& bitmap) OVERRIDE { return 0; } virtual WebKit::WebGLId GenerateCompressedTexture(gfx::Size& size, int data_size, void* data) OVERRIDE { return 0; } virtual void DeleteTexture(WebKit::WebGLId texture_id) OVERRIDE {} virtual void CopyTextureToBitmap(WebKit::WebGLId texture_id, gfx::JavaBitmap& bitmap) OVERRIDE {} }; } // namespace // static AwContents* AwContents::FromWebContents(WebContents* web_contents) { return AwContentsUserData::GetContents(web_contents); } AwContents::AwContents(JNIEnv* env, jobject obj, jobject web_contents_delegate, bool private_browsing) : java_ref_(env, obj), web_contents_delegate_( new AwWebContentsDelegate(env, web_contents_delegate)), view_visible_(false), compositor_visible_(false), is_composite_pending_(false) { android_webview::AwBrowserDependencyFactory* dependency_factory = android_webview::AwBrowserDependencyFactory::GetInstance(); // TODO(joth): rather than create and set the WebContents here, expose the // factory method to java side and have that orchestrate the construction // order. SetWebContents(dependency_factory->CreateWebContents(private_browsing)); } void AwContents::SetWebContents(content::WebContents* web_contents) { web_contents_.reset(web_contents); web_contents_->SetUserData(kAwContentsUserDataKey, new AwContentsUserData(this)); web_contents_->SetDelegate(web_contents_delegate_.get()); render_view_host_ext_.reset(new AwRenderViewHostExt(web_contents_.get())); if (UseCompositorDirectDraw()) { compositor_.reset(content::Compositor::Create(this)); } else { LOG(WARNING) << "Running on unsupported device: using null Compositor"; compositor_.reset(new NullCompositor); } } void AwContents::SetWebContents(JNIEnv* env, jobject obj, jint new_wc) { SetWebContents(reinterpret_cast(new_wc)); } AwContents::~AwContents() { DCHECK(AwContents::FromWebContents(web_contents_.get()) == this); web_contents_->RemoveUserData(kAwContentsUserDataKey); if (find_helper_.get()) find_helper_->SetListener(NULL); } void AwContents::DrawGL(AwDrawGLInfo* draw_info) { // TODO(joth): Use the |draw_info| parameters. DLOG(INFO) << "Unimplemented AwContents::DrawGL params" << " clip_left=" << draw_info->clip_left << " clip_top=" << draw_info->clip_top << " clip_right=" << draw_info->clip_right << " clip_bottom=" << draw_info->clip_bottom; if (compositor_visible_ != view_visible_) { compositor_visible_ = view_visible_; compositor_->SetVisible(compositor_visible_); } if (!compositor_visible_ || draw_info->mode == AwDrawGLInfo::kModeProcess) return; DCHECK_EQ(draw_info->mode, AwDrawGLInfo::kModeDraw); compositor_->Composite(); is_composite_pending_ = false; } jint AwContents::GetWebContents(JNIEnv* env, jobject obj) { return reinterpret_cast(web_contents_.get()); } void AwContents::DidInitializeContentViewCore(JNIEnv* env, jobject obj, jint content_view_core) { ContentViewCore* core = reinterpret_cast(content_view_core); DCHECK(core == ContentViewCore::FromWebContents(web_contents_.get())); compositor_->SetRootLayer(core->GetLayer()); Invalidate(); } void AwContents::Destroy(JNIEnv* env, jobject obj) { delete this; } // static jint GetAwDrawGLFunction(JNIEnv* env, jclass) { return reinterpret_cast(&DrawGLFunction); } namespace { // |message| is passed as base::Owned, so it will automatically be deleted // when the callback goes out of scope. void DocumentHasImagesCallback(ScopedJavaGlobalRef* message, bool has_images) { Java_AwContents_onDocumentHasImagesResponse(AttachCurrentThread(), has_images, message->obj()); } } // namespace void AwContents::DocumentHasImages(JNIEnv* env, jobject obj, jobject message) { ScopedJavaGlobalRef* j_message = new ScopedJavaGlobalRef(); j_message->Reset(env, message); render_view_host_ext_->DocumentHasImages( base::Bind(&DocumentHasImagesCallback, base::Owned(j_message))); } namespace { void GenerateMHTMLCallback(ScopedJavaGlobalRef* callback, const FilePath& path, int64 size) { JNIEnv* env = AttachCurrentThread(); // Android files are UTF8, so the path conversion below is safe. Java_AwContents_generateMHTMLCallback( env, ConvertUTF8ToJavaString(env, path.AsUTF8Unsafe()).obj(), size, callback->obj()); } } // namespace void AwContents::GenerateMHTML(JNIEnv* env, jobject obj, jstring jpath, jobject callback) { ScopedJavaGlobalRef* j_callback = new ScopedJavaGlobalRef(); j_callback->Reset(env, callback); web_contents_->GenerateMHTML( FilePath(ConvertJavaStringToUTF8(env, jpath)), base::Bind(&GenerateMHTMLCallback, base::Owned(j_callback))); } void AwContents::RunJavaScriptDialog( content::JavaScriptMessageType message_type, const GURL& origin_url, const string16& message_text, const string16& default_prompt_text, const ScopedJavaLocalRef& js_result) { JNIEnv* env = AttachCurrentThread(); ScopedJavaLocalRef obj = java_ref_.get(env); if (obj.is_null()) return; ScopedJavaLocalRef jurl(ConvertUTF8ToJavaString( env, origin_url.spec())); ScopedJavaLocalRef jmessage(ConvertUTF16ToJavaString( env, message_text)); switch (message_type) { case content::JAVASCRIPT_MESSAGE_TYPE_ALERT: Java_AwContents_handleJsAlert( env, obj.obj(), jurl.obj(), jmessage.obj(), js_result.obj()); break; case content::JAVASCRIPT_MESSAGE_TYPE_CONFIRM: Java_AwContents_handleJsConfirm( env, obj.obj(), jurl.obj(), jmessage.obj(), js_result.obj()); break; case content::JAVASCRIPT_MESSAGE_TYPE_PROMPT: { ScopedJavaLocalRef jdefault_value( ConvertUTF16ToJavaString(env, default_prompt_text)); Java_AwContents_handleJsPrompt( env, obj.obj(), jurl.obj(), jmessage.obj(), jdefault_value.obj(), js_result.obj()); break; } default: NOTREACHED(); } } void AwContents::RunBeforeUnloadDialog( const GURL& origin_url, const string16& message_text, const ScopedJavaLocalRef& js_result) { JNIEnv* env = AttachCurrentThread(); ScopedJavaLocalRef obj = java_ref_.get(env); if (obj.is_null()) return; ScopedJavaLocalRef jurl(ConvertUTF8ToJavaString( env, origin_url.spec())); ScopedJavaLocalRef jmessage(ConvertUTF16ToJavaString( env, message_text)); Java_AwContents_handleJsBeforeUnload( env, obj.obj(), jurl.obj(), jmessage.obj(), js_result.obj()); } void AwContents::onReceivedHttpAuthRequest( const JavaRef& handler, const std::string& host, const std::string& realm) { JNIEnv* env = AttachCurrentThread(); ScopedJavaLocalRef jhost = ConvertUTF8ToJavaString(env, host); ScopedJavaLocalRef jrealm = ConvertUTF8ToJavaString(env, realm); Java_AwContents_onReceivedHttpAuthRequest(env, java_ref_.get(env).obj(), handler.obj(), jhost.obj(), jrealm.obj()); } void AwContents::SetIoThreadClient(JNIEnv* env, jobject obj, jobject client) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); AwContentsIoThreadClientImpl::Associate( web_contents_.get(), ScopedJavaLocalRef(env, client)); int child_id = web_contents_->GetRenderProcessHost()->GetID(); int route_id = web_contents_->GetRoutingID(); AwResourceDispatcherHostDelegate::OnIoThreadClientReady(child_id, route_id); } void AwContents::SetInterceptNavigationDelegate(JNIEnv* env, jobject obj, jobject delegate) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); InterceptNavigationDelegate::Associate( web_contents_.get(), make_scoped_ptr(new InterceptNavigationDelegate(env, delegate))); } static jint Init(JNIEnv* env, jobject obj, jobject web_contents_delegate, jboolean private_browsing) { AwContents* tab = new AwContents(env, obj, web_contents_delegate, private_browsing); return reinterpret_cast(tab); } bool RegisterAwContents(JNIEnv* env) { return RegisterNativesImpl(env) >= 0; } jint AwContents::FindAllSync(JNIEnv* env, jobject obj, jstring search_string) { return GetFindHelper()->FindAllSync( ConvertJavaStringToUTF16(env, search_string)); } void AwContents::FindAllAsync(JNIEnv* env, jobject obj, jstring search_string) { GetFindHelper()->FindAllAsync(ConvertJavaStringToUTF16(env, search_string)); } void AwContents::FindNext(JNIEnv* env, jobject obj, jboolean forward) { GetFindHelper()->FindNext(forward); } void AwContents::ClearMatches(JNIEnv* env, jobject obj) { GetFindHelper()->ClearMatches(); } void AwContents::ClearCache( JNIEnv* env, jobject obj, jboolean include_disk_files) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); render_view_host_ext_->ClearCache(); if (include_disk_files) { RemoveHttpDiskCache(web_contents_->GetBrowserContext(), web_contents_->GetRoutingID()); } } FindHelper* AwContents::GetFindHelper() { if (!find_helper_.get()) { find_helper_.reset(new FindHelper(web_contents_.get())); find_helper_->SetListener(this); } return find_helper_.get(); } void AwContents::OnFindResultReceived(int active_ordinal, int match_count, bool finished) { JNIEnv* env = AttachCurrentThread(); ScopedJavaLocalRef obj = java_ref_.get(env); if (obj.is_null()) return; Java_AwContents_onFindResultReceived( env, obj.obj(), active_ordinal, match_count, finished); } void AwContents::ScheduleComposite() { // TODO(joth): Call back out to framework attachFunctor (Java side) from here. Invalidate(); } void AwContents::Invalidate() { if (is_composite_pending_) return; is_composite_pending_ = true; JNIEnv* env = AttachCurrentThread(); ScopedJavaLocalRef obj = java_ref_.get(env); if (obj.is_null()) return; Java_AwContents_invalidate(env, obj.obj()); } void AwContents::OnSwapBuffersCompleted() { } base::android::ScopedJavaLocalRef AwContents::GetCertificate(JNIEnv* env, jobject obj) { content::NavigationEntry* entry = web_contents_->GetController().GetActiveEntry(); if (!entry) return ScopedJavaLocalRef(); // Get the certificate int cert_id = entry->GetSSL().cert_id; scoped_refptr cert; bool ok = content::CertStore::GetInstance()->RetrieveCert(cert_id, &cert); if (!ok) return ScopedJavaLocalRef(); // Convert the certificate and return it std::string der_string; net::X509Certificate::GetDEREncoded(cert->os_cert_handle(), &der_string); return base::android::ToJavaByteArray(env, reinterpret_cast(der_string.data()), der_string.length()); } void AwContents::RequestNewHitTestDataAt(JNIEnv* env, jobject obj, jint x, jint y) { render_view_host_ext_->RequestNewHitTestDataAt(x, y); } void AwContents::UpdateLastHitTestData(JNIEnv* env, jobject obj) { if (!render_view_host_ext_->HasNewHitTestData()) return; const AwHitTestData& data = render_view_host_ext_->GetLastHitTestData(); render_view_host_ext_->MarkHitTestDataRead(); // Make sure to null the Java object if data is empty/invalid. ScopedJavaLocalRef extra_data_for_type; if (data.extra_data_for_type.length()) extra_data_for_type = ConvertUTF8ToJavaString( env, data.extra_data_for_type); ScopedJavaLocalRef href; if (data.href.length()) href = ConvertUTF16ToJavaString(env, data.href); ScopedJavaLocalRef anchor_text; if (data.anchor_text.length()) anchor_text = ConvertUTF16ToJavaString(env, data.anchor_text); ScopedJavaLocalRef img_src; if (data.img_src.is_valid()) img_src = ConvertUTF8ToJavaString(env, data.img_src.spec()); Java_AwContents_updateHitTestData(env, obj, data.type, extra_data_for_type.obj(), href.obj(), anchor_text.obj(), img_src.obj()); } void AwContents::OnSizeChanged(JNIEnv* env, jobject obj, int w, int h, int ow, int oh) { compositor_->SetWindowBounds(gfx::Size(w, h)); } void AwContents::SetWindowViewVisibility(JNIEnv* env, jobject obj, bool window_visible, bool view_visible) { view_visible_ = window_visible && view_visible; Invalidate(); } void AwContents::OnAttachedToWindow(JNIEnv* env, jobject obj, int w, int h) { // Seed the Compositor size here, as we'll only receive OnSizeChanged calls // for a genuine change in size, not to set initial size. Note the |w| and |h| // passed here are the Java view size, NOT window size (which correctly maps // to the Compositor's "window" size). compositor_->SetWindowBounds(gfx::Size(w, h)); } void AwContents::OnDetachedFromWindow(JNIEnv* env, jobject obj) { view_visible_ = false; // TODO(joth): Request a DrawGL (kModeProcess) call, to tell the compositor. } base::android::ScopedJavaLocalRef AwContents::GetOpaqueState(JNIEnv* env, jobject obj) { // Required optimization in WebViewClassic to not save any state if // there has been no navigations. if (!web_contents_->GetController().GetEntryCount()) return ScopedJavaLocalRef(); Pickle pickle; if (!WriteToPickle(*web_contents_, &pickle)) { return ScopedJavaLocalRef(); } else { return base::android::ToJavaByteArray(env, reinterpret_cast(pickle.data()), pickle.size()); } } jboolean AwContents::RestoreFromOpaqueState( JNIEnv* env, jobject obj, jbyteArray state) { // TODO(boliu): This copy can be optimized out if this is a performance // problem. std::vector state_vector; base::android::JavaByteArrayToByteVector(env, state, &state_vector); Pickle pickle(reinterpret_cast(state_vector.begin()), state_vector.size()); PickleIterator iterator(pickle); return RestoreFromPickle(&iterator, web_contents_.get()); } void AwContents::SetPendingWebContentsForPopup( scoped_ptr pending) { if (pending_contents_.get()) { // TODO(benm): Support holding multiple pop up window requests. LOG(WARNING) << "Blocking popup window creation as an outstanding " << "popup window is still pending."; MessageLoop::current()->DeleteSoon(FROM_HERE, pending.release()); return; } pending_contents_ = pending.Pass(); } jint AwContents::ReleasePopupWebContents(JNIEnv* env, jobject obj) { return reinterpret_cast(pending_contents_.release()); } } // namespace android_webview