arch: split ConfigRepository, extract polling, consolidate conversions, decouple protocol
- Value↔JSON: From impls on domain Value behind `json` feature, delete 4 duplicate converters - ConfigRepository split into ConfigRepository (12), UserRepository (3), WidgetStateCache (2) - polling orchestration moved from bootstrap to application::polling_service - WidgetRenderer in client-domain owns scroll/cache, both clients use it - network loop consolidated into client-application::run_connection_loop - protocol crate drops domain dep, Wire↔Domain conversions move to adapters
This commit is contained in:
@@ -1,56 +1,26 @@
|
||||
use std::sync::mpsc;
|
||||
use std::thread;
|
||||
use client_domain::NetworkPort;
|
||||
use protocol::decode_server_message;
|
||||
use client_application::run_connection_loop;
|
||||
use super::RenderEvent;
|
||||
use crate::config::{NET_THREAD_STACK_SIZE, NET_POLL_INTERVAL, NET_RECONNECT_DELAY};
|
||||
use crate::adapters::network::Esp32Network;
|
||||
use log::*;
|
||||
|
||||
pub fn spawn(server_addr: String, tx: mpsc::Sender<RenderEvent>) {
|
||||
thread::Builder::new()
|
||||
.stack_size(NET_THREAD_STACK_SIZE)
|
||||
.name("net".into())
|
||||
.spawn(move || run(server_addr, tx))
|
||||
.spawn(move || {
|
||||
let mut net = Esp32Network::new();
|
||||
let tx_msg = tx.clone();
|
||||
let tx_status = tx.clone();
|
||||
run_connection_loop(
|
||||
&mut net,
|
||||
&server_addr,
|
||||
NET_POLL_INTERVAL,
|
||||
NET_RECONNECT_DELAY,
|
||||
move |msg| { let _ = tx_msg.send(RenderEvent::Server(msg)); },
|
||||
move |connected| { let _ = tx_status.send(RenderEvent::ConnectionStatus(connected)); },
|
||||
);
|
||||
})
|
||||
.expect("failed to spawn network thread");
|
||||
}
|
||||
|
||||
fn run(server_addr: String, tx: mpsc::Sender<RenderEvent>) {
|
||||
let mut net = Esp32Network::new();
|
||||
|
||||
loop {
|
||||
if !net.is_connected() {
|
||||
info!("Connecting to server {server_addr}...");
|
||||
match net.connect(&server_addr) {
|
||||
Ok(()) => {
|
||||
info!("Server connected");
|
||||
let _ = tx.send(RenderEvent::ConnectionStatus(true));
|
||||
}
|
||||
Err(e) => {
|
||||
error!("Connection failed: {e}, retrying...");
|
||||
let _ = tx.send(RenderEvent::ConnectionStatus(false));
|
||||
thread::sleep(NET_RECONNECT_DELAY);
|
||||
continue;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
match net.receive() {
|
||||
Ok(Some(payload)) => {
|
||||
match decode_server_message(&payload) {
|
||||
Ok(msg) => { let _ = tx.send(RenderEvent::Server(msg)); }
|
||||
Err(e) => error!("Decode error: {e}"),
|
||||
}
|
||||
}
|
||||
Ok(None) => {
|
||||
thread::sleep(NET_POLL_INTERVAL);
|
||||
}
|
||||
Err(e) => {
|
||||
error!("Receive error: {e}, reconnecting...");
|
||||
let _ = net.disconnect();
|
||||
let _ = tx.send(RenderEvent::ConnectionStatus(false));
|
||||
thread::sleep(NET_RECONNECT_DELAY);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,11 +1,10 @@
|
||||
use std::sync::mpsc;
|
||||
use std::time::{Duration, Instant};
|
||||
use std::collections::HashMap;
|
||||
use client_domain::{
|
||||
BoundingBox, Color, DisplayPort, FontMetrics, RenderEngine, ScrollState, ThemeConfig,
|
||||
BoundingBox, Color, DisplayPort, FontMetrics, RenderEngine, RepaintRequest, ThemeConfig,
|
||||
WidgetRenderer,
|
||||
};
|
||||
use client_application::{ClientApp, RepaintCommand};
|
||||
use domain::{DisplayHint, Value, WidgetError};
|
||||
use client_application::{ClientApp, RepaintCommand, conversions};
|
||||
use protocol::ServerMessage;
|
||||
use super::RenderEvent;
|
||||
use crate::config::RENDER_POLL_INTERVAL;
|
||||
@@ -18,12 +17,17 @@ const INDICATOR_MARGIN: u16 = 4;
|
||||
const COLOR_CONNECTED: Color = Color(0, 200, 0);
|
||||
const COLOR_DISCONNECTED: Color = Color(200, 0, 0);
|
||||
|
||||
struct WidgetCache {
|
||||
hint: DisplayHint,
|
||||
data: Vec<(String, Value)>,
|
||||
error: Option<WidgetError>,
|
||||
bounds: BoundingBox,
|
||||
scroll: ScrollState,
|
||||
fn to_repaint_requests(repaints: &[RepaintCommand]) -> Vec<RepaintRequest> {
|
||||
repaints
|
||||
.iter()
|
||||
.map(|cmd| RepaintRequest {
|
||||
widget_id: cmd.widget_id,
|
||||
bounds: cmd.bounds,
|
||||
display_hint: conversions::wire_to_display_hint(cmd.display_hint.clone()),
|
||||
data: cmd.state.data.iter().map(|kv| (kv.key.clone(), conversions::wire_to_value(kv.value.clone()))).collect(),
|
||||
error: cmd.state.error.as_ref().map(|e| conversions::wire_to_widget_error(e.clone())),
|
||||
})
|
||||
.collect()
|
||||
}
|
||||
|
||||
pub fn run(
|
||||
@@ -37,7 +41,7 @@ pub fn run(
|
||||
};
|
||||
let mut engine = RenderEngine::new(metrics, ThemeConfig::default());
|
||||
let mut app = ClientApp::new(screen);
|
||||
let mut widgets: HashMap<u16, WidgetCache> = HashMap::new();
|
||||
let mut renderer = WidgetRenderer::new();
|
||||
let mut first_update = true;
|
||||
let mut last_tick = Instant::now();
|
||||
let mut connected = false;
|
||||
@@ -47,7 +51,7 @@ pub fn run(
|
||||
display.flush().unwrap();
|
||||
|
||||
loop {
|
||||
let has_scrollers = widgets.values().any(|c| c.scroll.is_active());
|
||||
let has_scrollers = renderer.has_active_scrollers();
|
||||
let timeout = if has_scrollers { SCROLL_TICK } else { RENDER_POLL_INTERVAL };
|
||||
match rx.recv_timeout(timeout) {
|
||||
Ok(RenderEvent::ConnectionStatus(status)) => {
|
||||
@@ -71,14 +75,17 @@ pub fn run(
|
||||
display.fill_rect(screen, bg).unwrap();
|
||||
first_update = false;
|
||||
}
|
||||
for cmd in &repaints {
|
||||
let cache = update_cache(&engine, cmd);
|
||||
display.fill_rect(cache.bounds, bg).unwrap();
|
||||
draw_widget(&engine, &mut display, &cache);
|
||||
widgets.insert(cmd.widget_id, cache);
|
||||
|
||||
let requests = to_repaint_requests(&repaints);
|
||||
let updates = renderer.apply_repaints(&engine, requests);
|
||||
for update in &updates {
|
||||
display.fill_rect(update.bounds, bg).unwrap();
|
||||
for dc in &update.commands {
|
||||
display.draw_text_span(&dc.text, dc.x, dc.y, dc.color, dc.font).unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
if !repaints.is_empty() {
|
||||
if !updates.is_empty() {
|
||||
draw_indicator(&mut display, screen, connected);
|
||||
display.flush().unwrap();
|
||||
}
|
||||
@@ -94,16 +101,15 @@ pub fn run(
|
||||
let elapsed = now.duration_since(last_tick);
|
||||
last_tick = now;
|
||||
|
||||
let mut needs_flush = false;
|
||||
for cache in widgets.values_mut() {
|
||||
if cache.scroll.tick(elapsed) {
|
||||
let bg = engine.theme().background;
|
||||
display.fill_rect(cache.bounds, bg).unwrap();
|
||||
draw_widget(&engine, &mut display, cache);
|
||||
needs_flush = true;
|
||||
let scroll_updates = renderer.tick_scroll(&engine, elapsed);
|
||||
if !scroll_updates.is_empty() {
|
||||
let bg = engine.theme().background;
|
||||
for update in &scroll_updates {
|
||||
display.fill_rect(update.bounds, bg).unwrap();
|
||||
for dc in &update.commands {
|
||||
display.draw_text_span(&dc.text, dc.x, dc.y, dc.color, dc.font).unwrap();
|
||||
}
|
||||
}
|
||||
}
|
||||
if needs_flush {
|
||||
draw_indicator(&mut display, screen, connected);
|
||||
display.flush().unwrap();
|
||||
}
|
||||
@@ -116,41 +122,3 @@ fn draw_indicator(display: &mut Esp32DisplayAdapter, screen: BoundingBox, connec
|
||||
let y = screen.y + screen.height - INDICATOR_DIAMETER - INDICATOR_MARGIN;
|
||||
display.fill_circle(x, y, INDICATOR_DIAMETER, color).unwrap();
|
||||
}
|
||||
|
||||
fn update_cache(engine: &RenderEngine, cmd: &RepaintCommand) -> WidgetCache {
|
||||
let hint: DisplayHint = cmd.display_hint.clone().into();
|
||||
let data: Vec<(String, Value)> = cmd.state.data
|
||||
.iter()
|
||||
.map(|kv| (kv.key.clone(), kv.value.clone().into()))
|
||||
.collect();
|
||||
let error: Option<WidgetError> = cmd.state.error.as_ref().map(|e| e.clone().into());
|
||||
|
||||
let content_h = engine.content_height(&hint, &data, cmd.bounds.width, error.as_ref());
|
||||
let scroll = ScrollState::new(cmd.bounds.height, content_h);
|
||||
|
||||
WidgetCache {
|
||||
hint,
|
||||
data,
|
||||
error,
|
||||
bounds: cmd.bounds,
|
||||
scroll,
|
||||
}
|
||||
}
|
||||
|
||||
fn draw_widget(
|
||||
engine: &RenderEngine,
|
||||
display: &mut Esp32DisplayAdapter,
|
||||
cache: &WidgetCache,
|
||||
) {
|
||||
let draw_cmds = engine.render_widget(
|
||||
&cache.hint,
|
||||
&cache.data,
|
||||
cache.bounds,
|
||||
cache.scroll.offset(),
|
||||
cache.error.as_ref(),
|
||||
);
|
||||
|
||||
for dc in &draw_cmds {
|
||||
display.draw_text_span(&dc.text, dc.x, dc.y, dc.color, dc.font).unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user