1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
use crate::Context;
use sequent::SimulationError;
use revolver::command::{
ApplyCommandError, ApplyOutcome, Command, Description, Example, NamedCommandParser,
ParseCommandError,
};
use revolver::looper::Looper;
use revolver::terminal::{Terminal};
use std::borrow::Cow;
pub struct Jump {
location: usize,
}
impl<S: Clone, C: Context<S>, T: Terminal> Command<C, SimulationError<S>, T> for Jump {
fn apply(
&mut self,
looper: &mut Looper<C, SimulationError<S>, T>,
) -> Result<ApplyOutcome, ApplyCommandError<SimulationError<S>>> {
let (terminal, _, context) = looper.split();
context.sim().jump(self.location).map_err(ApplyCommandError::Application)?;
context.print_state(terminal)?;
Ok(ApplyOutcome::Applied)
}
}
pub struct Parser;
impl<S: Clone, C: Context<S>, T: Terminal> NamedCommandParser<C, SimulationError<S>, T> for Parser {
fn parse(
&self,
s: &str,
) -> Result<Box<dyn Command<C, SimulationError<S>, T>>, ParseCommandError> {
if s.is_empty() {
return Err(ParseCommandError(
"empty arguments to 'jump'".into(),
));
}
let location = s.parse().map_err(ParseCommandError::convert)?;
Ok(Box::new(Jump { location }))
}
fn shorthand(&self) -> Option<Cow<'static, str>> {
Some("j".into())
}
fn name(&self) -> Cow<'static, str> {
"jump".into()
}
fn description(&self) -> Description {
Description {
purpose: "Jumps to a specified cursor location in the timeline.".into(),
usage: "<location>".into(),
examples: vec![Example {
scenario: "jump to location 2".into(),
command: "2".into(),
}],
}
}
}
#[cfg(test)]
mod tests;