use crate::color::Color;
use crate::protocol::zwlr_layer_surface_v1;
use anyhow::{Context, Result};
use pangocairo::pango::FontDescription;
use serde::{de, Deserialize};
use std::collections::HashMap;
use std::fs::read_to_string;
use std::ops::Deref;
use std::path::{Path, PathBuf};
use std::{env, fmt};
#[derive(Deserialize, Debug)]
#[serde(deny_unknown_fields, default)]
pub struct Config {
pub command: Option<String>,
pub background: Color,
pub color: Color,
pub separator: Color,
pub tag_fg: Color,
pub tag_bg: Color,
pub tag_focused_fg: Color,
pub tag_focused_bg: Color,
pub tag_urgent_fg: Color,
pub tag_urgent_bg: Color,
pub tag_inactive_fg: Color,
pub tag_inactive_bg: Color,
pub font: Font,
pub height: u32,
pub margin_top: i32,
pub margin_bottom: i32,
pub margin_left: i32,
pub margin_right: i32,
pub separator_width: f64,
pub tags_r: f64,
pub tags_padding: f64,
pub blocks_r: f64,
pub blocks_overlap: f64,
pub position: Position,
pub hide_inactive_tags: bool,
pub invert_touchpad_scrolling: bool,
pub show_layout_name: bool,
pub blend: bool,
pub show_mode: bool,
pub wm: WmConfig,
pub output: HashMap<String, OutputOverrides>,
}
impl Default for Config {
fn default() -> Self {
Self {
command: None,
background: Color::from_rgba_hex(0x282828ff),
color: Color::from_rgba_hex(0xffffffff),
separator: Color::from_rgba_hex(0x9a8a62ff),
tag_fg: Color::from_rgba_hex(0xd79921ff),
tag_bg: Color::from_rgba_hex(0x282828ff),
tag_focused_fg: Color::from_rgba_hex(0x1d2021ff),
tag_focused_bg: Color::from_rgba_hex(0x689d68ff),
tag_urgent_fg: Color::from_rgba_hex(0x282828ff),
tag_urgent_bg: Color::from_rgba_hex(0xcc241dff),
tag_inactive_fg: Color::from_rgba_hex(0xd79921ff),
tag_inactive_bg: Color::from_rgba_hex(0x282828ff),
font: Font::new("monospace 10"),
height: 24,
margin_top: 0,
margin_bottom: 0,
margin_left: 0,
margin_right: 0,
separator_width: 2.0,
tags_r: 0.0,
tags_padding: 25.0,
blocks_r: 0.0,
blocks_overlap: 0.0,
position: Position::Top,
hide_inactive_tags: true,
invert_touchpad_scrolling: true,
show_layout_name: true,
blend: true,
show_mode: true,
wm: WmConfig {
river: RiverConfig { max_tag: 9 },
},
output: HashMap::new(),
}
}
}
impl Config {
pub fn new(path: Option<&Path>) -> Result<Self> {
let buf;
let path = match path {
Some(path) => Some(path),
None => {
buf = config_path();
buf.as_deref()
}
};
Ok(match path {
Some(config_path) => {
let config = read_to_string(config_path).context("Failed to read configuration")?;
toml::from_str(&config).context("Failed to deserialize configuration")?
}
None => {
eprintln!("Could not find the configuration path");
eprintln!("Using default configuration");
Self::default()
}
})
}
pub fn output_enabled(&self, output: &str) -> bool {
self.output
.get(output)
.and_then(|o| o.enable)
.unwrap_or(true)
}
}
fn config_dir() -> Option<PathBuf> {
env::var_os("XDG_CONFIG_HOME")
.map(PathBuf::from)
.or_else(|| Some(PathBuf::from(env::var_os("HOME")?).join(".config")))
}
fn config_path() -> Option<PathBuf> {
let mut path = config_dir()?;
path.push("i3bar-river");
path.push("config.toml");
path.exists().then_some(path)
}
#[derive(Deserialize, Debug, Clone, Copy, PartialEq, Eq)]
#[serde(rename_all = "lowercase")]
pub enum Position {
Top,
Bottom,
}
impl From<Position> for zwlr_layer_surface_v1::Anchor {
fn from(position: Position) -> Self {
match position {
Position::Top => Self::Top | Self::Left | Self::Right,
Position::Bottom => Self::Bottom | Self::Left | Self::Right,
}
}
}
#[derive(Debug, Deserialize)]
pub struct WmConfig {
pub river: RiverConfig,
}
#[derive(Debug, Deserialize)]
pub struct RiverConfig {
pub max_tag: u8,
}
#[derive(Debug, Deserialize)]
pub struct OutputOverrides {
#[serde(default)]
enable: Option<bool>,
}
#[derive(Debug)]
pub struct Font(pub FontDescription);
impl Font {
pub fn new(desc: &str) -> Self {
Self(FontDescription::from_string(desc))
}
}
impl Deref for Font {
type Target = FontDescription;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl<'de> de::Deserialize<'de> for Font {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: de::Deserializer<'de>,
{
struct FontVisitor;
impl<'de> de::Visitor<'de> for FontVisitor {
type Value = Font;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("font description")
}
fn visit_str<E>(self, s: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(Font::new(s))
}
}
deserializer.deserialize_str(FontVisitor)
}
}