use fltk::{ app, button::Button, enums::{Color, Key, Shortcut}, group::{Pack, PackType}, output::Output, prelude::*, window::Window, }; use std::ops::{Deref, DerefMut}; #[derive(Debug, Copy, Clone, PartialEq, Eq)] enum Ops { None, Add, Sub, Mul, Div, Eq, CE, C, Back, } #[derive(Debug, Copy, Clone)] enum Message { Number(i32), Op(Ops), Dot, } struct MyButton { b: Button, } impl MyButton { pub fn new(title: &'static str) -> MyButton { let mut b = Button::new(0, 0, 90, 0, title); b.set_label_size(20); b.set_compact(true); match title { "0" => { b.resize(0, 0, 90 * 2, 0); b.set_color(Color::Light3); b.set_shortcut(Shortcut::None | '0'); } "CE" => { b.set_color(Color::Red); b.set_shortcut(Shortcut::None | Key::Delete); } "x" | "/" | "+" | "-" | "=" | "C" | "@<-" => { b.set_color(Color::Yellow); let shortcut = if title == "x" { '*' } else { title.chars().next().unwrap() }; b.set_shortcut(Shortcut::None | shortcut); if shortcut == '@' { b.set_shortcut(Shortcut::None | Key::BackSpace); } if shortcut == '=' { b.set_shortcut(Shortcut::None | Key::Enter); } } _ => { b.set_color(Color::Light3); b.set_shortcut(Shortcut::None | title.chars().next().unwrap()); } } Self { b } } } impl Deref for MyButton { type Target = Button; fn deref(&self) -> &Self::Target { &self.b } } impl DerefMut for MyButton { fn deref_mut(&mut self) -> &mut Self::Target { &mut self.b } } fn main() { let app = app::App::default(); let win_w = 400; let win_h = 500; let border = 20; let but_row = 180; let mut operation = Ops::None; let mut txt = String::from("0"); let mut old_val = String::from("0"); let mut new_val: String; let mut wind = Window::default() .with_label("FLTK Calc") .with_size(win_w, win_h) .center_screen(); wind.set_color(Color::Light3); let mut out = Output::new(border, border, win_w - 40, 140, ""); out.set_text_size(36); out.set_value("0"); let vpack = Pack::new(border, but_row, win_w - 40, 300, ""); let mut hpack = Pack::new(0, 0, win_w - 40, 60, ""); let but_ce = MyButton::new("CE"); let but_c = MyButton::new("C"); let but_back = MyButton::new("@<-"); let but_div = MyButton::new("/"); hpack.end(); hpack.set_type(PackType::Horizontal); let mut hpack = Pack::new(0, 0, win_w - 40, 60, ""); let mut but7 = MyButton::new("7"); let mut but8 = MyButton::new("8"); let mut but9 = MyButton::new("9"); let but_mul = MyButton::new("x"); hpack.end(); hpack.set_type(PackType::Horizontal); let mut hpack = Pack::new(0, 0, win_w - 40, 60, ""); let mut but4 = MyButton::new("4"); let mut but5 = MyButton::new("5"); let mut but6 = MyButton::new("6"); let but_sub = MyButton::new("-"); hpack.end(); hpack.set_type(PackType::Horizontal); let mut hpack = Pack::new(0, 0, win_w - 40, 60, ""); let mut but1 = MyButton::new("1"); let mut but2 = MyButton::new("2"); let mut but3 = MyButton::new("3"); let but_add = MyButton::new("+"); hpack.end(); hpack.set_type(PackType::Horizontal); let mut hpack = Pack::new(0, 0, win_w - 40, 60, ""); let mut but_dot = MyButton::new("."); let mut but0 = MyButton::new("0"); let but_eq = MyButton::new("="); hpack.end(); hpack.set_type(PackType::Horizontal); vpack.end(); wind.make_resizable(false); wind.end(); wind.show_with_args(&["-scheme", "gtk+", "-nokbd"]); app::set_focus(&*but1); let but_vec = vec![ &mut but1, &mut but2, &mut but3, &mut but4, &mut but5, &mut but6, &mut but7, &mut but8, &mut but9, &mut but0, ]; let but_op_vec = vec![ but_add, but_sub, but_mul, but_div, but_c, but_ce, but_back, but_eq, ]; let (s, r) = app::channel::(); for but in but_vec { let label = but.label(); but.emit(s, Message::Number(label.parse().unwrap())); } for mut but in but_op_vec { let op = match but.label().as_str() { "+" => Ops::Add, "-" => Ops::Sub, "x" => Ops::Mul, "/" => Ops::Div, "=" => Ops::Eq, "CE" => Ops::CE, "C" => Ops::C, "@<-" => Ops::Back, _ => Ops::None, }; but.emit(s, Message::Op(op)); } but_dot.emit(s, Message::Dot); while app.wait() { if let Some(val) = r.recv() { match val { Message::Number(num) => { if out.value() == "0" { txt.clear(); } txt.push_str(&num.to_string()); out.set_value(txt.as_str()); } Message::Dot => { if operation == Ops::Eq { txt.clear(); operation = Ops::None; out.set_value("0."); txt.push_str("0."); } if !txt.contains('.') { txt.push('.'); out.set_value(txt.as_str()); } } Message::Op(op) => match op { Ops::Add | Ops::Sub | Ops::Div | Ops::Mul => { old_val.clear(); old_val.push_str(&out.value()); operation = op; out.set_value("0"); } Ops::Back => { let val = out.value(); txt.pop(); if val.len() > 1 { out.set_value(txt.as_str()); } else { out.set_value("0"); } } Ops::CE => { txt.clear(); old_val.clear(); txt.push('0'); out.set_value(txt.as_str()); } Ops::C => { txt.clear(); txt.push('0'); out.set_value(txt.as_str()); } Ops::Eq => { new_val = out.value(); let old: f64 = old_val.parse().unwrap(); let new: f64 = new_val.parse().unwrap(); let val = match operation { Ops::Div => old / new, Ops::Mul => old * new, Ops::Add => old + new, Ops::Sub => old - new, _ => new, }; operation = Ops::None; txt = String::from("0"); out.set_value(&val.to_string()); } _ => (), }, } } } }