use crate::{AsyncWindowContext, Bounds, Pixels, PlatformInputHandler, View, ViewContext}; use std::ops::Range; /// Implement this trait to allow views to handle textual input when implementing an editor, field, etc. /// /// Once your view `V` 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]. pub trait InputHandler: 'static + Sized { fn text_for_range(&mut self, range: Range, cx: &mut ViewContext) -> Option; fn selected_text_range(&mut self, cx: &mut ViewContext) -> Option>; fn marked_text_range(&self, cx: &mut ViewContext) -> Option>; fn unmark_text(&mut self, cx: &mut ViewContext); fn replace_text_in_range( &mut self, range: Option>, text: &str, cx: &mut ViewContext, ); fn replace_and_mark_text_in_range( &mut self, range: Option>, new_text: &str, new_selected_range: Option>, cx: &mut ViewContext, ); 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, cx: AsyncWindowContext, } impl ElementInputHandler { /// Used in [Element::paint] with the element's bounds and a view context for its /// containing view. pub fn new(element_bounds: Bounds, cx: &mut ViewContext) -> Self { ElementInputHandler { view: cx.view().clone(), element_bounds, cx: cx.to_async(), } } } impl PlatformInputHandler for ElementInputHandler { fn selected_text_range(&mut self) -> Option> { self.view .update(&mut self.cx, |view, cx| view.selected_text_range(cx)) .ok() .flatten() } fn marked_text_range(&mut self) -> Option> { self.view .update(&mut self.cx, |view, cx| view.marked_text_range(cx)) .ok() .flatten() } fn text_for_range(&mut self, range_utf16: Range) -> Option { self.view .update(&mut self.cx, |view, cx| { view.text_for_range(range_utf16, cx) }) .ok() .flatten() } fn replace_text_in_range(&mut self, replacement_range: Option>, text: &str) { self.view .update(&mut self.cx, |view, cx| { view.replace_text_in_range(replacement_range, text, cx) }) .ok(); } fn replace_and_mark_text_in_range( &mut self, range_utf16: Option>, new_text: &str, new_selected_range: Option>, ) { self.view .update(&mut self.cx, |view, cx| { view.replace_and_mark_text_in_range(range_utf16, new_text, new_selected_range, cx) }) .ok(); } fn unmark_text(&mut self) { self.view .update(&mut self.cx, |view, cx| view.unmark_text(cx)) .ok(); } fn bounds_for_range(&mut self, range_utf16: Range) -> Option> { self.view .update(&mut self.cx, |view, cx| { view.bounds_for_range(range_utf16, self.element_bounds, cx) }) .ok() .flatten() } }