//TODO: a lot of app state refresh depends on this so there needs // to be an extra AtomicBool for checking if notifications are enabled use notify_rust::{Hint, Notification, NotificationHandle}; use rog_aura::AuraEffect; use rog_dbus::{ zbus_anime::AnimeProxy, zbus_charge::ChargeProxy, zbus_led::LedProxy, zbus_profile::ProfileProxy, zbus_rogbios::RogBiosProxy, }; use rog_profiles::Profile; use smol::{future, Executor}; use std::{ error::Error, sync::{ atomic::{AtomicBool, Ordering}, Arc, Mutex, }, thread::spawn, }; use zbus::export::futures_util::StreamExt; const NOTIF_HEADER: &str = "ROG Control"; macro_rules! notify { ($notifier:ident, $last_notif:ident, $data:expr) => { if let Some(notif) = $last_notif.take() { notif.close(); } if let Ok(x) = $notifier($data) { $last_notif.replace(x); } }; } macro_rules! base_notification { ($body:expr) => { Notification::new() .summary(NOTIF_HEADER) .body($body) .timeout(2000) .show() }; } type SharedHandle = Arc>>; pub fn start_notifications( charge_notified: Arc, bios_notified: Arc, aura_notified: Arc, anime_notified: Arc, profiles_notified: Arc, _fans_notified: Arc, notifs_enabled: Arc, ) -> Result<(), Box> { let last_notification: SharedHandle = Arc::new(Mutex::new(None)); let executor = Executor::new(); // BIOS notif let last_notif = last_notification.clone(); let notifs_enabled1 = notifs_enabled.clone(); let bios_notified1 = bios_notified.clone(); // TODO: make a macro or generic function or something... executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = RogBiosProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_notify_post_boot_sound().await { p.for_each(|e| { if let Ok(out) = e.args() { if notifs_enabled1.load(Ordering::SeqCst) { if let Ok(ref mut lock) = last_notif.try_lock() { notify!(do_post_sound_notif, lock, &out.sound()); } } bios_notified1.store(true, Ordering::SeqCst); } future::ready(()) }) .await; }; }) .detach(); executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = RogBiosProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_notify_panel_overdrive().await { p.for_each(|_| { bios_notified.store(true, Ordering::SeqCst); future::ready(()) }) .await; }; }) .detach(); // Charge notif let last_notif = last_notification.clone(); let notifs_enabled1 = notifs_enabled.clone(); executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = ChargeProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_notify_charge().await { p.for_each(|e| { if let Ok(out) = e.args() { if notifs_enabled1.load(Ordering::SeqCst) { if let Ok(ref mut lock) = last_notif.try_lock() { notify!(do_charge_notif, lock, &out.limit); } } charge_notified.store(true, Ordering::SeqCst); } future::ready(()) }) .await; }; }) .detach(); // Profile notif let last_notif = last_notification.clone(); let notifs_enabled1 = notifs_enabled.clone(); executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = ProfileProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_notify_profile().await { p.for_each(|e| { if let Ok(out) = e.args() { if notifs_enabled1.load(Ordering::SeqCst) { if let Ok(ref mut lock) = last_notif.try_lock() { notify!(do_thermal_notif, lock, &out.profile); } } profiles_notified.store(true, Ordering::SeqCst); } future::ready(()) }) .await; }; }) .detach(); // LED notif let last_notif = last_notification.clone(); let aura_notif = aura_notified.clone(); let notifs_enabled1 = notifs_enabled.clone(); executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = LedProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_notify_led().await { p.for_each(|e| { if let Ok(out) = e.args() { if notifs_enabled1.load(Ordering::SeqCst) { if let Ok(ref mut lock) = last_notif.try_lock() { notify!(do_led_notif, lock, &out.data); } } aura_notif.store(true, Ordering::SeqCst); } future::ready(()) }) .await; }; }) .detach(); let aura_notif = aura_notified.clone(); executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = LedProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_notify_led().await { p.for_each(|_| { aura_notif.store(true, Ordering::SeqCst); future::ready(()) }) .await; }; }) .detach(); executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = LedProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_all_signals().await { p.for_each(|_| { aura_notified.store(true, Ordering::SeqCst); future::ready(()) }) .await; }; }) .detach(); executor .spawn(async move { let conn = zbus::Connection::system().await.unwrap(); let proxy = AnimeProxy::new(&conn).await.unwrap(); if let Ok(p) = proxy.receive_power_states().await { p.for_each(|_| { anime_notified.store(true, Ordering::SeqCst); future::ready(()) }) .await; }; }) .detach(); spawn(move || loop { smol::block_on(executor.tick()); }); Ok(()) } fn do_thermal_notif(profile: &Profile) -> Result> { let icon = match profile { Profile::Balanced => "asus_notif_yellow", Profile::Performance => "asus_notif_red", Profile::Quiet => "asus_notif_green", }; let profile: &str = (*profile).into(); let x = Notification::new() .summary("ASUS ROG") .body(&format!( "Thermal profile changed to {}", profile.to_uppercase(), )) .hint(Hint::Resident(true)) .timeout(2000) .hint(Hint::Category("device".into())) //.hint(Hint::Transient(true)) .icon(icon) .show()?; Ok(x) } fn do_led_notif(ledmode: &AuraEffect) -> Result { base_notification!(&format!( "Keyboard LED mode changed to {}", ledmode.mode_name() )) } fn do_charge_notif(limit: &u8) -> Result { base_notification!(&format!("Battery charge limit changed to {}", limit)) } fn do_post_sound_notif(on: &bool) -> Result { base_notification!(&format!("BIOS Post sound {}", on)) }