bitsy-parser/src/room.rs

238 lines
7.6 KiB
Rust
Raw Normal View History

2020-04-18 15:58:30 +00:00
use crate::{from_base36, optional_data_line, Exit, ExitInstance, Instance, Position, ToBase36};
use crate::game::{RoomType, RoomFormat};
use crate::exit::Transition;
2020-04-12 13:28:11 +00:00
#[derive(Debug, Eq, PartialEq)]
pub struct Room {
2020-04-13 12:46:33 +00:00
pub id: u64,
2020-04-18 14:50:01 +00:00
pub palette_id: Option<u64>, // optional in very early versions
pub name: Option<String>,
pub tiles: Vec<String>, // tile ids
pub items: Vec<Instance>,
pub exits: Vec<ExitInstance>,
pub endings: Vec<Instance>,
2020-04-18 13:39:17 +00:00
pub walls: Vec<u64>, // old way of handling walls...
2020-04-12 13:28:11 +00:00
}
2020-04-13 18:14:57 +00:00
impl Room {
fn name_line(&self) -> String {
optional_data_line("NAME", self.name.as_ref())
}
2020-04-18 13:39:17 +00:00
fn wall_line(&self) -> String {
if self.walls.len() > 0 {
let ids: Vec<String> = self.walls.iter().map(|&id| id.to_base36()).collect();
optional_data_line("WAL", Some(ids.join(",")))
} else {
"".to_string()
}
}
2020-04-18 14:50:01 +00:00
fn palette_line(&self) -> String {
if self.palette_id.is_some() {
optional_data_line("PAL", Some(self.palette_id.unwrap().to_base36()))
} else {
"".to_string()
}
}
2020-04-13 18:14:57 +00:00
}
2020-04-12 13:28:11 +00:00
impl From<String> for Room {
fn from(string: String) -> Room {
2020-04-18 14:33:04 +00:00
let string = string.replace("ROOM ", "");
let string = string.replace("SET ", "");
2020-04-12 13:28:11 +00:00
let mut lines: Vec<&str> = string.lines().collect();
2020-04-18 14:33:04 +00:00
let id = from_base36(&lines[0]);
2020-04-12 13:28:11 +00:00
let mut name = None;
2020-04-18 14:50:01 +00:00
let mut palette_id = None;
2020-04-12 13:28:11 +00:00
let mut items: Vec<Instance> = Vec::new();
let mut exits: Vec<ExitInstance> = Vec::new();
let mut endings: Vec<Instance> = Vec::new();
2020-04-18 13:39:17 +00:00
let mut walls: Vec<u64> = Vec::new();
2020-04-12 13:28:11 +00:00
loop {
let last_line = lines.pop().unwrap();
2020-04-18 13:39:17 +00:00
if last_line.starts_with("WAL") {
let last_line = last_line.replace("WAL ", "");
let ids: Vec<&str> = last_line.split(",").collect();
walls = ids.iter().map(|&id| from_base36(id)).collect();
} else if last_line.starts_with("NAME") {
2020-04-12 13:28:11 +00:00
name = Some(last_line.replace("NAME ", "").to_string());
} else if last_line.starts_with("PAL") {
2020-04-18 14:50:01 +00:00
palette_id = Some(from_base36(&last_line.replace("PAL ", "")));
2020-04-12 13:28:11 +00:00
} else if last_line.starts_with("ITM") {
let last_line = last_line.replace("ITM ", "");
let item_position: Vec<&str> = last_line.split(' ').collect();
let item_id = item_position[0];
let position = item_position[1];
2020-04-18 15:12:06 +00:00
let position = Position::from(position.to_string()).unwrap();
2020-04-12 13:28:11 +00:00
2020-04-18 15:58:30 +00:00
items.push(Instance {
position,
id: item_id.to_string(),
});
2020-04-12 13:28:11 +00:00
} else if last_line.starts_with("EXT") {
let last_line = last_line.replace("EXT ", "");
let parts: Vec<&str> = last_line.split(' ').collect();
2020-04-18 15:12:06 +00:00
let position = Position::from(parts[0].to_string()).unwrap();
2020-04-12 13:28:11 +00:00
let exit = Exit::from(format!("{} {}", parts[1], parts[2]));
let mut transition = None;
let mut dialogue_id = None;
let chunks = parts[3..].chunks(2);
for chunk in chunks {
if chunk[0] == "FX" {
transition = Some(Transition::from(chunk[1]));
} else if chunk[0] == "DLG" {
dialogue_id = Some(chunk[1].to_string());
}
}
exits.push(ExitInstance { position, exit, transition, dialogue_id });
2020-04-12 13:28:11 +00:00
} else if last_line.starts_with("END") {
let last_line = last_line.replace("END ", "");
let ending_position: Vec<&str> = last_line.split(' ').collect();
let ending = ending_position[0].to_string();
let position = ending_position[1].to_string();
2020-04-18 15:12:06 +00:00
let position = Position::from(position).unwrap();
2020-04-12 13:28:11 +00:00
2020-04-18 15:58:30 +00:00
endings.push(Instance {
position,
id: ending,
});
2020-04-12 13:28:11 +00:00
} else {
lines.push(last_line);
break;
}
}
let lines = &lines[1..];
let mut tiles: Vec<String> = Vec::new();
for line in lines.into_iter() {
2020-04-24 17:07:32 +00:00
let comma_separated = line.contains(","); // old room format?
let mut line: Vec<&str> = line
.split(if comma_separated {","} else {""})
2020-04-23 07:57:46 +00:00
.collect();
2020-04-12 13:28:11 +00:00
2020-04-24 17:07:32 +00:00
if ! comma_separated { line = line[1..].to_owned(); }
let line = line[..16].to_owned();
2020-04-12 13:28:11 +00:00
for tile_id in line {
tiles.push(tile_id.to_string());
}
}
2020-04-13 18:26:11 +00:00
items.reverse();
exits.reverse();
endings.reverse();
2020-04-18 15:58:30 +00:00
Room {
id,
palette_id,
name,
tiles,
items,
exits,
endings,
walls,
}
2020-04-12 13:28:11 +00:00
}
}
impl Room {
pub fn to_string(&self, room_format: RoomFormat, room_type: RoomType) -> String {
2020-04-12 13:28:11 +00:00
let mut tiles = String::new();
let mut items = String::new();
let mut exits = String::new();
let mut endings = String::new();
for line in self.tiles.chunks(16) {
2020-04-12 13:28:11 +00:00
for tile in line {
2020-04-24 17:07:32 +00:00
tiles.push_str(tile);
if room_format == RoomFormat::CommaSeparated {
tiles.push(',');
}
}
if room_format == RoomFormat::CommaSeparated {
tiles.pop(); // remove trailing comma
2020-04-12 13:28:11 +00:00
}
2020-04-24 17:07:32 +00:00
2020-04-12 13:28:11 +00:00
tiles.push_str("\n");
}
tiles.pop(); // remove trailing newline
for instance in &self.items {
2020-04-18 15:58:30 +00:00
items.push_str(&format!(
"\nITM {} {}",
instance.id,
instance.position.to_string()
));
2020-04-12 13:28:11 +00:00
}
for instance in &self.exits {
2020-04-18 15:58:30 +00:00
exits.push_str(&format!(
"\nEXT {} {}{}{}{}",
2020-04-18 15:58:30 +00:00
instance.position.to_string(),
instance.exit.to_string(),
if instance.transition.is_some() {
instance.transition.as_ref().unwrap().to_string()
} else {"".to_string()},
if instance.dialogue_id.is_some() {" DLG "} else {""},
instance.dialogue_id.as_ref().unwrap_or(&"".to_string()),
2020-04-18 15:58:30 +00:00
));
2020-04-12 13:28:11 +00:00
}
for instance in &self.endings {
2020-04-18 15:58:30 +00:00
endings.push_str(&format!(
"\nEND {} {}",
instance.id,
instance.position.to_string()
));
2020-04-12 13:28:11 +00:00
}
format!(
"{} {}\n{}{}{}{}{}{}{}",
room_type.to_string(),
self.id.to_base36(),
2020-04-12 13:28:11 +00:00
tiles,
2020-04-13 18:14:57 +00:00
self.name_line(),
2020-04-18 13:39:17 +00:00
self.wall_line(),
2020-04-12 13:28:11 +00:00
items,
exits,
endings,
2020-04-18 14:50:01 +00:00
self.palette_line()
2020-04-12 13:28:11 +00:00
)
}
}
2020-04-19 07:13:55 +00:00
#[cfg(test)]
mod test {
use crate::room::Room;
use crate::game::{RoomType, RoomFormat};
2020-04-19 07:13:55 +00:00
#[test]
fn test_room_from_string() {
assert_eq!(
Room::from(include_str!("test-resources/room").to_string()),
crate::mock::room()
);
}
#[test]
fn test_room_to_string() {
assert_eq!(
crate::mock::room().to_string(RoomFormat::CommaSeparated, RoomType::Room),
2020-04-19 07:13:55 +00:00
include_str!("test-resources/room").to_string()
);
}
2020-04-18 13:39:17 +00:00
2020-04-19 07:13:55 +00:00
#[test]
fn test_room_walls_array() {
let output = Room::from(include_str!("test-resources/room-with-walls").to_string());
2020-04-18 13:39:17 +00:00
2020-04-19 07:13:55 +00:00
assert_eq!(output.walls, vec![10, 15]);
}
2020-04-18 13:39:17 +00:00
}