add support for multiple <p>s in one block

This commit is contained in:
リキ萌え 2023-08-18 20:44:27 +02:00
parent 84ec91042e
commit 3203989cb2
4 changed files with 57 additions and 16 deletions

View file

@ -1,3 +1,5 @@
% always_expanded = true
id = "treehouse"
- # treehouse - # treehouse
- welcome to the treehouse! - welcome to the treehouse!
@ -98,3 +100,16 @@
- expanded - expanded
<svg width="16" height="16" xmlns="http://www.w3.org/2000/svg"><line x1="4" y1="8" x2="12" y2="8" stroke="currentColor" stroke-width="2"/></svg> <svg width="16" height="16" xmlns="http://www.w3.org/2000/svg"><line x1="4" y1="8" x2="12" y2="8" stroke="currentColor" stroke-width="2"/></svg>
+ some tests for multiple ps in one block
- here's a test for multiple paragraphs in one block
this should be working fine
- and this shouldn't be breaking yeah
- nor should
this be
- breaking

View file

@ -10,6 +10,9 @@ pub enum ParseErrorKind {
#[error("root branches must not be indented")] #[error("root branches must not be indented")]
RootIndentLevel, RootIndentLevel,
#[error("at least {expected} spaces of indentation were expected, but got {got}")]
InconsistentIndentation { got: usize, expected: usize },
} }
#[derive(Debug, Clone, PartialEq, Eq, thiserror::Error)] #[derive(Debug, Clone, PartialEq, Eq, thiserror::Error)]

View file

@ -1,4 +1,4 @@
use std::ops::Range; use std::{convert::identity, ops::Range};
use crate::{ParseError, ParseErrorKind}; use crate::{ParseError, ParseErrorKind};
@ -22,10 +22,10 @@ impl BranchKind {
#[derive(Debug, Clone, PartialEq, Eq)] #[derive(Debug, Clone, PartialEq, Eq)]
pub struct BranchEvent { pub struct BranchEvent {
pub indent_level: usize, pub indent_level: usize,
pub attributes: Range<usize>,
pub kind: BranchKind, pub kind: BranchKind,
pub kind_span: Range<usize>, pub kind_span: Range<usize>,
pub content: Range<usize>, pub content: Range<usize>,
pub attributes: Range<usize>,
} }
#[derive(Debug, Clone, PartialEq, Eq)] #[derive(Debug, Clone, PartialEq, Eq)]
@ -52,8 +52,8 @@ impl<'a> Parser<'a> {
count count
} }
fn eat_until(&mut self, c: char) { fn eat_until(&mut self, cond: impl Fn(char) -> bool) {
while self.current() != Some(c) { while self.current().map(&cond).is_some_and(|x| !x) {
self.advance(); self.advance();
} }
self.advance(); self.advance();
@ -66,6 +66,30 @@ impl<'a> Parser<'a> {
indent_level indent_level
} }
fn eat_indented_lines_until(
&mut self,
indent_level: usize,
cond: impl Fn(char) -> bool,
) -> Result<(), ParseError> {
loop {
self.eat_until(|c| c == '\n');
let before_indentation = self.position;
let line_indent_level = self.eat_as_long_as(' ');
let after_indentation = self.position;
if self.current().map(&cond).is_some_and(identity) || self.current().is_none() {
self.position = before_indentation;
break;
} else if !matches!(self.current(), Some('\n')) && line_indent_level < indent_level {
return Err(ParseErrorKind::InconsistentIndentation {
got: line_indent_level,
expected: indent_level,
}
.at(before_indentation..after_indentation));
}
}
Ok(())
}
pub fn next_branch(&mut self) -> Result<Option<BranchEvent>, ParseError> { pub fn next_branch(&mut self) -> Result<Option<BranchEvent>, ParseError> {
if self.current().is_none() { if self.current().is_none() {
return Ok(None); return Ok(None);
@ -73,9 +97,15 @@ impl<'a> Parser<'a> {
let indent_level = self.eat_as_long_as(' '); let indent_level = self.eat_as_long_as(' ');
// TODO: Configs let attributes = if self.current() == Some('%') {
let config_start = self.position; let start = self.position;
let config_end = self.position; self.advance();
self.eat_indented_lines_until(indent_level, |c| c == '-' || c == '+')?;
let end = self.position;
start..end
} else {
self.position..self.position
};
let kind_start = self.position; let kind_start = self.position;
let kind = match self.current() { let kind = match self.current() {
@ -87,18 +117,12 @@ impl<'a> Parser<'a> {
let kind_end = self.position; let kind_end = self.position;
let content_start = self.position; let content_start = self.position;
loop { self.eat_indented_lines_until(indent_level, |c| c == '-' || c == '+' || c == '%')?;
self.eat_until('\n');
if let Some('\n') | None = self.current() {
self.advance();
break;
}
}
let content_end = self.position; let content_end = self.position;
Ok(Some(BranchEvent { Ok(Some(BranchEvent {
indent_level, indent_level,
attributes: config_start..config_end, attributes,
kind, kind,
kind_span: kind_start..kind_end, kind_span: kind_start..kind_end,
content: content_start..content_end, content: content_start..content_end,

View file

@ -30,7 +30,6 @@ pub fn branch_to_html(s: &mut String, branch: &Branch, source: &str) {
} }
} }
dbg!(&line[space_count..]);
unindented_block_content.push_str(&line[space_count..]); unindented_block_content.push_str(&line[space_count..]);
unindented_block_content.push('\n'); unindented_block_content.push('\n');
} }