diff --git a/helix-view/src/coverage.rs b/helix-view/src/coverage.rs index cafca5d89..afe7b3abe 100644 --- a/helix-view/src/coverage.rs +++ b/helix-view/src/coverage.rs @@ -3,6 +3,7 @@ use serde::Deserialize; use std::collections::HashMap; use std::fs::File; use std::io::BufReader; +use std::time::SystemTime; pub struct Coverage { pub files: HashMap, @@ -10,6 +11,7 @@ pub struct Coverage { pub struct FileCoverage { pub lines: HashMap, + pub modified_time: Option, } #[derive(Deserialize, Debug)] @@ -18,6 +20,7 @@ struct RawCoverage { version: String, sources: Sources, packages: Packages, + modified_time: Option, } #[derive(Deserialize, Debug)] @@ -72,8 +75,10 @@ struct Line { pub fn parse(path: std::path::PathBuf) -> Option { let file = File::open(path).ok()?; + let metadata = file.metadata().ok()?; let reader = BufReader::new(file); - let tmp: RawCoverage = from_reader(reader).ok()?; + let mut tmp: RawCoverage = from_reader(reader).ok()?; + tmp.modified_time = metadata.modified().ok(); Some(tmp.into()) } @@ -89,7 +94,13 @@ impl From for Coverage { for source in &coverage.sources.source { let path: std::path::PathBuf = [&source.name, &class.filename].iter().collect(); if path.exists() { - files.insert(path, FileCoverage { lines }); + files.insert( + path, + FileCoverage { + lines, + modified_time: coverage.modified_time, + }, + ); break; } } diff --git a/helix-view/src/document.rs b/helix-view/src/document.rs index 5c68c62ac..cafa8904c 100644 --- a/helix-view/src/document.rs +++ b/helix-view/src/document.rs @@ -185,7 +185,7 @@ pub struct Document { // Last time we wrote to the file. This will carry the time the file was last opened if there // were no saves. - last_saved_time: SystemTime, + pub last_saved_time: SystemTime, last_saved_revision: usize, version: i32, // should be usize? diff --git a/helix-view/src/gutter.rs b/helix-view/src/gutter.rs index 2eaa0e0d6..9ca73ce8f 100644 --- a/helix-view/src/gutter.rs +++ b/helix-view/src/gutter.rs @@ -159,28 +159,35 @@ pub fn coverage<'doc>( path = tmp.into(); } } - if let Some(file) = cov.files.get(&path) { - let this_file = coverage::FileCoverage { - lines: file.lines.clone(), - }; - return Box::new( - move |line: usize, - _selected: bool, - _first_visual_line: bool, - out: &mut String| { - if let Some(line_coverage) = this_file.lines.get(&(line as u32)) { - let (icon, style) = if *line_coverage { - ("┃", covered) + if let Some(file_coverage) = cov.files.get(&path) { + if file_coverage + .modified_time + .is_some_and(|x| x > doc.last_saved_time) + { + // clone file coverage so it can be moved into the closure + let this_file = coverage::FileCoverage { + lines: file_coverage.lines.clone(), + modified_time: file_coverage.modified_time, + }; + return Box::new( + move |line: usize, + _selected: bool, + _first_visual_line: bool, + out: &mut String| { + if let Some(line_coverage) = this_file.lines.get(&(line as u32)) { + let (icon, style) = if *line_coverage { + ("┃", covered) + } else { + ("┃", not_covered) + }; + write!(out, "{}", icon).unwrap(); + Some(style) } else { - ("┃", not_covered) - }; - write!(out, "{}", icon).unwrap(); - Some(style) - } else { - None - } - }, - ); + None + } + }, + ); + } } } }