an-anime-game-launcher/src/main.rs
2024-03-24 11:23:15 +02:00

260 lines
8.5 KiB
Rust

use std::path::PathBuf;
use std::sync::atomic::{AtomicBool, Ordering};
use relm4::prelude::*;
use anime_launcher_sdk::config::ConfigExt;
use anime_launcher_sdk::genshin::config::{Config, Schema};
use anime_launcher_sdk::genshin::states::LauncherState;
use anime_launcher_sdk::genshin::consts::*;
use anime_launcher_sdk::anime_game_core::prelude::*;
use anime_launcher_sdk::anime_game_core::genshin::prelude::*;
use anime_launcher_sdk::sessions::SessionsExt;
use anime_launcher_sdk::genshin::sessions::Sessions;
use tracing_subscriber::prelude::*;
use tracing_subscriber::filter::*;
pub mod move_files;
pub mod i18n;
pub mod background;
pub mod ui;
use ui::main::*;
use ui::first_run::main::*;
pub const APP_ID: &str = "moe.launcher.an-anime-game-launcher";
pub const APP_RESOURCE_PATH: &str = "/moe/launcher/an-anime-game-launcher";
pub const APP_VERSION: &str = env!("CARGO_PKG_VERSION");
pub const APP_DEBUG: bool = cfg!(debug_assertions);
/// Sets to `true` when the `App` component is ready (fully initialized)
pub static READY: AtomicBool = AtomicBool::new(false);
// TODO: get rid of using this function in all the components' events
// e.g. by converting preferences pages into Relm4 Components
/// Check if the app is ready
pub fn is_ready() -> bool {
READY.load(Ordering::Relaxed)
}
lazy_static::lazy_static! {
/// Config loaded on the app's start. Use `Config::get()` to get up to date config instead.
/// This one is used to prepare some launcher UI components on start
pub static ref CONFIG: Schema = Config::get().expect("Failed to load config");
pub static ref GAME: Game = Game::new(CONFIG.game.path.for_edition(CONFIG.launcher.edition), CONFIG.launcher.edition);
/// Path to launcher folder. Standard is `$HOME/.local/share/anime-game-launcher`
pub static ref LAUNCHER_FOLDER: PathBuf = launcher_dir().expect("Failed to get launcher folder");
/// Path to launcher's cache folder. Standard is `$HOME/.cache/anime-game-launcher`
pub static ref CACHE_FOLDER: PathBuf = cache_dir().expect("Failed to get launcher's cache folder");
/// Path to `debug.log` file. Standard is `$HOME/.local/share/anime-game-launcher/debug.log`
pub static ref DEBUG_FILE: PathBuf = LAUNCHER_FOLDER.join("debug.log");
/// Path to `background` file. Standard is `$HOME/.local/share/anime-game-launcher/background`
pub static ref BACKGROUND_FILE: PathBuf = LAUNCHER_FOLDER.join("background");
/// Path to `.keep-background` file. Used to mark launcher that it shouldn't update background picture
///
/// Standard is `$HOME/.local/share/anime-game-launcher/.keep-background`
pub static ref KEEP_BACKGROUND_FILE: PathBuf = LAUNCHER_FOLDER.join(".keep-background");
/// Path to `.first-run` file. Used to mark launcher that it should run FirstRun window
///
/// Standard is `$HOME/.local/share/anime-game-launcher/.first-run`
pub static ref FIRST_RUN_FILE: PathBuf = LAUNCHER_FOLDER.join(".first-run");
/// Global app's css
static ref GLOBAL_CSS: String = format!("
progressbar > text {{
margin-bottom: 4px;
}}
window.classic-style {{
background: url(\"file://{}\");
background-repeat: no-repeat;
background-size: cover;
}}
window.classic-style progressbar {{
background-color: #00000020;
border-radius: 16px;
padding: 8px 16px;
}}
window.classic-style progressbar:hover {{
background-color: #00000060;
color: #ffffff;
transition-duration: 0.5s;
transition-timing-function: linear;
}}
.round-bin {{
border-radius: 24px;
}}
", BACKGROUND_FILE.to_string_lossy());
}
fn main() -> anyhow::Result<()> {
// Setup custom panic handler
human_panic::setup_panic!(human_panic::metadata!());
// Create launcher folder if it isn't
if !LAUNCHER_FOLDER.exists() {
std::fs::create_dir_all(LAUNCHER_FOLDER.as_path()).expect("Failed to create launcher folder");
// This one is kinda critical buy well, I can't do something with it
std::fs::write(FIRST_RUN_FILE.as_path(), "").expect("Failed to create .first-run file");
// Set initial launcher language based on system language
// CONFIG is initialized lazily so it will contain following changes as well
let mut config = Config::get().expect("Failed to get config");
config.launcher.language = i18n::format_lang(&i18n::get_default_lang());
Config::update_raw(config).expect("Failed to update config");
}
// Force debug output
let mut force_debug = false;
// Run the game
let mut run_game = false;
// Force run the game
let mut just_run_game = false;
// Force disable verbose tracing output in stdout
let mut no_verbose_tracing = false;
let args = std::env::args().collect::<Vec<_>>();
let mut gtk_args = Vec::new();
// Parse arguments
for i in 0..args.len() {
match args[i].as_str() {
"--debug" => force_debug = true,
"--run-game" => run_game = true,
"--just-run-game" => just_run_game = true,
"--no-verbose-tracing" => no_verbose_tracing = true,
"--session" => {
// Switch active session prior running the app
if let Some(session) = args.get(i + 1) {
Sessions::set_current(session.to_owned())?;
}
},
arg => gtk_args.push(arg.to_string())
}
}
// Prepare stdout logger
let stdout = tracing_subscriber::fmt::layer()
.pretty()
.with_filter({
if APP_DEBUG || force_debug {
LevelFilter::TRACE
} else {
LevelFilter::WARN
}
})
.with_filter(filter_fn(move |metadata| {
!metadata.target().contains("rustls") && !no_verbose_tracing
}));
// Prepare debug file logger
let file = std::fs::File::create(DEBUG_FILE.as_path())?;
let debug_log = tracing_subscriber::fmt::layer()
.pretty()
.with_ansi(false)
.with_writer(std::sync::Arc::new(file))
.with_filter(filter_fn(|metadata| {
!metadata.target().contains("rustls")
}));
tracing_subscriber::registry()
.with(stdout)
.with(debug_log)
.init();
tracing::info!("Starting application ({APP_VERSION})");
adw::init().expect("Libadwaita initialization failed");
// Register and include resources
gtk::gio::resources_register_include!("resources.gresource")
.expect("Failed to register resources");
// Set icons search path
gtk::IconTheme::for_display(&gtk::gdk::Display::default().unwrap())
.add_resource_path(&format!("{APP_RESOURCE_PATH}/icons"));
// Set application's title
gtk::glib::set_application_name("An Anime Game Launcher");
gtk::glib::set_program_name(Some("An Anime Game Launcher"));
// Set UI language
let lang = CONFIG.launcher.language.parse().expect("Wrong language format used in config");
i18n::set_lang(lang).expect("Failed to set launcher language");
tracing::info!("Set UI language to {}", i18n::get_lang());
// Run FirstRun window if .first-run file persist
if FIRST_RUN_FILE.exists() {
// Create the app
let app = RelmApp::new(APP_ID)
.with_args(gtk_args);
// Set global css
app.set_global_css(&GLOBAL_CSS);
// Show first run window
app.run::<FirstRunApp>(());
}
// Run the app if everything's ready
else {
if run_game || just_run_game {
let state = LauncherState::get_from_config(|_| {})
.expect("Failed to get launcher state");
match state {
LauncherState::Launch => {
anime_launcher_sdk::genshin::game::run().expect("Failed to run the game");
return Ok(());
}
LauncherState::PredownloadAvailable { .. } if just_run_game => {
anime_launcher_sdk::genshin::game::run().expect("Failed to run the game");
return Ok(());
}
_ => ()
}
}
// Create the app
let app = RelmApp::new(APP_ID)
.with_args(gtk_args);
// Set global css
app.set_global_css(&GLOBAL_CSS);
// Show main window
app.run::<App>(());
}
Ok(())
}