-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmain.py
234 lines (192 loc) · 7.49 KB
/
main.py
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
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
import logging
import sublime
import sublime_plugin
import os
import threading
import time
import subprocess
import fcntl
import functools
import pathlib
import signal
import itertools
import shutil
import contextlib
import uuid
from .helpers import settings_watcher, read_system_theme, parse_dbus_monitor
from .lifecycle import lifecycle, CycleStage
from .logger import logger
from typing import Optional, Any
plugin_settings_file = f"{__package__}.sublime-settings"
plugin_settings: Any
daemon = None
stop_daemon = True
class AutoDarkLinuxInputHandler(sublime_plugin.ListInputHandler):
def name(self):
# this is used to determine the name of the argument passed to the
# auto_dark_linux command when a list item is picked
return "new_mode"
def validate(self, text):
return True
def placeholder(self):
return "Choose mode"
def description(self, _: str, text: str):
# TODO: Test this
return f"{text} Mode"
def list_items(self, options=[("Light", "light"), ("Dark", "dark"), ("System", "system")]):
# TODO: Test this
mode = plugin_settings.get("auto_dark_mode", "system")
index = next((i for i, v in enumerate(options) if v[1] == mode.lower()))
return (options, index)
class AutoDarkLinuxCommand(sublime_plugin.ApplicationCommand):
def run(self, new_mode="system"):
logger.info(f"setting mode to '{new_mode}'")
current_mode = plugin_settings.get("auto_dark_mode", "system")
if new_mode is None or new_mode == "system":
plugin_settings["auto_dark_mode"] = "system"
else:
plugin_settings["auto_dark_mode"] = new_mode
sublime.save_settings(plugin_settings_file)
if new_mode == "system":
if (new_mode := read_system_theme()) is None:
return
if current_mode == "system":
current_mode = read_system_theme()
sublime.set_timeout(functools.partial(change_color_scheme, new_mode, current_mode))
def input(self, _):
return AutoDarkLinuxInputHandler()
def is_visible(self) -> bool:
return sublime.platform() == "linux"
def is_checked(self, new_mode: str):
return new_mode == plugin_settings.get("auto_dark_mode", "system")
def is_enabled(self) -> bool:
return shutil.which("busctl") is not None
class AutoDarkLinuxEventListener(sublime_plugin.EventListener):
def on_exit(self):
logger.info("Exiting. Cleaning up")
unmonitor()
logger.info("Exited")
@lifecycle(CycleStage.LOADED)
def plugin_loaded():
if sublime.platform() != "linux":
return sublime.message_dialog("AutoDarkLinux plugin only works on Linux")
if not shutil.which("busctl"):
return sublime.message_dialog(
"AutoDarkLinux plugin requires the 'busctl' command from systemd"
)
global plugin_settings
sublime.save_settings(plugin_settings_file)
plugin_settings = sublime.load_settings(plugin_settings_file)
@lifecycle(CycleStage.UNLOADED)
def plugin_unloaded():
# save settings before the plugin is unloaded
sublime.save_settings("Preferences.sublime-settings")
sublime.save_settings(plugin_settings_file)
unmonitor()
def monitor():
pid_file = pathlib.Path(sublime.cache_path()) / f"{__package__}/daemon.pid"
with contextlib.suppress(FileNotFoundError, ValueError, ProcessLookupError):
with pid_file.open() as pid:
daemon_pid = int(next(pid))
os.kill(daemon_pid, signal.SIGTERM)
pid_file.parent.mkdir(parents=True, exist_ok=True)
current_mode = read_system_theme()
with subprocess.Popen(
[
"/usr/bin/busctl",
"--user",
"--json=short",
"--match",
",".join(
[
"type='signal'",
"interface='org.freedesktop.portal.Settings'",
"path='/org/freedesktop/portal/desktop'",
"member='SettingChanged'",
"arg0='org.freedesktop.appearance'",
"arg1='color-scheme'",
]
),
"monitor",
],
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
) as proc:
assert proc.stdout is not None
fcntl.fcntl(proc.stdout, fcntl.F_SETFL, os.O_NONBLOCK)
pid_file.write_text(f"{proc.pid}")
logger.info("Daemon started with pid: %d", proc.pid)
global stop_daemon
while proc.poll() is None:
if stop_daemon:
proc.kill()
break
data = None
for data in itertools.takewhile(bool, map(proc.stdout.readline, itertools.repeat(-1))):
# ignore duplicate events, only take the last one
pass
if not data:
time.sleep(0.5)
continue
mode = parse_dbus_monitor(data.decode("utf-8").strip())
sublime.set_timeout(functools.partial(change_color_scheme, mode, current_mode))
current_mode = mode
else:
proc.kill()
unmonitor()
def unmonitor():
global daemon
if daemon is not None:
global stop_daemon
stop_daemon = True
if threading.current_thread() == threading.main_thread():
daemon.join()
daemon = None
logger.info("Daemon stopped")
def change_color_scheme(new_scheme: str, old_scheme: Optional[str] = None):
settings = sublime.load_settings("Preferences.sublime-settings")
ui_info = sublime.ui_info()
if (theme := settings.get(f"{new_scheme}_theme")) != ui_info.get("theme").get("resolved_value"):
settings["theme"] = theme
if (color_scheme := settings.get(f"{new_scheme}_color_scheme")) != ui_info.get(
"color_scheme"
).get("resolved_value"):
settings["color_scheme"] = color_scheme
if new_scheme != old_scheme:
logger.info(f"Color scheme change detected: previous={old_scheme}, new={new_scheme}")
sublime.save_settings("Preferences.sublime-settings")
@settings_watcher("auto_dark_mode")
def watch_auto_dark_mode(new_mode: str, old_mode: str = ""):
global daemon, stop_daemon
logger.info(f"Detected mode change. Previous={old_mode}, New={new_mode}")
unmonitor()
if new_mode == "system":
stop_daemon = False
daemon = threading.Thread(target=monitor, name="AutoDarkLinuxMonitor")
daemon.start()
else:
sublime.run_command("auto_dark_linux", args={"new_mode": new_mode})
@settings_watcher("debug")
def watch_debug(new_value: bool, old_value=False):
if new_value:
logger.setLevel(logging.INFO)
logger.info("Turned on debug logging")
else:
logger.info("Turning off debug logging...")
logger.setLevel(logging.WARNING)
@plugin_loaded.notify()
@plugin_unloaded.notify()
def listen_settings(life_cycle=CycleStage.NONE, tags=[uuid.uuid4(), uuid.uuid4()]):
# the use of mutable key 'tags' is to persist the value across
# multiple function calls
mode_tag, debug_tag = tags
if life_cycle == CycleStage.LOADED:
logger.info(f"Plugin load detected.")
watch_auto_dark_mode()
plugin_settings.add_on_change(str(mode_tag), watch_auto_dark_mode)
watch_debug()
plugin_settings.add_on_change(str(debug_tag), watch_debug)
elif life_cycle == CycleStage.UNLOADED:
logger.info(f"Plugin unloaded.")
plugin_settings.clear_on_change(str(mode_tag))
plugin_settings.clear_on_change(str(debug_tag))