// Copyright 2014 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 "content/renderer/java/gin_java_bridge_object.h" #include "content/common/gin_java_bridge_messages.h" #include "content/public/renderer/render_thread.h" #include "content/renderer/java/gin_java_function_invocation_helper.h" #include "gin/function_template.h" #include "third_party/WebKit/public/web/WebFrame.h" #include "third_party/WebKit/public/web/WebKit.h" namespace content { // static GinJavaBridgeObject* GinJavaBridgeObject::InjectNamed( blink::WebFrame* frame, const base::WeakPtr& dispatcher, const std::string& object_name, GinJavaBridgeDispatcher::ObjectID object_id) { v8::Isolate* isolate = blink::mainThreadIsolate(); v8::HandleScope handle_scope(isolate); v8::Local context = frame->mainWorldScriptContext(); if (context.IsEmpty()) return NULL; GinJavaBridgeObject* object = new GinJavaBridgeObject(isolate, dispatcher, object_id); v8::Context::Scope context_scope(context); v8::Local global = context->Global(); gin::Handle controller = gin::CreateHandle(isolate, object); // WrappableBase instance deletes itself in case of a wrapper // creation failure, thus there is no need to delete |object|. if (controller.IsEmpty()) return NULL; global->Set(gin::StringToV8(isolate, object_name), controller.ToV8()); return object; } // static GinJavaBridgeObject* GinJavaBridgeObject::InjectAnonymous( const base::WeakPtr& dispatcher, GinJavaBridgeDispatcher::ObjectID object_id) { return new GinJavaBridgeObject( blink::mainThreadIsolate(), dispatcher, object_id); } GinJavaBridgeObject::GinJavaBridgeObject( v8::Isolate* isolate, const base::WeakPtr& dispatcher, GinJavaBridgeDispatcher::ObjectID object_id) : gin::NamedPropertyInterceptor(isolate, this), dispatcher_(dispatcher), object_id_(object_id), frame_routing_id_(dispatcher_->routing_id()), template_cache_(isolate) { } GinJavaBridgeObject::~GinJavaBridgeObject() { if (dispatcher_) { dispatcher_->OnGinJavaBridgeObjectDeleted(this); } else { // A wrapper can outlive a render frame, and thus the dispatcher. // Note that we intercept GinJavaBridgeHostMsg messages in a browser filter // thus it's OK to send the message with a routing id of a ceased frame. RenderThread::Get()->Send(new GinJavaBridgeHostMsg_ObjectWrapperDeleted( frame_routing_id_, object_id_)); } } gin::ObjectTemplateBuilder GinJavaBridgeObject::GetObjectTemplateBuilder( v8::Isolate* isolate) { return gin::Wrappable::GetObjectTemplateBuilder(isolate) .AddNamedPropertyInterceptor(); } v8::Local GinJavaBridgeObject::GetNamedProperty( v8::Isolate* isolate, const std::string& property) { std::map::iterator method_pos = known_methods_.find(property); if (method_pos == known_methods_.end()) { if (!dispatcher_) { return v8::Local(); } known_methods_[property] = dispatcher_->HasJavaMethod(object_id_, property); } if (known_methods_[property]) return GetFunctionTemplate(isolate, property)->GetFunction(); else return v8::Local(); } std::vector GinJavaBridgeObject::EnumerateNamedProperties( v8::Isolate* isolate) { std::set method_names; if (dispatcher_) dispatcher_->GetJavaMethods(object_id_, &method_names); return std::vector (method_names.begin(), method_names.end()); } v8::Local GinJavaBridgeObject::GetFunctionTemplate( v8::Isolate* isolate, const std::string& name) { v8::Local function_template = template_cache_.Get(name); if (!function_template.IsEmpty()) return function_template; function_template = gin::CreateFunctionTemplate( isolate, base::Bind(&GinJavaFunctionInvocationHelper::Invoke, base::Owned(new GinJavaFunctionInvocationHelper( name, dispatcher_)))); template_cache_.Set(name, function_template); return function_template; } gin::WrapperInfo GinJavaBridgeObject::kWrapperInfo = {gin::kEmbedderNativeGin}; } // namespace content