marble-machinations/src/editor.rs

1040 lines
30 KiB
Rust
Raw Normal View History

use std::{
fs::{read_dir, read_to_string},
mem::transmute,
ops::Rem,
};
2024-10-06 00:24:11 +02:00
use raylib::prelude::*;
use crate::{
blueprint::Blueprint,
2024-10-13 01:23:56 +02:00
draw_scaled_texture, draw_usize, get_free_id, get_scroll,
2024-10-06 12:39:36 +02:00
level::Level,
2024-10-06 00:24:11 +02:00
marble_engine::{
2024-10-10 17:09:41 +02:00
board::Board,
pos::Pos,
2024-10-06 00:24:11 +02:00
tile::{Direction, GateType, MathOp, MirrorType, PTile, Tile, WireType},
Machine,
},
simple_button, simple_option_button, slider,
solution::{Score, Solution},
2024-10-13 12:45:58 +02:00
text_input, texture_option_button,
theme::*,
userdata_dir, Scroll, Textures,
2024-10-06 00:24:11 +02:00
};
2024-10-06 16:00:12 +02:00
const HEADER_HEIGHT: i32 = 40;
const FOOTER_HEIGHT: i32 = 95;
const SIDEBAR_WIDTH: i32 = 200 + 32 * 2 + 5 * 4;
const MAX_ZOOM_IN: i32 = 3;
const BOARD_MARGIN: isize = 3;
2024-10-06 16:00:12 +02:00
2024-10-06 00:24:11 +02:00
#[derive(Debug)]
2024-10-06 00:57:24 +02:00
pub struct Editor {
2024-10-06 00:24:11 +02:00
source_board: Board,
2024-10-06 12:39:36 +02:00
level: Level,
2024-10-06 00:24:11 +02:00
machine: Machine,
sim_state: SimState,
view_offset: Vector2,
zoom: i32,
output_as_text: bool,
input_as_text: bool,
active_tool: Tool,
tool_math: MathOp,
tool_gate: GateType,
tool_arrow: Direction,
tool_mirror: MirrorType,
tool_wire: WireType,
2024-10-06 00:24:11 +02:00
input_text_selected: bool,
new_blueprint_name: String,
blueprint_name_selected: bool,
2024-10-07 14:16:50 +02:00
sim_speed: u8,
2024-10-06 00:24:11 +02:00
time_since_step: f32,
2024-10-06 16:00:12 +02:00
exit_state: ExitState,
exit_menu: bool,
complete_popup: Popup,
2024-10-07 18:56:09 +02:00
// fail_popup: Popup,
score: Option<Score>,
blueprints: Vec<Blueprint>,
selected_blueprint: usize,
2024-10-10 21:56:36 +02:00
blueprint_scroll: usize,
}
#[derive(Debug, PartialEq)]
enum Popup {
Start,
Visible,
Dismissed,
2024-10-06 00:24:11 +02:00
}
#[derive(Debug, Clone)]
enum Tool {
None,
2024-10-07 16:51:56 +02:00
Erase,
2024-10-06 00:24:11 +02:00
SetTile(Tile),
Digits(Option<Pos>),
Math,
Gate,
Wire,
Arrow,
Mirror,
SelectArea(Selection),
Blueprint,
2024-10-06 00:24:11 +02:00
}
#[derive(Debug, Clone, Default)]
struct Selection {
area: Option<(Pos, Pos)>,
is_selecting: bool,
}
2024-10-06 00:24:11 +02:00
#[derive(Debug, Clone, PartialEq)]
enum SimState {
Editing,
Running,
Stepping,
}
2024-10-06 16:00:12 +02:00
#[derive(Debug, Clone, Copy)]
pub enum ExitState {
Dont,
ExitAndSave,
2024-10-06 16:29:45 +02:00
Save,
2024-10-06 16:00:12 +02:00
ExitNoSave,
}
2024-10-06 00:57:24 +02:00
impl Editor {
2024-10-06 16:00:12 +02:00
pub fn new(solution: Solution, level: Level) -> Self {
2024-10-06 00:24:11 +02:00
Self {
2024-10-06 16:00:12 +02:00
source_board: Board::parse(&solution.board),
machine: Machine::new_empty(level.inputs().to_owned(), 1),
2024-10-06 00:24:11 +02:00
sim_state: SimState::Editing,
view_offset: Vector2::zero(),
zoom: 1,
active_tool: Tool::None,
output_as_text: level.output_is_text(),
input_as_text: level.input_is_text(),
2024-10-06 00:24:11 +02:00
input_text_selected: false,
new_blueprint_name: String::new(),
blueprint_name_selected: false,
sim_speed: 3,
2024-10-06 00:24:11 +02:00
time_since_step: 0.,
tool_math: MathOp::Add,
tool_gate: GateType::Equal,
tool_arrow: Direction::Right,
tool_mirror: MirrorType::Forward,
tool_wire: WireType::Vertical,
2024-10-06 16:00:12 +02:00
level,
exit_state: ExitState::Dont,
exit_menu: false,
complete_popup: Popup::Start,
2024-10-07 18:56:09 +02:00
// fail_popup: Popup::Start,
score: solution.score,
blueprints: get_blueprints(),
selected_blueprint: usize::MAX,
2024-10-10 21:56:36 +02:00
blueprint_scroll: 0,
2024-10-06 00:24:11 +02:00
}
}
2024-10-06 16:00:12 +02:00
pub fn get_exit_state(&self) -> ExitState {
self.exit_state
}
pub fn level_id(&self) -> &str {
2024-10-06 23:37:21 +02:00
self.level.id()
2024-10-06 16:00:12 +02:00
}
pub fn source_board(&self) -> &Board {
&self.source_board
}
pub fn score(&self) -> Option<Score> {
self.score.clone()
}
fn pos_to_screen(&self, pos: Vector2) -> Vector2 {
pos * (16 << self.zoom) as f32 + self.view_offset
}
2024-10-06 00:24:11 +02:00
fn start_sim(&mut self) {
self.machine.reset();
self.machine.set_board(self.source_board.clone());
}
2024-10-07 13:16:26 +02:00
fn step_pressed(&mut self) {
2024-10-07 12:57:32 +02:00
match self.sim_state {
SimState::Editing => {
self.start_sim();
self.step();
}
SimState::Running => (),
SimState::Stepping => self.step(),
}
self.sim_state = SimState::Stepping;
}
fn step(&mut self) {
self.machine.step();
if self.complete_popup == Popup::Visible {
self.complete_popup = Popup::Dismissed;
}
2024-10-07 12:44:15 +02:00
if !self.level.outputs().is_empty()
&& self.level.outputs() == self.machine.output()
&& self.complete_popup == Popup::Start
{
self.complete_popup = Popup::Visible;
self.exit_state = ExitState::Save;
self.sim_state = SimState::Stepping;
self.score = Some(Score {
cycles: self.machine.step_count(),
tiles: self.source_board.count_tiles(),
area: 0,
});
}
}
2024-10-06 00:24:11 +02:00
fn rotate_tool(&mut self, shift: bool) {
if shift {
match &self.active_tool {
Tool::Math => self.tool_math.prev(),
Tool::Gate => self.tool_gate.prev(),
Tool::Arrow => self.tool_arrow = self.tool_arrow.left(),
Tool::Mirror => self.tool_mirror.flip(),
Tool::Wire => self.tool_wire.prev(),
_ => (),
2024-10-06 00:24:11 +02:00
}
} else {
match &self.active_tool {
Tool::Math => self.tool_math.next(),
Tool::Gate => self.tool_gate.next(),
Tool::Arrow => self.tool_arrow = self.tool_arrow.right(),
Tool::Mirror => self.tool_mirror.flip(),
Tool::Wire => self.tool_wire.next(),
_ => (),
2024-10-06 00:24:11 +02:00
}
}
}
pub fn center_view(&mut self, d: &RaylibHandle) {
let tile_size = (16 << self.zoom) as f32;
let tile_x = self.source_board.width() as f32 / 2. * tile_size;
let tile_y = self.source_board.height() as f32 / 2. * tile_size;
let screen_x = d.get_screen_width() as f32 / 2.;
let screen_y = d.get_screen_height() as f32 / 2.;
self.view_offset.x = (screen_x - tile_x).floor();
self.view_offset.y = (screen_y - tile_y).floor();
}
fn change_zoom_level(&mut self, d: &RaylibHandle, delta: i32) {
let tile_size = (16 << self.zoom) as f32;
let mouse_pos = d.get_mouse_position();
let tile_pos_of_mouse = (mouse_pos - self.view_offset) / tile_size;
self.zoom += delta;
let tile_size = (16 << self.zoom) as f32;
self.view_offset = mouse_pos - tile_pos_of_mouse * tile_size;
self.view_offset.x = self.view_offset.x.floor();
self.view_offset.y = self.view_offset.y.floor();
}
fn zoom_in(&mut self, d: &RaylibHandle) {
if self.zoom < MAX_ZOOM_IN {
self.change_zoom_level(d, 1);
}
}
fn zoom_out(&mut self, d: &RaylibHandle) {
if self.zoom > 0 {
self.change_zoom_level(d, -1);
}
}
fn save_blueprint(&mut self, selection: (Pos, Pos)) {
let min = selection.0.min(selection.1);
let max = selection.0.max(selection.1) + (1, 1).into();
let width = (max.x - min.x) as usize;
let height = (max.y - min.y) as usize;
let mut board = Board::new_empty(width, height);
for (target_x, x) in (min.x..=max.x).enumerate() {
for (target_y, y) in (min.y..=max.y).enumerate() {
if let Some(tile) = self.source_board.get(Pos { x, y }) {
board.set((target_x, target_y).into(), tile);
}
}
}
2024-10-10 17:21:18 +02:00
board.trim_size(0);
2024-10-13 01:23:56 +02:00
let id = get_free_id(&self.blueprints, Blueprint::id);
let mut blueprint = Blueprint::new(&board, id);
if !self.new_blueprint_name.is_empty() {
blueprint.name = self.new_blueprint_name.clone();
}
blueprint.save();
self.blueprints.push(blueprint);
self.active_tool = Tool::Blueprint;
}
fn grow_board_and_update_view(&mut self, pos: &mut Pos) {
let tile_size = (16 << self.zoom) as f32;
let (x, y) = self.source_board.grow_to_include(*pos);
if x != 0 || y != 0 {
self.view_offset.x -= x as f32 * tile_size;
self.view_offset.y -= y as f32 * tile_size;
pos.x += x;
pos.y += y;
match &mut self.active_tool {
Tool::Digits(Some(pos)) => {
pos.x += x;
pos.y += y;
}
Tool::SelectArea(Selection {
area: Some((start, end)),
is_selecting: _,
}) => {
start.x += x;
start.y += y;
end.x += x;
end.y += y;
}
_ => (),
}
}
}
fn set_tile(&mut self, mut pos: Pos, tile: Tile) {
let tile_size = (16 << self.zoom) as f32;
pos.x -= BOARD_MARGIN;
pos.y -= BOARD_MARGIN;
self.grow_board_and_update_view(&mut pos);
pos.x += BOARD_MARGIN * 2;
pos.y += BOARD_MARGIN * 2;
self.grow_board_and_update_view(&mut pos);
pos.x -= BOARD_MARGIN;
pos.y -= BOARD_MARGIN;
self.source_board.set(pos, tile);
2024-10-07 12:44:15 +02:00
if tile.is_blank() {
let (x, y) = self.source_board.trim_size(BOARD_MARGIN as usize);
self.view_offset.x += x as f32 * tile_size;
self.view_offset.y += y as f32 * tile_size;
2024-10-07 12:44:15 +02:00
}
}
pub fn update(&mut self, rl: &RaylibHandle) {
2024-10-06 16:00:12 +02:00
if rl.is_key_pressed(KeyboardKey::KEY_ESCAPE) {
self.sim_state = SimState::Editing;
self.complete_popup = Popup::Start;
2024-10-06 16:00:12 +02:00
}
2024-10-06 00:24:11 +02:00
if self.sim_state == SimState::Running {
self.time_since_step += rl.get_frame_time();
let step_size = 1. / (1 << self.sim_speed) as f32;
if self.time_since_step > step_size {
let step_count = (self.time_since_step / step_size) as u32;
for _ in 0..step_count {
if self.sim_state != SimState::Running {
// pause on level completion
break;
}
self.step();
}
self.time_since_step -= step_count as f32 * step_size;
2024-10-06 00:24:11 +02:00
}
}
if rl.is_key_pressed(KeyboardKey::KEY_SPACE) {
2024-10-07 12:57:32 +02:00
self.step_pressed()
2024-10-06 00:24:11 +02:00
}
if rl.is_key_pressed(KeyboardKey::KEY_ENTER) {
match self.sim_state {
SimState::Editing => {
self.start_sim();
self.sim_state = SimState::Running;
}
SimState::Running => {
self.sim_state = SimState::Editing;
self.complete_popup = Popup::Start;
}
2024-10-06 00:24:11 +02:00
SimState::Stepping => self.sim_state = SimState::Running,
}
}
2024-10-10 21:56:36 +02:00
let mouse_pos = rl.get_mouse_position();
if (self.active_tool != Tool::Blueprint
2024-10-10 21:56:36 +02:00
|| self.sim_state != SimState::Editing
|| mouse_pos.x > SIDEBAR_WIDTH as f32)
&& mouse_pos.y > HEADER_HEIGHT as f32
&& (mouse_pos.y as i32) < (rl.get_screen_height() - FOOTER_HEIGHT)
2024-10-10 21:56:36 +02:00
{
if rl.get_mouse_wheel_move() > 0. {
self.zoom_in(rl);
}
if rl.get_mouse_wheel_move() < 0. {
self.zoom_out(rl);
}
2024-10-06 00:24:11 +02:00
}
if rl.is_mouse_button_down(MouseButton::MOUSE_BUTTON_MIDDLE) {
self.view_offset += rl.get_mouse_delta();
2024-10-06 00:24:11 +02:00
}
if rl.is_mouse_button_pressed(MouseButton::MOUSE_BUTTON_RIGHT) {
self.center_view(rl);
2024-10-06 00:24:11 +02:00
}
2024-10-06 16:00:12 +02:00
if rl.is_key_pressed(KeyboardKey::KEY_R) {
self.rotate_tool(rl.is_key_down(KeyboardKey::KEY_LEFT_SHIFT));
}
2024-10-06 00:24:11 +02:00
}
fn draw_board(&self, d: &mut RaylibDrawHandle, textures: &Textures) {
if self.sim_state == SimState::Editing {
self.source_board
.draw(d, textures, self.view_offset, self.zoom);
} else {
self.machine
.board()
.draw(d, textures, self.view_offset, self.zoom);
self.machine
.draw_marble_values(d, textures, self.view_offset, self.zoom);
2024-10-06 00:24:11 +02:00
}
}
pub fn draw(&mut self, d: &mut RaylibDrawHandle, textures: &Textures) {
2024-10-06 12:39:36 +02:00
d.clear_background(Color::new(64, 64, 64, 255));
2024-10-06 00:24:11 +02:00
let tile_size = (16 << self.zoom) as f32;
let grid_spill_x = (self.view_offset.x).rem(tile_size) - tile_size;
let grid_spill_y = (self.view_offset.y).rem(tile_size) - tile_size;
d.gui_grid(
Rectangle::new(
grid_spill_x,
grid_spill_y,
d.get_screen_width() as f32 * 2.,
2024-10-06 16:00:12 +02:00
d.get_screen_height() as f32 * 2.,
2024-10-06 00:24:11 +02:00
),
None,
tile_size,
1,
);
2024-10-06 16:00:12 +02:00
self.draw_board(d, textures);
self.board_overlay(d, textures);
self.draw_bottom_bar(d, textures);
self.draw_top_bar(d, textures);
if self.active_tool == Tool::Blueprint {
let sidebar_height = d.get_screen_height() - FOOTER_HEIGHT - HEADER_HEIGHT - 40;
d.draw_rectangle(
0,
HEADER_HEIGHT + 20,
SIDEBAR_WIDTH,
sidebar_height,
Color::new(32, 32, 32, 255),
);
d.draw_text("Blueprints", 10, HEADER_HEIGHT + 30, 20, Color::WHITE);
let mut y = HEADER_HEIGHT + 60;
2024-10-10 21:56:36 +02:00
let blueprints_shown = (sidebar_height as usize - 45) / 37;
let end = self
.blueprints
.len()
.min(blueprints_shown + self.blueprint_scroll);
for (i, b) in self.blueprints[self.blueprint_scroll..end]
.iter_mut()
.enumerate()
{
let i = i + self.blueprint_scroll;
if simple_button(d, 5, y, 32, 32) {
b.remove_file();
self.blueprints.remove(i);
break;
}
2024-10-10 20:50:25 +02:00
draw_scaled_texture(d, textures.get("rubbish"), 5, y, 2.);
let is_selected = self.selected_blueprint == i;
let mut text_selected = is_selected && self.blueprint_name_selected;
text_input(
d,
Rectangle::new(42., y as f32, 200., 32.),
&mut b.name,
&mut text_selected,
32,
is_selected,
);
if is_selected {
self.blueprint_name_selected = text_selected;
}
simple_option_button(d, 42 + 205, y, 32, 32, i, &mut self.selected_blueprint);
d.draw_texture_ex(
textures.get("blueprint"),
Vector2::new((42 + 205) as f32, y as f32),
0.,
2.,
Color::new(255, 255, 255, if is_selected { 255 } else { 150 }),
);
y += 37;
}
}
if self.complete_popup == Popup::Visible {
2024-10-13 12:45:58 +02:00
let width = 320;
let height = 165;
let x = d.get_screen_width() / 2 - width / 2;
let y = d.get_screen_height() / 2 - height / 2;
2024-10-13 12:45:58 +02:00
d.draw_rectangle(x, y, width, height, BG_DARK);
d.draw_text("Level Complete!", x + 45, y + 10, 30, Color::LIME);
if let Some(score) = &self.score {
d.draw_text("cycles", x + 15, y + 45, 20, Color::WHITE);
2024-10-13 12:45:58 +02:00
draw_usize(d, textures, score.cycles, x + 10, y + 70, 6, 2);
d.draw_text("tiles", x + 145, y + 45, 20, Color::WHITE);
draw_usize(d, textures, score.tiles, x + 140, y + 70, 4, 2);
d.draw_text("area", x + 155 + 80, y + 45, 20, Color::WHITE);
draw_usize(d, textures, score.area, x + 150 + 80, y + 70, 4, 2);
}
if simple_button(d, x + 10, y + 110, 140, 45) {
self.complete_popup = Popup::Dismissed;
}
d.draw_text("continue\nediting", x + 15, y + 115, 20, Color::WHITE);
if simple_button(d, x + width / 2 + 5, y + 110, 140, 45) {
self.exit_state = ExitState::ExitAndSave;
}
d.draw_text(
"return to\nlevel list",
x + width / 2 + 10,
y + 115,
20,
Color::WHITE,
);
}
2024-10-06 16:00:12 +02:00
}
2024-10-07 01:16:52 +02:00
fn draw_top_bar(&mut self, d: &mut RaylibDrawHandle, textures: &Textures) {
2024-10-06 16:00:12 +02:00
// background
d.draw_rectangle(
0,
0,
d.get_screen_width(),
HEADER_HEIGHT,
Color::new(32, 32, 32, 255),
);
if self.exit_menu {
2024-10-07 18:56:09 +02:00
if simple_button(d, 5, 5, 75, 30) {
2024-10-06 16:00:12 +02:00
self.exit_menu = false;
}
d.draw_text("cancel", 10, 10, 20, Color::WHITE);
2024-10-07 18:56:09 +02:00
if simple_button(d, 85, 5, 60, 30) {
2024-10-06 16:00:12 +02:00
self.exit_state = ExitState::ExitAndSave;
}
2024-10-07 18:56:09 +02:00
d.draw_text("save", 90, 10, 20, Color::WHITE);
if simple_button(d, 150, 5, 80, 30) {
2024-10-06 16:00:12 +02:00
self.exit_state = ExitState::ExitNoSave;
}
2024-10-07 18:56:09 +02:00
d.draw_text("revert", 155, 10, 20, Color::WHITE);
2024-10-06 16:00:12 +02:00
} else {
2024-10-07 18:56:09 +02:00
if simple_button(d, 5, 5, 75, 30) {
2024-10-06 16:00:12 +02:00
self.exit_menu = true;
}
d.draw_text("exit", 10, 10, 20, Color::WHITE);
2024-10-07 18:56:09 +02:00
if simple_button(d, 85, 5, 60, 30) {
2024-10-06 16:29:45 +02:00
self.exit_state = ExitState::Save;
}
2024-10-07 18:56:09 +02:00
d.draw_text("save", 90, 10, 20, Color::WHITE);
2024-10-06 16:00:12 +02:00
}
2024-10-06 20:40:00 +02:00
2024-10-07 01:16:52 +02:00
match self.sim_state {
SimState::Editing => {
if simple_button(d, 260, 4, 32, 32) {
self.start_sim();
self.sim_state = SimState::Running;
}
draw_scaled_texture(d, textures.get("play"), 260, 4, 2.);
}
SimState::Running => {
if simple_button(d, 260, 4, 32, 32) {
self.sim_state = SimState::Stepping;
}
draw_scaled_texture(d, textures.get("pause"), 260, 4, 2.);
2024-10-07 12:57:32 +02:00
if simple_button(d, 296, 4, 32, 32) {
2024-10-07 01:16:52 +02:00
self.sim_state = SimState::Editing;
self.complete_popup = Popup::Start;
}
2024-10-07 12:57:32 +02:00
draw_scaled_texture(d, textures.get("stop"), 296, 4, 2.);
2024-10-07 01:16:52 +02:00
}
SimState::Stepping => {
if simple_button(d, 260, 4, 32, 32) {
self.sim_state = SimState::Running;
}
draw_scaled_texture(d, textures.get("play"), 260, 4, 2.);
2024-10-07 12:57:32 +02:00
if simple_button(d, 296, 4, 32, 32) {
2024-10-07 01:16:52 +02:00
self.sim_state = SimState::Editing;
self.complete_popup = Popup::Start;
}
2024-10-07 12:57:32 +02:00
draw_scaled_texture(d, textures.get("stop"), 296, 4, 2.);
2024-10-07 01:16:52 +02:00
}
}
2024-10-07 12:57:32 +02:00
if simple_button(d, 332, 4, 32, 32) {
self.step_pressed();
}
draw_scaled_texture(d, textures.get("step"), 332, 4, 2.);
2024-10-07 01:16:52 +02:00
d.draw_text("spd", 368, 6, 10, Color::WHITE);
draw_usize(d, textures, 1 << self.sim_speed, 388, 4, 3, 1);
slider(d, &mut self.sim_speed, 0, 9, 368, 24, 48, 12);
2024-10-07 14:16:50 +02:00
draw_usize(d, textures, self.machine.step_count(), 420, 4, 5, 2);
2024-10-07 13:16:26 +02:00
2024-10-07 17:29:56 +02:00
d.draw_text("input:", 523, 8, 10, Color::WHITE);
if simple_button(d, 520, 20, 35, 15) {
self.input_as_text = !self.input_as_text
}
let input_mode_text = if self.input_as_text { "text" } else { "bytes" };
d.draw_text(input_mode_text, 523, 23, 10, Color::WHITE);
let input_x = 560;
2024-10-06 20:40:00 +02:00
let width = d.get_screen_width();
2024-10-07 17:29:56 +02:00
if self.input_as_text {
2024-10-07 18:45:24 +02:00
let mut input_text = String::new();
for &byte in self.machine.input() {
if byte.is_ascii_graphic() || byte == b' ' {
input_text.push(byte as char);
} else {
input_text.push('\\');
}
}
2024-10-07 17:29:56 +02:00
if text_input(
d,
Rectangle::new(input_x as f32, 5., (width - input_x - 5) as f32, 30.),
&mut input_text,
&mut self.input_text_selected,
2024-10-08 22:24:41 +02:00
256,
2024-10-07 17:29:56 +02:00
self.level.is_sandbox(),
) {
self.machine.set_input(input_text.into_bytes());
}
} else {
let input_cell_width = 43;
let input_cells = (d.get_screen_width() - input_x) as usize / input_cell_width as usize;
let input_start = self.machine.input_index().saturating_sub(input_cells);
let input_end = input_start + input_cells;
for (box_index, index) in (input_start..input_end).enumerate() {
let x = input_x + input_cell_width * box_index as i32;
let byte = self.machine.input().get(index);
2024-10-13 12:45:58 +02:00
d.draw_rectangle(x, 5, input_cell_width - 5, 30, BG_WIDGET);
2024-10-07 17:29:56 +02:00
let color = if index < self.machine.input_index() {
d.draw_rectangle(x + 4, 25, input_cell_width - 13, 8, Color::LIME);
Color::LIGHTGREEN
} else {
Color::WHITE
};
if let Some(&byte) = byte {
let top_text = format!("{}", byte);
d.draw_text(&top_text, x + 2, 5, 20, color);
}
}
2024-10-06 20:40:00 +02:00
}
2024-10-06 16:00:12 +02:00
}
fn draw_bottom_bar(&mut self, d: &mut RaylibDrawHandle, textures: &Textures) {
let height = d.get_screen_height();
let footer_top = (height - FOOTER_HEIGHT) as f32;
// background
d.draw_rectangle(
0,
height - FOOTER_HEIGHT,
d.get_screen_width(),
FOOTER_HEIGHT,
Color::new(32, 32, 32, 255),
);
let mut hide_tile_tools = false;
if let Tool::SelectArea(Selection {
area: Some(selection),
is_selecting: _,
}) = self.active_tool
{
hide_tile_tools = true;
text_input(
d,
Rectangle::new(100., footer_top + 10., 240., 30.),
&mut self.new_blueprint_name,
&mut self.blueprint_name_selected,
32,
true,
);
if simple_button(d, 100, footer_top as i32 + 49, 40, 40) {
self.save_blueprint(selection);
}
draw_scaled_texture(d, textures.get("save"), 104, footer_top as i32 + 53, 2.);
if simple_button(d, 144, footer_top as i32 + 49, 40, 40) {
self.active_tool = Tool::SelectArea(Selection::default());
}
draw_scaled_texture(d, textures.get("cancel"), 148, footer_top as i32 + 53, 2.);
// if simple_button(d, 144, footer_top as i32 + 49, 40, 40) {
// self.active_tool = Tool::SelectArea(Selection::default());
// }
// draw_scaled_texture(
// d,
// textures.get("direction_up"),
// 148,
// footer_top as i32 + 53,
// 2.,
// );
}
let mouse_pos = d.get_mouse_position();
2024-10-06 00:24:11 +02:00
let mut tool_button = |(row, col): (i32, i32), texture: &str, tool_option: Tool| {
let border = 4.;
let gap = 2.;
let tex_size = 32.;
let button_size = tex_size + border * 2.;
let grid_size = button_size + gap * 2.;
let pos = Vector2 {
x: 100. + col as f32 * grid_size - if col < 0 { 10. } else { 0. },
y: footer_top + 5. + row as f32 * grid_size,
};
2024-10-06 12:39:36 +02:00
texture_option_button(
2024-10-06 00:24:11 +02:00
d,
pos,
2024-10-06 00:24:11 +02:00
textures.get(texture),
tool_option,
&mut self.active_tool,
tex_size,
2024-10-06 00:24:11 +02:00
border,
);
let bounds = Rectangle::new(pos.x, pos.y, button_size, button_size);
if bounds.check_collision_point_rec(mouse_pos) {
get_scroll(d)
} else {
None
}
2024-10-06 00:24:11 +02:00
};
2024-10-07 16:51:56 +02:00
tool_button((0, -2), "eraser", Tool::Erase);
tool_button((1, -2), "selection", Tool::SelectArea(Selection::default()));
tool_button((0, -1), "blueprint", Tool::Blueprint);
2024-10-06 00:24:11 +02:00
tool_button((1, -1), "transparent", Tool::None);
if !hide_tile_tools {
tool_button((0, 0), "block", Tool::SetTile(Tile::from_char('#')));
tool_button((0, 1), "bag_off", Tool::SetTile(Tile::from_char('B')));
tool_button((0, 2), "trigger_off", Tool::SetTile(Tile::from_char('*')));
tool_button((0, 3), "io_tile_off", Tool::SetTile(Tile::from_char('I')));
tool_button((0, 4), "flipper_off", Tool::SetTile(Tile::from_char('F')));
tool_button((0, 5), "digit_tool", Tool::Digits(None));
tool_button((1, 0), "marble", Tool::SetTile(Tile::from_char('o')));
match tool_button(
(1, 1),
&Tile::Powerable(PTile::Wire(self.tool_wire), false).texture(),
Tool::Wire,
) {
Some(Scroll::Down) => self.tool_wire.next(),
Some(Scroll::Up) => self.tool_wire.prev(),
None => (),
}
2024-10-06 00:24:11 +02:00
match tool_button((1, 2), &Tile::Arrow(self.tool_arrow).texture(), Tool::Arrow) {
Some(Scroll::Down) => self.tool_arrow = self.tool_arrow.right(),
Some(Scroll::Up) => self.tool_arrow = self.tool_arrow.left(),
None => (),
}
if tool_button(
(1, 3),
&Tile::Mirror(self.tool_mirror).texture(),
Tool::Mirror,
)
.is_some()
{
self.tool_mirror.flip()
}
match tool_button(
(1, 4),
&Tile::Powerable(PTile::Math(self.tool_math), false).texture(),
Tool::Math,
) {
Some(Scroll::Down) => self.tool_math.next(),
Some(Scroll::Up) => self.tool_math.prev(),
None => (),
}
match tool_button(
(1, 5),
&Tile::Powerable(PTile::Gate(self.tool_gate), false).texture(),
Tool::Gate,
) {
Some(Scroll::Down) => self.tool_gate.next(),
Some(Scroll::Up) => self.tool_gate.prev(),
None => (),
}
}
2024-10-06 19:51:09 +02:00
let output_x = 370;
let output_cell_width = 43;
2024-10-07 17:29:56 +02:00
let output_cells = (d.get_screen_width() - output_x) as usize / output_cell_width as usize;
2024-10-06 19:51:09 +02:00
let y = footer_top as i32 + 5;
if simple_button(d, output_x, y + 70, 65, 15) {
2024-10-06 19:51:09 +02:00
self.output_as_text = !self.output_as_text
}
let output_mode_text = if self.output_as_text {
"show bytes"
} else {
"show text"
};
d.draw_text(output_mode_text, output_x + 5, y + 72, 10, Color::WHITE);
let output_start = self.machine.output().len().saturating_sub(output_cells);
let output_end = output_start + output_cells;
2024-10-06 19:51:09 +02:00
for (box_index, index) in (output_start..output_end).enumerate() {
let x = output_x + output_cell_width * box_index as i32;
2024-10-06 19:51:09 +02:00
let expected_byte = self.level.outputs().get(index);
let real_byte = self.machine.output().get(index);
let (top_color, bottom_color) =
if let (Some(&real_byte), Some(&expected_byte)) = (real_byte, expected_byte) {
if expected_byte == real_byte {
(Color::GREEN, Color::DARKGREEN)
} else {
(Color::RED, Color::DARKRED)
}
} else {
(Color::DARKGRAY, Color::DIMGRAY)
};
2024-10-07 17:29:56 +02:00
d.draw_rectangle(x, y, output_cell_width - 5, 30, top_color);
d.draw_rectangle(x, y + 35, output_cell_width - 5, 30, bottom_color);
2024-10-06 19:51:09 +02:00
if let Some(&expected_byte) = expected_byte {
let top_text = if self.output_as_text
&& (expected_byte.is_ascii_graphic() || expected_byte.is_ascii_whitespace())
{
format!("{:?}", expected_byte as char)
} else {
format!("{}", expected_byte)
};
d.draw_text(&top_text, x + 2, y + 5, 20, Color::WHITE);
}
if let Some(&real_byte) = real_byte {
let bottom_text = if self.output_as_text
&& (real_byte.is_ascii_graphic() || real_byte.is_ascii_whitespace())
{
format!("{:?}", real_byte as char)
} else {
format!("{}", real_byte)
};
d.draw_text(&bottom_text, x + 2, y + 40, 20, Color::WHITE);
}
}
2024-10-06 16:00:12 +02:00
}
2024-10-06 00:24:11 +02:00
2024-10-06 16:00:12 +02:00
fn board_overlay(&mut self, d: &mut RaylibDrawHandle, textures: &Textures) {
let footer_top = (d.get_screen_height() - FOOTER_HEIGHT) as f32;
2024-10-06 00:24:11 +02:00
let mouse_pos = d.get_mouse_position();
2024-10-10 21:56:36 +02:00
let scroll_delta = d.get_mouse_wheel_move();
let tile_scale = (1 << self.zoom) as f32;
let tile_size = 16 << self.zoom;
2024-10-06 00:24:11 +02:00
if self.sim_state == SimState::Editing {
if let Tool::Digits(Some(pos)) = &mut self.active_tool {
let tile_screen_pos = pos.to_vec() * tile_size as f32 + self.view_offset;
2024-10-06 00:24:11 +02:00
d.draw_texture_ex(
textures.get("selection"),
tile_screen_pos,
0.,
tile_scale,
2024-10-06 00:24:11 +02:00
Color::new(255, 180, 20, 255),
);
if d.is_key_pressed(KeyboardKey::KEY_LEFT) {
pos.x -= 1;
}
if d.is_key_pressed(KeyboardKey::KEY_RIGHT) {
pos.x += 1;
}
if d.is_key_pressed(KeyboardKey::KEY_UP) {
pos.y -= 1;
}
if d.is_key_pressed(KeyboardKey::KEY_DOWN) {
pos.y += 1;
}
let pos = *pos;
for n in 0..10 {
if d.is_key_pressed(unsafe { transmute::<u32, KeyboardKey>(b'0' as u32 + n) }) {
self.set_tile(pos, Tile::Digit(n as u8));
}
}
2024-10-06 00:24:11 +02:00
}
2024-10-06 16:00:12 +02:00
if mouse_pos.y < footer_top && mouse_pos.y > HEADER_HEIGHT as f32 {
let tile_pos = (mouse_pos - self.view_offset) / tile_size as f32;
2024-10-06 00:24:11 +02:00
let tile_pos = Vector2::new(tile_pos.x.floor(), tile_pos.y.floor());
let tile_screen_pos = self.pos_to_screen(tile_pos);
2024-10-06 00:24:11 +02:00
if self.active_tool != Tool::None {
let tex = match &self.active_tool {
2024-10-06 00:24:11 +02:00
Tool::None => unreachable!(),
Tool::Erase => "cancel".into(),
2024-10-07 16:51:56 +02:00
Tool::SetTile(t) => t.texture(),
Tool::Math => format!("{}_off", self.tool_math.texture_name()),
Tool::Gate => format!("{}_off", self.tool_gate.texture_name()),
Tool::Wire => format!("{}_off", self.tool_wire.texture_name()),
Tool::Arrow => self.tool_arrow.arrow_tile_texture_name().into(),
Tool::Mirror => self.tool_mirror.texture_name().into(),
2024-10-06 00:24:11 +02:00
Tool::Digits(_) => "selection".into(),
Tool::SelectArea(selection) => {
if selection.is_selecting {
"transparent".into()
} else {
"area_full".into()
}
}
Tool::Blueprint => "transparent".into(),
2024-10-06 00:24:11 +02:00
};
d.draw_texture_ex(
textures.get(&tex),
tile_screen_pos,
0.,
tile_scale,
2024-10-06 00:24:11 +02:00
Color::new(255, 255, 255, 100),
);
}
if d.is_mouse_button_pressed(MouseButton::MOUSE_BUTTON_LEFT) {
let pos = tile_pos.into();
2024-10-06 00:24:11 +02:00
match self.active_tool {
Tool::None => (),
2024-10-07 16:51:56 +02:00
Tool::Erase => (),
Tool::SetTile(tile) => self.set_tile(pos, tile),
Tool::Math => {
self.set_tile(pos, Tile::Powerable(PTile::Math(self.tool_math), false))
}
Tool::Gate => {
self.set_tile(pos, Tile::Powerable(PTile::Gate(self.tool_gate), false))
}
Tool::Wire => {
self.set_tile(pos, Tile::Powerable(PTile::Wire(self.tool_wire), false))
}
Tool::Arrow => self.set_tile(pos, Tile::Arrow(self.tool_arrow)),
Tool::Mirror => self.set_tile(pos, Tile::Mirror(self.tool_mirror)),
2024-10-06 00:24:11 +02:00
Tool::Digits(_pos) => {
self.active_tool = Tool::Digits(Some(pos));
if let Some(tile) = self.source_board.get_mut(pos) {
2024-10-06 00:24:11 +02:00
if let Tile::Digit(_) = tile {
} else {
*tile = Tile::Digit(0);
}
}
}
Tool::Blueprint => {
if mouse_pos.x > SIDEBAR_WIDTH as f32 {
if let Some(bp) = self.blueprints.get(self.selected_blueprint) {
let board = bp.get_board().unwrap().clone();
let mut pos = pos;
self.grow_board_and_update_view(&mut pos);
self.grow_board_and_update_view(
&mut (pos + (board.width() - 1, board.height() - 1).into()),
);
for x in 0..board.width() {
for y in 0..board.height() {
let p = (x, y).into();
if let Some(tile) = board.get(p) {
self.source_board.set(p + pos, tile);
}
}
}
}
}
}
Tool::SelectArea(_) => (),
}
}
2024-10-07 17:29:56 +02:00
if d.is_mouse_button_down(MouseButton::MOUSE_BUTTON_LEFT)
&& self.active_tool == Tool::Erase
{
self.set_tile(tile_pos.into(), Tile::Blank)
2024-10-07 16:51:56 +02:00
}
if let Tool::SelectArea(selection) = &mut self.active_tool {
if d.is_mouse_button_down(MouseButton::MOUSE_BUTTON_LEFT) {
if selection.is_selecting {
if let Some((_start, end)) = &mut selection.area {
*end = tile_pos.into();
} else {
selection.area = Some((tile_pos.into(), tile_pos.into()));
}
} else {
selection.area = Some((tile_pos.into(), tile_pos.into()));
selection.is_selecting = true;
}
} else if d.is_mouse_button_released(MouseButton::MOUSE_BUTTON_LEFT) {
selection.is_selecting = false;
2024-10-06 00:24:11 +02:00
}
}
if let Tool::Blueprint = self.active_tool {
if let Some(bp) = self.blueprints.get_mut(self.selected_blueprint) {
let view_offset = Vector2::new(
self.view_offset.x.rem(tile_size as f32),
self.view_offset.y.rem(tile_size as f32),
);
let mut offset = mouse_pos - view_offset;
offset.x -= offset.x.rem(tile_size as f32);
offset.y -= offset.y.rem(tile_size as f32);
offset += view_offset;
bp.convert_board().draw(d, textures, offset, self.zoom);
}
2024-10-10 21:56:36 +02:00
if mouse_pos.x < SIDEBAR_WIDTH as f32 {
if scroll_delta < 0.
&& self.blueprint_scroll < self.blueprints.len().saturating_sub(5)
{
self.blueprint_scroll += 1;
} else if scroll_delta > 0. && self.blueprint_scroll > 0 {
self.blueprint_scroll -= 1;
}
}
}
2024-10-06 00:24:11 +02:00
}
// draw selection
if let Tool::SelectArea(Selection {
area: Some((start, end)),
is_selecting: _,
}) = self.active_tool
{
let min = start.min(end);
let max = start.max(end);
let p_min = self.pos_to_screen(min.to_vec());
let p_max = self.pos_to_screen(max.to_vec());
let tex = textures.get("area_corner");
d.draw_texture_ex(tex, p_min, 0., tile_scale, Color::WHITE);
let one_xy = Vector2::new(tile_size as f32, tile_size as f32);
d.draw_texture_ex(tex, p_max + one_xy, 180., tile_scale, Color::WHITE);
let top_right = Vector2::new(p_max.x + tile_size as f32, p_min.y);
d.draw_texture_ex(tex, top_right, 90., tile_scale, Color::WHITE);
let bot_left = Vector2::new(p_min.x, p_max.y + tile_size as f32);
d.draw_texture_ex(tex, bot_left, -90., tile_scale, Color::WHITE);
}
2024-10-06 00:24:11 +02:00
}
}
}
impl PartialEq for Tool {
fn eq(&self, other: &Self) -> bool {
match (self, other) {
(Self::SetTile(l0), Self::SetTile(r0)) => l0 == r0,
(Self::Digits(_), Self::Digits(_)) => true,
(Self::SelectArea(_), Self::SelectArea(_)) => true,
2024-10-06 00:24:11 +02:00
_ => ::core::mem::discriminant(self) == ::core::mem::discriminant(other),
}
}
}
fn get_blueprints() -> Vec<Blueprint> {
let mut blueprints = Vec::<Blueprint>::new();
let Ok(dir) = read_dir(userdata_dir().join("blueprints")) else {
return blueprints;
};
for d in dir.flatten() {
let l = read_to_string(d.path())
.ok()
.as_deref()
.and_then(|s| serde_json::from_str(s).ok());
if let Some(level) = l {
blueprints.push(level);
}
}
blueprints.sort_by(|a, b| a.name.cmp(&b.name));
blueprints
}