Skip to content

Commit

Permalink
feat(toggle-with-cycle): allows cycling option values at runtime
Browse files Browse the repository at this point in the history
  • Loading branch information
alevinval committed Jun 3, 2023
1 parent d511122 commit cc6e5ab
Showing 1 changed file with 37 additions and 12 deletions.
49 changes: 37 additions & 12 deletions helix-term/src/commands/typed.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@ use super::*;
use helix_core::{encoding, shellwords::Shellwords};
use helix_view::document::DEFAULT_LANGUAGE_NAME;
use helix_view::editor::{Action, CloseError, ConfigEvent};
use serde_json::Value;
use ui::completers::{self, Completer};

#[derive(Clone)]
Expand Down Expand Up @@ -1790,8 +1789,8 @@ fn toggle_option(
return Ok(());
}

if args.len() != 1 {
anyhow::bail!("Bad arguments. Usage: `:toggle key`");
if args.is_empty() {
anyhow::bail!("Bad arguments. Usage: `:toggle key [values]?`");
}
let key = &args[0].to_lowercase();

Expand All @@ -1801,22 +1800,48 @@ fn toggle_option(
let pointer = format!("/{}", key.replace('.', "/"));
let value = config.pointer_mut(&pointer).ok_or_else(key_error)?;

let Value::Bool(old_value) = *value else {
anyhow::bail!("Key `{}` is not toggle-able", key)
*value = match value.as_bool() {
Some(value) => serde_json::Value::Bool(!value),
None => {
if args.len() != 2 {
anyhow::bail!(
"Bad arguments. For non-boolean configurations use: `:toggle key val1,val2`"
);
}
if !value.is_string() {
anyhow::bail!("Bad configuration. Cannot cycle non-string configurations");
}

// The comma separated values to cycle
let first_arg = &args[1];
let field_error = |_| anyhow::anyhow!("Could not parse field `{}`", first_arg);

first_arg
.split_once(',')
.map(|(first, remaining)| {
Ok(serde_json::Value::String(
[first]
.into_iter()
.chain(remaining.split(','))
.skip_while(|e| e != value)
.nth(1)
.unwrap_or(first)
.to_string(),
))
})
.unwrap_or_else(|| first_arg.parse().map_err(field_error))?
}
};

let new_value = !old_value;
*value = Value::Bool(new_value);
// This unwrap should never fail because we only replace one boolean value
// with another, maintaining a valid json config
let config = serde_json::from_value(config).unwrap();
let status = format!("Option `{}` is now set to `{}`", key, value);
let config = serde_json::from_value(config)
.map_err(|_| anyhow::anyhow!("Could not parse field: `{:?}`", &args))?;

cx.editor
.config_events
.0
.send(ConfigEvent::Update(config))?;
cx.editor
.set_status(format!("Option `{}` is now set to `{}`", key, new_value));
cx.editor.set_status(status);
Ok(())
}

Expand Down

0 comments on commit cc6e5ab

Please sign in to comment.