Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

text cursor blink in TextEdit #4121

Closed
wants to merge 73 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
73 commits
Select commit Hold shift + click to select a range
b553092
Update visuals.rs
rustbasic Mar 2, 2024
11e31ea
Update builder.rs
rustbasic Mar 2, 2024
8dd93b4
Update visuals.rs
rustbasic Mar 2, 2024
e56ad08
Update builder.rs
rustbasic Mar 2, 2024
c6a0fca
Update builder.rs
rustbasic Mar 2, 2024
61362e5
Update builder.rs
rustbasic Mar 2, 2024
53991c4
Update builder.rs
rustbasic Mar 2, 2024
ae48093
Update visuals.rs
rustbasic Mar 2, 2024
f534359
Update visuals.rs
rustbasic Mar 2, 2024
86e0910
Update visuals.rs
rustbasic Mar 2, 2024
f748437
Update builder.rs
rustbasic Mar 2, 2024
443e68b
Merge branch 'emilk:master' into patch19
rustbasic Mar 5, 2024
5a55d0b
Merge branch 'emilk:master' into patch19
rustbasic Mar 8, 2024
790609c
Merge branch 'emilk:master' into patch19
rustbasic Mar 11, 2024
183500b
Merge branch 'emilk:master' into patch19
rustbasic Mar 12, 2024
8d21de6
Merge branch 'emilk:master' into patch19
rustbasic Mar 13, 2024
6b76c1c
Update state.rs
rustbasic Mar 13, 2024
e20c05f
Update builder.rs
rustbasic Mar 13, 2024
ff68bd2
Update builder.rs
rustbasic Mar 13, 2024
7ce3353
Update builder.rs
rustbasic Mar 14, 2024
35704f7
Update state.rs
rustbasic Mar 14, 2024
ee4a20b
Update builder.rs
rustbasic Mar 14, 2024
691308b
Update builder.rs
rustbasic Mar 14, 2024
bc1e54a
Update builder.rs
rustbasic Mar 14, 2024
ea247ec
Update builder.rs
rustbasic Mar 14, 2024
ded8e31
Update builder.rs
rustbasic Mar 14, 2024
c813943
Update builder.rs
rustbasic Mar 14, 2024
3c8bc78
Merge branch 'emilk:master' into patch19
rustbasic Mar 14, 2024
8cbe04b
Merge branch 'emilk:master' into patch19
rustbasic Mar 15, 2024
037c4ac
Merge branch 'emilk:master' into patch19
rustbasic Mar 18, 2024
ce54c56
Update style.rs
rustbasic Mar 20, 2024
0b3c88e
Update builder.rs
rustbasic Mar 20, 2024
043ac3f
Merge branch 'emilk:master' into patch19
rustbasic Mar 20, 2024
782fd9d
Merge branch 'emilk:master' into patch19
rustbasic Mar 20, 2024
01675d1
Update style.rs
rustbasic Mar 21, 2024
d5fcd5c
Merge branch 'emilk:master' into patch19
rustbasic Mar 21, 2024
55a7c5e
Update style.rs
rustbasic Mar 21, 2024
ea89cf6
Update builder.rs
rustbasic Mar 21, 2024
79eb765
Update visuals.rs
rustbasic Mar 21, 2024
8faf0f5
Update builder.rs
rustbasic Mar 21, 2024
894fc2f
Update style.rs
rustbasic Mar 21, 2024
6ca5eeb
Update visuals.rs
rustbasic Mar 21, 2024
f231579
Update style.rs
rustbasic Mar 21, 2024
14ef3f4
Update style.rs
rustbasic Mar 21, 2024
f62a193
Update visuals.rs
rustbasic Mar 21, 2024
0ba726e
Update visuals.rs
rustbasic Mar 21, 2024
79dddbf
Update visuals.rs
rustbasic Mar 21, 2024
08daf51
Update visuals.rs
rustbasic Mar 21, 2024
875114f
Merge branch 'emilk:master' into patch19
rustbasic Mar 22, 2024
eb64586
Merge branch 'emilk:master' into patch19
rustbasic Mar 25, 2024
c293e19
Merge branch 'master' into patch19
rustbasic Mar 27, 2024
956b853
Merge branch 'master' into patch19
rustbasic Mar 28, 2024
135b88d
Update style.rs
rustbasic Mar 29, 2024
9505776
Update visuals.rs
rustbasic Mar 29, 2024
9ee6ca6
Update style.rs
rustbasic Mar 29, 2024
ec0f56a
Merge branch 'emilk:master' into patch19
rustbasic Mar 29, 2024
55467b3
Update style.rs
rustbasic Mar 29, 2024
a1a3d11
Update visuals.rs
rustbasic Mar 29, 2024
0681300
Update builder.rs
rustbasic Mar 29, 2024
de4fa6e
Update visuals.rs
rustbasic Mar 29, 2024
0b4ea7f
Update style.rs
rustbasic Mar 29, 2024
1600d32
Update input_state.rs
rustbasic Mar 29, 2024
fbabc24
Update visuals.rs
rustbasic Mar 29, 2024
a2eee07
Update builder.rs
rustbasic Mar 29, 2024
82d3814
Update builder.rs
rustbasic Mar 29, 2024
24e86d1
Update style.rs
rustbasic Mar 29, 2024
d098a8f
Update visuals.rs
rustbasic Mar 29, 2024
27098a1
Update builder.rs
rustbasic Mar 29, 2024
e50f80d
Update visuals.rs
rustbasic Mar 29, 2024
7dd8226
Update input_state.rs
rustbasic Mar 29, 2024
94f8424
Merge branch 'emilk:master' into patch19
rustbasic Mar 30, 2024
f101c92
Merge branch 'emilk:master' into patch19
rustbasic Mar 30, 2024
837c277
Merge branch 'emilk:master' into patch19
rustbasic Mar 30, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions crates/egui/src/input_state.rs
Original file line number Diff line number Diff line change
Expand Up @@ -416,6 +416,11 @@ impl InputState {
.count()
}

