mirror of
https://github.com/servo/servo.git
synced 2025-08-15 02:15:33 +01:00
script/compositor: Send mouseleave
events when cursor moves between <iframe>
s (#38539)
Properly send `mouseleave` events when the cursor moves between `<iframe>`s. This allows a better handling of cursor changes and status text updates. Specifically, we do not need to continuously update the cursor and the value can be cached in the `Document`. In addition, status updates can now be sent properly when moving focus between `<iframe>`s. Note that style updates for `:hover` values are still broken, but less so than before. Now the hover state on the `Node` is updated, but for some reason the restyle isn't taking place properly. This maintains the status quo as far as behavior goes when hover moves between `<iframe>`s. This change also adds a helper data structure to `Document` which will eventually be responsible for event handling. Testing: Cursor and status change are currently very hard to test as the API test harness makes this difficult at the moment. Signed-off-by: Martin Robinson <mrobinson@igalia.com> Co-authored-by: Oriol Brufau <obrufau@igalia.com>
This commit is contained in:
parent
82ca2b92cd
commit
b75c3feb97
11 changed files with 307 additions and 210 deletions
|
@ -329,23 +329,23 @@ impl WebViewRenderer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn dispatch_point_input_event(&self, mut event: InputEvent) -> bool {
|
pub(crate) fn dispatch_input_event_with_hit_testing(&self, mut event: InputEvent) -> bool {
|
||||||
// Events that do not need to do hit testing are sent directly to the
|
let event_point = event.point();
|
||||||
// constellation to filter down.
|
let hit_test_result = match event_point {
|
||||||
let Some(point) = event.point() else {
|
Some(point) => {
|
||||||
return false;
|
let hit_test_result = self
|
||||||
};
|
.global
|
||||||
|
.borrow()
|
||||||
// If we can't find a pipeline to send this event to, we cannot continue.
|
.hit_test_at_point(point)
|
||||||
let Some(result) = self
|
.into_iter()
|
||||||
.global
|
.nth(0);
|
||||||
.borrow()
|
if hit_test_result.is_none() {
|
||||||
.hit_test_at_point(point)
|
warn!("Empty hit test result for input event, ignoring.");
|
||||||
.into_iter()
|
return false;
|
||||||
.nth(0)
|
}
|
||||||
else {
|
hit_test_result
|
||||||
warn!("Empty hit test result for input event, ignoring.");
|
},
|
||||||
return false;
|
None => None,
|
||||||
};
|
};
|
||||||
|
|
||||||
match event {
|
match event {
|
||||||
|
@ -353,7 +353,7 @@ impl WebViewRenderer {
|
||||||
touch_event.init_sequence_id(self.touch_handler.current_sequence_id);
|
touch_event.init_sequence_id(self.touch_handler.current_sequence_id);
|
||||||
},
|
},
|
||||||
InputEvent::MouseMove(_) => {
|
InputEvent::MouseMove(_) => {
|
||||||
self.global.borrow_mut().last_mouse_move_position = Some(point);
|
self.global.borrow_mut().last_mouse_move_position = event_point;
|
||||||
},
|
},
|
||||||
InputEvent::MouseLeave(_) => {
|
InputEvent::MouseLeave(_) => {
|
||||||
self.global.borrow_mut().last_mouse_move_position = None;
|
self.global.borrow_mut().last_mouse_move_position = None;
|
||||||
|
@ -363,7 +363,7 @@ impl WebViewRenderer {
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Err(error) = self.global.borrow().constellation_sender.send(
|
if let Err(error) = self.global.borrow().constellation_sender.send(
|
||||||
EmbedderToConstellationMessage::ForwardInputEvent(self.id, event, Some(result)),
|
EmbedderToConstellationMessage::ForwardInputEvent(self.id, event, hit_test_result),
|
||||||
) {
|
) {
|
||||||
warn!("Sending event to constellation failed ({error:?}).");
|
warn!("Sending event to constellation failed ({error:?}).");
|
||||||
false
|
false
|
||||||
|
@ -438,11 +438,11 @@ impl WebViewRenderer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
self.dispatch_point_input_event(event);
|
self.dispatch_input_event_with_hit_testing(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn send_touch_event(&mut self, event: TouchEvent) -> bool {
|
fn send_touch_event(&mut self, event: TouchEvent) -> bool {
|
||||||
self.dispatch_point_input_event(InputEvent::Touch(event))
|
self.dispatch_input_event_with_hit_testing(InputEvent::Touch(event))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn on_touch_event(&mut self, event: TouchEvent) {
|
pub(crate) fn on_touch_event(&mut self, event: TouchEvent) {
|
||||||
|
@ -706,13 +706,15 @@ impl WebViewRenderer {
|
||||||
/// <http://w3c.github.io/touch-events/#mouse-events>
|
/// <http://w3c.github.io/touch-events/#mouse-events>
|
||||||
fn simulate_mouse_click(&mut self, point: DevicePoint) {
|
fn simulate_mouse_click(&mut self, point: DevicePoint) {
|
||||||
let button = MouseButton::Left;
|
let button = MouseButton::Left;
|
||||||
self.dispatch_point_input_event(InputEvent::MouseMove(MouseMoveEvent::new(point)));
|
self.dispatch_input_event_with_hit_testing(InputEvent::MouseMove(MouseMoveEvent::new(
|
||||||
self.dispatch_point_input_event(InputEvent::MouseButton(MouseButtonEvent::new(
|
point,
|
||||||
|
)));
|
||||||
|
self.dispatch_input_event_with_hit_testing(InputEvent::MouseButton(MouseButtonEvent::new(
|
||||||
MouseButtonAction::Down,
|
MouseButtonAction::Down,
|
||||||
button,
|
button,
|
||||||
point,
|
point,
|
||||||
)));
|
)));
|
||||||
self.dispatch_point_input_event(InputEvent::MouseButton(MouseButtonEvent::new(
|
self.dispatch_input_event_with_hit_testing(InputEvent::MouseButton(MouseButtonEvent::new(
|
||||||
MouseButtonAction::Up,
|
MouseButtonAction::Up,
|
||||||
button,
|
button,
|
||||||
point,
|
point,
|
||||||
|
|
|
@ -2939,34 +2939,8 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let pipeline_id = match &hit_test_result {
|
let Some(webview) = self.webviews.get_mut(webview_id) else {
|
||||||
Some(hit_test) => hit_test.pipeline_id,
|
warn!("Got input event for unknown WebViewId: {webview_id:?}");
|
||||||
None => {
|
|
||||||
// If there's no hit test, send to the focused browsing context of the given webview.
|
|
||||||
let Some(browsing_context_id) = self
|
|
||||||
.webviews
|
|
||||||
.get(webview_id)
|
|
||||||
.map(|webview| webview.focused_browsing_context_id)
|
|
||||||
else {
|
|
||||||
warn!("Handling InputEvent for an unknown webview: {webview_id}");
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
|
|
||||||
let Some(pipeline_id) = self
|
|
||||||
.browsing_contexts
|
|
||||||
.get(&browsing_context_id)
|
|
||||||
.map(|context| context.pipeline_id)
|
|
||||||
else {
|
|
||||||
warn!("{browsing_context_id}: Got InputEvent for nonexistent browsing context");
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
|
|
||||||
pipeline_id
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
let Some(pipeline) = self.pipelines.get(&pipeline_id) else {
|
|
||||||
debug!("Got event for pipeline ({pipeline_id}) after closure");
|
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -2976,13 +2950,7 @@ where
|
||||||
active_keyboard_modifiers,
|
active_keyboard_modifiers,
|
||||||
event,
|
event,
|
||||||
};
|
};
|
||||||
|
webview.forward_input_event(event, &self.pipelines, &self.browsing_contexts);
|
||||||
if let Err(error) = pipeline
|
|
||||||
.event_loop
|
|
||||||
.send(ScriptThreadMessage::SendInputEvent(pipeline_id, event))
|
|
||||||
{
|
|
||||||
self.handle_send_error(pipeline_id, error);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[servo_tracing::instrument(skip_all)]
|
#[servo_tracing::instrument(skip_all)]
|
||||||
|
|
|
@ -2,9 +2,17 @@
|
||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
use base::id::BrowsingContextId;
|
use std::collections::HashMap;
|
||||||
use embedder_traits::Theme;
|
|
||||||
|
|
||||||
|
use base::id::{BrowsingContextId, PipelineId};
|
||||||
|
use embedder_traits::{InputEvent, MouseLeaveEvent, Theme};
|
||||||
|
use euclid::Point2D;
|
||||||
|
use log::warn;
|
||||||
|
use script_traits::{ConstellationInputEvent, ScriptThreadMessage};
|
||||||
|
use style_traits::CSSPixel;
|
||||||
|
|
||||||
|
use crate::browsingcontext::BrowsingContext;
|
||||||
|
use crate::pipeline::Pipeline;
|
||||||
use crate::session_history::JointSessionHistory;
|
use crate::session_history::JointSessionHistory;
|
||||||
|
|
||||||
/// The `Constellation`'s view of a `WebView` in the embedding layer. This tracks all of the
|
/// The `Constellation`'s view of a `WebView` in the embedding layer. This tracks all of the
|
||||||
|
@ -15,6 +23,14 @@ pub(crate) struct ConstellationWebView {
|
||||||
/// context.
|
/// context.
|
||||||
pub focused_browsing_context_id: BrowsingContextId,
|
pub focused_browsing_context_id: BrowsingContextId,
|
||||||
|
|
||||||
|
/// The [`BrowsingContextId`] of the currently hovered browsing context, to use for
|
||||||
|
/// knowing which frame is currently receiving cursor events.
|
||||||
|
pub hovered_browsing_context_id: Option<BrowsingContextId>,
|
||||||
|
|
||||||
|
/// The last mouse move point in the coordinate space of the Pipeline that it
|
||||||
|
/// happened int.
|
||||||
|
pub last_mouse_move_point: Point2D<f32, CSSPixel>,
|
||||||
|
|
||||||
/// The joint session history for this webview.
|
/// The joint session history for this webview.
|
||||||
pub session_history: JointSessionHistory,
|
pub session_history: JointSessionHistory,
|
||||||
|
|
||||||
|
@ -27,6 +43,8 @@ impl ConstellationWebView {
|
||||||
pub(crate) fn new(focused_browsing_context_id: BrowsingContextId) -> Self {
|
pub(crate) fn new(focused_browsing_context_id: BrowsingContextId) -> Self {
|
||||||
Self {
|
Self {
|
||||||
focused_browsing_context_id,
|
focused_browsing_context_id,
|
||||||
|
hovered_browsing_context_id: None,
|
||||||
|
last_mouse_move_point: Default::default(),
|
||||||
session_history: JointSessionHistory::new(),
|
session_history: JointSessionHistory::new(),
|
||||||
theme: Theme::Light,
|
theme: Theme::Light,
|
||||||
}
|
}
|
||||||
|
@ -42,4 +60,91 @@ impl ConstellationWebView {
|
||||||
pub(crate) fn theme(&self) -> Theme {
|
pub(crate) fn theme(&self) -> Theme {
|
||||||
self.theme
|
self.theme
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn target_pipeline_id_for_input_event(
|
||||||
|
&self,
|
||||||
|
event: &ConstellationInputEvent,
|
||||||
|
browsing_contexts: &HashMap<BrowsingContextId, BrowsingContext>,
|
||||||
|
) -> Option<PipelineId> {
|
||||||
|
if let Some(hit_test_result) = &event.hit_test_result {
|
||||||
|
return Some(hit_test_result.pipeline_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If there's no hit test, send the event to either the hovered or focused browsing context,
|
||||||
|
// depending on the event type.
|
||||||
|
let browsing_context_id = if matches!(event.event, InputEvent::MouseLeave(_)) {
|
||||||
|
self.hovered_browsing_context_id
|
||||||
|
.unwrap_or(self.focused_browsing_context_id)
|
||||||
|
} else {
|
||||||
|
self.focused_browsing_context_id
|
||||||
|
};
|
||||||
|
|
||||||
|
Some(browsing_contexts.get(&browsing_context_id)?.pipeline_id)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn forward_input_event(
|
||||||
|
&mut self,
|
||||||
|
event: ConstellationInputEvent,
|
||||||
|
pipelines: &HashMap<PipelineId, Pipeline>,
|
||||||
|
browsing_contexts: &HashMap<BrowsingContextId, BrowsingContext>,
|
||||||
|
) {
|
||||||
|
let Some(pipeline_id) = self.target_pipeline_id_for_input_event(&event, browsing_contexts)
|
||||||
|
else {
|
||||||
|
warn!("Unknown pipeline for input event. Ignoring.");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
let Some(pipeline) = pipelines.get(&pipeline_id) else {
|
||||||
|
warn!("Unknown pipeline id {pipeline_id:?} for input event. Ignoring.");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut update_hovered_browsing_context = |newly_hovered_browsing_context_id| {
|
||||||
|
let old_hovered_context_id = std::mem::replace(
|
||||||
|
&mut self.hovered_browsing_context_id,
|
||||||
|
newly_hovered_browsing_context_id,
|
||||||
|
);
|
||||||
|
if old_hovered_context_id == newly_hovered_browsing_context_id {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let Some(old_hovered_context_id) = old_hovered_context_id else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
let Some(pipeline) = browsing_contexts
|
||||||
|
.get(&old_hovered_context_id)
|
||||||
|
.and_then(|browsing_context| pipelines.get(&browsing_context.pipeline_id))
|
||||||
|
else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut synthetic_mouse_leave_event = event.clone();
|
||||||
|
synthetic_mouse_leave_event.event = InputEvent::MouseLeave(MouseLeaveEvent {
|
||||||
|
focus_moving_to_another_iframe: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
let _ = pipeline
|
||||||
|
.event_loop
|
||||||
|
.send(ScriptThreadMessage::SendInputEvent(
|
||||||
|
pipeline.id,
|
||||||
|
synthetic_mouse_leave_event,
|
||||||
|
));
|
||||||
|
};
|
||||||
|
|
||||||
|
if let InputEvent::MouseLeave(_) = &event.event {
|
||||||
|
update_hovered_browsing_context(None);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let InputEvent::MouseMove(_) = &event.event {
|
||||||
|
update_hovered_browsing_context(Some(pipeline.browsing_context_id));
|
||||||
|
self.last_mouse_move_point = event
|
||||||
|
.hit_test_result
|
||||||
|
.as_ref()
|
||||||
|
.expect("MouseMove events should always have hit tests.")
|
||||||
|
.point_in_viewport;
|
||||||
|
}
|
||||||
|
|
||||||
|
let _ = pipeline
|
||||||
|
.event_loop
|
||||||
|
.send(ScriptThreadMessage::SendInputEvent(pipeline.id, event));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,8 +30,8 @@ use dom_struct::dom_struct;
|
||||||
use embedder_traits::{
|
use embedder_traits::{
|
||||||
AllowOrDeny, AnimationState, ContextMenuResult, Cursor, EditingActionEvent, EmbedderMsg,
|
AllowOrDeny, AnimationState, ContextMenuResult, Cursor, EditingActionEvent, EmbedderMsg,
|
||||||
FocusSequenceNumber, ImeEvent, InputEvent, LoadStatus, MouseButton, MouseButtonAction,
|
FocusSequenceNumber, ImeEvent, InputEvent, LoadStatus, MouseButton, MouseButtonAction,
|
||||||
MouseButtonEvent, ScrollEvent, TouchEvent, TouchEventType, TouchId, UntrustedNodeAddress,
|
MouseButtonEvent, MouseLeaveEvent, ScrollEvent, TouchEvent, TouchEventType, TouchId,
|
||||||
WheelEvent,
|
UntrustedNodeAddress, WheelEvent,
|
||||||
};
|
};
|
||||||
use encoding_rs::{Encoding, UTF_8};
|
use encoding_rs::{Encoding, UTF_8};
|
||||||
use euclid::Point2D;
|
use euclid::Point2D;
|
||||||
|
@ -141,6 +141,7 @@ use crate::dom::cssstylesheet::CSSStyleSheet;
|
||||||
use crate::dom::customelementregistry::CustomElementDefinition;
|
use crate::dom::customelementregistry::CustomElementDefinition;
|
||||||
use crate::dom::customevent::CustomEvent;
|
use crate::dom::customevent::CustomEvent;
|
||||||
use crate::dom::datatransfer::DataTransfer;
|
use crate::dom::datatransfer::DataTransfer;
|
||||||
|
use crate::dom::document_event_handler::DocumentEventHandler;
|
||||||
use crate::dom::documentfragment::DocumentFragment;
|
use crate::dom::documentfragment::DocumentFragment;
|
||||||
use crate::dom::documentorshadowroot::{
|
use crate::dom::documentorshadowroot::{
|
||||||
DocumentOrShadowRoot, ServoStylesheetInDocument, StylesheetSource,
|
DocumentOrShadowRoot, ServoStylesheetInDocument, StylesheetSource,
|
||||||
|
@ -322,6 +323,8 @@ pub(crate) struct Document {
|
||||||
#[ignore_malloc_size_of = "defined in selectors"]
|
#[ignore_malloc_size_of = "defined in selectors"]
|
||||||
#[no_trace]
|
#[no_trace]
|
||||||
quirks_mode: Cell<QuirksMode>,
|
quirks_mode: Cell<QuirksMode>,
|
||||||
|
/// A helper used to process and store data related to input event handling.
|
||||||
|
event_handler: DocumentEventHandler,
|
||||||
/// Caches for the getElement methods
|
/// Caches for the getElement methods
|
||||||
id_map: DomRefCell<HashMapTracedValues<Atom, Vec<Dom<Element>>>>,
|
id_map: DomRefCell<HashMapTracedValues<Atom, Vec<Dom<Element>>>>,
|
||||||
name_map: DomRefCell<HashMapTracedValues<Atom, Vec<Dom<Element>>>>,
|
name_map: DomRefCell<HashMapTracedValues<Atom, Vec<Dom<Element>>>>,
|
||||||
|
@ -2001,7 +2004,6 @@ impl Document {
|
||||||
pub(crate) unsafe fn handle_mouse_move_event(
|
pub(crate) unsafe fn handle_mouse_move_event(
|
||||||
&self,
|
&self,
|
||||||
input_event: &ConstellationInputEvent,
|
input_event: &ConstellationInputEvent,
|
||||||
prev_mouse_over_target: &MutNullableDom<Element>,
|
|
||||||
can_gc: CanGc,
|
can_gc: CanGc,
|
||||||
) {
|
) {
|
||||||
// Ignore all incoming events without a hit test.
|
// Ignore all incoming events without a hit test.
|
||||||
|
@ -2021,7 +2023,9 @@ impl Document {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
let target_has_changed = prev_mouse_over_target
|
let target_has_changed = self
|
||||||
|
.event_handler
|
||||||
|
.current_hover_target
|
||||||
.get()
|
.get()
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.is_none_or(|old_target| old_target != &new_target);
|
.is_none_or(|old_target| old_target != &new_target);
|
||||||
|
@ -2030,7 +2034,7 @@ impl Document {
|
||||||
// dispatch mouseout to the previous one, mouseover to the new one.
|
// dispatch mouseout to the previous one, mouseover to the new one.
|
||||||
if target_has_changed {
|
if target_has_changed {
|
||||||
// Dispatch mouseout and mouseleave to previous target.
|
// Dispatch mouseout and mouseleave to previous target.
|
||||||
if let Some(old_target) = prev_mouse_over_target.get() {
|
if let Some(old_target) = self.event_handler.current_hover_target.get() {
|
||||||
let old_target_is_ancestor_of_new_target = old_target
|
let old_target_is_ancestor_of_new_target = old_target
|
||||||
.upcast::<Node>()
|
.upcast::<Node>()
|
||||||
.is_ancestor_of(new_target.upcast::<Node>());
|
.is_ancestor_of(new_target.upcast::<Node>());
|
||||||
|
@ -2078,9 +2082,6 @@ impl Document {
|
||||||
.inclusive_ancestors(ShadowIncluding::No)
|
.inclusive_ancestors(ShadowIncluding::No)
|
||||||
.filter_map(DomRoot::downcast::<Element>)
|
.filter_map(DomRoot::downcast::<Element>)
|
||||||
{
|
{
|
||||||
if element.hover_state() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
element.set_hover_state(true);
|
element.set_hover_state(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2094,7 +2095,9 @@ impl Document {
|
||||||
can_gc,
|
can_gc,
|
||||||
);
|
);
|
||||||
|
|
||||||
let moving_from = prev_mouse_over_target
|
let moving_from = self
|
||||||
|
.event_handler
|
||||||
|
.current_hover_target
|
||||||
.get()
|
.get()
|
||||||
.map(|old_target| DomRoot::from_ref(old_target.upcast::<Node>()));
|
.map(|old_target| DomRoot::from_ref(old_target.upcast::<Node>()));
|
||||||
let event_target = DomRoot::from_ref(new_target.upcast::<Node>());
|
let event_target = DomRoot::from_ref(new_target.upcast::<Node>());
|
||||||
|
@ -2120,56 +2123,107 @@ impl Document {
|
||||||
can_gc,
|
can_gc,
|
||||||
);
|
);
|
||||||
|
|
||||||
// If the target has changed then store the current mouse over target for next frame.
|
self.update_current_hover_target_and_status(Some(new_target));
|
||||||
if target_has_changed {
|
|
||||||
prev_mouse_over_target.set(Some(&new_target));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn set_cursor(&self, cursor: Cursor) {
|
fn update_current_hover_target_and_status(&self, new_hover_target: Option<DomRoot<Element>>) {
|
||||||
self.send_to_embedder(EmbedderMsg::SetCursor(self.webview_id(), cursor));
|
let previous_hover_target = self.event_handler.current_hover_target.get();
|
||||||
|
if previous_hover_target == new_hover_target {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.event_handler
|
||||||
|
.current_hover_target
|
||||||
|
.set(new_hover_target.as_deref());
|
||||||
|
|
||||||
|
// If the new hover target is an anchor with a status value, inform the embedder
|
||||||
|
// of the new value.
|
||||||
|
let window = self.window();
|
||||||
|
if let Some(anchor) = new_hover_target.and_then(|new_hover_target| {
|
||||||
|
new_hover_target
|
||||||
|
.upcast::<Node>()
|
||||||
|
.inclusive_ancestors(ShadowIncluding::No)
|
||||||
|
.filter_map(DomRoot::downcast::<HTMLAnchorElement>)
|
||||||
|
.next()
|
||||||
|
}) {
|
||||||
|
let status = anchor
|
||||||
|
.upcast::<Element>()
|
||||||
|
.get_attribute(&ns!(), &local_name!("href"))
|
||||||
|
.and_then(|href| {
|
||||||
|
let value = href.value();
|
||||||
|
let url = self.url();
|
||||||
|
url.join(&value).map(|url| url.to_string()).ok()
|
||||||
|
});
|
||||||
|
window.send_to_embedder(EmbedderMsg::Status(self.webview_id(), status));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// No state was set above, which means that the new value of the status in the embedder
|
||||||
|
// should be `None`. Set that now. If `previous_hover_target` is `None` that means this
|
||||||
|
// is the first mouse move event we are seeing after getting the cursor. In that case,
|
||||||
|
// we also clear the status.
|
||||||
|
if previous_hover_target.is_none_or(|previous_hover_target| {
|
||||||
|
previous_hover_target
|
||||||
|
.upcast::<Node>()
|
||||||
|
.inclusive_ancestors(ShadowIncluding::No)
|
||||||
|
.filter_map(DomRoot::downcast::<HTMLAnchorElement>)
|
||||||
|
.next()
|
||||||
|
.is_some()
|
||||||
|
}) {
|
||||||
|
window.send_to_embedder(EmbedderMsg::Status(window.webview_id(), None));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unsafe_code)]
|
#[allow(unsafe_code)]
|
||||||
pub(crate) fn handle_mouse_leave_event(
|
pub(crate) fn handle_mouse_leave_event(
|
||||||
&self,
|
&self,
|
||||||
input_event: &ConstellationInputEvent,
|
input_event: &ConstellationInputEvent,
|
||||||
|
mouse_leave_event: &MouseLeaveEvent,
|
||||||
can_gc: CanGc,
|
can_gc: CanGc,
|
||||||
) {
|
) {
|
||||||
// Ignore all incoming events without a hit test.
|
if let Some(current_hover_target) = self.event_handler.current_hover_target.get() {
|
||||||
let Some(hit_test_result) = self.window.hit_test_from_input_event(input_event) else {
|
let current_hover_target = current_hover_target.upcast::<Node>();
|
||||||
return;
|
for element in current_hover_target
|
||||||
};
|
.inclusive_ancestors(ShadowIncluding::No)
|
||||||
|
.filter_map(DomRoot::downcast::<Element>)
|
||||||
|
{
|
||||||
|
element.set_hover_state(false);
|
||||||
|
element.set_active_state(false);
|
||||||
|
}
|
||||||
|
|
||||||
self.window()
|
if let Some(hit_test_result) = self
|
||||||
.send_to_embedder(EmbedderMsg::Status(self.webview_id(), None));
|
.window()
|
||||||
|
.hit_test_from_point_in_viewport(self.event_handler.most_recent_mousemove_point)
|
||||||
for element in hit_test_result
|
{
|
||||||
.node
|
self.fire_mouse_event(
|
||||||
.inclusive_ancestors(ShadowIncluding::No)
|
current_hover_target.upcast(),
|
||||||
.filter_map(DomRoot::downcast::<Element>)
|
FireMouseEventType::Out,
|
||||||
{
|
EventBubbles::Bubbles,
|
||||||
element.set_hover_state(false);
|
EventCancelable::Cancelable,
|
||||||
element.set_active_state(false);
|
&hit_test_result,
|
||||||
|
input_event,
|
||||||
|
can_gc,
|
||||||
|
);
|
||||||
|
self.handle_mouse_enter_leave_event(
|
||||||
|
DomRoot::from_ref(current_hover_target),
|
||||||
|
None,
|
||||||
|
FireMouseEventType::Leave,
|
||||||
|
&hit_test_result,
|
||||||
|
input_event,
|
||||||
|
can_gc,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
self.fire_mouse_event(
|
self.event_handler.current_cursor.set(None);
|
||||||
hit_test_result.node.upcast(),
|
self.event_handler.current_hover_target.set(None);
|
||||||
FireMouseEventType::Out,
|
|
||||||
EventBubbles::Bubbles,
|
// If focus is moving to another frame, it will decide what the new status text is, but if
|
||||||
EventCancelable::Cancelable,
|
// this mouse leave event is leaving the WebView entirely, then clear it.
|
||||||
&hit_test_result,
|
if !mouse_leave_event.focus_moving_to_another_iframe {
|
||||||
input_event,
|
self.window()
|
||||||
can_gc,
|
.send_to_embedder(EmbedderMsg::Status(self.webview_id(), None));
|
||||||
);
|
}
|
||||||
self.handle_mouse_enter_leave_event(
|
|
||||||
hit_test_result.node.clone(),
|
|
||||||
None,
|
|
||||||
FireMouseEventType::Leave,
|
|
||||||
&hit_test_result,
|
|
||||||
input_event,
|
|
||||||
can_gc,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_mouse_enter_leave_event(
|
fn handle_mouse_enter_leave_event(
|
||||||
|
@ -4123,6 +4177,14 @@ impl Document {
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn set_cursor(&self, cursor: Cursor) {
|
||||||
|
if Some(cursor) == self.event_handler.current_cursor.get() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.event_handler.current_cursor.set(Some(cursor));
|
||||||
|
self.send_to_embedder(EmbedderMsg::SetCursor(self.webview_id(), cursor));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn is_character_value_key(key: &Key) -> bool {
|
fn is_character_value_key(key: &Key) -> bool {
|
||||||
|
@ -4321,6 +4383,7 @@ impl Document {
|
||||||
url: DomRefCell::new(url),
|
url: DomRefCell::new(url),
|
||||||
// https://dom.spec.whatwg.org/#concept-document-quirks
|
// https://dom.spec.whatwg.org/#concept-document-quirks
|
||||||
quirks_mode: Cell::new(QuirksMode::NoQuirks),
|
quirks_mode: Cell::new(QuirksMode::NoQuirks),
|
||||||
|
event_handler: DocumentEventHandler::default(),
|
||||||
id_map: DomRefCell::new(HashMapTracedValues::new()),
|
id_map: DomRefCell::new(HashMapTracedValues::new()),
|
||||||
name_map: DomRefCell::new(HashMapTracedValues::new()),
|
name_map: DomRefCell::new(HashMapTracedValues::new()),
|
||||||
// https://dom.spec.whatwg.org/#concept-document-encoding
|
// https://dom.spec.whatwg.org/#concept-document-encoding
|
||||||
|
@ -4821,14 +4884,14 @@ impl Document {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn element_state_will_change(&self, el: &Element) {
|
pub(crate) fn element_state_will_change(&self, element: &Element) {
|
||||||
let mut entry = self.ensure_pending_restyle(el);
|
let mut entry = self.ensure_pending_restyle(element);
|
||||||
if entry.snapshot.is_none() {
|
if entry.snapshot.is_none() {
|
||||||
entry.snapshot = Some(Snapshot::new());
|
entry.snapshot = Some(Snapshot::new());
|
||||||
}
|
}
|
||||||
let snapshot = entry.snapshot.as_mut().unwrap();
|
let snapshot = entry.snapshot.as_mut().unwrap();
|
||||||
if snapshot.state.is_none() {
|
if snapshot.state.is_none() {
|
||||||
snapshot.state = Some(el.state());
|
snapshot.state = Some(element.state());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
29
components/script/dom/document_event_handler.rs
Normal file
29
components/script/dom/document_event_handler.rs
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
|
use std::cell::Cell;
|
||||||
|
|
||||||
|
use embedder_traits::Cursor;
|
||||||
|
use euclid::Point2D;
|
||||||
|
use style_traits::CSSPixel;
|
||||||
|
|
||||||
|
use crate::dom::bindings::root::MutNullableDom;
|
||||||
|
use crate::dom::types::Element;
|
||||||
|
|
||||||
|
/// The [`DocumentEventHandler`] is a structure responsible for handling events for
|
||||||
|
/// the [`crate::Document`] and storing data related to event handling. It exists to
|
||||||
|
/// decrease the size of the [`crate::Document`] structure.
|
||||||
|
#[derive(Default, JSTraceable, MallocSizeOf)]
|
||||||
|
#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
|
||||||
|
pub(crate) struct DocumentEventHandler {
|
||||||
|
/// The element that is currently hovered by the cursor.
|
||||||
|
pub(crate) current_hover_target: MutNullableDom<Element>,
|
||||||
|
/// The most recent mouse movement point, used for processing `mouseleave` events.
|
||||||
|
#[no_trace]
|
||||||
|
pub(crate) most_recent_mousemove_point: Point2D<f32, CSSPixel>,
|
||||||
|
/// The currently set [`Cursor`] or `None` if the `Document` isn't being hovered
|
||||||
|
/// by the cursor.
|
||||||
|
#[no_trace]
|
||||||
|
pub(crate) current_cursor: Cell<Option<Cursor>>,
|
||||||
|
}
|
|
@ -5425,8 +5425,8 @@ impl Element {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn set_focus_state(&self, value: bool) {
|
pub(crate) fn set_focus_state(&self, value: bool) {
|
||||||
self.set_state(ElementState::FOCUS, value);
|
|
||||||
self.upcast::<Node>().dirty(NodeDamage::Other);
|
self.upcast::<Node>().dirty(NodeDamage::Other);
|
||||||
|
self.set_state(ElementState::FOCUS, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn hover_state(&self) -> bool {
|
pub(crate) fn hover_state(&self) -> bool {
|
||||||
|
|
|
@ -298,6 +298,7 @@ pub(crate) mod dissimilaroriginlocation;
|
||||||
pub(crate) mod dissimilaroriginwindow;
|
pub(crate) mod dissimilaroriginwindow;
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub(crate) mod document;
|
pub(crate) mod document;
|
||||||
|
pub(crate) mod document_event_handler;
|
||||||
pub(crate) mod documentfragment;
|
pub(crate) mod documentfragment;
|
||||||
pub(crate) mod documentorshadowroot;
|
pub(crate) mod documentorshadowroot;
|
||||||
pub(crate) mod documenttype;
|
pub(crate) mod documenttype;
|
||||||
|
|
|
@ -2599,21 +2599,25 @@ impl Window {
|
||||||
self.layout().query_elements_from_point(point, flags)
|
self.layout().query_elements_from_point(point, flags)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unsafe_code)]
|
|
||||||
pub(crate) fn hit_test_from_input_event(
|
pub(crate) fn hit_test_from_input_event(
|
||||||
&self,
|
&self,
|
||||||
input_event: &ConstellationInputEvent,
|
input_event: &ConstellationInputEvent,
|
||||||
) -> Option<HitTestResult> {
|
) -> Option<HitTestResult> {
|
||||||
let compositor_hit_test_result = input_event.hit_test_result.as_ref()?;
|
self.hit_test_from_point_in_viewport(
|
||||||
|
input_event.hit_test_result.as_ref()?.point_in_viewport,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(unsafe_code)]
|
||||||
|
pub(crate) fn hit_test_from_point_in_viewport(
|
||||||
|
&self,
|
||||||
|
point_in_frame: Point2D<f32, CSSPixel>,
|
||||||
|
) -> Option<HitTestResult> {
|
||||||
let result = self
|
let result = self
|
||||||
.elements_from_point_query(
|
.elements_from_point_query(point_in_frame.cast_unit(), ElementsFromPointFlags::empty())
|
||||||
compositor_hit_test_result.point_in_viewport.cast_unit(),
|
|
||||||
ElementsFromPointFlags::empty(),
|
|
||||||
)
|
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.nth(0)?;
|
.nth(0)?;
|
||||||
|
|
||||||
let point_in_frame = compositor_hit_test_result.point_in_viewport;
|
|
||||||
let point_relative_to_initial_containing_block =
|
let point_relative_to_initial_containing_block =
|
||||||
point_in_frame + self.scroll_offset().cast_unit();
|
point_in_frame + self.scroll_offset().cast_unit();
|
||||||
|
|
||||||
|
|
|
@ -49,15 +49,14 @@ use devtools_traits::{
|
||||||
};
|
};
|
||||||
use embedder_traits::user_content_manager::UserContentManager;
|
use embedder_traits::user_content_manager::UserContentManager;
|
||||||
use embedder_traits::{
|
use embedder_traits::{
|
||||||
EmbedderMsg, FocusSequenceNumber, InputEvent, JavaScriptEvaluationError,
|
FocusSequenceNumber, InputEvent, JavaScriptEvaluationError, JavaScriptEvaluationId,
|
||||||
JavaScriptEvaluationId, MediaSessionActionType, MouseButton, MouseButtonAction,
|
MediaSessionActionType, MouseButton, MouseButtonAction, MouseButtonEvent, Theme,
|
||||||
MouseButtonEvent, Theme, ViewportDetails, WebDriverScriptCommand,
|
ViewportDetails, WebDriverScriptCommand,
|
||||||
};
|
};
|
||||||
use euclid::Point2D;
|
use euclid::Point2D;
|
||||||
use euclid::default::Rect;
|
use euclid::default::Rect;
|
||||||
use fonts::{FontContext, SystemFontServiceProxy};
|
use fonts::{FontContext, SystemFontServiceProxy};
|
||||||
use headers::{HeaderMapExt, LastModified, ReferrerPolicy as ReferrerPolicyHeader};
|
use headers::{HeaderMapExt, LastModified, ReferrerPolicy as ReferrerPolicyHeader};
|
||||||
use html5ever::{local_name, ns};
|
|
||||||
use http::header::REFRESH;
|
use http::header::REFRESH;
|
||||||
use hyper_serde::Serde;
|
use hyper_serde::Serde;
|
||||||
use ipc_channel::ipc;
|
use ipc_channel::ipc;
|
||||||
|
@ -115,9 +114,7 @@ use crate::dom::bindings::conversions::{
|
||||||
use crate::dom::bindings::inheritance::Castable;
|
use crate::dom::bindings::inheritance::Castable;
|
||||||
use crate::dom::bindings::refcounted::Trusted;
|
use crate::dom::bindings::refcounted::Trusted;
|
||||||
use crate::dom::bindings::reflector::DomGlobal;
|
use crate::dom::bindings::reflector::DomGlobal;
|
||||||
use crate::dom::bindings::root::{
|
use crate::dom::bindings::root::{Dom, DomRoot, RootCollection, ThreadLocalStackRoots};
|
||||||
Dom, DomRoot, MutNullableDom, RootCollection, ThreadLocalStackRoots,
|
|
||||||
};
|
|
||||||
use crate::dom::bindings::settings_stack::AutoEntryScript;
|
use crate::dom::bindings::settings_stack::AutoEntryScript;
|
||||||
use crate::dom::bindings::str::DOMString;
|
use crate::dom::bindings::str::DOMString;
|
||||||
use crate::dom::bindings::trace::{HashMapTracedValues, JSTraceable};
|
use crate::dom::bindings::trace::{HashMapTracedValues, JSTraceable};
|
||||||
|
@ -130,11 +127,10 @@ use crate::dom::document::{
|
||||||
};
|
};
|
||||||
use crate::dom::element::Element;
|
use crate::dom::element::Element;
|
||||||
use crate::dom::globalscope::GlobalScope;
|
use crate::dom::globalscope::GlobalScope;
|
||||||
use crate::dom::htmlanchorelement::HTMLAnchorElement;
|
|
||||||
use crate::dom::htmliframeelement::HTMLIFrameElement;
|
use crate::dom::htmliframeelement::HTMLIFrameElement;
|
||||||
use crate::dom::htmlslotelement::HTMLSlotElement;
|
use crate::dom::htmlslotelement::HTMLSlotElement;
|
||||||
use crate::dom::mutationobserver::MutationObserver;
|
use crate::dom::mutationobserver::MutationObserver;
|
||||||
use crate::dom::node::{Node, NodeTraits, ShadowIncluding};
|
use crate::dom::node::NodeTraits;
|
||||||
use crate::dom::servoparser::{ParserContext, ServoParser};
|
use crate::dom::servoparser::{ParserContext, ServoParser};
|
||||||
use crate::dom::types::DebuggerGlobalScope;
|
use crate::dom::types::DebuggerGlobalScope;
|
||||||
#[cfg(feature = "webgpu")]
|
#[cfg(feature = "webgpu")]
|
||||||
|
@ -253,9 +249,6 @@ pub struct ScriptThread {
|
||||||
/// The JavaScript runtime.
|
/// The JavaScript runtime.
|
||||||
js_runtime: Rc<Runtime>,
|
js_runtime: Rc<Runtime>,
|
||||||
|
|
||||||
/// The topmost element over the mouse.
|
|
||||||
topmost_mouse_over_target: MutNullableDom<Element>,
|
|
||||||
|
|
||||||
/// List of pipelines that have been owned and closed by this script thread.
|
/// List of pipelines that have been owned and closed by this script thread.
|
||||||
#[no_trace]
|
#[no_trace]
|
||||||
closed_pipelines: DomRefCell<HashSet<PipelineId>>,
|
closed_pipelines: DomRefCell<HashSet<PipelineId>>,
|
||||||
|
@ -978,7 +971,6 @@ impl ScriptThread {
|
||||||
timer_scheduler: Default::default(),
|
timer_scheduler: Default::default(),
|
||||||
microtask_queue,
|
microtask_queue,
|
||||||
js_runtime,
|
js_runtime,
|
||||||
topmost_mouse_over_target: MutNullableDom::new(Default::default()),
|
|
||||||
closed_pipelines: DomRefCell::new(HashSet::new()),
|
closed_pipelines: DomRefCell::new(HashSet::new()),
|
||||||
mutation_observer_microtask_queued: Default::default(),
|
mutation_observer_microtask_queued: Default::default(),
|
||||||
mutation_observers: Default::default(),
|
mutation_observers: Default::default(),
|
||||||
|
@ -1053,59 +1045,7 @@ impl ScriptThread {
|
||||||
input_event: &ConstellationInputEvent,
|
input_event: &ConstellationInputEvent,
|
||||||
can_gc: CanGc,
|
can_gc: CanGc,
|
||||||
) {
|
) {
|
||||||
// Get the previous target temporarily
|
unsafe { document.handle_mouse_move_event(input_event, can_gc) }
|
||||||
let prev_mouse_over_target = self.topmost_mouse_over_target.get();
|
|
||||||
|
|
||||||
unsafe {
|
|
||||||
document.handle_mouse_move_event(input_event, &self.topmost_mouse_over_target, can_gc)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Short-circuit if nothing changed
|
|
||||||
if self.topmost_mouse_over_target.get() == prev_mouse_over_target {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut state_already_changed = false;
|
|
||||||
|
|
||||||
// Notify Constellation about the topmost anchor mouse over target.
|
|
||||||
let window = document.window();
|
|
||||||
if let Some(target) = self.topmost_mouse_over_target.get() {
|
|
||||||
if let Some(anchor) = target
|
|
||||||
.upcast::<Node>()
|
|
||||||
.inclusive_ancestors(ShadowIncluding::No)
|
|
||||||
.filter_map(DomRoot::downcast::<HTMLAnchorElement>)
|
|
||||||
.next()
|
|
||||||
{
|
|
||||||
let status = anchor
|
|
||||||
.upcast::<Element>()
|
|
||||||
.get_attribute(&ns!(), &local_name!("href"))
|
|
||||||
.and_then(|href| {
|
|
||||||
let value = href.value();
|
|
||||||
let url = document.url();
|
|
||||||
url.join(&value).map(|url| url.to_string()).ok()
|
|
||||||
});
|
|
||||||
let event = EmbedderMsg::Status(document.webview_id(), status);
|
|
||||||
window.send_to_embedder(event);
|
|
||||||
|
|
||||||
state_already_changed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// We might have to reset the anchor state
|
|
||||||
if !state_already_changed {
|
|
||||||
if let Some(target) = prev_mouse_over_target {
|
|
||||||
if target
|
|
||||||
.upcast::<Node>()
|
|
||||||
.inclusive_ancestors(ShadowIncluding::No)
|
|
||||||
.filter_map(DomRoot::downcast::<HTMLAnchorElement>)
|
|
||||||
.next()
|
|
||||||
.is_some()
|
|
||||||
{
|
|
||||||
let event = EmbedderMsg::Status(window.webview_id(), None);
|
|
||||||
window.send_to_embedder(event);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Process compositor events as part of a "update the rendering task".
|
/// Process compositor events as part of a "update the rendering task".
|
||||||
|
@ -1131,12 +1071,10 @@ impl ScriptThread {
|
||||||
document.handle_mouse_button_event(mouse_button_event, &event, can_gc);
|
document.handle_mouse_button_event(mouse_button_event, &event, can_gc);
|
||||||
},
|
},
|
||||||
InputEvent::MouseMove(_) => {
|
InputEvent::MouseMove(_) => {
|
||||||
// The event itself is unecessary here, because the point in the viewport is in the hit test.
|
|
||||||
self.process_mouse_move_event(&document, &event, can_gc);
|
self.process_mouse_move_event(&document, &event, can_gc);
|
||||||
},
|
},
|
||||||
InputEvent::MouseLeave(_) => {
|
InputEvent::MouseLeave(mouse_leave_event) => {
|
||||||
self.topmost_mouse_over_target.take();
|
document.handle_mouse_leave_event(&event, &mouse_leave_event, can_gc);
|
||||||
document.handle_mouse_leave_event(&event, can_gc);
|
|
||||||
},
|
},
|
||||||
InputEvent::Touch(touch_event) => {
|
InputEvent::Touch(touch_event) => {
|
||||||
let touch_result = document.handle_touch_event(touch_event, &event, can_gc);
|
let touch_result = document.handle_touch_event(touch_event, &event, can_gc);
|
||||||
|
@ -3062,13 +3000,6 @@ impl ScriptThread {
|
||||||
debug!("{id}: Clearing animations");
|
debug!("{id}: Clearing animations");
|
||||||
document.animations().clear();
|
document.animations().clear();
|
||||||
|
|
||||||
// We don't want to dispatch `mouseout` event pointing to non-existing element
|
|
||||||
if let Some(target) = self.topmost_mouse_over_target.get() {
|
|
||||||
if target.upcast::<Node>().owner_doc() == document {
|
|
||||||
self.topmost_mouse_over_target.set(None);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// We discard the browsing context after requesting layout shut down,
|
// We discard the browsing context after requesting layout shut down,
|
||||||
// to avoid running layout on detached iframes.
|
// to avoid running layout on detached iframes.
|
||||||
let window = document.window();
|
let window = document.window();
|
||||||
|
|
|
@ -43,7 +43,7 @@ impl InputEvent {
|
||||||
InputEvent::Keyboard(..) => None,
|
InputEvent::Keyboard(..) => None,
|
||||||
InputEvent::MouseButton(event) => Some(event.point),
|
InputEvent::MouseButton(event) => Some(event.point),
|
||||||
InputEvent::MouseMove(event) => Some(event.point),
|
InputEvent::MouseMove(event) => Some(event.point),
|
||||||
InputEvent::MouseLeave(event) => Some(event.point),
|
InputEvent::MouseLeave(_) => None,
|
||||||
InputEvent::Touch(event) => Some(event.point),
|
InputEvent::Touch(event) => Some(event.point),
|
||||||
InputEvent::Wheel(event) => Some(event.point),
|
InputEvent::Wheel(event) => Some(event.point),
|
||||||
InputEvent::Scroll(..) => None,
|
InputEvent::Scroll(..) => None,
|
||||||
|
@ -218,15 +218,9 @@ impl MouseMoveEvent {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, Deserialize, Serialize)]
|
#[derive(Clone, Copy, Debug, Default, Deserialize, Serialize)]
|
||||||
pub struct MouseLeaveEvent {
|
pub struct MouseLeaveEvent {
|
||||||
pub point: DevicePoint,
|
pub focus_moving_to_another_iframe: bool,
|
||||||
}
|
|
||||||
|
|
||||||
impl MouseLeaveEvent {
|
|
||||||
pub fn new(point: DevicePoint) -> Self {
|
|
||||||
Self { point }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The type of input represented by a multi-touch event.
|
/// The type of input represented by a multi-touch event.
|
||||||
|
|
|
@ -624,17 +624,17 @@ impl WindowPortsMethods for Window {
|
||||||
if webview.rect().contains(point) {
|
if webview.rect().contains(point) {
|
||||||
webview.notify_input_event(InputEvent::MouseMove(MouseMoveEvent::new(point)));
|
webview.notify_input_event(InputEvent::MouseMove(MouseMoveEvent::new(point)));
|
||||||
} else if webview.rect().contains(previous_point) {
|
} else if webview.rect().contains(previous_point) {
|
||||||
webview.notify_input_event(InputEvent::MouseLeave(MouseLeaveEvent::new(
|
webview.notify_input_event(InputEvent::MouseLeave(MouseLeaveEvent::default()));
|
||||||
previous_point,
|
|
||||||
)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
self.webview_relative_mouse_point.set(point);
|
self.webview_relative_mouse_point.set(point);
|
||||||
},
|
},
|
||||||
WindowEvent::CursorLeft { .. } => {
|
WindowEvent::CursorLeft { .. } => {
|
||||||
let point = self.webview_relative_mouse_point.get();
|
if webview
|
||||||
if webview.rect().contains(point) {
|
.rect()
|
||||||
webview.notify_input_event(InputEvent::MouseLeave(MouseLeaveEvent::new(point)));
|
.contains(self.webview_relative_mouse_point.get())
|
||||||
|
{
|
||||||
|
webview.notify_input_event(InputEvent::MouseLeave(MouseLeaveEvent::default()));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
WindowEvent::MouseWheel { delta, .. } => {
|
WindowEvent::MouseWheel { delta, .. } => {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue