add support for multiple <p>s in one block
This commit is contained in:
parent
84ec91042e
commit
3203989cb2
4 changed files with 57 additions and 16 deletions
|
@ -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
|
||||||
|
|
|
@ -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)]
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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');
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue