diff --git a/src/parsing/parser.rs b/src/parsing/parser.rs index a5e54960..ff5501e3 100644 --- a/src/parsing/parser.rs +++ b/src/parsing/parser.rs @@ -36,7 +36,7 @@ pub struct ParseState { #[derive(Debug, Clone, Eq, PartialEq)] struct StateLevel { context: ContextId, - prototype: Option, + prototype: Vec, captures: Option<(Region, String)>, } @@ -150,7 +150,7 @@ impl ParseState { pub fn new(syntax: &SyntaxReference) -> ParseState { let start_state = StateLevel { context: syntax.contexts["__start"].clone(), - prototype: None, + prototype: Vec::new(), captures: None, }; ParseState { @@ -309,7 +309,7 @@ impl ParseState { let context_chain = { let proto_start = self.proto_starts.last().cloned().unwrap_or(0); // Sublime applies with_prototypes from bottom to top - let with_prototypes = self.stack[proto_start..].iter().filter_map(|lvl| lvl.prototype.as_ref().map(|ctx| (true, ctx, lvl.captures.as_ref()))); + let with_prototypes = self.stack[proto_start..].iter().flat_map(|lvl| lvl.prototype.iter().map(|ctx| (true, ctx, None))); // TODO: replace None with lvl.captures.as_ref() let cur_prototype = prototype.into_iter().map(|ctx| (false, ctx, None)); let cur_context = Some((false, &cur_level.context, cur_level.captures.as_ref())).into_iter(); with_prototypes.chain(cur_prototype).chain(cur_context) @@ -604,11 +604,13 @@ impl ParseState { pat: &MatchPattern, syntax_set: &SyntaxSet ) -> bool { - let (ctx_refs, old_proto_id) = match pat.operation { + let (ctx_refs, old_proto_ids) = match pat.operation { MatchOperation::Push(ref ctx_refs) => (ctx_refs, None), MatchOperation::Set(ref ctx_refs) => { - let old_proto_id = self.stack.pop().and_then(|s| s.prototype); - (ctx_refs, old_proto_id) + // a `with_prototype` stays active when the context is `set` + // until the context layer in the stack (where the `with_prototype` + // was initially applied) is popped off. + (ctx_refs, self.stack.pop().map(|s| s.prototype)) } MatchOperation::Pop => { self.stack.pop(); @@ -617,29 +619,23 @@ impl ParseState { MatchOperation::None => return false, }; for (i, r) in ctx_refs.iter().enumerate() { - let proto_id = if i == ctx_refs.len() - 1 { - // a `with_prototype` stays active when the context is `set` - // until the context layer in the stack (where the `with_prototype` - // was initially applied) is popped off. + let mut proto_ids = old_proto_ids.clone().unwrap_or(Vec::new()); + if i == ctx_refs.len() - 1 { // if a with_prototype was specified, and multiple contexts were pushed, // then the with_prototype applies only to the last context pushed, i.e. // top most on the stack after all the contexts are pushed - this is also // referred to as the "target" of the push by sublimehq - see // https://forum.sublimetext.com/t/dev-build-3111/19240/17 for more info if let Some(ref p) = pat.with_prototype { - Some(p.id()) - } else { - old_proto_id + proto_ids.push(p.id().clone()); } - } else { - None - }; + } let context_id = r.id(); let context = syntax_set.get_context(&context_id); let captures = { let mut uses_backrefs = context.uses_backrefs; - if let Some(ref proto_id) = proto_id { - uses_backrefs = uses_backrefs || syntax_set.get_context(proto_id).uses_backrefs; + if !proto_ids.is_empty() { + uses_backrefs = uses_backrefs || proto_ids.iter().any(|id| syntax_set.get_context(id).uses_backrefs); } if uses_backrefs { Some((regions.clone(), line.to_owned())) @@ -649,7 +645,7 @@ impl ParseState { }; self.stack.push(StateLevel { context: context_id, - prototype: proto_id, + prototype: proto_ids, captures, }); } @@ -1404,6 +1400,35 @@ contexts: expect_scope_stacks_with_syntax("testfoo", &["", /*"",*/ "", ""], syntax); } + #[test] + fn can_parse_two_with_prototypes_at_same_stack_level() { + let syntax = r#" +%YAML 1.2 +--- +# See http://www.sublimetext.com/docs/3/syntax.html +scope: source.example-wp +contexts: + main: + - match: a + scope: a + push: + - match: b + scope: b + set: + - match: c + scope: c + with_prototype: + - match: '2' + scope: '2' + with_prototype: + - match: '1' + scope: '1' +"#; + + let syntax = SyntaxDefinition::load_from_str(&syntax, true, None).unwrap(); + expect_scope_stacks_with_syntax("abc12", &["<1>", "<2>"], syntax); + } + fn expect_scope_stacks(line_without_newline: &str, expect: &[&str], syntax: &str) { println!("Parsing with newlines"); let line_with_newline = format!("{}\n", line_without_newline);