233 lines
8.4 KiB
Rust
233 lines
8.4 KiB
Rust
use cosmic_comp_config::output::randr;
|
|
use cosmic_config::CosmicConfigEntry;
|
|
use kdl::KdlDocument;
|
|
use std::{
|
|
collections::BTreeMap,
|
|
fs,
|
|
path::{Path, PathBuf},
|
|
};
|
|
|
|
pub use cosmic_applets_config::time::TimeAppletConfig;
|
|
pub use cosmic_bg_config::{Color, Source as BgSource, state::State as BgState};
|
|
pub use cosmic_comp_config::{CosmicCompConfig, XkbConfig, ZoomConfig};
|
|
pub use cosmic_theme::{Theme, ThemeBuilder};
|
|
|
|
#[derive(Clone, Debug, Default, serde::Deserialize, serde::Serialize)]
|
|
pub struct UserData {
|
|
pub uid: u32,
|
|
pub name: String,
|
|
pub full_name: String,
|
|
pub icon_opt: Option<Vec<u8>>,
|
|
pub theme_opt: Option<Theme>,
|
|
pub theme_builder_opt: Option<ThemeBuilder>,
|
|
pub bg_state: BgState,
|
|
pub bg_path_data: BTreeMap<PathBuf, Vec<u8>>,
|
|
pub xkb_config_opt: Option<XkbConfig>,
|
|
pub time_applet_config: TimeAppletConfig,
|
|
pub accessibility_zoom: ZoomConfig,
|
|
pub kdl_output_lists: Vec<String>,
|
|
}
|
|
|
|
impl UserData {
|
|
pub fn load_wallpapers_as_user(&mut self) {
|
|
//TODO: reload changed background files?
|
|
self.bg_path_data.retain(|path, _| {
|
|
self.bg_state
|
|
.wallpapers
|
|
.iter()
|
|
.any(|(_, source)| match source {
|
|
BgSource::Path(source_path) => source_path == path,
|
|
_ => false,
|
|
})
|
|
});
|
|
for (_, source) in self.bg_state.wallpapers.iter() {
|
|
match source {
|
|
//TODO: do not reread duplicate paths, cache data by path?
|
|
BgSource::Path(path) => {
|
|
if !self.bg_path_data.contains_key(path) {
|
|
match fs::read(&path) {
|
|
Ok(bytes) => {
|
|
self.bg_path_data.insert(path.clone(), bytes);
|
|
}
|
|
Err(err) => {
|
|
tracing::error!("failed to read wallpaper {:?}: {:?}", path, err);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// Other types not supported
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn load_config_as_user(&mut self) {
|
|
self.icon_opt = None;
|
|
self.theme_opt = None;
|
|
self.theme_builder_opt = None;
|
|
self.bg_state = Default::default();
|
|
self.xkb_config_opt = None;
|
|
self.time_applet_config = Default::default();
|
|
|
|
//TODO: use accountsservice?
|
|
//IMPORTANT: This file is owned by root and safe to read (it won't be a link to /etc/shadow for example)
|
|
// It may not exist if the user uses one of the system icons. In that case, we should read the
|
|
// information in /var/lib/AccountsService/users, and then read the icon path as the user
|
|
let icon_path = Path::new("/var/lib/AccountsService/icons").join(&self.name);
|
|
if icon_path.is_file() {
|
|
match fs::read(&icon_path) {
|
|
Ok(icon_data) => {
|
|
self.icon_opt = Some(icon_data);
|
|
}
|
|
Err(err) => {
|
|
tracing::error!("failed to read icon {:?}: {:?}", icon_path, err);
|
|
}
|
|
}
|
|
}
|
|
|
|
let mut is_dark = true;
|
|
match cosmic_theme::ThemeMode::config() {
|
|
Ok(helper) => match cosmic_theme::ThemeMode::get_entry(&helper) {
|
|
Ok(theme_mode) => {
|
|
is_dark = theme_mode.is_dark;
|
|
}
|
|
Err((errs, theme_mode)) => {
|
|
tracing::error!("failed to load cosmic-theme config: {:?}", errs);
|
|
is_dark = theme_mode.is_dark;
|
|
}
|
|
},
|
|
Err(err) => {
|
|
tracing::error!("failed to create cosmic-theme mode helper: {:?}", err);
|
|
}
|
|
}
|
|
|
|
match if is_dark {
|
|
cosmic_theme::Theme::dark_config()
|
|
} else {
|
|
cosmic_theme::Theme::light_config()
|
|
} {
|
|
Ok(helper) => match cosmic_theme::Theme::get_entry(&helper) {
|
|
Ok(theme) => {
|
|
self.theme_opt = Some(theme);
|
|
}
|
|
Err((errs, theme)) => {
|
|
tracing::error!("failed to load cosmic-theme config: {:?}", errs);
|
|
self.theme_opt = Some(theme);
|
|
}
|
|
},
|
|
Err(err) => {
|
|
tracing::error!("failed to create cosmic-theme config helper: {:?}", err);
|
|
}
|
|
}
|
|
|
|
match if is_dark {
|
|
cosmic_theme::ThemeBuilder::dark_config()
|
|
} else {
|
|
cosmic_theme::ThemeBuilder::light_config()
|
|
} {
|
|
Ok(helper) => match cosmic_theme::ThemeBuilder::get_entry(&helper) {
|
|
Ok(theme) => {
|
|
self.theme_builder_opt = Some(theme);
|
|
}
|
|
Err((errs, theme)) => {
|
|
tracing::error!("failed to load cosmic-theme builder config: {:?}", errs);
|
|
self.theme_builder_opt = Some(theme);
|
|
}
|
|
},
|
|
Err(err) => {
|
|
tracing::error!(
|
|
"failed to create cosmic-theme builder config helper: {:?}",
|
|
err
|
|
);
|
|
}
|
|
}
|
|
|
|
//TODO: fallback to background config if background state is not set?
|
|
match cosmic_bg_config::state::State::state() {
|
|
Ok(helper) => match cosmic_bg_config::state::State::get_entry(&helper) {
|
|
Ok(state) => {
|
|
self.bg_state = state;
|
|
}
|
|
Err((errs, state)) => {
|
|
tracing::error!("failed to load cosmic-bg state: {:?}", errs);
|
|
self.bg_state = state;
|
|
}
|
|
},
|
|
Err(err) => {
|
|
tracing::error!("failed to create cosmic-bg state helper: {:?}", err);
|
|
}
|
|
}
|
|
self.load_wallpapers_as_user();
|
|
|
|
match cosmic_config::Config::new("com.system76.CosmicComp", CosmicCompConfig::VERSION) {
|
|
Ok(config_handler) => {
|
|
match CosmicCompConfig::get_entry(&config_handler) {
|
|
Ok(config) => {
|
|
self.xkb_config_opt = Some(config.xkb_config);
|
|
self.accessibility_zoom = config.accessibility_zoom;
|
|
}
|
|
Err((errs, config)) => {
|
|
tracing::error!("errors loading cosmic-comp config: {:?}", errs);
|
|
self.xkb_config_opt = Some(config.xkb_config);
|
|
self.accessibility_zoom = config.accessibility_zoom;
|
|
}
|
|
};
|
|
}
|
|
Err(err) => {
|
|
tracing::error!("failed to create cosmic-comp config handler: {}", err);
|
|
}
|
|
};
|
|
|
|
let xdg = xdg::BaseDirectories::new();
|
|
self.kdl_output_lists = xdg
|
|
.get_state_home()
|
|
.map(|mut s| {
|
|
s.push("cosmic-comp/outputs.ron");
|
|
let lists = randr::load_outputs(Some(&s));
|
|
lists
|
|
.into_iter()
|
|
.map(|l| KdlDocument::from(l).to_string())
|
|
.collect()
|
|
})
|
|
.unwrap_or_default();
|
|
|
|
match cosmic_config::Config::new("com.system76.CosmicAppletTime", TimeAppletConfig::VERSION)
|
|
{
|
|
Ok(config_handler) => match TimeAppletConfig::get_entry(&config_handler) {
|
|
Ok(config) => {
|
|
self.time_applet_config = config;
|
|
}
|
|
Err((errs, config)) => {
|
|
tracing::error!("failed to load time applet config: {:?}", errs);
|
|
self.time_applet_config = config;
|
|
}
|
|
},
|
|
Err(err) => {
|
|
tracing::error!(
|
|
"failed to create CosmicAppletTime config handler: {:?}",
|
|
err
|
|
);
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
impl From<pwd::Passwd> for UserData {
|
|
fn from(user: pwd::Passwd) -> Self {
|
|
let mut full_name = user
|
|
.gecos
|
|
.as_ref()
|
|
.and_then(|gecos| gecos.split(',').next())
|
|
.map(|x| x.to_string())
|
|
.unwrap_or_default();
|
|
if full_name.is_empty() {
|
|
full_name = user.name.clone();
|
|
}
|
|
Self {
|
|
uid: user.uid,
|
|
name: user.name.clone(),
|
|
full_name,
|
|
..Default::default()
|
|
}
|
|
}
|
|
}
|