From bbb6ea9c321eee169fcc346d63dfec86a6ecba57 Mon Sep 17 00:00:00 2001 From: Bryan Van de Ven Date: Thu, 5 Oct 2023 15:25:48 -0700 Subject: [PATCH 1/7] Add support for panning with arrow keys --- Cargo.lock | 70 ++++++++++++- Cargo.toml | 1 + src/app.rs | 35 +++++++ src/timestamp.rs | 267 +++++++++++++++++++++++++++-------------------- 4 files changed, 258 insertions(+), 115 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index afbe68d..a3d08f5 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1669,7 +1669,7 @@ dependencies = [ "bytemuck", "byteorder", "color_quant", - "num-rational", + "num-rational 0.4.1", "num-traits", "png", ] @@ -1803,6 +1803,7 @@ dependencies = [ "env_logger", "getrandom", "log", + "percentage", "rand", "rayon", "reqwest", @@ -2066,6 +2067,41 @@ dependencies = [ "minimal-lexical", ] +[[package]] +name = "num" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8536030f9fea7127f841b45bb6243b27255787fb4eb83958aa1ef9d2fdc0c36" +dependencies = [ + "num-bigint", + "num-complex", + "num-integer", + "num-iter", + "num-rational 0.2.4", + "num-traits", +] + +[[package]] +name = "num-bigint" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "090c7f9998ee0ff65aa5b723e4009f7b217707f1fb5ea551329cc4d6231fb304" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-complex" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6b19411a9719e753aff12e5187b74d60d3dc449ec3f4dc21e3989c3f554bc95" +dependencies = [ + "autocfg", + "num-traits", +] + [[package]] name = "num-integer" version = "0.1.45" @@ -2076,6 +2112,29 @@ dependencies = [ "num-traits", ] +[[package]] +name = "num-iter" +version = "0.1.43" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7d03e6c028c5dc5cac6e2dec0efda81fc887605bb3d884578bb6d6bf7514e252" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-rational" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c000134b5dbf44adc5cb772486d335293351644b801551abe8f75c84cfa4aef" +dependencies = [ + "autocfg", + "num-bigint", + "num-integer", + "num-traits", +] + [[package]] name = "num-rational" version = "0.4.1" @@ -2301,6 +2360,15 @@ version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9b2a4787296e9989611394c33f193f676704af1686e70b8f8033ab5ba9a35a94" +[[package]] +name = "percentage" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2fd23b938276f14057220b707937bcb42fa76dda7560e57a2da30cb52d557937" +dependencies = [ + "num", +] + [[package]] name = "pin-project-lite" version = "0.2.9" diff --git a/Cargo.toml b/Cargo.toml index c2ba4d5..ca10ba1 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -11,6 +11,7 @@ client = ["dep:reqwest", "dep:url"] server = ["dep:actix-cors", "dep:actix-web"] [dependencies] +percentage = "0.1.0" egui = "0.22.0" egui_extras = "0.22.0" eframe = { version = "0.22.0", default-features = false, features = [ diff --git a/src/app.rs b/src/app.rs index c24d166..f2bb8aa 100644 --- a/src/app.rs +++ b/src/app.rs @@ -8,6 +8,7 @@ use egui::{ Align2, Color32, NumExt, Pos2, Rect, RichText, ScrollArea, Slider, Stroke, TextStyle, Vec2, }; use egui_extras::{Column, TableBuilder}; +use percentage::{Percentage, PercentageInteger}; use serde::{Deserialize, Serialize}; use crate::data::{ @@ -1669,6 +1670,11 @@ impl Window { } } +enum PanDirection { + Left, + Right, +} + impl ProfApp { /// Called once before the first frame. pub fn new( @@ -1711,6 +1717,19 @@ impl ProfApp { result } + fn pan(cx: &mut Context, percent: PercentageInteger, dir: PanDirection) { + if percent.value() == 0 { + return; + } + + let duration: i64 = percent.apply_to(cx.view_interval.duration_ns()); + let sign: i64 = match dir { + PanDirection::Left => -1, + PanDirection::Right => 1, + }; + cx.view_interval = cx.view_interval.translate(duration * sign) + } + fn zoom(cx: &mut Context, interval: Interval) { if cx.view_interval == interval { return; @@ -1783,6 +1802,7 @@ impl ProfApp { UndoZoom, RedoZoom, ResetZoom, + Pan(PercentageInteger, PanDirection), ExpandVertical, ShrinkVertical, ResetVertical, @@ -1814,8 +1834,20 @@ impl ProfApp { } else { Actions::NoAction } + } else if i.modifiers.shift { + if i.key_pressed(egui::Key::ArrowLeft) { + Actions::Pan(Percentage::from(1), PanDirection::Left) + } else if i.key_pressed(egui::Key::ArrowRight) { + Actions::Pan(Percentage::from(1), PanDirection::Right) + } else { + Actions::NoAction + } } else if i.key_pressed(egui::Key::H) { Actions::ToggleControls + } else if i.key_pressed(egui::Key::ArrowLeft) { + Actions::Pan(Percentage::from(5), PanDirection::Left) + } else if i.key_pressed(egui::Key::ArrowRight) { + Actions::Pan(Percentage::from(5), PanDirection::Right) } else { Actions::NoAction } @@ -1826,6 +1858,7 @@ impl ProfApp { Actions::UndoZoom => ProfApp::undo_zoom(cx), Actions::RedoZoom => ProfApp::redo_zoom(cx), Actions::ResetZoom => ProfApp::zoom(cx, cx.total_interval), + Actions::Pan(percent, dir) => ProfApp::pan(cx, percent, dir), Actions::ExpandVertical => ProfApp::multiply_scale_factor(cx, 2.0), Actions::ShrinkVertical => ProfApp::multiply_scale_factor(cx, 0.5), Actions::ResetVertical => ProfApp::reset_scale_factor(cx), @@ -1966,6 +1999,8 @@ impl ProfApp { }); }; show_row("Zoom to Interval", "Click and Drag"); + show_row("Pan 5%", "Left/Right Arrow"); + show_row("Pan 1%", "Shift + Left/Right Arrow"); show_row("Zoom In", "Ctrl + Plus/Equals"); show_row("Zoom Out", "Ctrl + Minus"); show_row("Undo Zoom", "Ctrl + Left Arrow"); diff --git a/src/timestamp.rs b/src/timestamp.rs index 0d470e8..c492d54 100644 --- a/src/timestamp.rs +++ b/src/timestamp.rs @@ -164,148 +164,187 @@ impl Interval { stop: Timestamp(self.stop.0 + duration_ns), } } + // Translate interval by duration_ns on both sides. + pub fn translate(self, duration_ns: i64) -> Self { + Self { + start: Timestamp(self.start.0 + duration_ns), + stop: Timestamp(self.stop.0 + duration_ns), + } + } } #[cfg(test)] mod tests { use super::*; - #[test] - fn test_s() { - assert_eq!(Timestamp::parse("123.4 s"), Ok(Timestamp(123_400_000_000))); - } + mod timestamp { + use super::*; - #[test] - fn test_ms() { - assert_eq!(Timestamp::parse("234.5 ms"), Ok(Timestamp(234_500_000))); - } + #[test] + fn test_s() { + assert_eq!(Timestamp::parse("123.4 s"), Ok(Timestamp(123_400_000_000))); + } - #[test] - fn test_us() { - assert_eq!(Timestamp::parse("345.6 us"), Ok(Timestamp(345_600))); - } + #[test] + fn test_ms() { + assert_eq!(Timestamp::parse("234.5 ms"), Ok(Timestamp(234_500_000))); + } - #[test] - fn test_ns() { - assert_eq!(Timestamp::parse("567.0 ns"), Ok(Timestamp(567))); - } + #[test] + fn test_us() { + assert_eq!(Timestamp::parse("345.6 us"), Ok(Timestamp(345_600))); + } - #[test] - fn test_s_upper() { - assert_eq!(Timestamp::parse("123.4 S"), Ok(Timestamp(123_400_000_000))); - } + #[test] + fn test_ns() { + assert_eq!(Timestamp::parse("567.0 ns"), Ok(Timestamp(567))); + } - #[test] - fn test_ms_upper() { - assert_eq!(Timestamp::parse("234.5 MS"), Ok(Timestamp(234_500_000))); - } + #[test] + fn test_s_upper() { + assert_eq!(Timestamp::parse("123.4 S"), Ok(Timestamp(123_400_000_000))); + } - #[test] - fn test_us_upper() { - assert_eq!(Timestamp::parse("345.6 US"), Ok(Timestamp(345_600))); - } + #[test] + fn test_ms_upper() { + assert_eq!(Timestamp::parse("234.5 MS"), Ok(Timestamp(234_500_000))); + } - #[test] - fn test_ns_upper() { - assert_eq!(Timestamp::parse("567.0 NS"), Ok(Timestamp(567))); - } + #[test] + fn test_us_upper() { + assert_eq!(Timestamp::parse("345.6 US"), Ok(Timestamp(345_600))); + } - #[test] - fn test_s_nospace() { - assert_eq!(Timestamp::parse("123.4s"), Ok(Timestamp(123_400_000_000))); - } + #[test] + fn test_ns_upper() { + assert_eq!(Timestamp::parse("567.0 NS"), Ok(Timestamp(567))); + } - #[test] - fn test_ms_nospace() { - assert_eq!(Timestamp::parse("234.5ms"), Ok(Timestamp(234_500_000))); - } + #[test] + fn test_s_nospace() { + assert_eq!(Timestamp::parse("123.4s"), Ok(Timestamp(123_400_000_000))); + } - #[test] - fn test_us_nospace() { - assert_eq!(Timestamp::parse("345.6us"), Ok(Timestamp(345_600))); - } + #[test] + fn test_ms_nospace() { + assert_eq!(Timestamp::parse("234.5ms"), Ok(Timestamp(234_500_000))); + } - #[test] - fn test_ns_nospace() { - assert_eq!(Timestamp::parse("567.0ns"), Ok(Timestamp(567))); - } + #[test] + fn test_us_nospace() { + assert_eq!(Timestamp::parse("345.6us"), Ok(Timestamp(345_600))); + } - #[test] - fn test_s_spaces() { - assert_eq!( - Timestamp::parse(" 123.4 s "), - Ok(Timestamp(123_400_000_000)) - ); - } + #[test] + fn test_ns_nospace() { + assert_eq!(Timestamp::parse("567.0ns"), Ok(Timestamp(567))); + } - #[test] - fn test_ms_spaces() { - assert_eq!( - Timestamp::parse(" 234.5 ms "), - Ok(Timestamp(234_500_000)) - ); - } + #[test] + fn test_s_spaces() { + assert_eq!( + Timestamp::parse(" 123.4 s "), + Ok(Timestamp(123_400_000_000)) + ); + } - #[test] - fn test_us_spaces() { - assert_eq!(Timestamp::parse(" 345.6 us "), Ok(Timestamp(345_600))); - } + #[test] + fn test_ms_spaces() { + assert_eq!( + Timestamp::parse(" 234.5 ms "), + Ok(Timestamp(234_500_000)) + ); + } - #[test] - fn test_ns_spaces() { - assert_eq!(Timestamp::parse(" 567.0 ns "), Ok(Timestamp(567))); - } + #[test] + fn test_us_spaces() { + assert_eq!(Timestamp::parse(" 345.6 us "), Ok(Timestamp(345_600))); + } - #[test] - fn test_no_unit() { - assert_eq!(Timestamp::parse("500.0"), Err(TimestampParseError::NoUnit)); - } + #[test] + fn test_ns_spaces() { + assert_eq!(Timestamp::parse(" 567.0 ns "), Ok(Timestamp(567))); + } - #[test] - fn test_no_value() { - assert_eq!( - Timestamp::parse("ms"), - Err(TimestampParseError::InvalidValue) - ); - } + #[test] + fn test_no_unit() { + assert_eq!(Timestamp::parse("500.0"), Err(TimestampParseError::NoUnit)); + } - #[test] - fn test_invalid_unit() { - assert_eq!( - Timestamp::parse("500.0 foo"), - Err(TimestampParseError::InvalidUnit) - ); - } + #[test] + fn test_no_value() { + assert_eq!( + Timestamp::parse("ms"), + Err(TimestampParseError::InvalidValue) + ); + } - #[test] - fn test_invalid_value() { - assert_eq!( - Timestamp::parse("foo ms"), - Err(TimestampParseError::InvalidValue) - ); - } + #[test] + fn test_invalid_unit() { + assert_eq!( + Timestamp::parse("500.0 foo"), + Err(TimestampParseError::InvalidUnit) + ); + } - #[test] - fn test_invalid_value2() { - assert_eq!( - Timestamp::parse("500.0.0 ms"), - Err(TimestampParseError::InvalidValue) - ); - } + #[test] + fn test_invalid_value() { + assert_eq!( + Timestamp::parse("foo ms"), + Err(TimestampParseError::InvalidValue) + ); + } + + #[test] + fn test_invalid_value2() { + assert_eq!( + Timestamp::parse("500.0.0 ms"), + Err(TimestampParseError::InvalidValue) + ); + } - #[test] - fn test_invalid_value3() { - assert_eq!( - Timestamp::parse("500.0.0"), - Err(TimestampParseError::NoUnit) - ); + #[test] + fn test_invalid_value3() { + assert_eq!( + Timestamp::parse("500.0.0"), + Err(TimestampParseError::NoUnit) + ); + } + + #[test] + fn test_extra() { + assert_eq!( + Timestamp::parse("500.0 ms asdfadf"), + Err(TimestampParseError::InvalidUnit) + ); + } } - #[test] - fn test_extra() { - assert_eq!( - Timestamp::parse("500.0 ms asdfadf"), - Err(TimestampParseError::InvalidUnit) - ); + mod interval { + use super::*; + + #[test] + fn test_translate_positive() { + let start: Timestamp = Timestamp::parse("234.5 ms").unwrap(); + let end: Timestamp = Timestamp::parse("235.5 ms").unwrap(); + let expected_start: Timestamp = Timestamp(start.0 + 250); + let expected_end: Timestamp = Timestamp(end.0 + 250); + assert_eq!( + Interval::new(start, end).translate(250), + Interval::new(expected_start, expected_end) + ) + } + + #[test] + fn test_translate_negative() { + let start: Timestamp = Timestamp::parse("234.5 ms").unwrap(); + let end: Timestamp = Timestamp::parse("235.5 ms").unwrap(); + let expected_start: Timestamp = Timestamp(start.0 - 250); + let expected_end: Timestamp = Timestamp(end.0 - 250); + assert_eq!( + Interval::new(start, end).translate(-250), + Interval::new(expected_start, expected_end) + ) + } } } From a0d572e7d68c75f6aab9a3a522bf521c909b7cb4 Mon Sep 17 00:00:00 2001 From: Bryan Van de Ven Date: Thu, 5 Oct 2023 16:12:21 -0700 Subject: [PATCH 2/7] DRY --- src/app.rs | 25 ++++++++++++------------- 1 file changed, 12 insertions(+), 13 deletions(-) diff --git a/src/app.rs b/src/app.rs index f2bb8aa..a3043e2 100644 --- a/src/app.rs +++ b/src/app.rs @@ -1717,6 +1717,13 @@ impl ProfApp { result } + fn update_interval_state(cx: &mut Context) { + cx.interval_state.start_buffer = cx.view_interval.start.to_string(); + cx.interval_state.stop_buffer = cx.view_interval.stop.to_string(); + cx.interval_state.start_error = None; + cx.interval_state.stop_error = None; + } + fn pan(cx: &mut Context, percent: PercentageInteger, dir: PanDirection) { if percent.value() == 0 { return; @@ -1727,7 +1734,8 @@ impl ProfApp { PanDirection::Left => -1, PanDirection::Right => 1, }; - cx.view_interval = cx.view_interval.translate(duration * sign) + cx.view_interval = cx.view_interval.translate(duration * sign); + ProfApp::update_interval_state(cx); } fn zoom(cx: &mut Context, interval: Interval) { @@ -1739,10 +1747,7 @@ impl ProfApp { cx.zoom_state.levels.truncate(cx.zoom_state.index + 1); cx.zoom_state.levels.push(cx.view_interval); cx.zoom_state.index = cx.zoom_state.levels.len() - 1; - cx.interval_state.start_buffer = cx.view_interval.start.to_string(); - cx.interval_state.stop_buffer = cx.view_interval.stop.to_string(); - cx.interval_state.start_error = None; - cx.interval_state.stop_error = None; + ProfApp::update_interval_state(cx); } fn undo_zoom(cx: &mut Context) { @@ -1751,10 +1756,7 @@ impl ProfApp { } cx.zoom_state.index -= 1; cx.view_interval = cx.zoom_state.levels[cx.zoom_state.index]; - cx.interval_state.start_buffer = cx.view_interval.start.to_string(); - cx.interval_state.stop_buffer = cx.view_interval.stop.to_string(); - cx.interval_state.start_error = None; - cx.interval_state.stop_error = None; + ProfApp::update_interval_state(cx); } fn redo_zoom(cx: &mut Context) { @@ -1763,10 +1765,7 @@ impl ProfApp { } cx.zoom_state.index += 1; cx.view_interval = cx.zoom_state.levels[cx.zoom_state.index]; - cx.interval_state.start_buffer = cx.view_interval.start.to_string(); - cx.interval_state.stop_buffer = cx.view_interval.stop.to_string(); - cx.interval_state.start_error = None; - cx.interval_state.stop_error = None; + ProfApp::update_interval_state(cx); } fn zoom_in(cx: &mut Context) { From 7f54477441f4ddf2f72154961ac8d0d8c039f329 Mon Sep 17 00:00:00 2001 From: Bryan Van de Ven Date: Fri, 6 Oct 2023 11:05:57 -0700 Subject: [PATCH 3/7] some renaming for generalization --- src/app.rs | 71 +++++++++++++++++++++++++++++------------------------- 1 file changed, 38 insertions(+), 33 deletions(-) diff --git a/src/app.rs b/src/app.rs index a3043e2..c6cfebb 100644 --- a/src/app.rs +++ b/src/app.rs @@ -244,9 +244,9 @@ struct Context { show_controls: bool, #[serde(skip)] - zoom_state: ZoomState, + view_interval_history: ZoomState, #[serde(skip)] - interval_state: IntervalSelectState, + interval_select_state: IntervalSelectState, } #[derive(Default, Deserialize, Serialize)] @@ -1450,62 +1450,65 @@ impl Window { let start_res = ui .horizontal(|ui| { ui.label("Start:"); - ui.text_edit_singleline(&mut cx.interval_state.start_buffer) + ui.text_edit_singleline(&mut cx.interval_select_state.start_buffer) }) .inner; - if let Some(error) = cx.interval_state.start_error { + if let Some(error) = cx.interval_select_state.start_error { ui.label(RichText::new(error.to_string()).color(Color32::RED)); } let stop_res = ui .horizontal(|ui| { ui.label("Stop:"); - ui.text_edit_singleline(&mut cx.interval_state.stop_buffer) + ui.text_edit_singleline(&mut cx.interval_select_state.stop_buffer) }) .inner; - if let Some(error) = cx.interval_state.stop_error { + if let Some(error) = cx.interval_select_state.stop_error { ui.label(RichText::new(error.to_string()).color(Color32::RED)); } if start_res.lost_focus() - && cx.interval_state.start_buffer != cx.view_interval.start.to_string() + && cx.interval_select_state.start_buffer != cx.view_interval.start.to_string() { - match Timestamp::parse(&cx.interval_state.start_buffer) { + match Timestamp::parse(&cx.interval_select_state.start_buffer) { Ok(start) => { // validate timestamp if start > cx.view_interval.stop { - cx.interval_state.start_error = Some(IntervalSelectError::StartAfterStop); + cx.interval_select_state.start_error = + Some(IntervalSelectError::StartAfterStop); return; } if start > cx.total_interval.stop { - cx.interval_state.start_error = Some(IntervalSelectError::StartAfterEnd); + cx.interval_select_state.start_error = + Some(IntervalSelectError::StartAfterEnd); return; } let target = Interval::new(start, cx.view_interval.stop); ProfApp::zoom(cx, target); } Err(e) => { - cx.interval_state.start_error = Some(e.into()); + cx.interval_select_state.start_error = Some(e.into()); } } } if stop_res.lost_focus() - && cx.interval_state.stop_buffer != cx.view_interval.stop.to_string() + && cx.interval_select_state.stop_buffer != cx.view_interval.stop.to_string() { - match Timestamp::parse(&cx.interval_state.stop_buffer) { + match Timestamp::parse(&cx.interval_select_state.stop_buffer) { Ok(stop) => { // validate timestamp if stop < cx.view_interval.start { - cx.interval_state.stop_error = Some(IntervalSelectError::StopBeforeStart); + cx.interval_select_state.stop_error = + Some(IntervalSelectError::StopBeforeStart); return; } let target = Interval::new(cx.view_interval.start, stop); ProfApp::zoom(cx, target); } Err(e) => { - cx.interval_state.stop_error = Some(e.into()); + cx.interval_select_state.stop_error = Some(e.into()); } } } @@ -1717,11 +1720,11 @@ impl ProfApp { result } - fn update_interval_state(cx: &mut Context) { - cx.interval_state.start_buffer = cx.view_interval.start.to_string(); - cx.interval_state.stop_buffer = cx.view_interval.stop.to_string(); - cx.interval_state.start_error = None; - cx.interval_state.stop_error = None; + fn update_interval_select_state(cx: &mut Context) { + cx.interval_select_state.start_buffer = cx.view_interval.start.to_string(); + cx.interval_select_state.stop_buffer = cx.view_interval.stop.to_string(); + cx.interval_select_state.start_error = None; + cx.interval_select_state.stop_error = None; } fn pan(cx: &mut Context, percent: PercentageInteger, dir: PanDirection) { @@ -1735,7 +1738,7 @@ impl ProfApp { PanDirection::Right => 1, }; cx.view_interval = cx.view_interval.translate(duration * sign); - ProfApp::update_interval_state(cx); + ProfApp::update_interval_select_state(cx); } fn zoom(cx: &mut Context, interval: Interval) { @@ -1744,28 +1747,30 @@ impl ProfApp { } cx.view_interval = interval; - cx.zoom_state.levels.truncate(cx.zoom_state.index + 1); - cx.zoom_state.levels.push(cx.view_interval); - cx.zoom_state.index = cx.zoom_state.levels.len() - 1; - ProfApp::update_interval_state(cx); + cx.view_interval_history + .levels + .truncate(cx.view_interval_history.index + 1); + cx.view_interval_history.levels.push(cx.view_interval); + cx.view_interval_history.index = cx.view_interval_history.levels.len() - 1; + ProfApp::update_interval_select_state(cx); } fn undo_zoom(cx: &mut Context) { - if cx.zoom_state.index == 0 { + if cx.view_interval_history.index == 0 { return; } - cx.zoom_state.index -= 1; - cx.view_interval = cx.zoom_state.levels[cx.zoom_state.index]; - ProfApp::update_interval_state(cx); + cx.view_interval_history.index -= 1; + cx.view_interval = cx.view_interval_history.levels[cx.view_interval_history.index]; + ProfApp::update_interval_select_state(cx); } fn redo_zoom(cx: &mut Context) { - if cx.zoom_state.index + 1 >= cx.zoom_state.levels.len() { + if cx.view_interval_history.index + 1 >= cx.view_interval_history.levels.len() { return; } - cx.zoom_state.index += 1; - cx.view_interval = cx.zoom_state.levels[cx.zoom_state.index]; - ProfApp::update_interval_state(cx); + cx.view_interval_history.index += 1; + cx.view_interval = cx.view_interval_history.levels[cx.view_interval_history.index]; + ProfApp::update_interval_select_state(cx); } fn zoom_in(cx: &mut Context) { From 5c0d57a8d4d463a3185ac7c598084d00fa16c729 Mon Sep 17 00:00:00 2001 From: Bryan Van de Ven Date: Fri, 6 Oct 2023 11:07:36 -0700 Subject: [PATCH 4/7] remove superfluous types --- src/app.rs | 4 ++-- src/timestamp.rs | 16 ++++++++-------- 2 files changed, 10 insertions(+), 10 deletions(-) diff --git a/src/app.rs b/src/app.rs index c6cfebb..66bb691 100644 --- a/src/app.rs +++ b/src/app.rs @@ -1732,8 +1732,8 @@ impl ProfApp { return; } - let duration: i64 = percent.apply_to(cx.view_interval.duration_ns()); - let sign: i64 = match dir { + let duration = percent.apply_to(cx.view_interval.duration_ns()); + let sign = match dir { PanDirection::Left => -1, PanDirection::Right => 1, }; diff --git a/src/timestamp.rs b/src/timestamp.rs index c492d54..8b4b09c 100644 --- a/src/timestamp.rs +++ b/src/timestamp.rs @@ -325,10 +325,10 @@ mod tests { #[test] fn test_translate_positive() { - let start: Timestamp = Timestamp::parse("234.5 ms").unwrap(); - let end: Timestamp = Timestamp::parse("235.5 ms").unwrap(); - let expected_start: Timestamp = Timestamp(start.0 + 250); - let expected_end: Timestamp = Timestamp(end.0 + 250); + let start = Timestamp::parse("234.5 ms").unwrap(); + let end = Timestamp::parse("235.5 ms").unwrap(); + let expected_start = Timestamp(start.0 + 250); + let expected_end = Timestamp(end.0 + 250); assert_eq!( Interval::new(start, end).translate(250), Interval::new(expected_start, expected_end) @@ -337,10 +337,10 @@ mod tests { #[test] fn test_translate_negative() { - let start: Timestamp = Timestamp::parse("234.5 ms").unwrap(); - let end: Timestamp = Timestamp::parse("235.5 ms").unwrap(); - let expected_start: Timestamp = Timestamp(start.0 - 250); - let expected_end: Timestamp = Timestamp(end.0 - 250); + let start = Timestamp::parse("234.5 ms").unwrap(); + let end = Timestamp::parse("235.5 ms").unwrap(); + let expected_start = Timestamp(start.0 - 250); + let expected_end = Timestamp(end.0 - 250); assert_eq!( Interval::new(start, end).translate(-250), Interval::new(expected_start, expected_end) From 49254cb80c8c65894caedac2f44643e48c15d444 Mon Sep 17 00:00:00 2001 From: Bryan Van de Ven Date: Fri, 6 Oct 2023 14:35:19 -0700 Subject: [PATCH 5/7] support undo/redo for zoom and (coalescing) pan --- src/app.rs | 51 ++++++++++++++++++++++++++++++++++++++------------- 1 file changed, 38 insertions(+), 13 deletions(-) diff --git a/src/app.rs b/src/app.rs index 66bb691..441605e 100644 --- a/src/app.rs +++ b/src/app.rs @@ -159,9 +159,16 @@ struct Window { config: Config, } +#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Deserialize, Serialize)] +enum IntervalOrigin { + Zoom, + Pan, +} + #[derive(Debug, Clone, Default, Deserialize, Serialize)] -struct ZoomState { +struct IntervalState { levels: Vec, + origins: Vec, index: usize, } @@ -244,7 +251,7 @@ struct Context { show_controls: bool, #[serde(skip)] - view_interval_history: ZoomState, + view_interval_history: IntervalState, #[serde(skip)] interval_select_state: IntervalSelectState, } @@ -1727,6 +1734,27 @@ impl ProfApp { cx.interval_select_state.stop_error = None; } + fn update_view_interval(cx: &mut Context, interval: Interval, origin: IntervalOrigin) { + cx.view_interval = interval; + + let history = &mut cx.view_interval_history; + let index = history.index; + + // Only keep at most one Pan origin in a row + if history.levels.len() > 0 { + if history.origins[index] == origin && origin == IntervalOrigin::Pan { + history.levels.truncate(index); + history.origins.truncate(index); + } + } + + history.levels.truncate(index + 1); + history.levels.push(interval); + history.origins.truncate(index + 1); + history.origins.push(origin); + history.index = history.levels.len() - 1; + } + fn pan(cx: &mut Context, percent: PercentageInteger, dir: PanDirection) { if percent.value() == 0 { return; @@ -1737,7 +1765,9 @@ impl ProfApp { PanDirection::Left => -1, PanDirection::Right => 1, }; - cx.view_interval = cx.view_interval.translate(duration * sign); + let interval = cx.view_interval.translate(duration * sign); + + ProfApp::update_view_interval(cx, interval, IntervalOrigin::Pan); ProfApp::update_interval_select_state(cx); } @@ -1746,16 +1776,11 @@ impl ProfApp { return; } - cx.view_interval = interval; - cx.view_interval_history - .levels - .truncate(cx.view_interval_history.index + 1); - cx.view_interval_history.levels.push(cx.view_interval); - cx.view_interval_history.index = cx.view_interval_history.levels.len() - 1; + ProfApp::update_view_interval(cx, interval, IntervalOrigin::Zoom); ProfApp::update_interval_select_state(cx); } - fn undo_zoom(cx: &mut Context) { + fn undo_pan_zoom(cx: &mut Context) { if cx.view_interval_history.index == 0 { return; } @@ -1764,7 +1789,7 @@ impl ProfApp { ProfApp::update_interval_select_state(cx); } - fn redo_zoom(cx: &mut Context) { + fn redo_pan_zoom(cx: &mut Context) { if cx.view_interval_history.index + 1 >= cx.view_interval_history.levels.len() { return; } @@ -1859,8 +1884,8 @@ impl ProfApp { match action { Actions::ZoomIn => ProfApp::zoom_in(cx), Actions::ZoomOut => ProfApp::zoom_out(cx), - Actions::UndoZoom => ProfApp::undo_zoom(cx), - Actions::RedoZoom => ProfApp::redo_zoom(cx), + Actions::UndoZoom => ProfApp::undo_pan_zoom(cx), + Actions::RedoZoom => ProfApp::redo_pan_zoom(cx), Actions::ResetZoom => ProfApp::zoom(cx, cx.total_interval), Actions::Pan(percent, dir) => ProfApp::pan(cx, percent, dir), Actions::ExpandVertical => ProfApp::multiply_scale_factor(cx, 2.0), From 2ba4bc01dd073bd7baafaeabc08c93f40a327e0a Mon Sep 17 00:00:00 2001 From: Bryan Van de Ven Date: Fri, 6 Oct 2023 14:40:33 -0700 Subject: [PATCH 6/7] update command help too --- src/app.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/app.rs b/src/app.rs index 441605e..745c005 100644 --- a/src/app.rs +++ b/src/app.rs @@ -2032,9 +2032,9 @@ impl ProfApp { show_row("Pan 1%", "Shift + Left/Right Arrow"); show_row("Zoom In", "Ctrl + Plus/Equals"); show_row("Zoom Out", "Ctrl + Minus"); - show_row("Undo Zoom", "Ctrl + Left Arrow"); - show_row("Redo Zoom", "Ctrl + Right Arrow"); - show_row("Reset Zoom", "Ctrl + 0"); + show_row("Undo Pan/Zoom", "Ctrl + Left Arrow"); + show_row("Redo Pan/Zoom", "Ctrl + Right Arrow"); + show_row("Reset Pan/Zoom", "Ctrl + 0"); show_row("Expand Vertical Spacing", "Ctrl + Alt + Plus/Equals"); show_row("Shrink Vertical Spacing", "Ctrl + Alt + Minus"); show_row("Reset Vertical Spacing", "Ctrl + Alt + 0"); From ba4b2da303e96557cc9159aa2845189f1516ca61 Mon Sep 17 00:00:00 2001 From: Bryan Van de Ven Date: Fri, 6 Oct 2023 15:18:47 -0700 Subject: [PATCH 7/7] lint --- src/app.rs | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/src/app.rs b/src/app.rs index 745c005..999f700 100644 --- a/src/app.rs +++ b/src/app.rs @@ -1741,11 +1741,12 @@ impl ProfApp { let index = history.index; // Only keep at most one Pan origin in a row - if history.levels.len() > 0 { - if history.origins[index] == origin && origin == IntervalOrigin::Pan { - history.levels.truncate(index); - history.origins.truncate(index); - } + if !history.levels.is_empty() + && history.origins[index] == IntervalOrigin::Pan + && origin == IntervalOrigin::Pan + { + history.levels.truncate(index); + history.origins.truncate(index); } history.levels.truncate(index + 1);