renderer_client_base.cc 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569
  1. // Copyright (c) 2017 GitHub, Inc.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/renderer/renderer_client_base.h"
  5. #include <memory>
  6. #include <string>
  7. #include <utility>
  8. #include <vector>
  9. #include "base/command_line.h"
  10. #include "base/strings/string_split.h"
  11. #include "base/strings/stringprintf.h"
  12. #include "components/network_hints/renderer/web_prescient_networking_impl.h"
  13. #include "content/common/buildflags.h"
  14. #include "content/public/common/content_constants.h"
  15. #include "content/public/common/content_switches.h"
  16. #include "content/public/renderer/render_frame.h"
  17. #include "content/public/renderer/render_thread.h"
  18. #include "content/public/renderer/render_view.h"
  19. #include "electron/buildflags/buildflags.h"
  20. #include "printing/buildflags/buildflags.h"
  21. #include "shell/browser/api/electron_api_protocol.h"
  22. #include "shell/common/api/electron_api_native_image.h"
  23. #include "shell/common/color_util.h"
  24. #include "shell/common/gin_helper/dictionary.h"
  25. #include "shell/common/node_includes.h"
  26. #include "shell/common/node_util.h"
  27. #include "shell/common/options_switches.h"
  28. #include "shell/common/world_ids.h"
  29. #include "shell/renderer/api/context_bridge/object_cache.h"
  30. #include "shell/renderer/api/electron_api_context_bridge.h"
  31. #include "shell/renderer/browser_exposed_renderer_interfaces.h"
  32. #include "shell/renderer/content_settings_observer.h"
  33. #include "shell/renderer/electron_api_service_impl.h"
  34. #include "shell/renderer/electron_autofill_agent.h"
  35. #include "third_party/blink/public/common/associated_interfaces/associated_interface_registry.h"
  36. #include "third_party/blink/public/common/web_preferences/web_preferences.h"
  37. #include "third_party/blink/public/platform/media/multi_buffer_data_source.h"
  38. #include "third_party/blink/public/web/blink.h"
  39. #include "third_party/blink/public/web/web_custom_element.h" // NOLINT(build/include_alpha)
  40. #include "third_party/blink/public/web/web_frame_widget.h"
  41. #include "third_party/blink/public/web/web_local_frame.h"
  42. #include "third_party/blink/public/web/web_plugin_params.h"
  43. #include "third_party/blink/public/web/web_script_source.h"
  44. #include "third_party/blink/public/web/web_security_policy.h"
  45. #include "third_party/blink/public/web/web_view.h"
  46. #include "third_party/blink/renderer/platform/weborigin/scheme_registry.h" // nogncheck
  47. #if defined(OS_MAC)
  48. #include "base/strings/sys_string_conversions.h"
  49. #endif
  50. #if defined(OS_WIN)
  51. #include <shlobj.h>
  52. #endif
  53. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  54. #include "components/spellcheck/renderer/spellcheck.h"
  55. #include "components/spellcheck/renderer/spellcheck_provider.h"
  56. #endif
  57. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  58. #include "shell/common/electron_constants.h"
  59. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  60. #if BUILDFLAG(ENABLE_PLUGINS)
  61. #include "shell/renderer/pepper_helper.h"
  62. #endif // BUILDFLAG(ENABLE_PLUGINS)
  63. #if BUILDFLAG(ENABLE_PRINTING)
  64. #include "components/printing/renderer/print_render_frame_helper.h"
  65. #include "printing/metafile_agent.h" // nogncheck
  66. #include "shell/renderer/printing/print_render_frame_helper_delegate.h"
  67. #endif // BUILDFLAG(ENABLE_PRINTING)
  68. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  69. #include "base/strings/utf_string_conversions.h"
  70. #include "content/public/common/webplugininfo.h"
  71. #include "extensions/common/constants.h"
  72. #include "extensions/common/extensions_client.h"
  73. #include "extensions/renderer/dispatcher.h"
  74. #include "extensions/renderer/extension_frame_helper.h"
  75. #include "extensions/renderer/guest_view/mime_handler_view/mime_handler_view_container_manager.h"
  76. #include "shell/common/extensions/electron_extensions_client.h"
  77. #include "shell/renderer/extensions/electron_extensions_renderer_client.h"
  78. #endif // BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  79. namespace electron {
  80. content::RenderFrame* GetRenderFrame(v8::Local<v8::Object> value);
  81. namespace {
  82. void SetIsWebView(v8::Isolate* isolate, v8::Local<v8::Object> object) {
  83. gin_helper::Dictionary dict(isolate, object);
  84. dict.SetHidden("isWebView", true);
  85. }
  86. std::vector<std::string> ParseSchemesCLISwitch(base::CommandLine* command_line,
  87. const char* switch_name) {
  88. std::string custom_schemes = command_line->GetSwitchValueASCII(switch_name);
  89. return base::SplitString(custom_schemes, ",", base::TRIM_WHITESPACE,
  90. base::SPLIT_WANT_NONEMPTY);
  91. }
  92. // static
  93. RendererClientBase* g_renderer_client_base = nullptr;
  94. } // namespace
  95. RendererClientBase::RendererClientBase() {
  96. auto* command_line = base::CommandLine::ForCurrentProcess();
  97. // Parse --service-worker-schemes=scheme1,scheme2
  98. std::vector<std::string> service_worker_schemes_list =
  99. ParseSchemesCLISwitch(command_line, switches::kServiceWorkerSchemes);
  100. for (const std::string& scheme : service_worker_schemes_list)
  101. electron::api::AddServiceWorkerScheme(scheme);
  102. // Parse --standard-schemes=scheme1,scheme2
  103. std::vector<std::string> standard_schemes_list =
  104. ParseSchemesCLISwitch(command_line, switches::kStandardSchemes);
  105. for (const std::string& scheme : standard_schemes_list)
  106. url::AddStandardScheme(scheme.c_str(), url::SCHEME_WITH_HOST);
  107. // Parse --cors-schemes=scheme1,scheme2
  108. std::vector<std::string> cors_schemes_list =
  109. ParseSchemesCLISwitch(command_line, switches::kCORSSchemes);
  110. for (const std::string& scheme : cors_schemes_list)
  111. url::AddCorsEnabledScheme(scheme.c_str());
  112. // Parse --streaming-schemes=scheme1,scheme2
  113. std::vector<std::string> streaming_schemes_list =
  114. ParseSchemesCLISwitch(command_line, switches::kStreamingSchemes);
  115. for (const std::string& scheme : streaming_schemes_list)
  116. blink::AddStreamingScheme(scheme.c_str());
  117. // Parse --secure-schemes=scheme1,scheme2
  118. std::vector<std::string> secure_schemes_list =
  119. ParseSchemesCLISwitch(command_line, switches::kSecureSchemes);
  120. for (const std::string& scheme : secure_schemes_list)
  121. url::AddSecureScheme(scheme.data());
  122. // We rely on the unique process host id which is notified to the
  123. // renderer process via command line switch from the content layer,
  124. // if this switch is removed from the content layer for some reason,
  125. // we should define our own.
  126. DCHECK(command_line->HasSwitch(::switches::kRendererClientId));
  127. renderer_client_id_ =
  128. command_line->GetSwitchValueASCII(::switches::kRendererClientId);
  129. g_renderer_client_base = this;
  130. }
  131. RendererClientBase::~RendererClientBase() {
  132. g_renderer_client_base = nullptr;
  133. }
  134. // static
  135. RendererClientBase* RendererClientBase::Get() {
  136. DCHECK(g_renderer_client_base);
  137. return g_renderer_client_base;
  138. }
  139. void RendererClientBase::BindProcess(v8::Isolate* isolate,
  140. gin_helper::Dictionary* process,
  141. content::RenderFrame* render_frame) {
  142. auto context_id = base::StringPrintf(
  143. "%s-%" PRId64, renderer_client_id_.c_str(), ++next_context_id_);
  144. process->SetReadOnly("isMainFrame", render_frame->IsMainFrame());
  145. process->SetReadOnly("contextIsolated",
  146. render_frame->GetBlinkPreferences().context_isolation);
  147. process->SetReadOnly("contextId", context_id);
  148. }
  149. void RendererClientBase::RenderThreadStarted() {
  150. auto* command_line = base::CommandLine::ForCurrentProcess();
  151. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  152. auto* thread = content::RenderThread::Get();
  153. extensions_client_.reset(CreateExtensionsClient());
  154. extensions::ExtensionsClient::Set(extensions_client_.get());
  155. extensions_renderer_client_ =
  156. std::make_unique<ElectronExtensionsRendererClient>();
  157. extensions::ExtensionsRendererClient::Set(extensions_renderer_client_.get());
  158. thread->AddObserver(extensions_renderer_client_->GetDispatcher());
  159. #endif
  160. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  161. // Enables printing from Chrome PDF viewer.
  162. pdf_print_client_ = std::make_unique<ChromePDFPrintClient>();
  163. pdf::PepperPDFHost::SetPrintClient(pdf_print_client_.get());
  164. #endif
  165. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  166. spellcheck_ = std::make_unique<SpellCheck>(this);
  167. #endif
  168. blink::WebCustomElement::AddEmbedderCustomElementName("webview");
  169. blink::WebCustomElement::AddEmbedderCustomElementName("browserplugin");
  170. WTF::String extension_scheme(extensions::kExtensionScheme);
  171. // Extension resources are HTTP-like and safe to expose to the fetch API. The
  172. // rules for the fetch API are consistent with XHR.
  173. blink::SchemeRegistry::RegisterURLSchemeAsSupportingFetchAPI(
  174. extension_scheme);
  175. // Extension resources, when loaded as the top-level document, should bypass
  176. // Blink's strict first-party origin checks.
  177. blink::SchemeRegistry::RegisterURLSchemeAsFirstPartyWhenTopLevel(
  178. extension_scheme);
  179. blink::SchemeRegistry::RegisterURLSchemeAsBypassingContentSecurityPolicy(
  180. extension_scheme);
  181. std::vector<std::string> fetch_enabled_schemes =
  182. ParseSchemesCLISwitch(command_line, switches::kFetchSchemes);
  183. for (const std::string& scheme : fetch_enabled_schemes) {
  184. blink::WebSecurityPolicy::RegisterURLSchemeAsSupportingFetchAPI(
  185. blink::WebString::FromASCII(scheme));
  186. }
  187. std::vector<std::string> service_worker_schemes =
  188. ParseSchemesCLISwitch(command_line, switches::kServiceWorkerSchemes);
  189. for (const std::string& scheme : service_worker_schemes)
  190. blink::WebSecurityPolicy::RegisterURLSchemeAsAllowingServiceWorkers(
  191. blink::WebString::FromASCII(scheme));
  192. std::vector<std::string> csp_bypassing_schemes =
  193. ParseSchemesCLISwitch(command_line, switches::kBypassCSPSchemes);
  194. for (const std::string& scheme : csp_bypassing_schemes)
  195. blink::SchemeRegistry::RegisterURLSchemeAsBypassingContentSecurityPolicy(
  196. WTF::String::FromUTF8(scheme.data(), scheme.length()));
  197. // Allow file scheme to handle service worker by default.
  198. // FIXME(zcbenz): Can this be moved elsewhere?
  199. blink::WebSecurityPolicy::RegisterURLSchemeAsAllowingServiceWorkers("file");
  200. blink::SchemeRegistry::RegisterURLSchemeAsSupportingFetchAPI("file");
  201. #if defined(OS_WIN)
  202. // Set ApplicationUserModelID in renderer process.
  203. std::wstring app_id =
  204. command_line->GetSwitchValueNative(switches::kAppUserModelId);
  205. if (!app_id.empty()) {
  206. SetCurrentProcessExplicitAppUserModelID(app_id.c_str());
  207. }
  208. #endif
  209. }
  210. void RendererClientBase::ExposeInterfacesToBrowser(mojo::BinderMap* binders) {
  211. // NOTE: Do not add binders directly within this method. Instead, modify the
  212. // definition of |ExposeElectronRendererInterfacesToBrowser()| to ensure
  213. // security review coverage.
  214. ExposeElectronRendererInterfacesToBrowser(this, binders);
  215. }
  216. void RendererClientBase::RenderFrameCreated(
  217. content::RenderFrame* render_frame) {
  218. #if defined(TOOLKIT_VIEWS)
  219. new AutofillAgent(render_frame,
  220. render_frame->GetAssociatedInterfaceRegistry());
  221. #endif
  222. #if BUILDFLAG(ENABLE_PLUGINS)
  223. new PepperHelper(render_frame);
  224. #endif
  225. new ContentSettingsObserver(render_frame);
  226. #if BUILDFLAG(ENABLE_PRINTING)
  227. new printing::PrintRenderFrameHelper(
  228. render_frame,
  229. std::make_unique<electron::PrintRenderFrameHelperDelegate>());
  230. #endif
  231. // Note: ElectronApiServiceImpl has to be created now to capture the
  232. // DidCreateDocumentElement event.
  233. new ElectronApiServiceImpl(render_frame, this);
  234. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  235. auto* dispatcher = extensions_renderer_client_->GetDispatcher();
  236. // ExtensionFrameHelper destroys itself when the RenderFrame is destroyed.
  237. new extensions::ExtensionFrameHelper(render_frame, dispatcher);
  238. dispatcher->OnRenderFrameCreated(render_frame);
  239. render_frame->GetAssociatedInterfaceRegistry()->AddInterface(
  240. base::BindRepeating(
  241. &extensions::MimeHandlerViewContainerManager::BindReceiver,
  242. render_frame->GetRoutingID()));
  243. #endif
  244. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  245. if (render_frame->GetBlinkPreferences().enable_spellcheck)
  246. new SpellCheckProvider(render_frame, spellcheck_.get(), this);
  247. #endif
  248. }
  249. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  250. void RendererClientBase::GetInterface(
  251. const std::string& interface_name,
  252. mojo::ScopedMessagePipeHandle interface_pipe) {
  253. // TODO(crbug.com/977637): Get rid of the use of this implementation of
  254. // |service_manager::LocalInterfaceProvider|. This was done only to avoid
  255. // churning spellcheck code while eliminating the "chrome" and
  256. // "chrome_renderer" services. Spellcheck is (and should remain) the only
  257. // consumer of this implementation.
  258. content::RenderThread::Get()->BindHostReceiver(
  259. mojo::GenericPendingReceiver(interface_name, std::move(interface_pipe)));
  260. }
  261. #endif
  262. void RendererClientBase::DidClearWindowObject(
  263. content::RenderFrame* render_frame) {
  264. // Make sure every page will get a script context created.
  265. render_frame->GetWebFrame()->ExecuteScript(blink::WebScriptSource("void 0"));
  266. }
  267. bool RendererClientBase::OverrideCreatePlugin(
  268. content::RenderFrame* render_frame,
  269. const blink::WebPluginParams& params,
  270. blink::WebPlugin** plugin) {
  271. if (params.mime_type.Utf8() == content::kBrowserPluginMimeType ||
  272. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  273. params.mime_type.Utf8() == kPdfPluginMimeType ||
  274. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  275. render_frame->GetBlinkPreferences().enable_plugins)
  276. return false;
  277. *plugin = nullptr;
  278. return true;
  279. }
  280. void RendererClientBase::AddSupportedKeySystems(
  281. std::vector<std::unique_ptr<::media::KeySystemProperties>>* key_systems) {
  282. #if defined(WIDEVINE_CDM_AVAILABLE)
  283. key_systems_provider_.AddSupportedKeySystems(key_systems);
  284. #endif
  285. }
  286. bool RendererClientBase::IsKeySystemsUpdateNeeded() {
  287. #if defined(WIDEVINE_CDM_AVAILABLE)
  288. return key_systems_provider_.IsKeySystemsUpdateNeeded();
  289. #else
  290. return false;
  291. #endif
  292. }
  293. void RendererClientBase::DidSetUserAgent(const std::string& user_agent) {
  294. #if BUILDFLAG(ENABLE_PRINTING)
  295. printing::SetAgent(user_agent);
  296. #endif
  297. }
  298. bool RendererClientBase::IsPluginHandledExternally(
  299. content::RenderFrame* render_frame,
  300. const blink::WebElement& plugin_element,
  301. const GURL& original_url,
  302. const std::string& mime_type) {
  303. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  304. DCHECK(plugin_element.HasHTMLTagName("object") ||
  305. plugin_element.HasHTMLTagName("embed"));
  306. // TODO(nornagon): this info should be shared with the data in
  307. // electron_content_client.cc / ComputeBuiltInPlugins.
  308. content::WebPluginInfo info;
  309. info.type = content::WebPluginInfo::PLUGIN_TYPE_BROWSER_PLUGIN;
  310. const char16_t kPDFExtensionPluginName[] = u"Chromium PDF Viewer";
  311. info.name = kPDFExtensionPluginName;
  312. info.path = base::FilePath::FromUTF8Unsafe(extension_misc::kPdfExtensionId);
  313. info.background_color = content::WebPluginInfo::kDefaultBackgroundColor;
  314. info.mime_types.emplace_back("application/pdf", "pdf",
  315. "Portable Document Format");
  316. return extensions::MimeHandlerViewContainerManager::Get(
  317. content::RenderFrame::FromWebFrame(
  318. plugin_element.GetDocument().GetFrame()),
  319. true /* create_if_does_not_exist */)
  320. ->CreateFrameContainer(plugin_element, original_url, mime_type, info);
  321. #else
  322. return false;
  323. #endif
  324. }
  325. bool RendererClientBase::IsOriginIsolatedPepperPlugin(
  326. const base::FilePath& plugin_path) {
  327. // Isolate all Pepper plugins, including the PDF plugin.
  328. return true;
  329. }
  330. std::unique_ptr<blink::WebPrescientNetworking>
  331. RendererClientBase::CreatePrescientNetworking(
  332. content::RenderFrame* render_frame) {
  333. return std::make_unique<network_hints::WebPrescientNetworkingImpl>(
  334. render_frame);
  335. }
  336. void RendererClientBase::RunScriptsAtDocumentStart(
  337. content::RenderFrame* render_frame) {
  338. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  339. extensions_renderer_client_.get()->RunScriptsAtDocumentStart(render_frame);
  340. #endif
  341. }
  342. void RendererClientBase::RunScriptsAtDocumentIdle(
  343. content::RenderFrame* render_frame) {
  344. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  345. extensions_renderer_client_.get()->RunScriptsAtDocumentIdle(render_frame);
  346. #endif
  347. }
  348. void RendererClientBase::RunScriptsAtDocumentEnd(
  349. content::RenderFrame* render_frame) {
  350. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  351. extensions_renderer_client_.get()->RunScriptsAtDocumentEnd(render_frame);
  352. #endif
  353. }
  354. bool RendererClientBase::AllowScriptExtensionForServiceWorker(
  355. const url::Origin& script_origin) {
  356. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  357. return script_origin.scheme() == extensions::kExtensionScheme;
  358. #else
  359. return false;
  360. #endif
  361. }
  362. void RendererClientBase::DidInitializeServiceWorkerContextOnWorkerThread(
  363. blink::WebServiceWorkerContextProxy* context_proxy,
  364. const GURL& service_worker_scope,
  365. const GURL& script_url) {
  366. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  367. extensions_renderer_client_->GetDispatcher()
  368. ->DidInitializeServiceWorkerContextOnWorkerThread(
  369. context_proxy, service_worker_scope, script_url);
  370. #endif
  371. }
  372. void RendererClientBase::WillEvaluateServiceWorkerOnWorkerThread(
  373. blink::WebServiceWorkerContextProxy* context_proxy,
  374. v8::Local<v8::Context> v8_context,
  375. int64_t service_worker_version_id,
  376. const GURL& service_worker_scope,
  377. const GURL& script_url) {
  378. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  379. extensions_renderer_client_->GetDispatcher()
  380. ->WillEvaluateServiceWorkerOnWorkerThread(
  381. context_proxy, v8_context, service_worker_version_id,
  382. service_worker_scope, script_url);
  383. #endif
  384. }
  385. void RendererClientBase::DidStartServiceWorkerContextOnWorkerThread(
  386. int64_t service_worker_version_id,
  387. const GURL& service_worker_scope,
  388. const GURL& script_url) {
  389. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  390. extensions_renderer_client_->GetDispatcher()
  391. ->DidStartServiceWorkerContextOnWorkerThread(
  392. service_worker_version_id, service_worker_scope, script_url);
  393. #endif
  394. }
  395. void RendererClientBase::WillDestroyServiceWorkerContextOnWorkerThread(
  396. v8::Local<v8::Context> context,
  397. int64_t service_worker_version_id,
  398. const GURL& service_worker_scope,
  399. const GURL& script_url) {
  400. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  401. extensions_renderer_client_->GetDispatcher()
  402. ->WillDestroyServiceWorkerContextOnWorkerThread(
  403. context, service_worker_version_id, service_worker_scope, script_url);
  404. #endif
  405. }
  406. v8::Local<v8::Context> RendererClientBase::GetContext(
  407. blink::WebLocalFrame* frame,
  408. v8::Isolate* isolate) const {
  409. auto* render_frame = content::RenderFrame::FromWebFrame(frame);
  410. DCHECK(render_frame);
  411. if (render_frame && render_frame->GetBlinkPreferences().context_isolation)
  412. return frame->GetScriptContextFromWorldId(isolate,
  413. WorldIDs::ISOLATED_WORLD_ID);
  414. else
  415. return frame->MainWorldScriptContext();
  416. }
  417. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  418. extensions::ExtensionsClient* RendererClientBase::CreateExtensionsClient() {
  419. return new ElectronExtensionsClient;
  420. }
  421. #endif
  422. bool RendererClientBase::IsWebViewFrame(
  423. v8::Handle<v8::Context> context,
  424. content::RenderFrame* render_frame) const {
  425. auto* isolate = context->GetIsolate();
  426. if (render_frame->IsMainFrame())
  427. return false;
  428. gin::Dictionary window_dict(
  429. isolate, GetContext(render_frame->GetWebFrame(), isolate)->Global());
  430. v8::Local<v8::Object> frame_element;
  431. if (!window_dict.Get("frameElement", &frame_element))
  432. return false;
  433. gin_helper::Dictionary frame_element_dict(isolate, frame_element);
  434. bool is_webview = false;
  435. return frame_element_dict.GetHidden("isWebView", &is_webview) && is_webview;
  436. }
  437. void RendererClientBase::SetupMainWorldOverrides(
  438. v8::Handle<v8::Context> context,
  439. content::RenderFrame* render_frame) {
  440. auto prefs = render_frame->GetBlinkPreferences();
  441. // We only need to run the isolated bundle if webview is enabled
  442. if (!prefs.webview_tag)
  443. return;
  444. // Setup window overrides in the main world context
  445. // Wrap the bundle into a function that receives the isolatedApi as
  446. // an argument.
  447. auto* isolate = context->GetIsolate();
  448. v8::HandleScope handle_scope(isolate);
  449. v8::Context::Scope context_scope(context);
  450. gin_helper::Dictionary isolated_api = gin::Dictionary::CreateEmpty(isolate);
  451. isolated_api.SetMethod("allowGuestViewElementDefinition",
  452. &AllowGuestViewElementDefinition);
  453. isolated_api.SetMethod("setIsWebView", &SetIsWebView);
  454. auto source_context = GetContext(render_frame->GetWebFrame(), isolate);
  455. gin_helper::Dictionary global(isolate, source_context->Global());
  456. v8::Local<v8::Value> guest_view_internal;
  457. if (global.GetHidden("guestViewInternal", &guest_view_internal)) {
  458. api::context_bridge::ObjectCache object_cache;
  459. auto result = api::PassValueToOtherContext(
  460. source_context, context, guest_view_internal, &object_cache, false, 0);
  461. if (!result.IsEmpty()) {
  462. isolated_api.Set("guestViewInternal", result.ToLocalChecked());
  463. }
  464. }
  465. std::vector<v8::Local<v8::String>> isolated_bundle_params = {
  466. node::FIXED_ONE_BYTE_STRING(isolate, "isolatedApi")};
  467. std::vector<v8::Local<v8::Value>> isolated_bundle_args = {
  468. isolated_api.GetHandle()};
  469. util::CompileAndCall(context, "electron/js2c/isolated_bundle",
  470. &isolated_bundle_params, &isolated_bundle_args, nullptr);
  471. }
  472. // static
  473. void RendererClientBase::AllowGuestViewElementDefinition(
  474. v8::Isolate* isolate,
  475. v8::Local<v8::Object> context,
  476. v8::Local<v8::Function> register_cb) {
  477. v8::HandleScope handle_scope(isolate);
  478. v8::Context::Scope context_scope(context->CreationContext());
  479. blink::WebCustomElement::EmbedderNamesAllowedScope embedder_names_scope;
  480. content::RenderFrame* render_frame = GetRenderFrame(context);
  481. if (!render_frame)
  482. return;
  483. render_frame->GetWebFrame()->RequestExecuteV8Function(
  484. context->CreationContext(), register_cb, v8::Null(isolate), 0, nullptr,
  485. nullptr);
  486. }
  487. } // namespace electron