Compare commits

..

3 commits

3 changed files with 49 additions and 7 deletions

View file

@ -3,7 +3,7 @@ use std::thread;
use std::time::Duration; use std::time::Duration;
use eframe::egui; use eframe::egui;
use renrs::{State, Command}; use renrs::State;
use sfml::{ use sfml::{
graphics::{Color, RenderTarget, RenderWindow, RectangleShape, Transformable}, graphics::{Color, RenderTarget, RenderWindow, RectangleShape, Transformable},
@ -41,7 +41,7 @@ impl App {
} }
fn next(&mut self) { fn next(&mut self) {
if let Some(Command::Say { name, text }) = self.state.next_command() { if let Some(renrs::Event::Say { name, text }) = self.state.next() {
self.text = match name { self.text = match name {
Some(name) => format!("{name}: {text}"), Some(name) => format!("{name}: {text}"),
None => text, None => text,

View file

@ -14,6 +14,7 @@ pub enum Token {
Str(String), Str(String),
Array(Vec<Token>), Array(Vec<Token>),
Boolean(bool), Boolean(bool),
Number(f64),
} }
impl Token { impl Token {
@ -23,6 +24,7 @@ impl Token {
Str(_) => "String", Str(_) => "String",
Array(_) => "Array", Array(_) => "Array",
Boolean(_) => "Boolean", Boolean(_) => "Boolean",
Number(_) => "Number",
} }
} }
} }
@ -31,13 +33,28 @@ use Token::*;
// Parsed script commands // Parsed script commands
#[derive(Debug)] #[derive(Debug)]
pub enum Command { #[allow(dead_code)]
enum Command {
Say { name: Option<String>, text: String }, Say { name: Option<String>, text: String },
Eat { food: String, politely: bool }, Eat { food: String, politely: bool },
} }
use Command::*; use Command::*;
impl Command {
fn is_blocking(&self) -> bool {
match self {
Say { .. } => true,
_ => false,
}
}
}
#[derive(Debug)]
pub enum Event {
Say { name: Option<String>, text: String },
}
// Tokenize raw script string // Tokenize raw script string
fn tokenize(script: &str) -> Vec<Vec<Token>> { fn tokenize(script: &str) -> Vec<Vec<Token>> {
let file = RpyParser::parse(Rule::file, script) let file = RpyParser::parse(Rule::file, script)
@ -95,6 +112,7 @@ fn parse_pair(pair: pest::iterators::Pair<Rule>) -> Token {
"False" => false, "False" => false,
_ => unreachable!(), _ => unreachable!(),
}), }),
Rule::number => Token::Number(contents.as_str().parse().unwrap()),
Rule::keyword => Token::Keyword(contents.as_str().to_owned()), Rule::keyword => Token::Keyword(contents.as_str().to_owned()),
__ => unreachable!(), __ => unreachable!(),
} }
@ -122,6 +140,11 @@ fn parse_file(file_path: PathBuf) -> Vec<Command> {
let token_lines = tokenize_file(file_path); let token_lines = tokenize_file(file_path);
let mut commands = Vec::new(); let mut commands = Vec::new();
for line in token_lines { for line in token_lines {
macro_rules! unknown {
() => {
panic!("Unknown command {}", describe_line(&line))
};
}
commands.push(match line.as_slice() { commands.push(match line.as_slice() {
[Str(text)] => Say { [Str(text)] => Say {
name: None, name: None,
@ -135,10 +158,10 @@ fn parse_file(file_path: PathBuf) -> Vec<Command> {
food: food.to_owned(), food: food.to_owned(),
politely: match tail { politely: match tail {
[Boolean(politely)] => *politely, [Boolean(politely)] => *politely,
_ => false, _ => unknown!(),
}, },
}, },
_ => panic!("Unknown command {}", describe_line(&line)), _ => unknown!(),
}); });
} }
commands commands
@ -159,7 +182,19 @@ impl State {
} }
} }
pub fn next_command(&mut self) -> Option<Command> { pub fn next(&mut self) -> Option<Event> {
while let Some(command) = self.next_command() {
if command.is_blocking() {
return Some(match command {
Say { name, text } => Event::Say { name, text },
_ => unimplemented!(),
})
}
}
None
}
fn next_command(&mut self) -> Option<Command> {
if self.command_queue.len() == 0 { if self.command_queue.len() == 0 {
None None
} else { } else {

View file

@ -8,7 +8,7 @@ char = { !NEWLINE ~ ANY }
// http://pest.rs/book/grammars/syntax.html#atomic // http://pest.rs/book/grammars/syntax.html#atomic
inner = @{ char* } inner = @{ char* }
token = { string | array | boolean | keyword } token = { string | array | boolean | number | keyword }
// KEYWORDS // KEYWORDS
// has to be atomic for no implicit separate (spaces) // has to be atomic for no implicit separate (spaces)
@ -37,6 +37,13 @@ array = {
// BOOLEAN // BOOLEAN
boolean = { "True" | "False" } boolean = { "True" | "False" }
// NUMBER
number = @{
"-"?
~ ("0" | ASCII_NONZERO_DIGIT ~ ASCII_DIGIT*)
~ ("." ~ ASCII_DIGIT*)?
}
// comments are a # followed by // comments are a # followed by
// any number of non-newline characters // any number of non-newline characters
COMMENT = _{ "#" ~ char* } COMMENT = _{ "#" ~ char* }