fmtview 0.3.1

Fast CLI viewer for highlighting, search, and diffs across JSON, JSONL, markup, Markdown, TOML, text, and Jinja
Documentation
use anyhow::Result;

use crate::{
    load::ViewFile,
    syntax::{MarkdownFenceState, SyntaxKind},
};

const MARKDOWN_SYNTAX_CHECKPOINT_INTERVAL_LINES: usize = 512;
const MARKDOWN_SYNTAX_SCAN_CHUNK_LINES: usize = 512;

#[derive(Debug, Default)]
pub(in crate::viewer) struct MarkdownSyntaxCache {
    checkpoints: Vec<MarkdownSyntaxCheckpoint>,
}

#[derive(Debug, Clone, Copy)]
struct MarkdownSyntaxCheckpoint {
    line: usize,
    state: MarkdownFenceState,
}

impl MarkdownSyntaxCache {
    pub(in crate::viewer) fn line_modes(
        &mut self,
        file: &dyn ViewFile,
        start: usize,
        lines: &[String],
        mode: SyntaxKind,
    ) -> Result<Option<Vec<SyntaxKind>>> {
        if mode != SyntaxKind::Markdown || lines.is_empty() {
            return Ok(None);
        }

        let mut state = self.state_before(file, start)?;
        let modes = lines
            .iter()
            .map(|line| {
                let line_mode = state.line_syntax(line);
                state.advance(line);
                line_mode
            })
            .collect();

        self.remember_interval_checkpoint(start.saturating_add(lines.len()), state);
        Ok(Some(modes))
    }

    fn state_before(&mut self, file: &dyn ViewFile, target: usize) -> Result<MarkdownFenceState> {
        let (mut line, mut state) = self
            .checkpoint_before(target)
            .map(|checkpoint| (checkpoint.line, checkpoint.state))
            .unwrap_or((0, MarkdownFenceState::default()));

        while line < target {
            let count = target
                .saturating_sub(line)
                .min(MARKDOWN_SYNTAX_SCAN_CHUNK_LINES);
            let lines = file.read_window(line, count)?;
            if lines.is_empty() {
                break;
            }

            for source_line in &lines {
                self.remember_interval_checkpoint(line, state);
                state.advance(source_line);
                line += 1;
                if line >= target {
                    break;
                }
            }

            if lines.len() < count {
                break;
            }
        }

        self.remember_interval_checkpoint(line, state);
        Ok(state)
    }

    fn checkpoint_before(&self, target: usize) -> Option<MarkdownSyntaxCheckpoint> {
        self.checkpoints
            .iter()
            .rev()
            .find(|checkpoint| checkpoint.line <= target)
            .copied()
    }

    fn remember_interval_checkpoint(&mut self, line: usize, state: MarkdownFenceState) {
        if line % MARKDOWN_SYNTAX_CHECKPOINT_INTERVAL_LINES != 0 {
            return;
        }

        match self
            .checkpoints
            .binary_search_by_key(&line, |checkpoint| checkpoint.line)
        {
            Ok(position) => self.checkpoints[position].state = state,
            Err(position) => self
                .checkpoints
                .insert(position, MarkdownSyntaxCheckpoint { line, state }),
        }
    }

    #[cfg(test)]
    pub(in crate::viewer) fn checkpoint_count(&self) -> usize {
        self.checkpoints.len()
    }
}