mirror of
https://github.com/servo/servo.git
synced 2025-08-06 06:00:15 +01:00
servoshell: Do not focus and raise new auxiliary WebDriver-created WebView
s (#37284)
For Desktop port of `request_open_auxiliary_webview`, stay on the original WebView if the request originates WebDriver. This is to make sure `webdriver_server::handle_new_window` does not focus the new window, according to spec. Seec7eba2dbba/tests/wpt/tests/webdriver/tests/classic/new_window/new_window.py (L31-L37)
**To clarify**: this won't change the behaviour when user interacts, but only affects WebDriver [New Window](https://w3c.github.io/webdriver/#new-window). Testing: `./mach test-wpt -r --log-raw "D:/servo log/all.txt" ./tests/wpt/tests/webdriver/tests/classic --product servodriver` based on96b0973037
--------- Signed-off-by: Euclid Ye <yezhizhenjiakang@gmail.com>
This commit is contained in:
parent
0fa3de3937
commit
a3c792e5aa
27 changed files with 76 additions and 158 deletions
|
@ -4702,25 +4702,38 @@ where
|
|||
WebDriverCommandMsg::CloseWebView(webview_id) => {
|
||||
self.handle_close_top_level_browsing_context(webview_id);
|
||||
},
|
||||
WebDriverCommandMsg::NewWebView(webview_id, sender, load_sender) => {
|
||||
let (chan, port) = match ipc::channel() {
|
||||
WebDriverCommandMsg::NewWebView(
|
||||
originating_webview_id,
|
||||
response_sender,
|
||||
load_status_sender,
|
||||
) => {
|
||||
let (embedder_sender, receiver) = match ipc::channel() {
|
||||
Ok(result) => result,
|
||||
Err(error) => return warn!("Failed to create channel: {error:?}"),
|
||||
};
|
||||
self.embedder_proxy
|
||||
.send(EmbedderMsg::AllowOpeningWebView(webview_id, chan));
|
||||
let (webview_id, viewport_details) = match port.recv() {
|
||||
Ok(Some((webview_id, viewport_details))) => (webview_id, viewport_details),
|
||||
self.embedder_proxy.send(EmbedderMsg::AllowOpeningWebView(
|
||||
originating_webview_id,
|
||||
embedder_sender,
|
||||
));
|
||||
let (new_webview_id, viewport_details) = match receiver.recv() {
|
||||
Ok(Some((new_webview_id, viewport_details))) => {
|
||||
(new_webview_id, viewport_details)
|
||||
},
|
||||
Ok(None) => return warn!("Embedder refused to allow opening webview"),
|
||||
Err(error) => return warn!("Failed to receive webview id: {error:?}"),
|
||||
};
|
||||
self.handle_new_top_level_browsing_context(
|
||||
ServoUrl::parse_with_base(None, "about:blank").expect("Infallible parse"),
|
||||
webview_id,
|
||||
new_webview_id,
|
||||
viewport_details,
|
||||
Some(load_sender),
|
||||
Some(load_status_sender),
|
||||
);
|
||||
let _ = sender.send(webview_id);
|
||||
if let Err(error) = response_sender.send(new_webview_id) {
|
||||
error!(
|
||||
"WebDriverCommandMsg::NewWebView: IPC error when sending new_webview_id \
|
||||
to webdriver server: {error}"
|
||||
);
|
||||
}
|
||||
},
|
||||
WebDriverCommandMsg::FocusWebView(webview_id) => {
|
||||
self.handle_focus_web_view(webview_id);
|
||||
|
|
|
@ -466,26 +466,40 @@ impl WindowProxy {
|
|||
features: DOMString,
|
||||
can_gc: CanGc,
|
||||
) -> Fallible<Option<DomRoot<WindowProxy>>> {
|
||||
// Step 4.
|
||||
// Step 5. If target is the empty string, then set target to "_blank".
|
||||
let non_empty_target = match target.as_ref() {
|
||||
"" => DOMString::from("_blank"),
|
||||
_ => target,
|
||||
};
|
||||
// Step 5
|
||||
// Step 6. Let tokenizedFeatures be the result of tokenizing features.
|
||||
let tokenized_features = tokenize_open_features(features);
|
||||
// Step 7-9
|
||||
// Step 7 - 8.
|
||||
// If tokenizedFeatures["noreferrer"] exists, then set noreferrer to
|
||||
// the result of parsing tokenizedFeatures["noreferrer"] as a boolean feature.
|
||||
let noreferrer = parse_open_feature_boolean(&tokenized_features, "noreferrer");
|
||||
|
||||
// Step 9. Let noopener be the result of getting noopener for window
|
||||
// open with sourceDocument, tokenizedFeatures, and urlRecord.
|
||||
let noopener = if noreferrer {
|
||||
true
|
||||
} else {
|
||||
parse_open_feature_boolean(&tokenized_features, "noopener")
|
||||
};
|
||||
// Step 10, 11
|
||||
// (TODO) Step 10. Remove tokenizedFeatures["noopener"] and tokenizedFeatures["noreferrer"].
|
||||
|
||||
// (TODO) Step 11. Let referrerPolicy be the empty string.
|
||||
// (TODO) Step 12. If noreferrer is true, then set referrerPolicy to "no-referrer".
|
||||
|
||||
// Step 13 - 14
|
||||
// Let targetNavigable and windowType be the result of applying the rules for
|
||||
// choosing a navigable given target, sourceDocument's node navigable, and noopener.
|
||||
// If targetNavigable is null, then return null.
|
||||
let (chosen, new) = match self.choose_browsing_context(non_empty_target, noopener) {
|
||||
(Some(chosen), new) => (chosen, new),
|
||||
(None, _) => return Ok(None),
|
||||
};
|
||||
// TODO Step 12, set up browsing context features.
|
||||
// TODO Step 15.2, Set up browsing context features for targetNavigable's
|
||||
// active browsing context given tokenizedFeatures.
|
||||
let target_document = match chosen.document() {
|
||||
Some(target_document) => target_document,
|
||||
None => return Ok(None),
|
||||
|
@ -496,7 +510,7 @@ impl WindowProxy {
|
|||
false
|
||||
};
|
||||
let target_window = target_document.window();
|
||||
// Step 13, and 14.4, will have happened elsewhere,
|
||||
// Step 15.3 and 15.4 will have happened elsewhere,
|
||||
// since we've created a new browsing context and loaded it with about:blank.
|
||||
if !url.is_empty() {
|
||||
let existing_document = self
|
||||
|
@ -504,18 +518,18 @@ impl WindowProxy {
|
|||
.get()
|
||||
.and_then(ScriptThread::find_document)
|
||||
.unwrap();
|
||||
// Step 14.1
|
||||
let url = match existing_document.url().join(&url) {
|
||||
Ok(url) => url,
|
||||
Err(_) => return Err(Error::Syntax),
|
||||
};
|
||||
// Step 14.3
|
||||
let referrer = if noreferrer {
|
||||
Referrer::NoReferrer
|
||||
} else {
|
||||
target_window.as_global_scope().get_referrer()
|
||||
};
|
||||
// Step 14.5
|
||||
// Step 15.5 Otherwise, navigate targetNavigable to urlRecord using sourceDocument,
|
||||
// with referrerPolicy set to referrerPolicy and exceptionsEnabled set to true.
|
||||
// FIXME: referrerPolicy may not be used properly here. exceptionsEnabled not used.
|
||||
let referrer_policy = target_document.get_referrer_policy();
|
||||
let pipeline_id = target_window.pipeline_id();
|
||||
let secure = target_window.as_global_scope().is_secure_context();
|
||||
|
@ -534,14 +548,13 @@ impl WindowProxy {
|
|||
} else {
|
||||
NavigationHistoryBehavior::Push
|
||||
};
|
||||
|
||||
target_window.load_url(history_handling, false, load_data, can_gc);
|
||||
}
|
||||
// Step 17 (Dis-owning has been done in create_auxiliary_browsing_context).
|
||||
if noopener {
|
||||
// Step 15 (Dis-owning has been done in create_auxiliary_browsing_context).
|
||||
return Ok(None);
|
||||
}
|
||||
// Step 17.
|
||||
// Step 18
|
||||
Ok(target_document.browsing_context())
|
||||
}
|
||||
|
||||
|
|
|
@ -79,6 +79,8 @@ fn find_node_by_unique_id(
|
|||
match documents.find_document(pipeline) {
|
||||
Some(doc) => find_node_by_unique_id_in_document(&doc, node_id),
|
||||
None => {
|
||||
// FIXME: This is unreacheable!! Because we already early return in Constellation
|
||||
// To be Fixed soon
|
||||
if ScriptThread::has_node_id(pipeline, &node_id) {
|
||||
Err(ErrorStatus::StaleElementReference)
|
||||
} else {
|
||||
|
|
|
@ -949,6 +949,7 @@ impl Handler {
|
|||
)))
|
||||
}
|
||||
|
||||
/// <https://w3c.github.io/webdriver/#new-window>
|
||||
fn handle_new_window(
|
||||
&mut self,
|
||||
_parameters: &NewWindowParameters,
|
||||
|
@ -956,11 +957,16 @@ impl Handler {
|
|||
let (sender, receiver) = ipc::channel().unwrap();
|
||||
|
||||
let session = self.session().unwrap();
|
||||
// Step 2. (TODO) If session's current top-level browsing context is no longer open,
|
||||
// return error with error code no such window.
|
||||
|
||||
let cmd_msg = WebDriverCommandMsg::NewWebView(
|
||||
session.webview_id,
|
||||
sender,
|
||||
self.load_status_sender.clone(),
|
||||
);
|
||||
// Step 5. Create a new top-level browsing context by running the window open steps.
|
||||
// This MUST be done without invoking the focusing steps.
|
||||
self.constellation_chan
|
||||
.send(EmbedderToConstellationMessage::WebDriverCommand(cmd_msg))
|
||||
.unwrap();
|
||||
|
@ -968,8 +974,6 @@ impl Handler {
|
|||
let mut handle = self.session.as_ref().unwrap().id.to_string();
|
||||
if let Ok(new_webview_id) = receiver.recv() {
|
||||
let session = self.session_mut().unwrap();
|
||||
session.webview_id = new_webview_id;
|
||||
session.browsing_context_id = BrowsingContextId::from(new_webview_id);
|
||||
let new_handle = Uuid::new_v4().to_string();
|
||||
handle = new_handle.clone();
|
||||
session.window_handles.insert(new_webview_id, new_handle);
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue