mirror of
https://github.com/caelestia-dots/cli.git
synced 2026-06-08 00:09:29 -05:00
b00c601d0a
LSP was screaming at me so I decided to just address it to get it off my screen. + Fixed the type hints := Modified and added type hints for certain functions and variables in most of the files in the utils/ folder (and some in the subcommands/ folder) for clarity and so pyright's type checker wouldn't cry. :+ To resolve certain type issues, I had to add a bit more tiny additional code such as, additional checks if a variable is None, a tiny class in utils/material/generator.py to resolve the constructor usage mismatch between what the DynamicScheme accepts and what the code actually passes, and etc. - Renamed certain functions and variables for clarity and also for some to not collide with pre-existing definitions from well-known library imports. + PIL has reorganized their code a bit, so the code is made to reflect their new definitions. = Reorganized the single import statement for "colourfulness" in utils/wallpaper.py to be close to the top. (I think that's it) Side Effects?: Everything should work the same as no logic change was done whatsover (unless we consider the added if statements for type checking as a logic change). I've tested it, everything seems to be in urdir.
165 lines
5.5 KiB
Python
165 lines
5.5 KiB
Python
import json
|
|
import shlex
|
|
import shutil
|
|
from argparse import Namespace
|
|
from collections import ChainMap
|
|
from typing import Any, Callable, cast
|
|
|
|
from caelestia.utils import hypr
|
|
from caelestia.utils.paths import user_config_path
|
|
|
|
|
|
def is_subset(superset, subset):
|
|
for key, value in subset.items():
|
|
if key not in superset:
|
|
return False
|
|
|
|
if isinstance(value, dict):
|
|
if not is_subset(superset[key], value):
|
|
return False
|
|
|
|
elif isinstance(value, str):
|
|
if value not in superset[key]:
|
|
return False
|
|
|
|
elif isinstance(value, list):
|
|
if not set(value) <= set(superset[key]):
|
|
return False
|
|
elif isinstance(value, set):
|
|
if not value <= superset[key]:
|
|
return False
|
|
|
|
else:
|
|
if not value == superset[key]:
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
class DeepChainMap(ChainMap):
|
|
def __getitem__(self, key):
|
|
values = (mapping[key] for mapping in self.maps if key in mapping)
|
|
try:
|
|
first = next(values)
|
|
except StopIteration:
|
|
return self.__missing__(key)
|
|
if isinstance(first, dict):
|
|
return self.__class__(first, *values)
|
|
return first
|
|
|
|
def __repr__(self):
|
|
return repr(dict(self))
|
|
|
|
|
|
class Command:
|
|
args: Namespace
|
|
cfg: dict[str, dict[str, dict[str, Any]]] | DeepChainMap
|
|
clients: list[dict[str, Any]] | None = None
|
|
|
|
def __init__(self, args: Namespace) -> None:
|
|
self.args = args
|
|
|
|
self.cfg = {
|
|
"communication": {
|
|
"discord": {
|
|
"enable": True,
|
|
"match": [{"class": "discord"}],
|
|
"command": ["discord"],
|
|
"move": True,
|
|
},
|
|
"whatsapp": {
|
|
"enable": True,
|
|
"match": [{"class": "whatsapp"}],
|
|
"move": True,
|
|
},
|
|
},
|
|
"music": {
|
|
"spotify": {
|
|
"enable": True,
|
|
"match": [{"class": "Spotify"}, {"initialTitle": "Spotify"}, {"initialTitle": "Spotify Free"}],
|
|
"command": ["spicetify", "watch", "-s"],
|
|
"move": True,
|
|
},
|
|
"feishin": {
|
|
"enable": True,
|
|
"match": [{"class": "feishin"}],
|
|
"move": True,
|
|
},
|
|
},
|
|
"sysmon": {
|
|
"btop": {
|
|
"enable": True,
|
|
"match": [{"class": "btop", "title": "btop", "workspace": {"name": "special:sysmon"}}],
|
|
"command": ["foot", "-a", "btop", "-T", "btop", "fish", "-C", "exec btop"],
|
|
},
|
|
},
|
|
"todo": {
|
|
"todoist": {
|
|
"enable": True,
|
|
"match": [{"class": "Todoist"}],
|
|
"command": ["todoist"],
|
|
"move": True,
|
|
},
|
|
},
|
|
}
|
|
try:
|
|
self.cfg = DeepChainMap(json.loads(user_config_path.read_text())["toggles"], self.cfg)
|
|
except (FileNotFoundError, json.JSONDecodeError, KeyError):
|
|
pass
|
|
|
|
def run(self) -> None:
|
|
if self.args.workspace == "specialws":
|
|
self.specialws()
|
|
return
|
|
|
|
spawned = False
|
|
if self.args.workspace in self.cfg:
|
|
for client in self.cfg[self.args.workspace].values():
|
|
if "enable" in client and client["enable"] and self.handle_client_config(client):
|
|
spawned = True
|
|
|
|
if not spawned:
|
|
hypr.dispatch("togglespecialworkspace", self.args.workspace)
|
|
|
|
def get_clients(self) -> list[dict[str, Any]]:
|
|
if self.clients is None:
|
|
self.clients = cast(list[dict[str, Any]], hypr.message("clients"))
|
|
return self.clients
|
|
|
|
def move_client(self, selector: Callable, workspace: str) -> None:
|
|
for client in self.get_clients():
|
|
if selector(client) and client["workspace"]["name"] != f"special:{workspace}":
|
|
hypr.dispatch("movetoworkspacesilent", f"special:{workspace},address:{client['address']}")
|
|
|
|
def spawn_client(self, selector: Callable, spawn: list[str]) -> bool:
|
|
if (spawn[0].endswith(".desktop") or shutil.which(spawn[0])) and not any(
|
|
selector(client) for client in self.get_clients()
|
|
):
|
|
hypr.dispatch("exec", f"[workspace special:{self.args.workspace}] app2unit -- {shlex.join(spawn)}")
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
def handle_client_config(self, client: dict[str, Any]) -> bool:
|
|
def selector(c: dict[str, Any]) -> bool:
|
|
# Each match is or, inside matches is and
|
|
for match in client["match"]:
|
|
if is_subset(c, match):
|
|
return True
|
|
return False
|
|
|
|
spawned = False
|
|
if "command" in client and client["command"]:
|
|
spawned = self.spawn_client(selector, client["command"])
|
|
if "move" in client and client["move"]:
|
|
self.move_client(selector, self.args.workspace)
|
|
|
|
return spawned
|
|
|
|
def specialws(self) -> None:
|
|
monitors = cast(list[dict[str, Any]], hypr.message("monitors"))
|
|
target = next((m for m in monitors if m.get("focused")), None)
|
|
if target:
|
|
special = target.get("specialWorkspace", {}).get("name", "")[8:] or "special"
|
|
hypr.dispatch("togglespecialworkspace", special)
|