// Copyright (c) 2011 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 kSearchProviderApiV1 = "var external;" "if (!external)" " external = {};" "external.AddSearchProvider = function(name) {" " native function NativeAddSearchProvider();" " NativeAddSearchProvider(name);" "};"; static const char* const kSearchProviderApiV2 = "var external;" "if (!external)" " external = {};" "external.AddSearchProvider = function(name, default_provider) {" " native function NativeAddSearchProvider();" " NativeAddSearchProvider(name, default_provider);" "};" "external.IsSearchProviderInstalled = function(name) {" " native function NativeIsSearchProviderInstalled();" " return NativeIsSearchProviderInstalled(name);" "};"; #undef SEARCH_PROVIDER_API_V1 const char* const kExternalExtensionName = "v8/External"; // Should the new api's "IsSearchProviderInstalled and InstallSearchProvider // with an extra parameter to indicate if the provider should be the default" // be available? static bool EnableSearchProviderV2() { #if defined(OS_WIN) return true; #else return CommandLine::ForCurrentProcess()->HasSwitch( switches::kEnableSearchProviderApiV2); #endif } 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); // 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, !EnableSearchProviderV2() ? kSearchProviderApiV1 : kSearchProviderApiV2) { } 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::frameForEnteredContext(); 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(); search_provider::OSDDType provider_type = ((args.Length() < 2) || !args[1]->BooleanValue()) ? search_provider::EXPLICIT_PROVIDER : search_provider::EXPLICIT_DEFAULT_PROVIDER; RenderView* render_view = GetRenderView(); if (!render_view) return v8::Undefined(); if (provider_type != search_provider::EXPLICIT_DEFAULT_PROVIDER || render_view->GetWebView()->mainFrame()->isProcessingUserGesture()) { GURL osd_url(name); if (!osd_url.is_empty()) { render_view->Send(new ChromeViewHostMsg_PageHasOSDD( render_view->GetRoutingId(), render_view->GetPageId(), osd_url, provider_type)); } } 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::frameForEnteredContext(); 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