This commit is contained in:
Kenzie Raditya Tirtarahardja 2025-06-04 15:04:12 +08:00 committed by GitHub
commit ff50ea5c5b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 78 additions and 55 deletions

View file

@ -2280,15 +2280,6 @@ impl ScriptThread {
can_gc, can_gc,
) )
}, },
WebDriverScriptCommand::FocusElement(element_id, reply) => {
webdriver_handlers::handle_focus_element(
&documents,
pipeline_id,
element_id,
reply,
can_gc,
)
},
WebDriverScriptCommand::ElementClick(element_id, reply) => { WebDriverScriptCommand::ElementClick(element_id, reply) => {
webdriver_handlers::handle_element_click( webdriver_handlers::handle_element_click(
&documents, &documents,
@ -2394,6 +2385,20 @@ impl ScriptThread {
WebDriverScriptCommand::GetTitle(reply) => { WebDriverScriptCommand::GetTitle(reply) => {
webdriver_handlers::handle_get_title(&documents, pipeline_id, reply) webdriver_handlers::handle_get_title(&documents, pipeline_id, reply)
}, },
WebDriverScriptCommand::WillSendKeys(
strict_file_interactability,
text,
element_id,
reply,
) => webdriver_handlers::handle_will_send_keys(
&documents,
pipeline_id,
element_id,
text,
strict_file_interactability,
reply,
can_gc,
),
_ => (), _ => (),
} }
} }

View file

@ -843,24 +843,66 @@ pub(crate) fn handle_find_element_elements_tag_name(
.unwrap(); .unwrap();
} }
pub(crate) fn handle_focus_element( pub(crate) fn handle_will_send_keys(
documents: &DocumentCollection, documents: &DocumentCollection,
pipeline: PipelineId, pipeline: PipelineId,
element_id: String, element_id: String,
reply: IpcSender<Result<(), ErrorStatus>>, text: String,
strict_file_interactability: bool,
reply: IpcSender<Result<bool, ErrorStatus>>,
can_gc: CanGc, can_gc: CanGc,
) { ) {
reply reply
.send( .send(
find_node_by_unique_id(documents, pipeline, element_id).and_then(|node| { find_node_by_unique_id(documents, pipeline, element_id).and_then(|node| {
match node.downcast::<HTMLElement>() { // Step 6: Let file be true if element is input element
Some(element) => { // in the file upload state, or false otherwise
// Need a way to find if this actually succeeded let file_input = node
element.Focus(can_gc); .downcast::<HTMLInputElement>()
Ok(()) .filter(|&input_element| input_element.input_type() == InputType::File);
},
None => Err(ErrorStatus::UnknownError), // Step 7: If file is false or the session's strict file interactability
if file_input.is_none() || strict_file_interactability {
match node.downcast::<HTMLElement>() {
Some(element) => {
// Need a way to find if this actually succeeded
element.Focus(can_gc);
},
None => return Err(ErrorStatus::UnknownError),
}
} }
// Step 8 (file input)
if let Some(file_input) = file_input {
// Step 8.1: Let files be the result of splitting text
// on the newline (\n) character.
let files: Vec<DOMString> = text.split("\n").map(|s| s.into()).collect();
// Step 8.2
if files.is_empty() {
return Err(ErrorStatus::InvalidArgument);
}
// Step 8.3 - 8.4
if !file_input.Multiple() && files.len() > 1 {
return Err(ErrorStatus::InvalidArgument);
}
// Step 8.5
// TODO: Should return invalid argument error if file doesn't exist
// Step 8.6 - 8.7
// Input and change event already fired in `htmlinputelement.rs`.
file_input.SelectFiles(files, can_gc);
// Step 8.8
return Ok(false);
}
// TODO: Check non-typeable form control
// TODO: Check content editable
Ok(true)
}), }),
) )
.unwrap(); .unwrap();

View file

@ -130,7 +130,6 @@ pub enum WebDriverScriptCommand {
IpcSender<Result<Vec<String>, ErrorStatus>>, IpcSender<Result<Vec<String>, ErrorStatus>>,
), ),
FindElementElementsTagName(String, String, IpcSender<Result<Vec<String>, ErrorStatus>>), FindElementElementsTagName(String, String, IpcSender<Result<Vec<String>, ErrorStatus>>),
FocusElement(String, IpcSender<Result<(), ErrorStatus>>),
ElementClick(String, IpcSender<Result<Option<String>, ErrorStatus>>), ElementClick(String, IpcSender<Result<Option<String>, ErrorStatus>>),
GetActiveElement(IpcSender<Option<String>>), GetActiveElement(IpcSender<Option<String>>),
GetComputedRole(String, IpcSender<Result<Option<String>, ErrorStatus>>), GetComputedRole(String, IpcSender<Result<Option<String>, ErrorStatus>>),
@ -161,6 +160,8 @@ pub enum WebDriverScriptCommand {
IsEnabled(String, IpcSender<Result<bool, ErrorStatus>>), IsEnabled(String, IpcSender<Result<bool, ErrorStatus>>),
IsSelected(String, IpcSender<Result<bool, ErrorStatus>>), IsSelected(String, IpcSender<Result<bool, ErrorStatus>>),
GetTitle(IpcSender<String>), GetTitle(IpcSender<String>),
/// Match the element type before sending the event for webdriver `element send keys`.
WillSendKeys(bool, String, String, IpcSender<Result<bool, ErrorStatus>>),
} }
#[derive(Debug, Deserialize, Serialize)] #[derive(Debug, Deserialize, Serialize)]

