events: implement property events and state management

This commit is contained in:
Victoria Fischer 2024-12-14 14:47:23 +01:00
parent da2a8f5bfc
commit f180171ee7

View File

@ -1,113 +1,24 @@
use core::fmt::Debug;
use core::fmt::{Debug, Display};
use std::{collections::{LinkedList, VecDeque}, sync::{Arc, Mutex}};
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum InputEvent {
PowerOn,
PowerOff,
NetworkActivity,
NetworkOnline,
NetworkOffline
}
use rgb::Rgb;
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum Variant {
SignedByte(i8),
Byte(u8),
UInt(u32),
Int(i32),
BigUInt(u64),
BigInt(i64),
Boolean(bool)
Boolean(bool),
String(String),
RGB(Rgb<u8>)
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum Event {
ReadyToRock,
Tick,
StartThing(&'static str),
StopThing(&'static str),
Input(InputEvent),
PropertyChange(&'static str, Variant)
}
pub struct SystemState {
key: &'static str,
value: Variant,
values: Vec::<Box<SystemState>>
}
impl SystemState {
pub fn new() -> Self {
SystemState {
key: "",
value: Variant::Byte(0),
values: Vec::new()
}
}
fn get_key(&self, key: &'static str) -> Option<&Self> {
if key == self.key {
Some(self)
} else {
for next in self.values.iter() {
match next.get_key(key) {
None => (),
Some(next_val) => return Some(next_val)
}
}
return None
}
}
fn get_key_mut(&mut self, key: &'static str) -> Option<&mut Self> {
if key == self.key {
Some(self)
} else {
for next in self.values.iter_mut() {
match next.get_key_mut(key) {
None => (),
Some(next_val) => return Some(next_val)
}
}
return None
}
}
pub fn get(&self, key: &'static str) -> Option<Variant> {
match self.get_key(key) {
None => None,
Some(v) => Some(v.value)
}
}
pub fn set<V>(&mut self, key: &'static str, value: V) where Variant: From<V> {
match self.get_key_mut(key) {
None => self.values.push(Box::new(SystemState {
value: value.into(),
key: key,
values: Vec::new()
})),
Some(found_key) => {
found_key.value = value.into()
}
}
}
}
impl Event {
pub fn new_tick() -> Self {
Event::Tick
}
pub fn new_property_change<T>(key: &'static str, data: T) -> Self where Variant: From<T> {
Event::PropertyChange(key, Variant::from(data))
}
pub fn new_ready_to_rock() -> Self {
Event::ReadyToRock
}
pub fn new_input_event(event: InputEvent) -> Self {
Event::Input(event)
impl From<u8> for Variant {
fn from(value: u8) -> Self {
Variant::Byte(value)
}
}
@ -120,6 +31,21 @@ impl Into<u8> for Variant {
}
}
impl From<u64> for Variant {
fn from(value: u64) -> Self {
Variant::BigUInt(value)
}
}
impl Into<u64> for Variant {
fn into(self) -> u64 {
match self {
Variant::BigUInt(b) => b,
_ => 0
}
}
}
impl From<bool> for Variant {
fn from(value: bool) -> Self {
Variant::Boolean(value)
@ -132,32 +58,197 @@ impl From<i64> for Variant {
}
}
impl From<u8> for Variant {
fn from(value: u8) -> Self {
Variant::Byte(value)
impl From<u32> for Variant {
fn from(value: u32) -> Self {
Variant::UInt(value)
}
}
impl From<i32> for Variant {
fn from(value: i32) -> Self {
Variant::Int(value)
}
}
impl From<String> for Variant {
fn from(value: String) -> Self {
Variant::String(value)
}
}
impl<'a> From<&'a str> for Variant {
fn from(value: &'a str) -> Self {
Variant::String(value.to_owned())
}
}
impl From<Rgb<u8>> for Variant {
fn from(value: Rgb<u8>) -> Self {
Variant::RGB(value)
}
}
impl From<i8> for Variant {
fn from(value: i8) -> Self {
Variant::SignedByte(value)
}
}
impl Display for Variant {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Variant::BigInt(i) => (i as &dyn Display).fmt(f),
Variant::BigUInt(i) => (i as &dyn Display).fmt(f),
Variant::Boolean(i) => (i as &dyn Display).fmt(f),
Variant::Byte(i) => (i as &dyn Display).fmt(f),
Variant::Int(i) => (i as &dyn Display).fmt(f),
Variant::String(i) => (i as &dyn Display).fmt(f),
Variant::UInt(i) => (i as &dyn Display).fmt(f),
Variant::SignedByte(b) => (b as &dyn Display).fmt(f),
Variant::RGB(rgb) => f.write_fmt(format_args!("[{}, {}, {}]", rgb.r, rgb.g, rgb.b)),
}
}
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum Event {
ReadyToRock,
Tick,
StartThing(&'static str),
StopThing(&'static str),
PropertyChange(&'static str, Variant)
}
impl Event {
pub const fn new_tick() -> Self {
Event::Tick
}
pub fn new_property_change<T>(key: &'static str, data: T) -> Self where Variant: From<T> {
Event::PropertyChange(key, Variant::from(data))
}
pub const fn new_ready_to_rock() -> Self {
Event::ReadyToRock
}
pub const fn new_start_thing(name: &'static str) -> Self {
Event::StartThing(name)
}
pub const fn new_stop_thing(name: &'static str) -> Self {
Event::StopThing(name)
}
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct Property {
key: &'static str,
value: Variant
}
impl Property {
pub const fn new(key: &'static str, value: Variant) -> Self {
Property {
key,
value: value
}
}
}
#[derive(Debug, Clone)]
pub struct Properties {
contents: LinkedList<Property>
}
impl Properties {
fn new() -> Self {
Properties {
contents: LinkedList::new()
}
}
fn get_key(&self, key: &'static str) -> Option<&Property> {
for next in self.contents.iter() {
if next.key == key {
return Some(next);
}
}
return None;
}
fn get_key_mut(&mut self, key: &'static str) -> Option<&mut Property> {
for next in self.contents.iter_mut() {
if next.key == key {
return Some(next);
}
}
return None;
}
pub fn get(&self, key: &'static str) -> Option<Variant> {
match self.get_key(key) {
None => None,
Some(v) => Some(v.value.clone())
}
}
pub fn set<V>(&mut self, key: &'static str, value: V) -> bool where Variant: From<V> {
let as_variant: Variant = value.into();
match self.get_key_mut(key) {
None => {
self.contents.push_front(Property::new(key, as_variant));
return true
},
Some(found_key) => {
if found_key.value != as_variant {
found_key.value = as_variant;
return true
}
}
}
return false
}
}
#[derive(Clone)]
pub struct EventBus {
pending: Vec<Event>
pending: Arc<Mutex<VecDeque<Event>>>,
props: Arc<Mutex<Properties>>
}
impl EventBus {
pub fn new() -> Self {
EventBus {
pending: Vec::new()
pending: Arc::new(Mutex::new(VecDeque::with_capacity(32))),
props: Arc::new(Mutex::new(Properties::new()))
}
}
pub fn next(&mut self) -> Event {
if self.pending.len() == 0 {
Event::new_tick()
} else {
self.pending.pop().unwrap()
let next = self.pending.lock().unwrap().pop_front().unwrap_or(Event::new_tick());
match next {
Event::PropertyChange(key, value) => {
if self.props.lock().unwrap().set(key, value.clone()) {
log::trace!("prop-update key={} value={:?}", key, value);
Event::PropertyChange(key, value)
} else {
Event::new_tick()
}
},
_ => next
}
}
pub fn push(&mut self, event: Event) {
self.pending.push(event);
self.pending.lock().unwrap().push_back(event);
}
pub fn property(&self, key: &'static str) -> Option<Variant> {
self.props.lock().unwrap().get(key)
}
}