fractal/src/application.rs

281 lines
8.5 KiB
Rust

use std::{borrow::Cow, fmt};
use gettextrs::gettext;
use gio::{ApplicationFlags, Settings};
use gtk::{gio, glib, glib::clone, prelude::*, subclass::prelude::*};
use ruma::{OwnedRoomId, RoomId};
use tracing::{debug, info};
use crate::{config, session_list::SessionList, spawn, system_settings::SystemSettings, Window};
mod imp {
use adw::subclass::prelude::AdwApplicationImpl;
use super::*;
#[derive(Debug)]
pub struct Application {
/// The application settings.
pub settings: Settings,
/// The system settings.
pub system_settings: SystemSettings,
/// The list of logged-in sessions.
pub session_list: SessionList,
}
impl Default for Application {
fn default() -> Self {
Self {
settings: Settings::new(config::APP_ID),
system_settings: Default::default(),
session_list: Default::default(),
}
}
}
#[glib::object_subclass]
impl ObjectSubclass for Application {
const NAME: &'static str = "Application";
type Type = super::Application;
type ParentType = adw::Application;
}
impl ObjectImpl for Application {
fn constructed(&self) {
self.parent_constructed();
let app = self.obj();
app.set_up_gactions();
app.set_up_accels();
self.session_list
.connect_error_notify(clone!(@weak app => move |session_list| {
if let Some(message) = session_list.error() {
let window = app.present_main_window();
window.show_secret_error(&message);
}
}));
spawn!(
clone!(@weak self.session_list as session_list => async move {
session_list.restore_sessions().await;
})
);
}
}
impl ApplicationImpl for Application {
fn activate(&self) {
debug!("Application::activate");
self.obj().present_main_window();
}
fn startup(&self) {
self.parent_startup();
}
}
impl GtkApplicationImpl for Application {}
impl AdwApplicationImpl for Application {}
}
glib::wrapper! {
pub struct Application(ObjectSubclass<imp::Application>)
@extends gio::Application, gtk::Application, adw::Application, @implements gio::ActionMap, gio::ActionGroup;
}
impl Application {
pub fn new() -> Self {
glib::Object::builder()
.property("application-id", Some(config::APP_ID))
.property("flags", ApplicationFlags::default())
.property("resource-base-path", Some("/org/gnome/Fractal/"))
.build()
}
/// Get or create the main window and make sure it is visible.
///
/// Returns the main window.
fn present_main_window(&self) -> Window {
let window = if let Some(window) = self.active_window().and_downcast() {
window
} else {
Window::new(self)
};
window.present();
window
}
/// The application settings.
pub fn settings(&self) -> Settings {
self.imp().settings.clone()
}
/// The system settings.
pub fn system_settings(&self) -> SystemSettings {
self.imp().system_settings.clone()
}
/// The list of logged-in sessions.
pub fn session_list(&self) -> &SessionList {
&self.imp().session_list
}
/// Set up the application actions.
fn set_up_gactions(&self) {
self.add_action_entries([
// Quit
gio::ActionEntry::builder("quit")
.activate(|app: &Application, _, _| {
if let Some(window) = app.active_window() {
// This is needed to trigger the delete event
// and saving the window state
window.close();
}
app.quit();
})
.build(),
// About
gio::ActionEntry::builder("about")
.activate(|app: &Application, _, _| {
app.show_about_dialog();
})
.build(),
// Show a room for a session. This is the action triggered when clicking a
// notification.
gio::ActionEntry::builder("show-room")
.parameter_type(Some(&AppShowRoomPayload::static_variant_type()))
.activate(|app: &Application, _, v| {
if let Some(payload) = v.and_then(|v| v.get::<AppShowRoomPayload>()) {
app.present_main_window()
.show_room(&payload.session_id, &payload.room_id);
}
})
.build(),
]);
}
/// Sets up keyboard shortcuts for application and window actions.
fn set_up_accels(&self) {
self.set_accels_for_action("app.quit", &["<Control>q"]);
self.set_accels_for_action("win.show-help-overlay", &["<Control>question"]);
}
fn show_about_dialog(&self) {
let dialog = adw::AboutWindow::builder()
.application_name("Fractal")
.application_icon(config::APP_ID)
.developer_name(gettext("The Fractal Team"))
.license_type(gtk::License::Gpl30)
.website("https://gitlab.gnome.org/GNOME/fractal/")
.issue_url("https://gitlab.gnome.org/GNOME/fractal/-/issues")
.support_url("https://matrix.to/#/#fractal:gnome.org")
.version(config::VERSION)
.modal(true)
.copyright(gettext("© 2017-2023 The Fractal Team"))
.developers(vec![
"Alejandro Domínguez".to_string(),
"Alexandre Franke".to_string(),
"Bilal Elmoussaoui".to_string(),
"Christopher Davis".to_string(),
"Daniel García Moreno".to_string(),
"Eisha Chen-yen-su".to_string(),
"Jordan Petridis".to_string(),
"Julian Sparber".to_string(),
"Kévin Commaille".to_string(),
"Saurav Sachidanand".to_string(),
])
.designers(vec!["Tobias Bernard".to_string()])
.translator_credits(gettext("translator-credits"))
.build();
dialog.set_transient_for(self.active_window().as_ref());
// This can't be added via the builder
dialog.add_credit_section(Some(&gettext("Name by")), &["Regina Bíró"]);
dialog.present();
}
pub fn run(&self) {
info!("Fractal ({})", config::APP_ID);
info!("Version: {} ({})", config::VERSION, config::PROFILE);
info!("Datadir: {}", config::PKGDATADIR);
ApplicationExtManual::run(self);
}
}
impl Default for Application {
fn default() -> Self {
gio::Application::default()
.and_downcast::<Application>()
.unwrap()
}
}
/// The profile that was built.
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
#[allow(dead_code)]
pub enum AppProfile {
/// A stable release.
Stable,
/// A beta release.
Beta,
/// A development release.
Devel,
}
impl AppProfile {
/// The string representation of this `AppProfile`.
pub fn as_str(&self) -> &str {
match self {
Self::Stable => "stable",
Self::Beta => "beta",
Self::Devel => "devel",
}
}
/// Whether this `AppProfile` should use the `.devel` CSS class on windows.
pub fn should_use_devel_class(&self) -> bool {
matches!(self, Self::Devel)
}
}
impl fmt::Display for AppProfile {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_str(self.as_str())
}
}
#[derive(Debug, Clone)]
pub struct AppShowRoomPayload {
pub session_id: String,
pub room_id: OwnedRoomId,
}
impl glib::StaticVariantType for AppShowRoomPayload {
fn static_variant_type() -> Cow<'static, glib::VariantTy> {
<(String, String)>::static_variant_type()
}
}
impl glib::ToVariant for AppShowRoomPayload {
fn to_variant(&self) -> glib::Variant {
(&self.session_id, self.room_id.as_str()).to_variant()
}
}
impl glib::FromVariant for AppShowRoomPayload {
fn from_variant(variant: &glib::Variant) -> Option<Self> {
let (session_id, room_id) = variant.get::<(String, String)>()?;
let room_id = RoomId::parse(room_id).ok()?;
Some(Self {
session_id,
room_id,
})
}
}