// Copyright (c) 2013 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "atom/renderer/atom_render_view_observer.h" #include <string> #include <vector> // Put this before event_emitter_caller.h to have string16 support. #include "atom/common/native_mate_converters/string16_converter.h" #include "atom/common/api/api_messages.h" #include "atom/common/api/event_emitter_caller.h" #include "atom/common/native_mate_converters/value_converter.h" #include "atom/common/node_includes.h" #include "atom/common/options_switches.h" #include "atom/renderer/atom_renderer_client.h" #include "base/command_line.h" #include "base/strings/string_number_conversions.h" #include "content/public/renderer/render_view.h" #include "ipc/ipc_message_macros.h" #include "net/base/net_module.h" #include "net/grit/net_resources.h" #include "third_party/WebKit/public/web/WebDraggableRegion.h" #include "third_party/WebKit/public/web/WebDocument.h" #include "third_party/WebKit/public/web/WebFrame.h" #include "third_party/WebKit/public/web/WebLocalFrame.h" #include "third_party/WebKit/public/web/WebKit.h" #include "third_party/WebKit/public/web/WebScopedUserGesture.h" #include "third_party/WebKit/public/web/WebScriptSource.h" #include "third_party/WebKit/public/web/WebView.h" #include "ui/base/resource/resource_bundle.h" namespace atom { namespace { bool GetIPCObject(v8::Isolate* isolate, v8::Local<v8::Context> context, v8::Local<v8::Object>* ipc) { v8::Local<v8::String> key = mate::StringToV8(isolate, "ipc"); v8::Local<v8::Value> value = context->Global()->GetHiddenValue(key); if (value.IsEmpty() || !value->IsObject()) return false; *ipc = value->ToObject(); return true; } std::vector<v8::Local<v8::Value>> ListValueToVector( v8::Isolate* isolate, const base::ListValue& list) { v8::Local<v8::Value> array = mate::ConvertToV8(isolate, list); std::vector<v8::Local<v8::Value>> result; mate::ConvertFromV8(isolate, array, &result); return result; } base::StringPiece NetResourceProvider(int key) { if (key == IDR_DIR_HEADER_HTML) { base::StringPiece html_data = ui::ResourceBundle::GetSharedInstance().GetRawDataResource( IDR_DIR_HEADER_HTML); return html_data; } return base::StringPiece(); } } // namespace AtomRenderViewObserver::AtomRenderViewObserver( content::RenderView* render_view, AtomRendererClient* renderer_client) : content::RenderViewObserver(render_view), renderer_client_(renderer_client), document_created_(false) { // Initialise resource for directory listing. net::NetModule::SetResourceProvider(NetResourceProvider); } AtomRenderViewObserver::~AtomRenderViewObserver() { } void AtomRenderViewObserver::DidCreateDocumentElement( blink::WebLocalFrame* frame) { document_created_ = true; // Read --zoom-factor from command line. std::string zoom_factor_str = base::CommandLine::ForCurrentProcess()-> GetSwitchValueASCII(switches::kZoomFactor);; if (zoom_factor_str.empty()) return; double zoom_factor; if (!base::StringToDouble(zoom_factor_str, &zoom_factor)) return; double zoom_level = blink::WebView::zoomFactorToZoomLevel(zoom_factor); frame->view()->setZoomLevel(zoom_level); } void AtomRenderViewObserver::DraggableRegionsChanged(blink::WebFrame* frame) { blink::WebVector<blink::WebDraggableRegion> webregions = frame->document().draggableRegions(); std::vector<DraggableRegion> regions; for (size_t i = 0; i < webregions.size(); ++i) { DraggableRegion region; region.bounds = webregions[i].bounds; region.draggable = webregions[i].draggable; regions.push_back(region); } Send(new AtomViewHostMsg_UpdateDraggableRegions(routing_id(), regions)); } bool AtomRenderViewObserver::OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(AtomRenderViewObserver, message) IPC_MESSAGE_HANDLER(AtomViewMsg_Message, OnBrowserMessage) IPC_MESSAGE_HANDLER(AtomViewMsg_ExecuteJavaScript, OnJavaScriptExecuteRequest) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void AtomRenderViewObserver::OnBrowserMessage(const base::string16& channel, const base::ListValue& args) { if (!document_created_) return; if (!render_view()->GetWebView()) return; blink::WebFrame* frame = render_view()->GetWebView()->mainFrame(); if (!frame || frame->isWebRemoteFrame()) return; v8::Isolate* isolate = blink::mainThreadIsolate(); v8::HandleScope handle_scope(isolate); v8::Local<v8::Context> context = frame->mainWorldScriptContext(); v8::Context::Scope context_scope(context); v8::Local<v8::Object> ipc; if (GetIPCObject(isolate, context, &ipc)) { mate::EmitEvent(isolate, ipc, channel, ListValueToVector(isolate, args)); } } void AtomRenderViewObserver::OnJavaScriptExecuteRequest( const base::string16& code, bool has_user_gesture) { if (!document_created_) return; if (!render_view()->GetWebView()) return; scoped_ptr<blink::WebScopedUserGesture> gesture( has_user_gesture ? new blink::WebScopedUserGesture : nullptr); v8::Isolate* isolate = blink::mainThreadIsolate(); v8::HandleScope handle_scope(isolate); blink::WebFrame* frame = render_view()->GetWebView()->mainFrame(); frame->executeScriptAndReturnValue(blink::WebScriptSource(code)); } } // namespace atom