use crate::{Bounds, InputHandler, Pixels, UTF16Selection, View, ViewContext, WindowContext}; use std::ops::Range; /// Implement this trait to allow views to handle textual input when implementing an editor, field, etc. /// /// Once your view implements this trait, you can use it to construct an [`ElementInputHandler`]. /// This input handler can then be assigned during paint by calling [`WindowContext::handle_input`]. /// /// See [`InputHandler`] for details on how to implement each method. pub trait ViewInputHandler: 'static + Sized { /// See [`InputHandler::text_for_range`] for details fn text_for_range( &mut self, range: Range, adjusted_range: &mut Option>, cx: &mut ViewContext, ) -> Option; /// See [`InputHandler::selected_text_range`] for details fn selected_text_range( &mut self, ignore_disabled_input: bool, cx: &mut ViewContext, ) -> Option; /// See [`InputHandler::marked_text_range`] for details fn marked_text_range(&self, cx: &mut ViewContext) -> Option>; /// See [`InputHandler::unmark_text`] for details fn unmark_text(&mut self, cx: &mut ViewContext); /// See [`InputHandler::replace_text_in_range`] for details fn replace_text_in_range( &mut self, range: Option>, text: &str, cx: &mut ViewContext, ); /// See [`InputHandler::replace_and_mark_text_in_range`] for details fn replace_and_mark_text_in_range( &mut self, range: Option>, new_text: &str, new_selected_range: Option>, cx: &mut ViewContext, ); /// See [`InputHandler::bounds_for_range`] for details fn bounds_for_range( &mut self, range_utf16: Range, element_bounds: Bounds, cx: &mut ViewContext, ) -> Option>; } /// The canonical implementation of [`PlatformInputHandler`]. Call [`WindowContext::handle_input`] /// with an instance during your element's paint. pub struct ElementInputHandler { view: View, element_bounds: Bounds, } impl ElementInputHandler { /// Used in [`Element::paint`][element_paint] with the element's bounds and a view context for its /// containing view. /// /// [element_paint]: crate::Element::paint pub fn new(element_bounds: Bounds, view: View) -> Self { ElementInputHandler { view, element_bounds, } } } impl InputHandler for ElementInputHandler { fn selected_text_range( &mut self, ignore_disabled_input: bool, cx: &mut WindowContext, ) -> Option { self.view.update(cx, |view, cx| { view.selected_text_range(ignore_disabled_input, cx) }) } fn marked_text_range(&mut self, cx: &mut WindowContext) -> Option> { self.view.update(cx, |view, cx| view.marked_text_range(cx)) } fn text_for_range( &mut self, range_utf16: Range, adjusted_range: &mut Option>, cx: &mut WindowContext, ) -> Option { self.view.update(cx, |view, cx| { view.text_for_range(range_utf16, adjusted_range, cx) }) } fn replace_text_in_range( &mut self, replacement_range: Option>, text: &str, cx: &mut WindowContext, ) { self.view.update(cx, |view, cx| { view.replace_text_in_range(replacement_range, text, cx) }); } fn replace_and_mark_text_in_range( &mut self, range_utf16: Option>, new_text: &str, new_selected_range: Option>, cx: &mut WindowContext, ) { self.view.update(cx, |view, cx| { view.replace_and_mark_text_in_range(range_utf16, new_text, new_selected_range, cx) }); } fn unmark_text(&mut self, cx: &mut WindowContext) { self.view.update(cx, |view, cx| view.unmark_text(cx)); } fn bounds_for_range( &mut self, range_utf16: Range, cx: &mut WindowContext, ) -> Option> { self.view.update(cx, |view, cx| { view.bounds_for_range(range_utf16, self.element_bounds, cx) }) } }