use std::{
cell::{
Ref,
RefCell,
},
rc::Rc,
};
use dioxus::prelude::*;
use freya_elements::{
self as dioxus_elements,
events::{
keyboard::Key,
KeyboardData,
MouseEvent,
},
};
use freya_hooks::{
use_applied_theme,
use_editable,
use_focus,
use_platform,
EditableConfig,
EditableEvent,
EditableMode,
InputTheme,
InputThemeWith,
TextEditor,
};
use winit::window::CursorIcon;
use crate::ScrollView;
#[derive(Default, Clone, PartialEq)]
pub enum InputMode {
#[default]
Shown,
Hidden(char),
}
impl InputMode {
pub fn new_password() -> Self {
Self::Hidden('*')
}
}
#[derive(Debug, Default, PartialEq, Clone, Copy)]
pub enum InputStatus {
#[default]
Idle,
Hovering,
}
#[derive(Clone)]
pub struct InputValidator {
valid: Rc<RefCell<bool>>,
text: Rc<RefCell<String>>,
}
impl InputValidator {
pub fn new(text: String) -> Self {
Self {
valid: Rc::new(RefCell::new(true)),
text: Rc::new(RefCell::new(text)),
}
}
pub fn text(&self) -> Ref<String> {
self.text.borrow()
}
pub fn set_valid(&self, is_valid: bool) {
*self.valid.borrow_mut() = is_valid;
}
pub fn is_valid(&self) -> bool {
*self.valid.borrow()
}
}
#[derive(Props, Clone, PartialEq)]
pub struct InputProps {
pub theme: Option<InputThemeWith>,
pub placeholder: Option<String>,
pub value: String,
pub onchange: EventHandler<String>,
#[props(default = InputMode::Shown, into)]
pub mode: InputMode,
#[props(default = false)]
pub auto_focus: bool,
pub onvalidate: Option<EventHandler<InputValidator>>,
}
#[allow(non_snake_case)]
pub fn Input(
InputProps {
theme,
value,
onchange,
mode,
placeholder,
auto_focus,
onvalidate,
}: InputProps,
) -> Element {
let platform = use_platform();
let mut status = use_signal(InputStatus::default);
let mut editable = use_editable(
|| EditableConfig::new(value.to_string()),
EditableMode::MultipleLinesSingleEditor,
);
let theme = use_applied_theme!(&theme, input);
let mut focus = use_focus();
let is_focused = focus.is_focused();
let display_placeholder = value.is_empty() && placeholder.is_some() && !is_focused;
if &value != editable.editor().read().rope() {
editable.editor_mut().write().set(&value);
editable.editor_mut().write().editor_history().clear();
}
use_drop(move || {
if *status.peek() == InputStatus::Hovering {
platform.set_cursor(CursorIcon::default());
}
});
let onkeydown = move |e: Event<KeyboardData>| {
if e.data.key != Key::Enter && e.data.key != Key::Tab {
e.stop_propagation();
editable.process_event(&EditableEvent::KeyDown(e.data));
let text = editable.editor().peek().to_string();
let apply_change = if let Some(onvalidate) = onvalidate {
let editor = editable.editor_mut();
let mut editor = editor.write();
let validator = InputValidator::new(text.clone());
onvalidate(validator.clone());
let is_valid = validator.is_valid();
if !is_valid {
let undo_result = editor.undo();
if let Some(idx) = undo_result {
editor.set_cursor_pos(idx);
}
editor.editor_history().clear_redos();
}
is_valid
} else {
true
};
if apply_change {
onchange.call(text);
}
}
};
let onkeyup = move |e: Event<KeyboardData>| {
e.stop_propagation();
editable.process_event(&EditableEvent::KeyUp(e.data));
};
let oninputmousedown = move |e: MouseEvent| {
if !display_placeholder {
editable.process_event(&EditableEvent::MouseDown(e.data, 0));
}
focus.focus();
};
let onmousedown = move |e: MouseEvent| {
e.stop_propagation();
if !display_placeholder {
editable.process_event(&EditableEvent::MouseDown(e.data, 0));
}
focus.focus();
};
let onmousemove = move |e: MouseEvent| {
editable.process_event(&EditableEvent::MouseMove(e.data, 0));
};
let onmouseenter = move |_| {
platform.set_cursor(CursorIcon::Text);
*status.write() = InputStatus::Hovering;
};
let onmouseleave = move |_| {
platform.set_cursor(CursorIcon::default());
*status.write() = InputStatus::default();
};
let onglobalclick = move |_| match *status.read() {
InputStatus::Idle if focus.is_focused() => {
focus.unfocus();
editable.process_event(&EditableEvent::Click);
}
InputStatus::Hovering => {
editable.process_event(&EditableEvent::Click);
}
_ => {}
};
let a11y_id = focus.attribute();
let cursor_reference = editable.cursor_attr();
let highlights = editable.highlights_attr(0);
let (background, cursor_char) = if focus.is_focused() {
(
theme.hover_background,
editable.editor().read().cursor_pos().to_string(),
)
} else {
(theme.background, "none".to_string())
};
let InputTheme {
border_fill,
width,
margin,
corner_radius,
font_theme,
placeholder_font_theme,
shadow,
..
} = theme;
let color = if display_placeholder {
placeholder_font_theme.color
} else {
font_theme.color
};
let text = match (mode, placeholder) {
(_, Some(placeholder)) if display_placeholder => placeholder,
(InputMode::Hidden(ch), _) => ch.to_string().repeat(value.len()),
(InputMode::Shown, _) => value,
};
rsx!(
rect {
width: "{width}",
direction: "vertical",
color: "{color}",
background: "{background}",
border: "1 inner {border_fill}",
shadow: "{shadow}",
corner_radius: "{corner_radius}",
margin: "{margin}",
main_align: "center",
cursor_reference,
a11y_id,
a11y_role: "text-input",
a11y_auto_focus: "{auto_focus}",
onkeydown,
onkeyup,
overflow: "clip",
onmousedown: oninputmousedown,
onmouseenter,
onmouseleave,
ScrollView {
height: "auto",
direction: "horizontal",
show_scrollbar: false,
paragraph {
min_width: "1",
margin: "8 12",
onglobalclick,
onmousedown,
onmousemove,
cursor_id: "0",
cursor_index: "{cursor_char}",
cursor_mode: "editable",
cursor_color: "{color}",
max_lines: "1",
highlights,
text {
"{text}"
}
}
}
}
)
}
#[cfg(test)]
mod test {
use freya::prelude::*;
use freya_testing::prelude::*;
#[tokio::test]
pub async fn input() {
fn input_app() -> Element {
let mut value = use_signal(|| "Hello, Worl".to_string());
rsx!(Input {
value: value.read().clone(),
onchange: move |new_value| {
value.set(new_value);
}
})
}
let mut utils = launch_test(input_app);
let root = utils.root();
let text = root.get(0).get(0).get(0).get(0).get(0).get(0);
utils.wait_for_update().await;
assert_eq!(text.get(0).text(), Some("Hello, Worl"));
assert_eq!(utils.focus_id(), ACCESSIBILITY_ROOT_ID);
utils.push_event(PlatformEvent::Mouse {
name: EventName::MouseDown,
cursor: (115., 25.).into(),
button: Some(MouseButton::Left),
});
utils.wait_for_update().await;
utils.wait_for_update().await;
utils.wait_for_update().await;
assert_ne!(utils.focus_id(), ACCESSIBILITY_ROOT_ID);
utils.push_event(PlatformEvent::Keyboard {
name: EventName::KeyDown,
key: Key::Character("d".to_string()),
code: Code::KeyD,
modifiers: Modifiers::default(),
});
utils.wait_for_update().await;
assert_eq!(text.get(0).text(), Some("Hello, World"));
}
#[tokio::test]
pub async fn validate() {
fn input_app() -> Element {
let mut value = use_signal(|| "A".to_string());
rsx!(Input {
value: value.read().clone(),
onvalidate: |validator: InputValidator| {
if validator.text().len() > 3 {
validator.set_valid(false)
}
},
onchange: move |new_value| {
value.set(new_value);
}
},)
}
let mut utils = launch_test(input_app);
let root = utils.root();
let text = root.get(0).get(0).get(0).get(0).get(0).get(0);
utils.wait_for_update().await;
assert_eq!(text.get(0).text(), Some("A"));
utils.push_event(PlatformEvent::Mouse {
name: EventName::MouseDown,
cursor: (115., 25.).into(),
button: Some(MouseButton::Left),
});
utils.wait_for_update().await;
utils.wait_for_update().await;
utils.wait_for_update().await;
assert_ne!(utils.focus_id(), ACCESSIBILITY_ROOT_ID);
for c in ['B', 'C', 'D', 'E', 'F', 'G'] {
utils.push_event(PlatformEvent::Keyboard {
name: EventName::KeyDown,
key: Key::Character(c.to_string()),
code: Code::Unidentified,
modifiers: Modifiers::default(),
});
utils.wait_for_update().await;
}
assert_eq!(text.get(0).text(), Some("ABC"));
}
}