Want a Matrix client that uses Vim keybindings (#1)

This commit is contained in:
Ulyssa 2022-12-29 18:00:59 -08:00
parent 704f631d54
commit 262c96b62f
No known key found for this signature in database
GPG key ID: 1B3965A3D18B9B64
22 changed files with 9050 additions and 7 deletions

202
src/commands.rs Normal file
View file

@ -0,0 +1,202 @@
use modalkit::{
editing::{action::WindowAction, base::OpenTarget},
env::vim::command::{CommandContext, CommandDescription},
input::commands::{CommandError, CommandResult, CommandStep},
input::InputContext,
};
use crate::base::{
IambAction,
IambId,
ProgramCommand,
ProgramCommands,
ProgramContext,
VerifyAction,
};
type ProgContext = CommandContext<ProgramContext>;
type ProgResult = CommandResult<ProgramCommand>;
fn iamb_verify(desc: CommandDescription, ctx: &mut ProgContext) -> ProgResult {
let mut args = desc.arg.strings()?;
match args.len() {
0 => {
let open = WindowAction::Switch(OpenTarget::Application(IambId::VerifyList));
let step = CommandStep::Continue(open.into(), ctx.context.take());
return Ok(step);
},
1 => {
return Result::Err(CommandError::InvalidArgument);
},
2 => {
let act = match args[0].as_str() {
"accept" => VerifyAction::Accept,
"cancel" => VerifyAction::Cancel,
"confirm" => VerifyAction::Confirm,
"mismatch" => VerifyAction::Mismatch,
"request" => {
let iact = IambAction::VerifyRequest(args.remove(1));
let step = CommandStep::Continue(iact.into(), ctx.context.take());
return Ok(step);
},
_ => return Result::Err(CommandError::InvalidArgument),
};
let vact = IambAction::Verify(act, args.remove(1));
let step = CommandStep::Continue(vact.into(), ctx.context.take());
return Ok(step);
},
_ => {
return Result::Err(CommandError::InvalidArgument);
},
}
}
fn iamb_dms(desc: CommandDescription, ctx: &mut ProgContext) -> ProgResult {
if !desc.arg.text.is_empty() {
return Result::Err(CommandError::InvalidArgument);
}
let open = WindowAction::Switch(OpenTarget::Application(IambId::DirectList));
let step = CommandStep::Continue(open.into(), ctx.context.take());
return Ok(step);
}
fn iamb_rooms(desc: CommandDescription, ctx: &mut ProgContext) -> ProgResult {
if !desc.arg.text.is_empty() {
return Result::Err(CommandError::InvalidArgument);
}
let open = WindowAction::Switch(OpenTarget::Application(IambId::RoomList));
let step = CommandStep::Continue(open.into(), ctx.context.take());
return Ok(step);
}
fn iamb_spaces(desc: CommandDescription, ctx: &mut ProgContext) -> ProgResult {
if !desc.arg.text.is_empty() {
return Result::Err(CommandError::InvalidArgument);
}
let open = WindowAction::Switch(OpenTarget::Application(IambId::SpaceList));
let step = CommandStep::Continue(open.into(), ctx.context.take());
return Ok(step);
}
fn iamb_welcome(desc: CommandDescription, ctx: &mut ProgContext) -> ProgResult {
if !desc.arg.text.is_empty() {
return Result::Err(CommandError::InvalidArgument);
}
let open = WindowAction::Switch(OpenTarget::Application(IambId::Welcome));
let step = CommandStep::Continue(open.into(), ctx.context.take());
return Ok(step);
}
fn iamb_join(desc: CommandDescription, ctx: &mut ProgContext) -> ProgResult {
let mut args = desc.arg.filenames()?;
if args.len() != 1 {
return Result::Err(CommandError::InvalidArgument);
}
let open = WindowAction::Switch(args.remove(0));
let step = CommandStep::Continue(open.into(), ctx.context.take());
return Ok(step);
}
fn add_iamb_commands(cmds: &mut ProgramCommands) {
cmds.add_command(ProgramCommand { names: vec!["dms".into()], f: iamb_dms });
cmds.add_command(ProgramCommand { names: vec!["join".into()], f: iamb_join });
cmds.add_command(ProgramCommand { names: vec!["rooms".into()], f: iamb_rooms });
cmds.add_command(ProgramCommand { names: vec!["spaces".into()], f: iamb_spaces });
cmds.add_command(ProgramCommand { names: vec!["verify".into()], f: iamb_verify });
cmds.add_command(ProgramCommand { names: vec!["welcome".into()], f: iamb_welcome });
}
pub fn setup_commands() -> ProgramCommands {
let mut cmds = ProgramCommands::default();
add_iamb_commands(&mut cmds);
return cmds;
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_cmd_verify() {
let mut cmds = setup_commands();
let ctx = ProgramContext::default();
let res = cmds.input_cmd(":verify", ctx.clone()).unwrap();
let act = WindowAction::Switch(OpenTarget::Application(IambId::VerifyList));
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds.input_cmd(":verify request @user1:example.com", ctx.clone()).unwrap();
let act = IambAction::VerifyRequest("@user1:example.com".into());
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds
.input_cmd(":verify accept @user1:example.com/FOOBAR", ctx.clone())
.unwrap();
let act = IambAction::Verify(VerifyAction::Accept, "@user1:example.com/FOOBAR".into());
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds
.input_cmd(":verify mismatch @user2:example.com/QUUXBAZ", ctx.clone())
.unwrap();
let act = IambAction::Verify(VerifyAction::Mismatch, "@user2:example.com/QUUXBAZ".into());
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds
.input_cmd(":verify cancel @user3:example.com/MYDEVICE", ctx.clone())
.unwrap();
let act = IambAction::Verify(VerifyAction::Cancel, "@user3:example.com/MYDEVICE".into());
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds
.input_cmd(":verify confirm @user4:example.com/GOODDEV", ctx.clone())
.unwrap();
let act = IambAction::Verify(VerifyAction::Confirm, "@user4:example.com/GOODDEV".into());
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds.input_cmd(":verify confirm", ctx.clone());
assert_eq!(res, Err(CommandError::InvalidArgument));
let res = cmds.input_cmd(":verify cancel @user4:example.com MYDEVICE", ctx.clone());
assert_eq!(res, Err(CommandError::InvalidArgument));
let res = cmds.input_cmd(":verify mismatch a b c d e f", ctx.clone());
assert_eq!(res, Err(CommandError::InvalidArgument));
}
#[test]
fn test_cmd_join() {
let mut cmds = setup_commands();
let ctx = ProgramContext::default();
let res = cmds.input_cmd("join #foobar:example.com", ctx.clone()).unwrap();
let act = WindowAction::Switch(OpenTarget::Name("#foobar:example.com".into()));
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds.input_cmd("join #", ctx.clone()).unwrap();
let act = WindowAction::Switch(OpenTarget::Alternate);
assert_eq!(res, vec![(act.into(), ctx.clone())]);
let res = cmds.input_cmd("join", ctx.clone());
assert_eq!(res, Err(CommandError::InvalidArgument));
let res = cmds.input_cmd("join foo bar", ctx.clone());
assert_eq!(res, Err(CommandError::InvalidArgument));
}
}