Skip to content

Commit f7ec8f2

Browse files
authored
Use runtime knowledge of OS for OS-specific text editing (#3840)
How text editing behaves depends on what OS we are running on. `target_os` is a compile-time check, but `ctx.os()` is a runtime check, that works also on web.
1 parent b766a48 commit f7ec8f2

File tree

4 files changed

+46
-14
lines changed

4 files changed

+46
-14
lines changed

crates/egui/src/widgets/label.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -377,7 +377,7 @@ fn process_selection_key_events(
377377
}
378378

379379
event => {
380-
cursor_range.on_event(event, galley, widget_id);
380+
cursor_range.on_event(ui.ctx().os(), event, galley, widget_id);
381381
}
382382
}
383383
}

crates/egui/src/widgets/text_edit/builder.rs

+9-4
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,9 @@ use std::sync::Arc;
22

33
use epaint::text::{cursor::*, Galley, LayoutJob};
44

5-
use crate::{output::OutputEvent, text_edit::cursor_interaction::cursor_rect, *};
5+
use crate::{
6+
os::OperatingSystem, output::OutputEvent, text_edit::cursor_interaction::cursor_rect, *,
7+
};
68

79
use super::{
810
cursor_interaction::{ccursor_next_word, ccursor_previous_word, find_line_start},
@@ -773,6 +775,8 @@ fn events(
773775
char_limit: usize,
774776
event_filter: EventFilter,
775777
) -> (bool, CursorRange) {
778+
let os = ui.ctx().os();
779+
776780
let mut cursor_range = state.cursor.range(galley).unwrap_or(default_cursor_range);
777781

778782
// We feed state to the undoer both before and after handling input
@@ -794,7 +798,7 @@ fn events(
794798
for event in &events {
795799
let did_mutate_text = match event {
796800
// First handle events that only changes the selection cursor, not the text:
797-
event if cursor_range.on_event(event, galley, id) => None,
801+
event if cursor_range.on_event(os, event, galley, id) => None,
798802

799803
Event::Copy => {
800804
if cursor_range.is_empty() {
@@ -909,7 +913,7 @@ fn events(
909913
key,
910914
pressed: true,
911915
..
912-
} => check_for_mutating_key_press(&mut cursor_range, text, galley, modifiers, *key),
916+
} => check_for_mutating_key_press(os, &mut cursor_range, text, galley, modifiers, *key),
913917

914918
Event::CompositionStart => {
915919
state.has_ime = true;
@@ -1143,6 +1147,7 @@ fn delete_paragraph_after_cursor(
11431147

11441148
/// Returns `Some(new_cursor)` if we did mutate `text`.
11451149
fn check_for_mutating_key_press(
1150+
os: OperatingSystem,
11461151
cursor_range: &mut CursorRange,
11471152
text: &mut dyn TextBuffer,
11481153
galley: &Galley,
@@ -1166,7 +1171,7 @@ fn check_for_mutating_key_press(
11661171
Some(CCursorRange::one(ccursor))
11671172
}
11681173

1169-
Key::Delete if !modifiers.shift || !cfg!(target_os = "windows") => {
1174+
Key::Delete if !modifiers.shift || os != OperatingSystem::Windows => {
11701175
let ccursor = if modifiers.mac_cmd {
11711176
delete_paragraph_after_cursor(text, galley, cursor_range)
11721177
} else if let Some(cursor) = cursor_range.single() {

crates/egui/src/widgets/text_edit/cursor_range.rs

+27-9
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
use epaint::{text::cursor::*, Galley};
22

3-
use crate::{Event, Id, Key, Modifiers};
3+
use crate::{os::OperatingSystem, Event, Id, Key, Modifiers};
44

55
use super::cursor_interaction::{ccursor_next_word, ccursor_previous_word, slice_char_range};
66

@@ -115,7 +115,13 @@ impl CursorRange {
115115
/// Check for key presses that are moving the cursor.
116116
///
117117
/// Returns `true` if we did mutate `self`.
118-
pub fn on_key_press(&mut self, galley: &Galley, modifiers: &Modifiers, key: Key) -> bool {
118+
pub fn on_key_press(
119+
&mut self,
120+
os: OperatingSystem,
121+
galley: &Galley,
122+
modifiers: &Modifiers,
123+
key: Key,
124+
) -> bool {
119125
match key {
120126
Key::A if modifiers.command => {
121127
*self = Self::select_all(galley);
@@ -137,17 +143,17 @@ impl CursorRange {
137143
| Key::ArrowDown
138144
| Key::Home
139145
| Key::End => {
140-
move_single_cursor(&mut self.primary, galley, key, modifiers);
146+
move_single_cursor(os, &mut self.primary, galley, key, modifiers);
141147
if !modifiers.shift {
142148
self.secondary = self.primary;
143149
}
144150
true
145151
}
146152

147153
Key::P | Key::N | Key::B | Key::F | Key::A | Key::E
148-
if cfg!(target_os = "macos") && modifiers.ctrl && !modifiers.shift =>
154+
if os == OperatingSystem::Mac && modifiers.ctrl && !modifiers.shift =>
149155
{
150-
move_single_cursor(&mut self.primary, galley, key, modifiers);
156+
move_single_cursor(os, &mut self.primary, galley, key, modifiers);
151157
self.secondary = self.primary;
152158
true
153159
}
@@ -159,14 +165,20 @@ impl CursorRange {
159165
/// Check for events that modify the cursor range.
160166
///
161167
/// Returns `true` if such an event was found and handled.
162-
pub fn on_event(&mut self, event: &Event, galley: &Galley, _widget_id: Id) -> bool {
168+
pub fn on_event(
169+
&mut self,
170+
os: OperatingSystem,
171+
event: &Event,
172+
galley: &Galley,
173+
_widget_id: Id,
174+
) -> bool {
163175
match event {
164176
Event::Key {
165177
modifiers,
166178
key,
167179
pressed: true,
168180
..
169-
} => self.on_key_press(galley, modifiers, *key),
181+
} => self.on_key_press(os, galley, modifiers, *key),
170182

171183
#[cfg(feature = "accesskit")]
172184
Event::AccessKitActionRequest(accesskit::ActionRequest {
@@ -287,8 +299,14 @@ fn ccursor_from_accesskit_text_position(
287299
// ----------------------------------------------------------------------------
288300

289301
/// Move a text cursor based on keyboard
290-
fn move_single_cursor(cursor: &mut Cursor, galley: &Galley, key: Key, modifiers: &Modifiers) {
291-
if cfg!(target_os = "macos") && modifiers.ctrl && !modifiers.shift {
302+
fn move_single_cursor(
303+
os: OperatingSystem,
304+
cursor: &mut Cursor,
305+
galley: &Galley,
306+
key: Key,
307+
modifiers: &Modifiers,
308+
) {
309+
if os == OperatingSystem::Mac && modifiers.ctrl && !modifiers.shift {
292310
match key {
293311
Key::A => *cursor = galley.cursor_begin_of_row(cursor),
294312
Key::E => *cursor = galley.cursor_end_of_row(cursor),

scripts/lint.py

+9
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,15 @@ def lint_lines(filepath, lines_in):
6666
)
6767
lines_out.append("#[inline]")
6868

69+
if (
70+
"(target_os" in line
71+
and filepath.startswith("./crates/egui/")
72+
and filepath != "./crates/egui/src/os.rs"
73+
):
74+
errors.append(
75+
f"{filepath}:{line_nr}: Don't use `target_os` - use ctx.os() instead."
76+
)
77+
6978
lines_out.append(line)
7079

7180
prev_line = line

0 commit comments

Comments
 (0)