1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
use std::fmt;

use crate::compare::MatchGraphBuilder;

use super::{GrammarItemName, GrammarProc, Grammar};

pub enum GrammarProcTemplate<PN, TN> {
	Token(TN),
	Proc(PN),
	Sequence(Vec<GrammarProcTemplate<PN, TN>>),
	Choice(Vec<GrammarProcTemplate<PN, TN>>),
	Repeat(u32, Box<GrammarProcTemplate<PN, TN>>),
	Option(Box<GrammarProcTemplate<PN, TN>>)
}

impl<PN: PartialEq + Copy + fmt::Debug + fmt::Display, TN: PartialEq + Copy + fmt::Display> GrammarProcTemplate<PN, TN> {
	pub fn resolve(&self, name: PN) -> GrammarProc<PN, TN> {
		let mut graph_builder = MatchGraphBuilder::new();
		self.resolve_part(&mut graph_builder);
		GrammarProc::new(name, graph_builder.complete())
	}

	fn resolve_part(&self, graph_builder: &mut MatchGraphBuilder<GrammarItemName<PN, TN>>){
		match self {
			Self::Token(token_name) => graph_builder.append(GrammarItemName::Terminal(*token_name), None),
			Self::Proc(proc_name) => graph_builder.append(GrammarItemName::NonTerminal(*proc_name), None),
			Self::Sequence(items) => {
				for item in items { item.resolve_part(graph_builder); }
			},
			Self::Choice(options) => {
				graph_builder.push_return();
				for option in options {
					option.resolve_part(graph_builder);
					graph_builder.end_choice_path();
				}
				graph_builder.pop_choice();
			},
			Self::Repeat(min, item) => {
				graph_builder.push_return();
				item.resolve_part(graph_builder);
				graph_builder.duplicate(*min);
				graph_builder.pop_repeat();
			},
			Self::Option(item) => {
				graph_builder.push_return();
				item.resolve_part(graph_builder);
				graph_builder.pop_optional();
			}
		}
	}
}

pub struct GrammarTemplate<PN, TN> {
	proc_templates: Vec<(PN, GrammarProcTemplate<PN, TN>)>
}

impl<PN: PartialEq + Copy + fmt::Debug + fmt::Display, TN: PartialEq + Copy + fmt::Display> GrammarTemplate<PN, TN> {
	pub fn new(proc_templates: Vec<(PN, GrammarProcTemplate<PN, TN>)>) -> Self {
		GrammarTemplate { proc_templates }
	}

	pub fn resolve(&self) -> Grammar<PN, TN> {
		Grammar::new(self.proc_templates.iter().map(|(name, template)| template.resolve(*name)).collect())
	}
}