mirror of
https://gitlab.com/asus-linux/asusctl.git
synced 2026-02-06 00:15:04 +01:00
218 lines
6.7 KiB
Rust
218 lines
6.7 KiB
Rust
use eframe::{IconData, NativeOptions};
|
|
use log::{error, info, LevelFilter};
|
|
use rog_aura::layouts::KeyLayout;
|
|
use rog_control_center::notify::EnabledNotifications;
|
|
use rog_control_center::tray::init_tray;
|
|
use rog_control_center::{
|
|
config::Config, error::Result, get_ipc_file, notify::start_notifications, on_tmp_dir_exists,
|
|
print_versions, startup_error::AppErrorShow, system_state::SystemState, RogApp,
|
|
RogDbusClientBlocking, SHOWING_GUI, SHOW_GUI,
|
|
};
|
|
use rog_platform::supported::SupportedFunctions;
|
|
use std::sync::Mutex;
|
|
use std::{
|
|
fs::OpenOptions,
|
|
io::{Read, Write},
|
|
path::PathBuf,
|
|
sync::Arc,
|
|
};
|
|
use tokio::runtime::Runtime;
|
|
|
|
#[cfg(not(feature = "mocking"))]
|
|
const DATA_DIR: &str = "/usr/share/rog-gui/";
|
|
#[cfg(feature = "mocking")]
|
|
const DATA_DIR: &str = env!("CARGO_MANIFEST_DIR");
|
|
const BOARD_NAME: &str = "/sys/class/dmi/id/board_name";
|
|
const APP_ICON_PATH: &str = "/usr/share/icons/hicolor/512x512/apps/rog-control-center.png";
|
|
|
|
fn main() -> Result<()> {
|
|
print_versions();
|
|
let mut logger = env_logger::Builder::new();
|
|
logger
|
|
.target(env_logger::Target::Stdout)
|
|
.format(|buf, record| writeln!(buf, "{}: {}", record.level(), record.args()))
|
|
.filter(None, LevelFilter::Info)
|
|
.init();
|
|
|
|
// start tokio
|
|
let rt = Runtime::new().expect("Unable to create Runtime");
|
|
// Enter the runtime so that `tokio::spawn` is available immediately.
|
|
let _enter = rt.enter();
|
|
|
|
let native_options = eframe::NativeOptions {
|
|
vsync: true,
|
|
decorated: true,
|
|
transparent: false,
|
|
min_window_size: Some(egui::vec2(840.0, 600.0)),
|
|
max_window_size: Some(egui::vec2(840.0, 600.0)),
|
|
run_and_return: true,
|
|
icon_data: Some(load_icon()),
|
|
..Default::default()
|
|
};
|
|
|
|
let (dbus, _) = RogDbusClientBlocking::new()
|
|
.map_err(|e| {
|
|
eframe::run_native(
|
|
"ROG Control Center",
|
|
native_options.clone(),
|
|
Box::new(move |_| Box::new(AppErrorShow::new(e.to_string()))),
|
|
);
|
|
})
|
|
.unwrap();
|
|
|
|
let supported = match dbus.proxies().supported().supported_functions() {
|
|
Ok(s) => s,
|
|
Err(e) => {
|
|
eframe::run_native(
|
|
"ROG Control Center",
|
|
native_options.clone(),
|
|
Box::new(move |_| Box::new(AppErrorShow::new(e.to_string()))),
|
|
);
|
|
SupportedFunctions::default()
|
|
}
|
|
};
|
|
|
|
// Startup
|
|
let mut config = Config::load()?;
|
|
let mut start_closed = config.startup_in_background;
|
|
|
|
if config.startup_in_background {
|
|
config.run_in_background = true;
|
|
let tmp = config.enabled_notifications.clone(); // ends up being a double clone, oh well.
|
|
config.save(&tmp)?;
|
|
}
|
|
let enabled_notifications = EnabledNotifications::tokio_mutex(&config);
|
|
|
|
// Find and load a matching layout for laptop
|
|
let mut file = OpenOptions::new()
|
|
.read(true)
|
|
.open(PathBuf::from(BOARD_NAME))
|
|
.map_err(|e| {
|
|
println!("{BOARD_NAME}, {e}");
|
|
e
|
|
})?;
|
|
let mut board_name = String::new();
|
|
file.read_to_string(&mut board_name)?;
|
|
|
|
#[cfg(feature = "mocking")]
|
|
{
|
|
board_name = "gl504".to_string();
|
|
path.pop();
|
|
path.push("rog-aura");
|
|
path.push("data");
|
|
}
|
|
|
|
let layout = KeyLayout::find_layout(board_name.as_str(), PathBuf::from(DATA_DIR))
|
|
.map_err(|e| {
|
|
println!("{BOARD_NAME}, {e}");
|
|
})
|
|
.unwrap_or_else(|_| KeyLayout::ga401_layout());
|
|
|
|
// tmp-dir must live to the end of program life
|
|
let _tmp_dir = match tempfile::Builder::new()
|
|
.prefix("rog-gui")
|
|
.rand_bytes(0)
|
|
.tempdir()
|
|
{
|
|
Ok(tmp) => tmp,
|
|
Err(_) => on_tmp_dir_exists().unwrap(),
|
|
};
|
|
|
|
let states = setup_page_state_and_notifs(layout, enabled_notifications, &supported)?;
|
|
|
|
init_tray(supported, states.clone());
|
|
|
|
loop {
|
|
if !start_closed {
|
|
start_app(states.clone(), native_options.clone())?;
|
|
}
|
|
|
|
let config = Config::load()?;
|
|
if !config.run_in_background {
|
|
break;
|
|
}
|
|
|
|
if config.run_in_background {
|
|
let mut buf = [0u8; 4];
|
|
// blocks until it is read, typically the read will happen after a second
|
|
// process writes to the IPC (so there is data to actually read)
|
|
if get_ipc_file()?.read(&mut buf).is_ok() && buf[0] == SHOW_GUI {
|
|
start_closed = false;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
// loop {
|
|
// // This is just a blocker to idle and ensure the reator reacts
|
|
// sleep(Duration::from_millis(1000)).await;
|
|
// }
|
|
Ok(())
|
|
}
|
|
|
|
fn setup_page_state_and_notifs(
|
|
keyboard_layout: KeyLayout,
|
|
enabled_notifications: Arc<Mutex<EnabledNotifications>>,
|
|
supported: &SupportedFunctions,
|
|
) -> Result<Arc<Mutex<SystemState>>> {
|
|
let page_states = Arc::new(Mutex::new(SystemState::new(
|
|
keyboard_layout,
|
|
enabled_notifications.clone(),
|
|
supported,
|
|
)?));
|
|
|
|
start_notifications(page_states.clone(), enabled_notifications)?;
|
|
|
|
Ok(page_states)
|
|
}
|
|
|
|
fn start_app(states: Arc<Mutex<SystemState>>, native_options: NativeOptions) -> Result<()> {
|
|
let mut ipc_file = get_ipc_file()?;
|
|
ipc_file.write_all(&[SHOWING_GUI])?;
|
|
eframe::run_native(
|
|
"ROG Control Center",
|
|
native_options,
|
|
Box::new(move |cc| Box::new(RogApp::new(Config::load().unwrap(), states, cc).unwrap())),
|
|
);
|
|
Ok(())
|
|
}
|
|
|
|
/// Bah.. the icon dosn't work on wayland anyway, but we'll leave it in for now.
|
|
fn load_icon() -> IconData {
|
|
let path = PathBuf::from(APP_ICON_PATH);
|
|
let mut buf = Vec::new();
|
|
let mut rgba = Vec::new();
|
|
let mut height = 512;
|
|
let mut width = 512;
|
|
if path.exists() {
|
|
if let Ok(mut file) = OpenOptions::new()
|
|
.read(true)
|
|
.open(path)
|
|
.map_err(|e| error!("Error opening app icon: {e:?}"))
|
|
{
|
|
file.read_to_end(&mut buf)
|
|
.map_err(|e| error!("Error reading app icon: {e:?}"))
|
|
.ok();
|
|
|
|
let data = std::io::Cursor::new(buf);
|
|
let decoder = png_pong::Decoder::new(data).unwrap().into_steps();
|
|
let png_pong::Step { raster, delay: _ } = decoder.last().unwrap().unwrap();
|
|
|
|
if let png_pong::PngRaster::Rgba8(ras) = raster {
|
|
rgba = ras.as_u8_slice().to_vec();
|
|
width = ras.width();
|
|
height = ras.height();
|
|
info!("Loaded app icon. Not actually supported in Wayland yet");
|
|
}
|
|
}
|
|
} else {
|
|
error!("Missing {APP_ICON_PATH}");
|
|
}
|
|
|
|
IconData {
|
|
height,
|
|
width,
|
|
rgba,
|
|
}
|
|
}
|