// 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 "chrome/renderer/external_extension.h" #include "base/command_line.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/render_messages.h" #include "chrome/common/search_provider.h" #include "content/public/renderer/render_view.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" #include "v8/include/v8.h" using WebKit::WebFrame; using WebKit::WebView; using content::RenderView; namespace extensions_v8 { static const char* const kSearchProviderApi = "var external;" "if (!external)" " external = {};" "external.AddSearchProvider = function(name) {" " native function NativeAddSearchProvider();" " NativeAddSearchProvider(name);" "};" "external.IsSearchProviderInstalled = function(name) {" " native function NativeIsSearchProviderInstalled();" " return NativeIsSearchProviderInstalled(name);" "};"; const char* const kExternalExtensionName = "v8/External"; class ExternalExtensionWrapper : public v8::Extension { public: ExternalExtensionWrapper(); // Allows v8's javascript code to call the native functions defined // in this class for window.external. virtual v8::Handle GetNativeFunction( v8::Handle name) OVERRIDE; // Helper function to find the RenderView. May return NULL. static RenderView* GetRenderView(); // Implementation of window.external.AddSearchProvider. static v8::Handle AddSearchProvider(const v8::Arguments& args); // Implementation of window.external.IsSearchProviderInstalled. static v8::Handle IsSearchProviderInstalled( const v8::Arguments& args); private: DISALLOW_COPY_AND_ASSIGN(ExternalExtensionWrapper); }; ExternalExtensionWrapper::ExternalExtensionWrapper() : v8::Extension( kExternalExtensionName, kSearchProviderApi) { } v8::Handle ExternalExtensionWrapper::GetNativeFunction( v8::Handle name) { if (name->Equals(v8::String::New("NativeAddSearchProvider"))) return v8::FunctionTemplate::New(AddSearchProvider); if (name->Equals(v8::String::New("NativeIsSearchProviderInstalled"))) return v8::FunctionTemplate::New(IsSearchProviderInstalled); return v8::Handle(); } // static RenderView* ExternalExtensionWrapper::GetRenderView() { WebFrame* webframe = WebFrame::frameForCurrentContext(); DCHECK(webframe) << "There should be an active frame since we just got " "a native function called."; if (!webframe) return NULL; WebView* webview = webframe->view(); if (!webview) return NULL; // can happen during closing return RenderView::FromWebView(webview); } // static v8::Handle ExternalExtensionWrapper::AddSearchProvider( const v8::Arguments& args) { if (!args.Length()) return v8::Undefined(); std::string name = std::string(*v8::String::Utf8Value(args[0])); if (!name.length()) return v8::Undefined(); RenderView* render_view = GetRenderView(); if (!render_view) return v8::Undefined(); GURL osd_url(name); if (!osd_url.is_empty() && osd_url.is_valid()) { render_view->Send(new ChromeViewHostMsg_PageHasOSDD( render_view->GetRoutingID(), render_view->GetPageId(), osd_url, search_provider::EXPLICIT_PROVIDER)); } return v8::Undefined(); } // static v8::Handle ExternalExtensionWrapper::IsSearchProviderInstalled( const v8::Arguments& args) { if (!args.Length()) return v8::Undefined(); v8::String::Utf8Value utf8name(args[0]); if (!utf8name.length()) return v8::Undefined(); std::string name = std::string(*utf8name); RenderView* render_view = GetRenderView(); if (!render_view) return v8::Undefined(); WebFrame* webframe = WebFrame::frameForCurrentContext(); if (!webframe) return v8::Undefined(); search_provider::InstallState install = search_provider::DENIED; GURL inquiry_url = GURL(name); if (!inquiry_url.is_empty()) { render_view->Send(new ChromeViewHostMsg_GetSearchProviderInstallState( render_view->GetRoutingID(), webframe->document().url(), inquiry_url, &install)); } if (install == search_provider::DENIED) { // FIXME: throw access denied exception. return v8::ThrowException(v8::Exception::Error(v8::String::Empty())); } return v8::Integer::New(install); } v8::Extension* ExternalExtension::Get() { return new ExternalExtensionWrapper(); } } // namespace extensions_v8