diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/compiler.rs | 38 | ||||
| -rw-r--r-- | src/construct/label_end.rs | 17 | ||||
| -rw-r--r-- | src/content/flow.rs | 12 | ||||
| -rw-r--r-- | src/parser.rs | 20 | ||||
| -rw-r--r-- | src/util/edit_map.rs | 19 | 
5 files changed, 60 insertions, 46 deletions
| diff --git a/src/compiler.rs b/src/compiler.rs index bb7aedc..6f4d1a6 100644 --- a/src/compiler.rs +++ b/src/compiler.rs @@ -91,6 +91,14 @@ struct Media {      title: Option<String>,  } +/// To do. +#[derive(Debug, Clone, PartialEq)] +struct DefinitionInfo { +    id: Option<String>, +    destination: Option<String>, +    title: Option<String>, +} +  /// Configuration (optional).  #[derive(Default, Debug)]  pub struct Options { @@ -226,17 +234,39 @@ pub fn compile(events: &[Event], codes: &[Code], options: &Options) -> String {      let mut media_stack: Vec<Media> = vec![];      // let mut slurp_all_line_endings = false; +    let mut definition: Option<DefinitionInfo> = None; + +    // To do: actually do a compile pass, so that `buffer`, `resume`, etc can be used.      while index < events.len() {          let event = &events[index]; -        if event.event_type == EventType::Exit +        // Find the used line ending style. +        if line_ending_inferred.is_none() +            && event.event_type == EventType::Exit              && (event.token_type == TokenType::BlankLineEnding                  || event.token_type == TokenType::CodeTextLineEnding                  || event.token_type == TokenType::LineEnding)          {              let codes = codes_from_span(codes, &from_exit_event(events, index));              line_ending_inferred = Some(LineEnding::from_code(*codes.first().unwrap())); -            break; +        } + +        if event.event_type == EventType::Enter { +            if event.token_type == TokenType::Definition { +                definition = Some(DefinitionInfo { +                    id: None, +                    destination: None, +                    title: None, +                }); +            } +        } else if event.token_type == TokenType::Definition { +            definition = None; +        } else if event.token_type == TokenType::DefinitionLabelString +            || event.token_type == TokenType::DefinitionDestinationString +            || event.token_type == TokenType::DefinitionTitleString +        { +            let slice = serialize(codes, &from_exit_event(events, index), false); +            println!("set: {:?} {:?}", slice, definition);          }          index += 1; @@ -250,7 +280,7 @@ pub fn compile(events: &[Event], codes: &[Code], options: &Options) -> String {          LineEnding::LineFeed      }; -    index = 0; +    let mut index = 0;      while index < events.len() {          let event = &events[index]; @@ -483,7 +513,7 @@ pub fn compile(events: &[Event], codes: &[Code], options: &Options) -> String {                      let label = media.label.unwrap();                      let buf = buf_tail_mut(buffers);                      // To do: get from definition. -                    let destination = media.destination.unwrap(); +                    let destination = media.destination.unwrap_or_else(|| "".to_string());                      let title = if let Some(title) = media.title {                          format!(" title=\"{}\"", title)                      } else { diff --git a/src/construct/label_end.rs b/src/construct/label_end.rs index 581e5e9..f43a740 100644 --- a/src/construct/label_end.rs +++ b/src/construct/label_end.rs @@ -413,13 +413,8 @@ pub fn start(tokenizer: &mut Tokenizer, code: Code) -> StateFnResult {  /// [a]: z  /// ```  fn after(tokenizer: &mut Tokenizer, code: Code, info: Info) -> StateFnResult { -    // let label_start = tokenizer -    //     .label_start_stack -    //     .get_mut(info.label_start_index) -    //     .unwrap(); -    // To do: figure out if defined or not. -    let defined = false; -    println!("to do: is `{:?}` defined?", info); +    let defined = tokenizer.parse_state.definitions.contains(&info.media.id); +      match code {          // Resource (`[asd](fgh)`)?          Code::Char('(') => tokenizer.attempt(resource, move |is_ok| { @@ -487,10 +482,6 @@ fn reference_not_full(tokenizer: &mut Tokenizer, code: Code, info: Info) -> Stat  /// [a]: z  /// ```  fn ok(tokenizer: &mut Tokenizer, code: Code, mut info: Info) -> StateFnResult { -    println!( -        "ok res, ref full, ref, collapsed, or ref shortcut: {:?}", -        info.media -    );      // Remove this one and everything after it.      let mut left: Vec<LabelStart> = tokenizer          .label_start_stack @@ -725,8 +716,8 @@ fn full_reference_after(tokenizer: &mut Tokenizer, code: Code) -> StateFnResult          },          false,      )); -    println!("to do: is `{:?}` defined?", id); -    let defined = false; +    let defined = tokenizer.parse_state.definitions.contains(&id); +    // To do: set `id` on the media somehow?      if defined {          (State::Ok, Some(vec![code])) diff --git a/src/content/flow.rs b/src/content/flow.rs index 546712f..0d3ede0 100644 --- a/src/content/flow.rs +++ b/src/content/flow.rs @@ -33,12 +33,13 @@ use crate::util::{      normalize_identifier::normalize_identifier,      span::{from_exit_event, serialize},  }; +use std::collections::HashSet;  /// Turn `codes` as the flow content type into events. -pub fn flow(parse_state: &ParseState, point: Point, index: usize) -> Vec<Event> { +pub fn flow(parse_state: &mut ParseState, point: Point, index: usize) -> Vec<Event> {      let mut tokenizer = Tokenizer::new(point, index, parse_state); -      tokenizer.push(&parse_state.codes, Box::new(start), true); +    let mut next_definitions: HashSet<String> = HashSet::new();      let mut index = 0; @@ -48,15 +49,14 @@ pub fn flow(parse_state: &ParseState, point: Point, index: usize) -> Vec<Event>          if event.event_type == EventType::Exit              && event.token_type == TokenType::DefinitionLabelString          { -            let id = normalize_identifier( +            next_definitions.insert(normalize_identifier(                  serialize(                      &parse_state.codes,                      &from_exit_event(&tokenizer.events, index),                      false,                  )                  .as_str(), -            ); -            println!("to do: use definition identifier {:?}", id); +            ));          }          index += 1; @@ -64,6 +64,8 @@ pub fn flow(parse_state: &ParseState, point: Point, index: usize) -> Vec<Event>      let mut result = (tokenizer.events, false); +    parse_state.definitions = next_definitions; +      while !result.1 {          result = subtokenize(result.0, parse_state);      } diff --git a/src/parser.rs b/src/parser.rs index 32b7f36..f11f0d1 100644 --- a/src/parser.rs +++ b/src/parser.rs @@ -4,24 +4,32 @@  use crate::content::flow::flow;  use crate::tokenizer::{as_codes, Code, Event, Point}; +/// To do: could we do without `HashSet`, so we don’t need `std`? +use std::collections::HashSet; + +/// Information needed, in all content types, when parsing markdown. +/// +/// Importantly, this contains a set of known definitions. +/// It also references the input value as [`Code`][]s. +#[derive(Debug)]  pub struct ParseState { -    /// To do. +    /// List of codes.      pub codes: Vec<Code>, -    /// To do. -    pub definitions: Vec<String>, +    /// Set of defined identifiers. +    pub definitions: HashSet<String>,  }  /// Turn a string of markdown into events.  ///  /// Passes the codes back so the compiler can access the source.  pub fn parse(value: &str) -> (Vec<Event>, Vec<Code>) { -    let parse_state = ParseState { +    let mut parse_state = ParseState {          codes: as_codes(value), -        definitions: vec![], +        definitions: HashSet::new(),      };      let events = flow( -        &parse_state, +        &mut parse_state,          Point {              line: 1,              column: 1, diff --git a/src/util/edit_map.rs b/src/util/edit_map.rs index 8136306..db9c887 100644 --- a/src/util/edit_map.rs +++ b/src/util/edit_map.rs @@ -71,31 +71,14 @@ impl EditMap {          assert!(!self.consumed, "cannot consume after consuming");          self.consumed = true; -        let mut index = 0; - -        while index < events.len() { -            let event = &events[index]; -            println!( -                "ev: {:?} {:?} {:?} {:?} {:?} {:?}", -                index, -                event.event_type, -                event.token_type, -                event.content_type, -                event.previous, -                event.next -            ); -            index += 1; -        } -          indices.sort_unstable();          let mut jumps: Vec<(usize, isize)> = vec![];          let mut index_into_indices = 0; -        let mut shift: isize = 0; +        let mut shift = 0;          while index_into_indices < indices.len() {              let index = *indices[index_into_indices];              let edit = self.map.get(&index).unwrap(); -            println!("?? {:?} {:?} {:?}", shift, edit.1.len(), edit.0);              #[allow(clippy::pedantic)]              let next = shift + (edit.1.len() as isize) - (edit.0 as isize); | 
