mirror of
https://github.com/helix-editor/helix.git
synced 2025-04-04 19:37:54 +03:00
Merge 2234d94b50
into 7ebf650029
This commit is contained in:
commit
f5552a9dc8
13 changed files with 197 additions and 60 deletions
44
Cargo.lock
generated
44
Cargo.lock
generated
|
@ -236,25 +236,6 @@ version = "0.8.20"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80"
|
checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossterm"
|
|
||||||
version = "0.28.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "829d955a0bb380ef178a640b91779e3987da38c9aea133b20614cfed8cdea9c6"
|
|
||||||
dependencies = [
|
|
||||||
"bitflags",
|
|
||||||
"crossterm_winapi",
|
|
||||||
"filedescriptor",
|
|
||||||
"futures-core",
|
|
||||||
"libc",
|
|
||||||
"mio",
|
|
||||||
"parking_lot",
|
|
||||||
"rustix 0.38.44",
|
|
||||||
"signal-hook",
|
|
||||||
"signal-hook-mio",
|
|
||||||
"winapi",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crossterm_winapi"
|
name = "crossterm_winapi"
|
||||||
version = "0.9.1"
|
version = "0.9.1"
|
||||||
|
@ -1354,6 +1335,25 @@ dependencies = [
|
||||||
"url",
|
"url",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "helix-crossterm"
|
||||||
|
version = "0.1.0-beta.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "8938f95c101672e5084b377daac1f78ed5964c2a75046fc0d29d66cbed975f8c"
|
||||||
|
dependencies = [
|
||||||
|
"bitflags",
|
||||||
|
"crossterm_winapi",
|
||||||
|
"filedescriptor",
|
||||||
|
"futures-core",
|
||||||
|
"libc",
|
||||||
|
"mio",
|
||||||
|
"parking_lot",
|
||||||
|
"rustix 0.38.44",
|
||||||
|
"signal-hook",
|
||||||
|
"signal-hook-mio",
|
||||||
|
"winapi",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "helix-dap"
|
name = "helix-dap"
|
||||||
version = "25.1.1"
|
version = "25.1.1"
|
||||||
|
@ -1464,12 +1464,12 @@ dependencies = [
|
||||||
"arc-swap",
|
"arc-swap",
|
||||||
"chrono",
|
"chrono",
|
||||||
"content_inspector",
|
"content_inspector",
|
||||||
"crossterm",
|
|
||||||
"fern",
|
"fern",
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"grep-regex",
|
"grep-regex",
|
||||||
"grep-searcher",
|
"grep-searcher",
|
||||||
"helix-core",
|
"helix-core",
|
||||||
|
"helix-crossterm",
|
||||||
"helix-dap",
|
"helix-dap",
|
||||||
"helix-event",
|
"helix-event",
|
||||||
"helix-loader",
|
"helix-loader",
|
||||||
|
@ -1508,8 +1508,8 @@ version = "25.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bitflags",
|
"bitflags",
|
||||||
"cassowary",
|
"cassowary",
|
||||||
"crossterm",
|
|
||||||
"helix-core",
|
"helix-core",
|
||||||
|
"helix-crossterm",
|
||||||
"helix-view",
|
"helix-view",
|
||||||
"log",
|
"log",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
|
@ -1542,9 +1542,9 @@ dependencies = [
|
||||||
"bitflags",
|
"bitflags",
|
||||||
"chardetng",
|
"chardetng",
|
||||||
"clipboard-win",
|
"clipboard-win",
|
||||||
"crossterm",
|
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"helix-core",
|
"helix-core",
|
||||||
|
"helix-crossterm",
|
||||||
"helix-dap",
|
"helix-dap",
|
||||||
"helix-event",
|
"helix-event",
|
||||||
"helix-loader",
|
"helix-loader",
|
||||||
|
|
|
@ -47,6 +47,7 @@ unicode-segmentation = "1.2"
|
||||||
ropey = { version = "1.6.1", default-features = false, features = ["simd"] }
|
ropey = { version = "1.6.1", default-features = false, features = ["simd"] }
|
||||||
foldhash = "0.1"
|
foldhash = "0.1"
|
||||||
parking_lot = "0.12"
|
parking_lot = "0.12"
|
||||||
|
crossterm = { package = "helix-crossterm", version = "0.1.0-beta.1" }
|
||||||
|
|
||||||
[workspace.package]
|
[workspace.package]
|
||||||
version = "25.1.1"
|
version = "25.1.1"
|
||||||
|
|
|
@ -55,7 +55,7 @@ once_cell = "1.21"
|
||||||
|
|
||||||
tokio = { version = "1", features = ["rt", "rt-multi-thread", "io-util", "io-std", "time", "process", "macros", "fs", "parking_lot"] }
|
tokio = { version = "1", features = ["rt", "rt-multi-thread", "io-util", "io-std", "time", "process", "macros", "fs", "parking_lot"] }
|
||||||
tui = { path = "../helix-tui", package = "helix-tui", default-features = false, features = ["crossterm"] }
|
tui = { path = "../helix-tui", package = "helix-tui", default-features = false, features = ["crossterm"] }
|
||||||
crossterm = { version = "0.28", features = ["event-stream"] }
|
crossterm = { workspace = true, features = ["event-stream"] }
|
||||||
signal-hook = "0.3"
|
signal-hook = "0.3"
|
||||||
tokio-stream = "0.1"
|
tokio-stream = "0.1"
|
||||||
futures-util = { version = "0.3", features = ["std", "async-await"], default-features = false }
|
futures-util = { version = "0.3", features = ["std", "async-await"], default-features = false }
|
||||||
|
@ -96,7 +96,7 @@ signal-hook-tokio = { version = "0.3", features = ["futures-v0_3"] }
|
||||||
libc = "0.2.171"
|
libc = "0.2.171"
|
||||||
|
|
||||||
[target.'cfg(target_os = "macos")'.dependencies]
|
[target.'cfg(target_os = "macos")'.dependencies]
|
||||||
crossterm = { version = "0.28", features = ["event-stream", "use-dev-tty", "libc"] }
|
crossterm = { workspace = true, features = ["event-stream", "use-dev-tty", "libc"] }
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
helix-loader = { path = "../helix-loader" }
|
helix-loader = { path = "../helix-loader" }
|
||||||
|
|
|
@ -68,6 +68,9 @@ pub struct Application {
|
||||||
signals: Signals,
|
signals: Signals,
|
||||||
jobs: Jobs,
|
jobs: Jobs,
|
||||||
lsp_progress: LspProgressMap,
|
lsp_progress: LspProgressMap,
|
||||||
|
|
||||||
|
/// The theme mode (light/dark) detected from the terminal, if available.
|
||||||
|
theme_mode: Option<theme::Mode>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "integration")]
|
#[cfg(feature = "integration")]
|
||||||
|
@ -109,6 +112,7 @@ impl Application {
|
||||||
#[cfg(feature = "integration")]
|
#[cfg(feature = "integration")]
|
||||||
let backend = TestBackend::new(120, 150);
|
let backend = TestBackend::new(120, 150);
|
||||||
|
|
||||||
|
let theme_mode = backend.get_theme_mode();
|
||||||
let terminal = Terminal::new(backend)?;
|
let terminal = Terminal::new(backend)?;
|
||||||
let area = terminal.size().expect("couldn't get terminal size");
|
let area = terminal.size().expect("couldn't get terminal size");
|
||||||
let mut compositor = Compositor::new(area);
|
let mut compositor = Compositor::new(area);
|
||||||
|
@ -123,12 +127,15 @@ impl Application {
|
||||||
})),
|
})),
|
||||||
handlers,
|
handlers,
|
||||||
);
|
);
|
||||||
Self::load_configured_theme(&mut editor, &config.load());
|
Self::load_configured_theme(&mut editor, &config.load(), theme_mode);
|
||||||
|
|
||||||
let keys = Box::new(Map::new(Arc::clone(&config), |config: &Config| {
|
let keys = Box::new(Map::new(Arc::clone(&config), |config: &Config| {
|
||||||
&config.keys
|
&config.keys
|
||||||
}));
|
}));
|
||||||
let editor_view = Box::new(ui::EditorView::new(Keymaps::new(keys)));
|
let editor_view = Box::new(ui::EditorView::new(
|
||||||
|
Keymaps::new(keys),
|
||||||
|
Map::new(Arc::clone(&config), |config: &Config| &config.theme),
|
||||||
|
));
|
||||||
compositor.push(editor_view);
|
compositor.push(editor_view);
|
||||||
|
|
||||||
if args.load_tutor {
|
if args.load_tutor {
|
||||||
|
@ -242,6 +249,7 @@ impl Application {
|
||||||
signals,
|
signals,
|
||||||
jobs: Jobs::new(),
|
jobs: Jobs::new(),
|
||||||
lsp_progress: LspProgressMap::new(),
|
lsp_progress: LspProgressMap::new(),
|
||||||
|
theme_mode,
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(app)
|
Ok(app)
|
||||||
|
@ -408,7 +416,7 @@ impl Application {
|
||||||
.map_err(|err| anyhow::anyhow!("Failed to load config: {}", err))?;
|
.map_err(|err| anyhow::anyhow!("Failed to load config: {}", err))?;
|
||||||
self.refresh_language_config()?;
|
self.refresh_language_config()?;
|
||||||
// Refresh theme after config change
|
// Refresh theme after config change
|
||||||
Self::load_configured_theme(&mut self.editor, &default_config);
|
Self::load_configured_theme(&mut self.editor, &default_config, self.theme_mode);
|
||||||
self.terminal
|
self.terminal
|
||||||
.reconfigure(default_config.editor.clone().into())?;
|
.reconfigure(default_config.editor.clone().into())?;
|
||||||
// Store new config
|
// Store new config
|
||||||
|
@ -427,12 +435,13 @@ impl Application {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Load the theme set in configuration
|
/// Load the theme set in configuration
|
||||||
fn load_configured_theme(editor: &mut Editor, config: &Config) {
|
fn load_configured_theme(editor: &mut Editor, config: &Config, mode: Option<theme::Mode>) {
|
||||||
let true_color = config.editor.true_color || crate::true_color();
|
let true_color = config.editor.true_color || crate::true_color();
|
||||||
let theme = config
|
let theme = config
|
||||||
.theme
|
.theme
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.and_then(|theme| {
|
.and_then(|theme_config| {
|
||||||
|
let theme = theme_config.choose(mode);
|
||||||
editor
|
editor
|
||||||
.theme_loader
|
.theme_loader
|
||||||
.load(theme)
|
.load(theme)
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use crate::keymap;
|
use crate::keymap;
|
||||||
use crate::keymap::{merge_keys, KeyTrie};
|
use crate::keymap::{merge_keys, KeyTrie};
|
||||||
use helix_loader::merge_toml_values;
|
use helix_loader::merge_toml_values;
|
||||||
use helix_view::document::Mode;
|
use helix_view::{document::Mode, theme};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::fmt::Display;
|
use std::fmt::Display;
|
||||||
|
@ -11,7 +11,7 @@ use toml::de::Error as TomlError;
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
pub theme: Option<String>,
|
pub theme: Option<theme::Config>,
|
||||||
pub keys: HashMap<Mode, KeyTrie>,
|
pub keys: HashMap<Mode, KeyTrie>,
|
||||||
pub editor: helix_view::editor::Config,
|
pub editor: helix_view::editor::Config,
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,7 @@ pub struct Config {
|
||||||
#[derive(Debug, Clone, PartialEq, Deserialize)]
|
#[derive(Debug, Clone, PartialEq, Deserialize)]
|
||||||
#[serde(deny_unknown_fields)]
|
#[serde(deny_unknown_fields)]
|
||||||
pub struct ConfigRaw {
|
pub struct ConfigRaw {
|
||||||
pub theme: Option<String>,
|
pub theme: Option<theme::Config>,
|
||||||
pub keys: Option<HashMap<Mode, KeyTrie>>,
|
pub keys: Option<HashMap<Mode, KeyTrie>>,
|
||||||
pub editor: Option<toml::Value>,
|
pub editor: Option<toml::Value>,
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,6 +13,7 @@ use crate::{
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use arc_swap::access::DynAccess;
|
||||||
use helix_core::{
|
use helix_core::{
|
||||||
diagnostic::NumberOrString,
|
diagnostic::NumberOrString,
|
||||||
graphemes::{next_grapheme_boundary, prev_grapheme_boundary},
|
graphemes::{next_grapheme_boundary, prev_grapheme_boundary},
|
||||||
|
@ -29,7 +30,7 @@ use helix_view::{
|
||||||
graphics::{Color, CursorKind, Modifier, Rect, Style},
|
graphics::{Color, CursorKind, Modifier, Rect, Style},
|
||||||
input::{KeyEvent, MouseButton, MouseEvent, MouseEventKind},
|
input::{KeyEvent, MouseButton, MouseEvent, MouseEventKind},
|
||||||
keyboard::{KeyCode, KeyModifiers},
|
keyboard::{KeyCode, KeyModifiers},
|
||||||
Document, Editor, Theme, View,
|
theme, Document, Editor, Theme, View,
|
||||||
};
|
};
|
||||||
use std::{mem::take, num::NonZeroUsize, path::PathBuf, rc::Rc};
|
use std::{mem::take, num::NonZeroUsize, path::PathBuf, rc::Rc};
|
||||||
|
|
||||||
|
@ -37,6 +38,7 @@ use tui::{buffer::Buffer as Surface, text::Span};
|
||||||
|
|
||||||
pub struct EditorView {
|
pub struct EditorView {
|
||||||
pub keymaps: Keymaps,
|
pub keymaps: Keymaps,
|
||||||
|
theme_config: Box<dyn DynAccess<Option<theme::Config>>>,
|
||||||
on_next_key: Option<(OnKeyCallback, OnKeyCallbackKind)>,
|
on_next_key: Option<(OnKeyCallback, OnKeyCallbackKind)>,
|
||||||
pseudo_pending: Vec<KeyEvent>,
|
pseudo_pending: Vec<KeyEvent>,
|
||||||
pub(crate) last_insert: (commands::MappableCommand, Vec<InsertEvent>),
|
pub(crate) last_insert: (commands::MappableCommand, Vec<InsertEvent>),
|
||||||
|
@ -58,9 +60,13 @@ pub enum InsertEvent {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl EditorView {
|
impl EditorView {
|
||||||
pub fn new(keymaps: Keymaps) -> Self {
|
pub fn new(
|
||||||
|
keymaps: Keymaps,
|
||||||
|
theme_config: impl DynAccess<Option<theme::Config>> + 'static,
|
||||||
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
keymaps,
|
keymaps,
|
||||||
|
theme_config: Box::new(theme_config),
|
||||||
on_next_key: None,
|
on_next_key: None,
|
||||||
pseudo_pending: Vec::new(),
|
pseudo_pending: Vec::new(),
|
||||||
last_insert: (commands::MappableCommand::normal_mode, Vec::new()),
|
last_insert: (commands::MappableCommand::normal_mode, Vec::new()),
|
||||||
|
@ -1535,6 +1541,18 @@ impl Component for EditorView {
|
||||||
self.terminal_focused = false;
|
self.terminal_focused = false;
|
||||||
EventResult::Consumed(None)
|
EventResult::Consumed(None)
|
||||||
}
|
}
|
||||||
|
Event::ThemeModeChanged(theme_mode) => {
|
||||||
|
if let Some(theme_config) = self.theme_config.load().as_ref() {
|
||||||
|
let theme_name = theme_config.choose(Some(*theme_mode));
|
||||||
|
match context.editor.theme_loader.load(theme_name) {
|
||||||
|
Ok(theme) => context.editor.set_theme(theme),
|
||||||
|
Err(err) => {
|
||||||
|
log::warn!("failed to load theme `{}` - {}", theme_name, err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
EventResult::Consumed(None)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -21,7 +21,7 @@ helix-core = { path = "../helix-core" }
|
||||||
bitflags.workspace = true
|
bitflags.workspace = true
|
||||||
cassowary = "0.3"
|
cassowary = "0.3"
|
||||||
unicode-segmentation.workspace = true
|
unicode-segmentation.workspace = true
|
||||||
crossterm = { version = "0.28", optional = true }
|
crossterm = { workspace = true, optional = true }
|
||||||
termini = "1.0"
|
termini = "1.0"
|
||||||
once_cell = "1.21"
|
once_cell = "1.21"
|
||||||
log = "~0.4"
|
log = "~0.4"
|
||||||
|
|
|
@ -2,9 +2,10 @@ use crate::{backend::Backend, buffer::Cell, terminal::Config};
|
||||||
use crossterm::{
|
use crossterm::{
|
||||||
cursor::{Hide, MoveTo, SetCursorStyle, Show},
|
cursor::{Hide, MoveTo, SetCursorStyle, Show},
|
||||||
event::{
|
event::{
|
||||||
DisableBracketedPaste, DisableFocusChange, DisableMouseCapture, EnableBracketedPaste,
|
DisableBracketedPaste, DisableFocusChange, DisableMouseCapture, DisableThemeModeUpdates,
|
||||||
EnableFocusChange, EnableMouseCapture, KeyboardEnhancementFlags,
|
EnableBracketedPaste, EnableFocusChange, EnableMouseCapture, EnableThemeModeUpdates,
|
||||||
PopKeyboardEnhancementFlags, PushKeyboardEnhancementFlags,
|
KeyboardEnhancementFlags, PopKeyboardEnhancementFlags, PushKeyboardEnhancementFlags,
|
||||||
|
SynchronizedOutputMode, TerminalFeatures,
|
||||||
},
|
},
|
||||||
execute, queue,
|
execute, queue,
|
||||||
style::{
|
style::{
|
||||||
|
@ -17,14 +18,18 @@ use crossterm::{
|
||||||
use helix_view::{
|
use helix_view::{
|
||||||
editor::Config as EditorConfig,
|
editor::Config as EditorConfig,
|
||||||
graphics::{Color, CursorKind, Modifier, Rect, UnderlineStyle},
|
graphics::{Color, CursorKind, Modifier, Rect, UnderlineStyle},
|
||||||
|
theme,
|
||||||
};
|
};
|
||||||
use once_cell::sync::OnceCell;
|
|
||||||
use std::{
|
use std::{
|
||||||
fmt,
|
fmt,
|
||||||
io::{self, Write},
|
io::{self, Write},
|
||||||
};
|
};
|
||||||
use termini::TermInfo;
|
use termini::TermInfo;
|
||||||
|
|
||||||
|
const KEYBOARD_ENHANCEMENT_FLAGS: KeyboardEnhancementFlags =
|
||||||
|
KeyboardEnhancementFlags::DISAMBIGUATE_ESCAPE_CODES
|
||||||
|
.union(KeyboardEnhancementFlags::REPORT_ALTERNATE_KEYS);
|
||||||
|
|
||||||
fn term_program() -> Option<String> {
|
fn term_program() -> Option<String> {
|
||||||
// Some terminals don't set $TERM_PROGRAM
|
// Some terminals don't set $TERM_PROGRAM
|
||||||
match std::env::var("TERM_PROGRAM") {
|
match std::env::var("TERM_PROGRAM") {
|
||||||
|
@ -98,7 +103,7 @@ impl Capabilities {
|
||||||
pub struct CrosstermBackend<W: Write> {
|
pub struct CrosstermBackend<W: Write> {
|
||||||
buffer: W,
|
buffer: W,
|
||||||
capabilities: Capabilities,
|
capabilities: Capabilities,
|
||||||
supports_keyboard_enhancement_protocol: OnceCell<bool>,
|
features: std::cell::Cell<Option<TerminalFeatures>>,
|
||||||
mouse_capture_enabled: bool,
|
mouse_capture_enabled: bool,
|
||||||
supports_bracketed_paste: bool,
|
supports_bracketed_paste: bool,
|
||||||
}
|
}
|
||||||
|
@ -115,27 +120,35 @@ where
|
||||||
CrosstermBackend {
|
CrosstermBackend {
|
||||||
buffer,
|
buffer,
|
||||||
capabilities: Capabilities::from_env_or_default(config),
|
capabilities: Capabilities::from_env_or_default(config),
|
||||||
supports_keyboard_enhancement_protocol: OnceCell::new(),
|
features: std::cell::Cell::new(None),
|
||||||
mouse_capture_enabled: false,
|
mouse_capture_enabled: false,
|
||||||
supports_bracketed_paste: true,
|
supports_bracketed_paste: true,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
fn features(&self) -> TerminalFeatures {
|
||||||
fn supports_keyboard_enhancement_protocol(&self) -> bool {
|
match self.features.get() {
|
||||||
*self.supports_keyboard_enhancement_protocol
|
Some(features) => features,
|
||||||
.get_or_init(|| {
|
None => {
|
||||||
use std::time::Instant;
|
use std::time::Instant;
|
||||||
|
|
||||||
let now = Instant::now();
|
let now = Instant::now();
|
||||||
let supported = matches!(terminal::supports_keyboard_enhancement(), Ok(true));
|
let features = terminal::terminal_features().unwrap_or_default();
|
||||||
log::debug!(
|
log::debug!(
|
||||||
"The keyboard enhancement protocol is {}supported in this terminal (checked in {:?})",
|
"Detected terminal features in {:?}. Enhanced keyboard support: {}, Synchronized output support: {}. Theme mode updates support: {}",
|
||||||
if supported { "" } else { "not " },
|
Instant::now().duration_since(now),
|
||||||
Instant::now().duration_since(now)
|
features.keyboard_enhancement_flags.is_some(),
|
||||||
|
features.synchronized_output_mode != SynchronizedOutputMode::NotSupported,
|
||||||
|
features.theme_mode.is_some(),
|
||||||
);
|
);
|
||||||
supported
|
self.features.set(Some(features));
|
||||||
})
|
features
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn supports_synchronized_output(&self) -> bool {
|
||||||
|
self.features().synchronized_output_mode != SynchronizedOutputMode::NotSupported
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,14 +189,28 @@ where
|
||||||
execute!(self.buffer, EnableMouseCapture)?;
|
execute!(self.buffer, EnableMouseCapture)?;
|
||||||
self.mouse_capture_enabled = true;
|
self.mouse_capture_enabled = true;
|
||||||
}
|
}
|
||||||
if self.supports_keyboard_enhancement_protocol() {
|
let features = self.features();
|
||||||
|
if features.theme_mode.is_some() {
|
||||||
|
execute!(self.buffer, EnableThemeModeUpdates)?;
|
||||||
|
}
|
||||||
|
if features.keyboard_enhancement_flags.is_some() {
|
||||||
execute!(
|
execute!(
|
||||||
self.buffer,
|
self.buffer,
|
||||||
PushKeyboardEnhancementFlags(
|
PushKeyboardEnhancementFlags(KEYBOARD_ENHANCEMENT_FLAGS)
|
||||||
KeyboardEnhancementFlags::DISAMBIGUATE_ESCAPE_CODES
|
|
||||||
| KeyboardEnhancementFlags::REPORT_ALTERNATE_KEYS
|
|
||||||
)
|
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
|
// If the terminal supports a limited subset of the keyboard enhancement protocol,
|
||||||
|
// turn it off. We need both `DISAMBIGUATE_ESCAPE_CODES` and `REPORT_ALTERNATE_KEYS`.
|
||||||
|
if let Ok(Some(flags)) = terminal::query_keyboard_enhancement_flags() {
|
||||||
|
if !flags.contains(KEYBOARD_ENHANCEMENT_FLAGS) {
|
||||||
|
log::info!("Turning off keyboard enhancement since the terminal didn't enable the required flags. Expected {KEYBOARD_ENHANCEMENT_FLAGS:?}, found {flags:?}");
|
||||||
|
self.features.set(Some(TerminalFeatures {
|
||||||
|
keyboard_enhancement_flags: None,
|
||||||
|
..features
|
||||||
|
}));
|
||||||
|
execute!(self.buffer, PopKeyboardEnhancementFlags)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -208,9 +235,13 @@ where
|
||||||
if config.enable_mouse_capture {
|
if config.enable_mouse_capture {
|
||||||
execute!(self.buffer, DisableMouseCapture)?;
|
execute!(self.buffer, DisableMouseCapture)?;
|
||||||
}
|
}
|
||||||
if self.supports_keyboard_enhancement_protocol() {
|
let features = self.features();
|
||||||
|
if features.keyboard_enhancement_flags.is_some() {
|
||||||
execute!(self.buffer, PopKeyboardEnhancementFlags)?;
|
execute!(self.buffer, PopKeyboardEnhancementFlags)?;
|
||||||
}
|
}
|
||||||
|
if features.theme_mode.is_some() {
|
||||||
|
execute!(self.buffer, DisableThemeModeUpdates)?;
|
||||||
|
}
|
||||||
if self.supports_bracketed_paste {
|
if self.supports_bracketed_paste {
|
||||||
execute!(self.buffer, DisableBracketedPaste,)?;
|
execute!(self.buffer, DisableBracketedPaste,)?;
|
||||||
}
|
}
|
||||||
|
@ -231,6 +262,7 @@ where
|
||||||
// disable without calling enable previously
|
// disable without calling enable previously
|
||||||
let _ = execute!(stdout, DisableMouseCapture);
|
let _ = execute!(stdout, DisableMouseCapture);
|
||||||
let _ = execute!(stdout, PopKeyboardEnhancementFlags);
|
let _ = execute!(stdout, PopKeyboardEnhancementFlags);
|
||||||
|
let _ = execute!(stdout, DisableThemeModeUpdates);
|
||||||
let _ = execute!(stdout, DisableBracketedPaste);
|
let _ = execute!(stdout, DisableBracketedPaste);
|
||||||
execute!(stdout, DisableFocusChange, terminal::LeaveAlternateScreen)?;
|
execute!(stdout, DisableFocusChange, terminal::LeaveAlternateScreen)?;
|
||||||
terminal::disable_raw_mode()
|
terminal::disable_raw_mode()
|
||||||
|
@ -240,6 +272,10 @@ where
|
||||||
where
|
where
|
||||||
I: Iterator<Item = (u16, u16, &'a Cell)>,
|
I: Iterator<Item = (u16, u16, &'a Cell)>,
|
||||||
{
|
{
|
||||||
|
if self.supports_synchronized_output() {
|
||||||
|
queue!(self.buffer, terminal::BeginSynchronizedUpdate)?;
|
||||||
|
}
|
||||||
|
|
||||||
let mut fg = Color::Reset;
|
let mut fg = Color::Reset;
|
||||||
let mut bg = Color::Reset;
|
let mut bg = Color::Reset;
|
||||||
let mut underline_color = Color::Reset;
|
let mut underline_color = Color::Reset;
|
||||||
|
@ -298,7 +334,13 @@ where
|
||||||
SetForegroundColor(CColor::Reset),
|
SetForegroundColor(CColor::Reset),
|
||||||
SetBackgroundColor(CColor::Reset),
|
SetBackgroundColor(CColor::Reset),
|
||||||
SetAttribute(CAttribute::Reset)
|
SetAttribute(CAttribute::Reset)
|
||||||
)
|
)?;
|
||||||
|
|
||||||
|
if self.supports_synchronized_output() {
|
||||||
|
execute!(self.buffer, terminal::EndSynchronizedUpdate)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn hide_cursor(&mut self) -> io::Result<()> {
|
fn hide_cursor(&mut self) -> io::Result<()> {
|
||||||
|
@ -338,6 +380,13 @@ where
|
||||||
fn flush(&mut self) -> io::Result<()> {
|
fn flush(&mut self) -> io::Result<()> {
|
||||||
self.buffer.flush()
|
self.buffer.flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_theme_mode(&self) -> Option<theme::Mode> {
|
||||||
|
self.features().theme_mode.map(|mode| match mode {
|
||||||
|
crossterm::event::ThemeMode::Light => theme::Mode::Light,
|
||||||
|
crossterm::event::ThemeMode::Dark => theme::Mode::Dark,
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
|
|
@ -2,7 +2,10 @@ use std::io;
|
||||||
|
|
||||||
use crate::{buffer::Cell, terminal::Config};
|
use crate::{buffer::Cell, terminal::Config};
|
||||||
|
|
||||||
use helix_view::graphics::{CursorKind, Rect};
|
use helix_view::{
|
||||||
|
graphics::{CursorKind, Rect},
|
||||||
|
theme,
|
||||||
|
};
|
||||||
|
|
||||||
#[cfg(feature = "crossterm")]
|
#[cfg(feature = "crossterm")]
|
||||||
mod crossterm;
|
mod crossterm;
|
||||||
|
@ -27,4 +30,5 @@ pub trait Backend {
|
||||||
fn clear(&mut self) -> Result<(), io::Error>;
|
fn clear(&mut self) -> Result<(), io::Error>;
|
||||||
fn size(&self) -> Result<Rect, io::Error>;
|
fn size(&self) -> Result<Rect, io::Error>;
|
||||||
fn flush(&mut self) -> Result<(), io::Error>;
|
fn flush(&mut self) -> Result<(), io::Error>;
|
||||||
|
fn get_theme_mode(&self) -> Option<theme::Mode>;
|
||||||
}
|
}
|
||||||
|
|
|
@ -164,4 +164,8 @@ impl Backend for TestBackend {
|
||||||
fn flush(&mut self) -> Result<(), io::Error> {
|
fn flush(&mut self) -> Result<(), io::Error> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_theme_mode(&self) -> Option<helix_view::theme::Mode> {
|
||||||
|
None
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ helix-vcs = { path = "../helix-vcs" }
|
||||||
|
|
||||||
bitflags.workspace = true
|
bitflags.workspace = true
|
||||||
anyhow = "1"
|
anyhow = "1"
|
||||||
crossterm = { version = "0.28", optional = true }
|
crossterm = { workspace = true, optional = true }
|
||||||
|
|
||||||
tempfile.workspace = true
|
tempfile.workspace = true
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@ use serde::de::{self, Deserialize, Deserializer};
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
|
|
||||||
pub use crate::keyboard::{KeyCode, KeyModifiers, MediaKeyCode, ModifierKeyCode};
|
pub use crate::keyboard::{KeyCode, KeyModifiers, MediaKeyCode, ModifierKeyCode};
|
||||||
|
use crate::theme;
|
||||||
|
|
||||||
#[derive(Debug, PartialOrd, PartialEq, Eq, Clone, Hash)]
|
#[derive(Debug, PartialOrd, PartialEq, Eq, Clone, Hash)]
|
||||||
pub enum Event {
|
pub enum Event {
|
||||||
|
@ -14,6 +15,7 @@ pub enum Event {
|
||||||
Mouse(MouseEvent),
|
Mouse(MouseEvent),
|
||||||
Paste(String),
|
Paste(String),
|
||||||
Resize(u16, u16),
|
Resize(u16, u16),
|
||||||
|
ThemeModeChanged(theme::Mode),
|
||||||
IdleTimeout,
|
IdleTimeout,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -468,6 +470,12 @@ impl From<crossterm::event::Event> for Event {
|
||||||
crossterm::event::Event::FocusGained => Self::FocusGained,
|
crossterm::event::Event::FocusGained => Self::FocusGained,
|
||||||
crossterm::event::Event::FocusLost => Self::FocusLost,
|
crossterm::event::Event::FocusLost => Self::FocusLost,
|
||||||
crossterm::event::Event::Paste(s) => Self::Paste(s),
|
crossterm::event::Event::Paste(s) => Self::Paste(s),
|
||||||
|
crossterm::event::Event::ThemeModeChanged(theme_mode) => {
|
||||||
|
Self::ThemeModeChanged(match theme_mode {
|
||||||
|
crossterm::event::ThemeMode::Light => theme::Mode::Light,
|
||||||
|
crossterm::event::ThemeMode::Dark => theme::Mode::Dark,
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,50 @@ pub static BASE16_DEFAULT_THEME: Lazy<Theme> = Lazy::new(|| Theme {
|
||||||
..Theme::from(BASE16_DEFAULT_THEME_DATA.clone())
|
..Theme::from(BASE16_DEFAULT_THEME_DATA.clone())
|
||||||
});
|
});
|
||||||
|
|
||||||
|
#[derive(Debug, PartialOrd, PartialEq, Eq, Clone, Copy, Hash)]
|
||||||
|
pub enum Mode {
|
||||||
|
Light,
|
||||||
|
Dark,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
|
pub struct Config {
|
||||||
|
light: String,
|
||||||
|
dark: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Config {
|
||||||
|
pub fn choose(&self, preference: Option<Mode>) -> &str {
|
||||||
|
match preference {
|
||||||
|
Some(Mode::Light) => &self.light,
|
||||||
|
Some(Mode::Dark) | None => &self.dark,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de> Deserialize<'de> for Config {
|
||||||
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
||||||
|
where
|
||||||
|
D: serde::Deserializer<'de>,
|
||||||
|
{
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
#[serde(untagged, deny_unknown_fields)]
|
||||||
|
enum InnerConfig {
|
||||||
|
Constant(String),
|
||||||
|
Adaptive { dark: String, light: String },
|
||||||
|
}
|
||||||
|
|
||||||
|
let inner = InnerConfig::deserialize(deserializer)?;
|
||||||
|
|
||||||
|
let (light, dark) = match inner {
|
||||||
|
InnerConfig::Constant(theme) => (theme.clone(), theme),
|
||||||
|
InnerConfig::Adaptive { light, dark } => (light, dark),
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(Self { light, dark })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct Loader {
|
pub struct Loader {
|
||||||
/// Theme directories to search from highest to lowest priority
|
/// Theme directories to search from highest to lowest priority
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue