reovim-driver-syntax 0.14.4

Syntax highlighting driver for reovim (trait definitions only)
Documentation
use {
    super::*,
    crate::{Annotation, SyntaxDriver, SyntaxEdit},
    std::ops::Range,
};

/// Mock driver for testing.
struct MockDriver;

impl SyntaxDriver for MockDriver {
    #[allow(clippy::unnecessary_literal_bound)]
    fn language(&self) -> &str {
        "mock"
    }

    fn parse(&mut self, _content: &str) {}

    fn update(&mut self, _content: &str, _edit: &SyntaxEdit) {}

    fn highlights(&self, _byte_range: Range<usize>) -> Vec<Annotation> {
        Vec::new()
    }

    fn is_parsed(&self) -> bool {
        false
    }
}

/// Mock factory for testing.
struct MockFactory {
    language: &'static str,
}

impl SyntaxDriverFactory for MockFactory {
    fn create(&self, language_id: &str) -> Option<Box<dyn SyntaxDriver>> {
        if language_id == self.language {
            Some(Box::new(MockDriver))
        } else {
            None
        }
    }

    fn supported_languages(&self) -> Vec<&str> {
        vec![self.language]
    }

    fn supports(&self, language_id: &str) -> bool {
        language_id == self.language
    }
}

#[test]
fn test_store_new_empty() {
    let store = SyntaxFactoryStore::new();
    assert!(store.is_empty());
    assert_eq!(store.len(), 0);
}

#[test]
fn test_store_add_and_find() {
    let store = SyntaxFactoryStore::new();

    store.add(Arc::new(MockFactory { language: "rust" }));
    store.add(Arc::new(MockFactory { language: "python" }));

    assert_eq!(store.len(), 2);
    assert!(store.find("rust").is_some());
    assert!(store.find("python").is_some());
    assert!(store.find("javascript").is_none());
}

#[test]
fn test_store_take_drains() {
    let store = SyntaxFactoryStore::new();

    store.add(Arc::new(MockFactory { language: "rust" }));
    store.add(Arc::new(MockFactory { language: "python" }));

    let factories = store.take_factories();
    assert_eq!(factories.len(), 2);

    // Store should now be empty
    assert!(store.is_empty());
    assert!(store.take_factories().is_empty());
}

#[test]
fn test_store_debug() {
    let store = SyntaxFactoryStore::new();
    store.add(Arc::new(MockFactory { language: "rust" }));

    let debug = format!("{store:?}");
    assert!(debug.contains("SyntaxFactoryStore"));
    assert!(debug.contains("count"));
}

#[test]
fn test_store_default() {
    let store = SyntaxFactoryStore::default();
    assert!(store.is_empty());
    assert_eq!(store.len(), 0);
}

#[test]
fn test_store_find_returns_none_on_empty() {
    let store = SyntaxFactoryStore::new();
    assert!(store.find("rust").is_none());
}

#[test]
fn test_store_service_impl() {
    fn accepts_service(_: &dyn reovim_kernel::api::v1::Service) {}
    let store = SyntaxFactoryStore::new();
    accepts_service(&store);
}

#[test]
fn test_store_debug_empty() {
    let store = SyntaxFactoryStore::new();
    let debug = format!("{store:?}");
    assert!(debug.contains("SyntaxFactoryStore"));
    assert!(debug.contains('0'));
}

#[test]
fn test_store_debug_with_items() {
    let store = SyntaxFactoryStore::new();
    store.add(Arc::new(MockFactory { language: "rust" }));
    store.add(Arc::new(MockFactory { language: "python" }));
    let debug = format!("{store:?}");
    assert!(debug.contains('2'));
}

#[test]
fn test_store_take_returns_empty_after_take() {
    let store = SyntaxFactoryStore::new();
    store.add(Arc::new(MockFactory { language: "rust" }));

    let first = store.take_factories();
    assert_eq!(first.len(), 1);

    let second = store.take_factories();
    assert!(second.is_empty());
}

#[test]
fn test_store_find_correct_factory() {
    let store = SyntaxFactoryStore::new();
    store.add(Arc::new(MockFactory { language: "rust" }));
    store.add(Arc::new(MockFactory { language: "python" }));

    let factory = store.find("python").unwrap();
    assert!(factory.supports("python"));
    assert!(!factory.supports("rust"));
}

#[test]
fn test_store_not_empty_after_add() {
    let store = SyntaxFactoryStore::new();
    assert!(store.is_empty());
    store.add(Arc::new(MockFactory { language: "rust" }));
    assert!(!store.is_empty());
}

#[test]
fn test_store_find_returns_first_match() {
    let store = SyntaxFactoryStore::new();
    store.add(Arc::new(MockFactory { language: "rust" }));
    store.add(Arc::new(MockFactory { language: "rust" })); // duplicate

    // Should find the first one
    let factory = store.find("rust").unwrap();
    assert!(factory.supports("rust"));
    assert_eq!(store.len(), 2);
}

#[test]
fn test_store_take_then_add_again() {
    let store = SyntaxFactoryStore::new();
    store.add(Arc::new(MockFactory { language: "rust" }));

    let taken = store.take_factories();
    assert_eq!(taken.len(), 1);
    assert!(store.is_empty());

    // Add new factories after take
    store.add(Arc::new(MockFactory { language: "python" }));
    assert_eq!(store.len(), 1);
    assert!(store.find("python").is_some());
    assert!(store.find("rust").is_none());
}

#[test]
fn test_mock_factory_create_supported() {
    let factory = MockFactory { language: "rust" };
    let driver = factory.create("rust");
    assert!(driver.is_some());
    let driver = driver.unwrap();
    assert_eq!(driver.language(), "mock");
}

#[test]
fn test_mock_factory_create_unsupported() {
    let factory = MockFactory { language: "rust" };
    let driver = factory.create("python");
    assert!(driver.is_none());
}

#[test]
fn test_mock_factory_supported_languages() {
    let factory = MockFactory { language: "rust" };
    let langs = factory.supported_languages();
    assert_eq!(langs, vec!["rust"]);
}

#[test]
fn test_mock_driver_methods() {
    let mut driver = MockDriver;
    assert_eq!(driver.language(), "mock");
    assert!(!driver.is_parsed());
    driver.parse("content");
    assert!(!driver.is_parsed()); // MockDriver always returns false

    let edit = SyntaxEdit {
        start_byte: 0,
        old_end_byte: 1,
        new_end_byte: 2,
        start_row: 0,
        start_col: 0,
        old_end_row: 0,
        old_end_col: 1,
        new_end_row: 0,
        new_end_col: 2,
    };
    driver.update("content", &edit);
    let highlights = driver.highlights(0..10);
    assert!(highlights.is_empty());
}