-
-
Notifications
You must be signed in to change notification settings - Fork 42
Expand file tree
/
Copy pathconfig.rs
More file actions
146 lines (137 loc) · 5.23 KB
/
config.rs
File metadata and controls
146 lines (137 loc) · 5.23 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
use std::{
fs::read_to_string,
os::fd::AsFd
};
use anyhow::Error;
use cairo::FontFace;
use crate::FunctionLayer;
use crate::fonts::{FontConfig, Pattern};
use freetype::Library as FtLibrary;
use input_linux::Key;
use nix::{
errno::Errno,
sys::inotify::{AddWatchFlags, InitFlags, Inotify, WatchDescriptor}
};
use serde::Deserialize;
const USER_CFG_PATH: &'static str = "/etc/tiny-dfr/config.toml";
pub struct Config {
pub show_button_outlines: bool,
pub enable_pixel_shift: bool,
pub font_face: FontFace,
pub adaptive_brightness: bool,
pub active_brightness: u32,
pub fn_toggle_layers: bool,
}
#[derive(Deserialize)]
#[serde(rename_all = "PascalCase")]
struct ConfigProxy {
media_layer_default: Option<bool>,
show_button_outlines: Option<bool>,
enable_pixel_shift: Option<bool>,
font_template: Option<String>,
adaptive_brightness: Option<bool>,
active_brightness: Option<u32>,
fn_toggle_layers: Option<bool>,
primary_layer_keys: Option<Vec<ButtonConfig>>,
media_layer_keys: Option<Vec<ButtonConfig>>
}
#[derive(Deserialize)]
#[serde(rename_all = "PascalCase")]
pub struct ButtonConfig {
#[serde(alias = "Svg")]
pub icon: Option<String>,
pub text: Option<String>,
pub action: Key
}
fn load_font(name: &str) -> FontFace {
let fontconfig = FontConfig::new();
let mut pattern = Pattern::new(name);
fontconfig.perform_substitutions(&mut pattern);
let pat_match = match fontconfig.match_pattern(&pattern) {
Ok(pat) => pat,
Err(_) => panic!("Unable to find specified font. If you are using the default config, make sure you have at least one font installed")
};
let file_name = pat_match.get_file_name();
let file_idx = pat_match.get_font_index();
let ft_library = FtLibrary::init().unwrap();
let face = ft_library.new_face(file_name, file_idx).unwrap();
FontFace::create_from_ft(&face).unwrap()
}
fn load_config() -> (Config, [FunctionLayer; 2]) {
let mut base = toml::from_str::<ConfigProxy>(&read_to_string("/usr/share/tiny-dfr/config.toml").unwrap()).unwrap();
let user = read_to_string(USER_CFG_PATH).map_err::<Error, _>(|e| e.into())
.and_then(|r| Ok(toml::from_str::<ConfigProxy>(&r)?));
if let Ok(user) = user {
base.media_layer_default = user.media_layer_default.or(base.media_layer_default);
base.show_button_outlines = user.show_button_outlines.or(base.show_button_outlines);
base.enable_pixel_shift = user.enable_pixel_shift.or(base.enable_pixel_shift);
base.font_template = user.font_template.or(base.font_template);
base.adaptive_brightness = user.adaptive_brightness.or(base.adaptive_brightness);
base.media_layer_keys = user.media_layer_keys.or(base.media_layer_keys);
base.primary_layer_keys = user.primary_layer_keys.or(base.primary_layer_keys);
base.active_brightness = user.active_brightness.or(base.active_brightness);
base.fn_toggle_layers = user.fn_toggle_layers.or(base.fn_toggle_layers);
};
let media_layer = FunctionLayer::with_config(base.media_layer_keys.unwrap());
let fkey_layer = FunctionLayer::with_config(base.primary_layer_keys.unwrap());
let layers = if base.media_layer_default.unwrap(){ [media_layer, fkey_layer] } else { [fkey_layer, media_layer] };
let cfg = Config {
show_button_outlines: base.show_button_outlines.unwrap(),
enable_pixel_shift: base.enable_pixel_shift.unwrap(),
adaptive_brightness: base.adaptive_brightness.unwrap(),
fn_toggle_layers: base.fn_toggle_layers.unwrap(),
font_face: load_font(&base.font_template.unwrap()),
active_brightness: base.active_brightness.unwrap()
};
(cfg, layers)
}
pub struct ConfigManager {
inotify_fd: Inotify,
watch_desc: Option<WatchDescriptor>
}
fn arm_inotify(inotify_fd: &Inotify) -> Option<WatchDescriptor> {
let flags = AddWatchFlags::IN_MOVED_TO | AddWatchFlags::IN_CLOSE | AddWatchFlags::IN_ONESHOT;
match inotify_fd.add_watch(USER_CFG_PATH, flags) {
Ok(wd) => Some(wd),
Err(Errno::ENOENT) => None,
e => Some(e.unwrap())
}
}
impl ConfigManager {
pub fn new() -> ConfigManager {
let inotify_fd = Inotify::init(InitFlags::IN_NONBLOCK).unwrap();
let watch_desc = arm_inotify(&inotify_fd);
ConfigManager {
inotify_fd, watch_desc
}
}
pub fn load_config(&self) -> (Config, [FunctionLayer; 2]) {
load_config()
}
pub fn update_config(&mut self, cfg: &mut Config, layers: &mut [FunctionLayer; 2]) -> bool {
if self.watch_desc.is_none() {
self.watch_desc = arm_inotify(&self.inotify_fd);
return false;
}
let evts = match self.inotify_fd.read_events() {
Ok(e) => e,
Err(Errno::EAGAIN) => Vec::new(),
r => r.unwrap(),
};
let mut ret = false;
for evt in evts {
if evt.wd != self.watch_desc.unwrap() {
continue
}
let parts = load_config();
*cfg = parts.0;
*layers = parts.1;
ret = true;
self.watch_desc = arm_inotify(&self.inotify_fd);
}
ret
}
pub fn fd(&self) -> &impl AsFd {
&self.inotify_fd
}
}