diff --git a/src/config.rs b/src/config.rs index cfdcb05..ac78436 100644 --- a/src/config.rs +++ b/src/config.rs @@ -41,6 +41,10 @@ pub struct CliArgs { pub resource_dir: Option, /// Override the user agent pub user_agent: Option, + /// Script to run on document started to load + pub init_script: Option, + /// The directory to load userscripts from + pub userscripts_directory: Option, /// Initial window's zoom level pub zoom_level: Option, } @@ -101,6 +105,18 @@ fn parse_cli_args() -> Result { "Override the user agent", "'VersoView/1.0'", ); + opts.optopt( + "", + "init-script", + "Script to run on document started to load", + "console.log('hello world')", + ); + opts.optopt( + "", + "userscripts-directory", + "The directory to load userscripts from", + "resources/user-agent-js/", + ); opts.optopt( "w", @@ -173,6 +189,8 @@ fn parse_cli_args() -> Result { }; let user_agent = matches.opt_str("user-agent"); + let init_script = matches.opt_str("init-script"); + let userscripts_directory = matches.opt_str("userscripts-directory"); let mut window_attributes = winit::window::Window::default_attributes(); @@ -237,6 +255,8 @@ fn parse_cli_args() -> Result { devtools_port, profiler_settings, user_agent, + init_script, + userscripts_directory, zoom_level, }) } @@ -257,6 +277,10 @@ impl Config { opts.time_profiler_trace_path = profiler_settings.trace_path.clone(); } + if let Some(ref userscripts_directory) = args.userscripts_directory { + opts.userscripts = Some(userscripts_directory.clone()); + } + let resource_dir = args.resource_dir.clone().unwrap_or(resources_dir_path()); Self { diff --git a/src/verso.rs b/src/verso.rs index 75e5cb9..803b6cd 100644 --- a/src/verso.rs +++ b/src/verso.rs @@ -114,6 +114,7 @@ impl Verso { .clone() .unwrap_or_else(|| default_user_agent_string().to_string()) .into(); + let init_script = config.args.init_script.clone(); let zoom_level = config.args.zoom_level; config.init(); @@ -391,6 +392,8 @@ impl Verso { window.create_webview(&constellation_sender, initial_url.into()); } + window.set_init_script(init_script); + let mut windows = HashMap::new(); windows.insert(window.id(), (window, webrender_document)); diff --git a/src/webview.rs b/src/webview.rs index a870dff..0679b78 100644 --- a/src/webview.rs +++ b/src/webview.rs @@ -5,7 +5,9 @@ use crossbeam_channel::Sender; use embedder_traits::{CompositorEventVariant, EmbedderMsg, PromptDefinition}; use ipc_channel::ipc; use script_traits::{ - webdriver_msg::{WebDriverJSResult, WebDriverScriptCommand}, + webdriver_msg::{ + WebDriverJSError, WebDriverJSResult, WebDriverJSValue, WebDriverScriptCommand, + }, TraversalDirection, WebDriverCommandMsg, }; use servo_url::ServoUrl; @@ -27,7 +29,7 @@ pub struct WebView { } impl WebView { - /// Create a web view from Winit window. + /// Create a web view. pub fn new(webview_id: WebViewId, rect: DeviceIntRect) -> Self { Self { webview_id, rect } } @@ -68,8 +70,7 @@ impl Window { ) { log::trace!("Verso WebView {webview_id:?} is handling Embedder message: {message:?}",); match message { - EmbedderMsg::LoadStart - | EmbedderMsg::HeadParsed + EmbedderMsg::HeadParsed | EmbedderMsg::WebViewOpened(_) | EmbedderMsg::WebViewClosed(_) => { // Most WebView messages are ignored because it's done by compositor. @@ -83,6 +84,11 @@ impl Window { w ); } + EmbedderMsg::LoadStart => { + if let Some(init_script) = &self.init_script { + execute_script_async(&sender, &webview_id, init_script); + } + } EmbedderMsg::LoadComplete => { self.window.request_redraw(); send_to_constellation(sender, ConstellationMsg::FocusWebView(webview_id)); @@ -328,3 +334,47 @@ impl Window { false } } + +/// Blocking execute a script on this webview +pub fn execute_script( + constellation_sender: &Sender, + webview: &WebViewId, + js: impl ToString, +) -> Result { + let (result_sender, result_receiver) = ipc::channel::().unwrap(); + send_to_constellation( + constellation_sender, + ConstellationMsg::WebDriverCommand(script_traits::WebDriverCommandMsg::ScriptCommand( + webview.0, + WebDriverScriptCommand::ExecuteScript(js.to_string(), result_sender), + )), + ); + result_receiver.recv().unwrap() +} + +/// Execute a script asynchronous on this webview +pub fn execute_script_async( + constellation_sender: &Sender, + webview: &WebViewId, + js: impl ToString, +) { + execute_script_async_with_callback(constellation_sender, webview, js, |_| {}) +} + +/// Execute a script asynchronous on this webview with a callback processing the result +pub fn execute_script_async_with_callback( + constellation_sender: &Sender, + webview: &WebViewId, + js: impl ToString, + callback: impl FnOnce(Result) + Send + 'static, +) { + let (result_sender, result_receiver) = ipc::channel::().unwrap(); + send_to_constellation( + constellation_sender, + ConstellationMsg::WebDriverCommand(script_traits::WebDriverCommandMsg::ScriptCommand( + webview.0, + WebDriverScriptCommand::ExecuteAsyncScript(js.to_string(), result_sender), + )), + ); + std::thread::spawn(move || callback(result_receiver.recv().unwrap())); +} diff --git a/src/window.rs b/src/window.rs index d9bde9a..3c40fb8 100644 --- a/src/window.rs +++ b/src/window.rs @@ -52,6 +52,8 @@ pub struct Window { pub(crate) panel: Option, /// The WebView of this window. pub(crate) webview: Option, + /// Script to run on document started to load + pub(crate) init_script: Option, /// The mouse physical position in the web view. mouse_position: Cell>>, /// Modifiers state of the keyboard. @@ -114,6 +116,7 @@ impl Window { surface, panel: None, webview: None, + init_script: None, mouse_position: Default::default(), modifiers_state: Cell::new(ModifiersState::default()), history: vec![], @@ -157,6 +160,7 @@ impl Window { surface, panel: None, webview: None, + init_script: None, mouse_position: Default::default(), modifiers_state: Cell::new(ModifiersState::default()), history: vec![], @@ -226,6 +230,11 @@ impl Window { log::debug!("Verso Window {:?} adds webview {}", self.id(), webview_id); } + /// Set the init script that runs on document started to load. + pub fn set_init_script(&mut self, init_script: Option) { + self.init_script = init_script; + } + /// Handle Winit window event and return a boolean to indicate if the compositor should repaint immediately. pub fn handle_winit_window_event( &mut self,