Arduino-Pedal-Effects/rust/serial_reader/src/main.rs
2023-08-25 11:53:08 +02:00

165 lines
5.3 KiB
Rust

use std::{
sync::{
atomic::{AtomicBool, Ordering::SeqCst},
Mutex,
},
thread,
};
use gtk::{prelude::*, PolicyType, TextView};
use gtk::{Application, ApplicationWindow};
use gtk::{Button, Entry, Orientation, ScrolledWindow};
use glib::{self, source::Priority, ControlFlow};
mod port;
use port::*;
static CONNECTED: AtomicBool = AtomicBool::new(false);
static PORT: Mutex<Option<Port>> = Mutex::new(None);
fn main() {
let app = Application::builder()
.application_id("org.example.HelloWorld")
.build();
app.connect_activate(move |app| {
// We create the main window.
let window = ApplicationWindow::builder()
.application(app)
.default_width(320)
.default_height(200)
.title("Hello, World!")
.build();
let master_box = gtk::Box::new(Orientation::Vertical, 5);
let top_bar_box = gtk::Box::new(Orientation::Horizontal, 5);
let pid = Entry::builder().text("0x27dd").editable(true).build();
let vid = Entry::builder().text("0x16c0").editable(true).build();
let connect_button = Button::with_label("Connect");
let reset_button = Button::with_label("Reset");
let serial_info = TextView::new();
serial_info.set_editable(false);
let scrolled_window = ScrolledWindow::builder()
.vscrollbar_policy(PolicyType::Automatic)
.child(&serial_info)
.build();
top_bar_box.pack_end(&connect_button, true, true, 3);
top_bar_box.pack_end(&reset_button, true, true, 3);
top_bar_box.pack_end(&vid, true, true, 3);
top_bar_box.pack_end(&pid, true, true, 3);
master_box.pack_end(&scrolled_window, true, true, 3);
master_box.pack_end(&top_bar_box, false, true, 3);
let (tx, rx) = glib::MainContext::channel(Priority::DEFAULT);
std::thread::spawn(move || {
// loop forever
loop {
let mut port_lock = PORT.lock().unwrap();
if let Some(port) = &mut *port_lock {
// handle incoming message
match port.read() {
Ok(port_read) => match port_read {
SerialReadResult::Message(msg) => {
if !msg.is_empty() {
tx.send(msg).unwrap();
}
}
SerialReadResult::UtfConversion(err) => println!("{:?}", err),
SerialReadResult::Timeout => (),
},
Err(err) => println!("{err}"),
}
}
drop(port_lock);
thread::sleep(Duration::from_millis(10));
}
});
let buffer = serial_info.buffer().unwrap();
connect_button.connect_clicked(move |_| {
if !CONNECTED.load(SeqCst) {
let usb_id = UsbId {
vendor_id: u16::from_str_radix(
vid.buffer().text().trim_start_matches("0x"),
16,
)
.unwrap(),
product_id: u16::from_str_radix(
pid.buffer().text().trim_start_matches("0x"),
16,
)
.unwrap(),
};
let serialport_settings = SerialPortSettings {
baud_rate: 9600,
data_bits: DataBits::Eight,
parity: Parity::None,
stop_bits: StopBits::One,
flow_control: FlowControl::None,
timeout: Duration::from_millis(2500),
};
if let Ok(port) = Port::open(usb_id, serialport_settings, 50, 1) {
CONNECTED.store(true, SeqCst);
let (mut buffer_start, mut buffer_end) = buffer.bounds();
buffer.delete(&mut buffer_start, &mut buffer_end);
*PORT.lock().unwrap() = Some(port);
}
}
});
reset_button.connect_clicked(move |_| {
if CONNECTED.load(SeqCst) {
CONNECTED.store(false, SeqCst);
let mut port_lock = PORT.lock().unwrap();
if let Some(port) = &mut *port_lock {
if let Err(err) = port.write("reset") {
println!("{err}");
}
}
*port_lock = None;
}
});
let buffer = serial_info.buffer().unwrap();
rx.attach(None, move |message| {
if !message.is_empty() {
match message.parse::<u32>() {
Ok(n) => {
buffer.insert(&mut buffer.end_iter(), &format!("{:#02x}\n", n));
}
Err(_) => {
buffer.insert(&mut buffer.end_iter(), &format!("{}\n", message));
}
}
serial_info.scroll_to_iter(&mut buffer.end_iter(), 0.0, true, 0.0, 0.0);
}
ControlFlow::Continue
});
window.add(&master_box);
// Don't forget to make all widgets visible.
window.show_all();
});
app.run();
}