/// Tells whether there is a key currently being pressed down.
pub fn has_key_down(&self) -> bool {
!self.keys_down.is_empty()
}

/// Is the given key currently held down?
pub fn key_down(&self, desired_key: Key) -> bool {
self.keys_down.contains(&desired_key)
Expand Down
55 changes: 47 additions & 8 deletions crates/egui/src/style.rs
Original file line number Diff line number Diff line change
Expand Up @@ -647,6 +647,27 @@ pub struct Interaction {
pub multi_widget_text_select: bool,
}

/// Text Cursor Style
#[derive(Clone, Default, Debug, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Deserialize, serde::Serialize))]
#[cfg_attr(feature = "serde", serde(default))]
pub struct TextCursorStyle {
/// show where the text cursor would be if you clicked
pub preview: bool,

/// set the text cursor to blink
pub blink: bool,

/// set the text cursor on duration
pub on_duration: f64,

/// set the text cursor off duration
pub off_duration: f64,

/// This is the time when the key was last pressed.
pub last_edit_time: f64,
}

/// Controls the visual style (colors etc) of egui.
///
/// You can change the visuals of a [`Ui`] with [`Ui::visuals_mut`]
Expand Down Expand Up @@ -725,8 +746,8 @@ pub struct Visuals {
/// The color and width of the text cursor
pub text_cursor: Stroke,

/// show where the text cursor would be if you clicked
pub text_cursor_preview: bool,
/// Text Cursor Style for blink
pub text_cursor_style: TextCursorStyle,

/// Allow child widgets to be just on the border and still have a stroke with some thickness
pub clip_rect_margin: f32,
Expand Down Expand Up @@ -1095,7 +1116,13 @@ impl Visuals {
resize_corner_size: 12.0,

text_cursor: Stroke::new(2.0, Color32::from_rgb(192, 222, 255)),
text_cursor_preview: false,
text_cursor_style: TextCursorStyle {
preview: false,
blink: true,
on_duration: 1.0,
off_duration: 0.3,
last_edit_time: Default::default(),
},

clip_rect_margin: 3.0, // should be at least half the size of the widest frame stroke + max WidgetVisuals::expansion
button_frame: true,
Expand Down Expand Up @@ -1581,10 +1608,10 @@ impl Interaction {
impl Widgets {
pub fn ui(&mut self, ui: &mut crate::Ui) {
let Self {
active,
hovered,
inactive,
noninteractive,
inactive,
hovered,
active,
open,
} = self;

Expand Down Expand Up @@ -1738,7 +1765,7 @@ impl Visuals {

resize_corner_size,
text_cursor,
text_cursor_preview,
text_cursor_style,
clip_rect_margin,
button_frame,
collapsing_header_frame,
Expand Down Expand Up @@ -1843,7 +1870,19 @@ impl Visuals {

ui.collapsing("Misc", |ui| {
ui.add(Slider::new(resize_corner_size, 0.0..=20.0).text("resize_corner_size"));
ui.checkbox(text_cursor_preview, "Preview text cursor on hover");
ui.checkbox(
&mut text_cursor_style.preview,
"Preview text cursor on hover",
);
ui.checkbox(&mut text_cursor_style.blink, "text cursor to blink");
ui.add(
Slider::new(&mut text_cursor_style.on_duration, 0.0..=2.0)
.text("text cursor on duration"),
);
ui.add(
Slider::new(&mut text_cursor_style.off_duration, 0.0..=2.0)
.text("text cursor off duration"),
);
ui.add(Slider::new(clip_rect_margin, 0.0..=20.0).text("clip_rect_margin"));

ui.checkbox(button_frame, "Button has a frame");
Expand Down
43 changes: 43 additions & 0 deletions crates/egui/src/text_selection/visuals.rs
Original file line number Diff line number Diff line change
Expand Up @@ -73,3 +73,46 @@ pub fn paint_cursor(painter: &Painter, visuals: &Visuals, cursor_rect: Rect) {
);
}
}

/// Paint text cursor.
pub fn paint_text_cursor(
Comment on lines +77 to +78
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fn paint_text_cursor and fn paint_cursor have very similar names, but one does blinking, and one does not. We should document that. fn paint_cursor should probably be made non-pub too

ui: &mut Ui,
painter: &Painter,
primary_cursor_rect: Rect,
blink_pause: bool,
) {
let i_time = ui.input(|i| i.time);
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Currently the cursor still blinks while typing, which is quite annoying. If we instead reset a timer each time the user presses a key, it won't blink until text_cursor_on_duration after the user last stopped typing.

So, I think you were right to put this timer in TextEditState. We could put a last_edit_time: Option<f64> there fed by ui.input(|i| i.time), and then paint_text_cursor is fed with time_since_last_edit which will be used as the basis for the animation

let blink_mode = ui.visuals().text_cursor_style.blink;
let is_blink_mode = blink_mode && !blink_pause;

let on_duration = ui.visuals().text_cursor_style.on_duration;
let off_duration = ui.visuals().text_cursor_style.off_duration;
let total_duration = on_duration + off_duration;

let mut is_cursor_visible = true;

if is_blink_mode {
// Check if the text cursor is on/off based on the time
is_cursor_visible = (i_time % total_duration) < on_duration;
}

// Keep the text cursor visible for some time.
if i_time < ui.visuals().text_cursor_style.last_edit_time + 3.0 {
is_cursor_visible = true;
}

if is_cursor_visible {
paint_cursor(painter, ui.visuals(), primary_cursor_rect);
}

if blink_mode {
if is_cursor_visible {
ui.ctx()
.request_repaint_after(std::time::Duration::from_secs_f64(on_duration));
}
if !is_cursor_visible {
ui.ctx()
.request_repaint_after(std::time::Duration::from_secs_f64(off_duration));
}
}
}
20 changes: 17 additions & 3 deletions crates/egui/src/widgets/text_edit/builder.rs
Original file line number Diff line number Diff line change
Expand Up @@ -511,6 +511,7 @@ impl<'t> TextEdit<'t> {
}
});
let mut state = TextEditState::load(ui.ctx(), id).unwrap_or_default();
let save_ccursor_range = state.cursor.char_range();

// On touch screens (e.g. mobile in `eframe` web), should
// dragging select text, or scroll the enclosing [`ScrollArea`] (if any)?
Expand Down Expand Up @@ -544,11 +545,11 @@ impl<'t> TextEdit<'t> {
let cursor_at_pointer =
galley.cursor_from_pos(pointer_pos - rect.min + singleline_offset);

if ui.visuals().text_cursor_preview
if ui.visuals().text_cursor_style.preview
&& response.hovered()
&& ui.input(|i| i.pointer.is_moving())
{
// preview:
// text cursor preview:
let cursor_rect =
cursor_rect(rect.min, &galley, &cursor_at_pointer, row_height);
paint_cursor(&painter, ui.visuals(), cursor_rect);
Expand Down Expand Up @@ -684,7 +685,20 @@ impl<'t> TextEdit<'t> {
}

if text.is_mutable() {
paint_cursor(&painter, ui.visuals(), primary_cursor_rect);
let has_key_down = ui.ctx().input(|i| i.has_key_down());
if has_key_down {
ui.visuals_mut().text_cursor_style.last_edit_time =
ui.ctx().input(|i| i.time);
}
let blink_pause =
(save_ccursor_range != state.cursor.char_range()) || has_key_down;

text_selection::visuals::paint_text_cursor(
ui,
&painter,
primary_cursor_rect,
blink_pause,
);

if interactive {
// For IME, so only set it when text is editable and visible!
Expand Down
Loading