use super::*;
use crate::cells::cell_len;
fn make_console(width: usize) -> Console {
Console::builder()
.width(width)
.force_terminal(true)
.no_color(true)
.markup(false)
.build()
}
fn render_markdown(console: &Console, md: &Markdown) -> String {
let opts = console.options();
let segments = md.gilt_console(console, &opts);
segments.iter().map(|s| s.text.as_str()).collect()
}
fn render_segments(console: &Console, md: &Markdown) -> Vec<Segment> {
let opts = console.options();
md.gilt_console(console, &opts)
}
#[test]
fn test_simple_paragraph() {
let console = make_console(80);
let md = Markdown::new("Hello, world!");
let output = render_markdown(&console, &md);
assert!(output.contains("Hello, world!"));
}
#[test]
fn test_two_paragraphs() {
let console = make_console(80);
let md = Markdown::new("First paragraph.\n\nSecond paragraph.");
let output = render_markdown(&console, &md);
assert!(output.contains("First paragraph."));
assert!(output.contains("Second paragraph."));
}
#[test]
fn test_heading_h1() {
let console = make_console(80);
let md = Markdown::new("# Heading 1");
let output = render_markdown(&console, &md);
assert!(output.contains("Heading 1"));
}
#[test]
fn test_heading_h2() {
let console = make_console(80);
let md = Markdown::new("## Heading 2");
let output = render_markdown(&console, &md);
assert!(output.contains("Heading 2"));
}
#[test]
fn test_heading_h3() {
let console = make_console(80);
let md = Markdown::new("### Heading 3");
let output = render_markdown(&console, &md);
assert!(output.contains("Heading 3"));
}
#[test]
fn test_heading_h4() {
let console = make_console(80);
let md = Markdown::new("#### Heading 4");
let output = render_markdown(&console, &md);
assert!(output.contains("Heading 4"));
}
#[test]
fn test_heading_h5() {
let console = make_console(80);
let md = Markdown::new("##### Heading 5");
let output = render_markdown(&console, &md);
assert!(output.contains("Heading 5"));
}
#[test]
fn test_heading_h6() {
let console = make_console(80);
let md = Markdown::new("###### Heading 6");
let output = render_markdown(&console, &md);
assert!(output.contains("Heading 6"));
}
#[test]
fn test_headings_have_appropriate_styles() {
let console = make_console(80);
let md = Markdown::new("# Title");
let output = render_markdown(&console, &md);
assert!(output.contains("Title"));
assert!(output.contains('\u{2501}') || output.contains('-'));
}
#[test]
fn test_h1_has_rule_underline() {
let console = make_console(40);
let md = Markdown::new("# Big Title");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("Big Title"));
assert!(text.contains('\u{2501}'));
}
#[test]
fn test_h2_has_rule_underline() {
let console = make_console(40);
let md = Markdown::new("## Sub Title");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("Sub Title"));
assert!(text.contains('\u{2501}'));
}
#[test]
fn test_bold_text() {
let console = make_console(80);
let md = Markdown::new("This is **bold** text.");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("bold"));
assert!(text.contains("This is"));
assert!(text.contains("text."));
let bold_seg = segments.iter().find(|s| s.text == "bold");
assert!(bold_seg.is_some(), "Should have a segment with text 'bold'");
if let Some(seg) = bold_seg {
assert!(seg.style().is_some(), "Bold segment should have a style");
}
}
#[test]
fn test_italic_text() {
let console = make_console(80);
let md = Markdown::new("This is *italic* text.");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("italic"));
let italic_seg = segments.iter().find(|s| s.text == "italic");
assert!(
italic_seg.is_some(),
"Should have a segment with text 'italic'"
);
if let Some(seg) = italic_seg {
assert!(seg.style().is_some(), "Italic segment should have a style");
}
}
#[test]
fn test_bold_italic_combined() {
let console = make_console(80);
let md = Markdown::new("This is ***bold and italic*** text.");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("bold and italic"));
let combined_seg = segments.iter().find(|s| s.text.contains("bold and italic"));
assert!(combined_seg.is_some());
if let Some(seg) = combined_seg {
assert!(
seg.style().is_some(),
"Bold+italic segment should have a style"
);
}
}
#[test]
fn test_inline_code() {
let console = make_console(80);
let md = Markdown::new("Use `println!` to print.");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("println!"));
let code_seg = segments.iter().find(|s| s.text == "println!");
assert!(
code_seg.is_some(),
"Should have a segment with inline code text"
);
if let Some(seg) = code_seg {
assert!(seg.style().is_some(), "Inline code should have a style");
}
}
#[test]
fn test_code_block() {
let console = make_console(80);
let md = Markdown::new("```\nfn main() {\n println!(\"hello\");\n}\n```");
let output = render_markdown(&console, &md);
assert!(output.contains("fn main()"));
assert!(output.contains("println!"));
}
#[test]
fn test_code_block_with_language() {
let console = make_console(80);
let md = Markdown::new("```rust\nlet x = 42;\n```");
let output = render_markdown(&console, &md);
assert!(output.contains("let x = 42;"));
}
#[test]
fn test_link_with_url() {
let console = make_console(80);
let md = Markdown::new("[Rust](https://www.rust-lang.org)");
let output = render_markdown(&console, &md);
assert!(output.contains("Rust"));
assert!(output.contains("https://www.rust-lang.org"));
}
#[test]
fn test_link_without_url_display() {
let console = make_console(80);
let md = Markdown::new("[Rust](https://www.rust-lang.org)").with_hyperlinks(false);
let output = render_markdown(&console, &md);
assert!(output.contains("Rust"));
assert!(!output.contains("https://www.rust-lang.org"));
}
#[test]
fn test_unordered_list() {
let console = make_console(80);
let md = Markdown::new("- Item 1\n- Item 2\n- Item 3");
let output = render_markdown(&console, &md);
assert!(output.contains("Item 1"));
assert!(output.contains("Item 2"));
assert!(output.contains("Item 3"));
assert!(output.contains('\u{2022}'));
}
#[test]
fn test_ordered_list() {
let console = make_console(80);
let md = Markdown::new("1. First\n2. Second\n3. Third");
let output = render_markdown(&console, &md);
assert!(output.contains("First"));
assert!(output.contains("Second"));
assert!(output.contains("Third"));
assert!(output.contains("1."));
assert!(output.contains("2."));
assert!(output.contains("3."));
}
#[test]
fn test_nested_list() {
let console = make_console(80);
let md = Markdown::new("- Outer\n - Inner 1\n - Inner 2\n- Outer 2");
let output = render_markdown(&console, &md);
assert!(output.contains("Outer"));
assert!(output.contains("Inner 1"));
assert!(output.contains("Inner 2"));
assert!(output.contains("Outer 2"));
}
#[test]
fn test_block_quote() {
let console = make_console(80);
let md = Markdown::new("> This is a quote.");
let output = render_markdown(&console, &md);
assert!(output.contains("This is a quote."));
assert!(output.contains('\u{2502}'));
}
#[test]
fn test_horizontal_rule() {
let console = make_console(40);
let md = Markdown::new("Above\n\n---\n\nBelow");
let output = render_markdown(&console, &md);
assert!(output.contains("Above"));
assert!(output.contains("Below"));
assert!(output.contains('\u{2501}'));
}
#[test]
fn test_mixed_content() {
let console = make_console(80);
let md = Markdown::new("# Title\n\nA paragraph.\n\n- Item 1\n- Item 2\n\n```\ncode\n```");
let output = render_markdown(&console, &md);
assert!(output.contains("Title"));
assert!(output.contains("A paragraph."));
assert!(output.contains("Item 1"));
assert!(output.contains("Item 2"));
assert!(output.contains("code"));
}
#[test]
fn test_empty_markdown() {
let console = make_console(80);
let md = Markdown::new("");
let output = render_markdown(&console, &md);
assert!(output.is_empty() || output.trim().is_empty());
}
#[test]
fn test_table() {
let console = make_console(80);
let md = Markdown::new("| Name | Age |\n|------|-----|\n| Alice | 30 |\n| Bob | 25 |");
let output = render_markdown(&console, &md);
assert!(output.contains("Name"));
assert!(output.contains("Age"));
assert!(output.contains("Alice"));
assert!(output.contains("30"));
assert!(output.contains("Bob"));
assert!(output.contains("25"));
}
#[test]
fn test_table_with_alignment() {
let console = make_console(80);
let md = Markdown::new("| Left | Center | Right |\n|:-----|:------:|------:|\n| L | C | R |");
let output = render_markdown(&console, &md);
assert!(output.contains("Left"));
assert!(output.contains("Center"));
assert!(output.contains("Right"));
}
#[test]
fn test_renderable_integration() {
let console = Console::builder()
.width(60)
.force_terminal(true)
.no_color(true)
.markup(false)
.build();
let md = Markdown::new("Hello, **world**!");
let opts = console.options();
let segments = md.gilt_console(&console, &opts);
assert!(!segments.is_empty());
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("Hello,"));
assert!(text.contains("world"));
}
#[test]
fn test_renderable_through_console_render() {
let console = Console::builder()
.width(60)
.force_terminal(true)
.no_color(true)
.markup(false)
.build();
let md = Markdown::new("# Title\n\nParagraph text.");
let segments = console.render(&md, None);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("Title"));
assert!(text.contains("Paragraph text."));
}
#[test]
fn test_constructor_defaults() {
let md = Markdown::new("test");
assert_eq!(md.markup, "test");
assert_eq!(md.code_theme, "monokai");
assert!(md.inline_code_lexer.is_none());
assert!(md.inline_code_theme.is_none());
assert!(md.hyperlinks);
assert!(md.justify.is_none());
}
#[test]
fn test_builder_code_theme() {
let md = Markdown::new("test").with_code_theme("dracula");
assert_eq!(md.code_theme, "dracula");
}
#[test]
fn test_builder_hyperlinks() {
let md = Markdown::new("test").with_hyperlinks(false);
assert!(!md.hyperlinks);
}
#[test]
fn test_builder_justify() {
let md = Markdown::new("test").with_justify(JustifyMethod::Center);
assert_eq!(md.justify, Some(JustifyMethod::Center));
}
#[test]
fn test_strikethrough() {
let console = make_console(80);
let md = Markdown::new("This is ~~deleted~~ text.");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("deleted"));
}
#[test]
fn test_soft_break() {
let console = make_console(80);
let md = Markdown::new("Line one\nLine two");
let output = render_markdown(&console, &md);
assert!(output.contains("Line one"));
assert!(output.contains("Line two"));
}
#[test]
fn test_hard_break() {
let console = make_console(80);
let md = Markdown::new("Line one \nLine two");
let output = render_markdown(&console, &md);
assert!(output.contains("Line one"));
assert!(output.contains("Line two"));
}
#[test]
fn test_all_heading_levels() {
let console = make_console(80);
let md = Markdown::new("# H1\n\n## H2\n\n### H3\n\n#### H4\n\n##### H5\n\n###### H6");
let output = render_markdown(&console, &md);
assert!(output.contains("H1"));
assert!(output.contains("H2"));
assert!(output.contains("H3"));
assert!(output.contains("H4"));
assert!(output.contains("H5"));
assert!(output.contains("H6"));
}
#[test]
fn test_narrow_width() {
let console = make_console(20);
let md = Markdown::new("This is a paragraph with enough text to wrap.");
let output = render_markdown(&console, &md);
assert!(output.contains("This"));
for line in output.split('\n') {
if !line.is_empty() {
assert!(
cell_len(line) <= 20,
"Line exceeds width: '{}' ({} cells)",
line,
cell_len(line)
);
}
}
}
#[test]
fn test_code_block_has_panel_border() {
let console = make_console(40);
let md = Markdown::new("```\nhello\n```");
let output = render_markdown(&console, &md);
assert!(
output.contains('\u{2501}') || output.contains('\u{2503}') || output.contains('\u{250F}'),
"Code block should be wrapped in a panel border"
);
}
#[test]
fn test_list_with_inline_formatting() {
let console = make_console(80);
let md = Markdown::new("- **Bold item**\n- *Italic item*\n- `Code item`");
let output = render_markdown(&console, &md);
assert!(output.contains("Bold item"));
assert!(output.contains("Italic item"));
assert!(output.contains("Code item"));
}
#[test]
fn test_whitespace_only() {
let console = make_console(80);
let md = Markdown::new(" \n\n ");
let output = render_markdown(&console, &md);
assert!(output.trim().is_empty());
}
#[test]
fn test_blockquote_multiple_paragraphs() {
let console = make_console(80);
let md = Markdown::new("> First quote.\n>\n> Second quote.");
let output = render_markdown(&console, &md);
assert!(output.contains("First quote."));
assert!(output.contains("Second quote."));
}
#[test]
fn test_image() {
let console = make_console(80);
let md = Markdown::new("");
let output = render_markdown(&console, &md);
assert!(output.contains("Alt text"));
assert!(output.contains("https://example.com/image.png"));
}
#[test]
fn test_output_has_trailing_content() {
let console = make_console(80);
let md = Markdown::new("Hello");
let segments = render_segments(&console, &md);
assert!(!segments.is_empty());
}
#[test]
fn test_display_trait() {
let md = Markdown::new("# Hello\n\nWorld");
let s = format!("{}", md);
assert!(!s.is_empty());
}
#[test]
fn inline_code_in_table_cell_is_rendered() {
let console = make_console(80);
let md = Markdown::new("| Col | Code |\n|---|---|\n| a | `foo` |");
let output = render_markdown(&console, &md);
assert!(
output.contains("foo"),
"Expected 'foo' to appear in table output, got: {:?}",
output
);
}
#[test]
fn inline_code_in_table_cell_keeps_style() {
let console = make_console(80);
let md = Markdown::new("| Col | Code |\n|---|---|\n| a | `foo` |");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("foo"), "Expected 'foo' in output");
let code_seg = segments.iter().find(|s| s.text.contains("foo"));
assert!(code_seg.is_some(), "Expected a segment containing 'foo'");
assert!(
code_seg.unwrap().style.is_some(),
"Inline code in table cell must carry a style"
);
}
#[test]
fn inline_code_outside_table_still_works() {
let console = make_console(80);
let md = Markdown::new("Use `println!` to print.");
let segments = render_segments(&console, &md);
let text: String = segments.iter().map(|s| s.text.as_str()).collect();
assert!(text.contains("println!"));
let code_seg = segments.iter().find(|s| s.text == "println!");
assert!(
code_seg.is_some(),
"Should have a segment with text 'println!'"
);
assert!(
code_seg.unwrap().style.is_some(),
"Inline code outside table should still have a style"
);
}
#[test]
fn inline_code_between_text_in_table_cell_preserves_order() {
let console = make_console(80);
let md = Markdown::new("| x |\n|---|\n| pre `mid` post |");
let output = render_markdown(&console, &md);
let pre_pos = output.find("pre");
let mid_pos = output.find("mid");
let post_pos = output.find("post");
assert!(
pre_pos.is_some() && mid_pos.is_some() && post_pos.is_some(),
"Expected 'pre', 'mid', and 'post' in output, got: {:?}",
output
);
assert!(
pre_pos.unwrap() < mid_pos.unwrap() && mid_pos.unwrap() < post_pos.unwrap(),
"Expected order pre < mid < post, got positions: pre={:?} mid={:?} post={:?}",
pre_pos,
mid_pos,
post_pos
);
}