View file

@ -1619,14 +1619,23 @@ impl Handler {
let (sender, receiver) = ipc::channel().unwrap(); let (sender, receiver) = ipc::channel().unwrap();
let cmd = WebDriverScriptCommand::FocusElement(element.to_string(), sender); let cmd = WebDriverScriptCommand::WillSendKeys(
self.session()?.strict_file_interactability,
keys.text.to_string(),
element.to_string(),
sender,
);
let cmd_msg = WebDriverCommandMsg::ScriptCommand(browsing_context_id, cmd); let cmd_msg = WebDriverCommandMsg::ScriptCommand(browsing_context_id, cmd);
self.constellation_chan self.constellation_chan
.send(EmbedderToConstellationMessage::WebDriverCommand(cmd_msg)) .send(EmbedderToConstellationMessage::WebDriverCommand(cmd_msg))
.unwrap(); .unwrap();
// TODO: distinguish the not found and not focusable cases // TODO: distinguish the not found and not focusable cases
wait_for_script_response(receiver)?.map_err(|error| WebDriverError::new(error, ""))?; // File input and non-typeable form control should have
// been handled in `webdriver_handler.rs`.
if wait_for_script_response(receiver)?.map_err(|error| WebDriverError::new(error, ""))? {
return Ok(WebDriverResponse::Void);
}
let input_events = send_keys(&keys.text); let input_events = send_keys(&keys.text);

View file

@ -1,7 +1,4 @@
[events.py] [events.py]
[test_file_upload]
expected: FAIL
[test_form_control_send_text[input\]] [test_form_control_send_text[input\]]
expected: FAIL expected: FAIL

View file

@ -1,50 +1,19 @@
[file_upload.py] [file_upload.py]
expected: TIMEOUT
[test_empty_text] [test_empty_text]
expected: FAIL expected: FAIL
[test_multiple_files]
expected: FAIL
[test_multiple_files_last_path_not_found] [test_multiple_files_last_path_not_found]
expected: FAIL expected: FAIL
[test_multiple_files_without_multiple_attribute]
expected: FAIL
[test_multiple_files_send_twice] [test_multiple_files_send_twice]
expected: FAIL expected: FAIL
[test_multiple_files_reset_with_element_clear] [test_multiple_files_reset_with_element_clear]
expected: FAIL expected: FAIL
[test_single_file]
expected: FAIL
[test_single_file_replaces_without_multiple_attribute]
expected: FAIL
[test_single_file_appends_with_multiple_attribute] [test_single_file_appends_with_multiple_attribute]
expected: FAIL expected: FAIL
[test_transparent]
expected: FAIL
[test_obscured]
expected: FAIL
[test_outside_viewport]
expected: FAIL
[test_hidden]
expected: FAIL
[test_display_none]
expected: FAIL
[test_not_focused]
expected: FAIL
[test_focused] [test_focused]
expected: ERROR expected: ERROR