Files
niritings/nirimod/pages/startup.py
2026-05-29 00:41:12 +00:00

172 lines
5.6 KiB
Python

"""Startup Programs page."""
from __future__ import annotations
import shlex
import gi
gi.require_version("Gtk", "4.0")
gi.require_version("Adw", "1")
from gi.repository import Adw, Gtk, GLib
from nirimod.kdl_parser import KdlNode
from nirimod.pages.base import BasePage
class StartupPage(BasePage):
def build(self) -> Gtk.Widget:
tb, header, _, content = self._make_toolbar_page("Startup Programs")
self._content = content
self.refresh()
return tb
def refresh(self):
self._rebuild()
def _get_entries(self) -> list[KdlNode]:
return [
n
for n in self._nodes
if n.name in ("spawn-at-startup", "spawn-sh-at-startup")
]
def _rebuild(self):
# Clear existing content
while True:
child = self._content.get_first_child()
if child is None:
break
self._content.remove(child)
entries = self._get_entries()
if not entries:
status = Adw.StatusPage(
title="No Startup Programs",
description="Programs added here will launch automatically when niri starts.",
icon_name="applications-system-symbolic",
)
add_btn = Gtk.Button(label="Add Program")
add_btn.add_css_class("pill")
add_btn.add_css_class("suggested-action")
add_btn.set_halign(Gtk.Align.CENTER)
add_btn.connect("clicked", self._on_add)
box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=0)
box.set_valign(Gtk.Align.CENTER)
box.set_vexpand(True)
box.append(status)
box.append(add_btn)
self._content.append(box)
else:
grp = Adw.PreferencesGroup(
title="Startup Programs",
description=f"{len(entries)} program{'s' if len(entries) != 1 else ''} configured to launch",
)
for i, entry in enumerate(entries):
row = self._make_row(entry, i)
grp.add(row)
self._content.append(grp)
# Also add a convenient button at the bottom
add_btn = Gtk.Button(label="Add Another Program")
add_btn.add_css_class("pill")
add_btn.set_halign(Gtk.Align.CENTER)
add_btn.set_margin_top(16)
add_btn.connect("clicked", self._on_add)
self._content.append(add_btn)
def _make_row(self, node: KdlNode, idx: int) -> Adw.ActionRow:
cmd = " ".join(str(a) for a in node.args)
is_sh = "sh" in node.name
cmd_str = GLib.markup_escape_text(cmd) if cmd else "(empty)"
row = Adw.ActionRow(
title=cmd_str or "(empty)",
subtitle="Via shell (spawn-sh-at-startup)" if is_sh else "Launched directly",
)
row.set_activatable(True)
row.connect("activated", lambda *_, i=idx: self._on_edit(i))
del_btn = Gtk.Button(icon_name="user-trash-symbolic")
del_btn.set_valign(Gtk.Align.CENTER)
del_btn.add_css_class("flat")
del_btn.add_css_class("error")
del_btn.set_tooltip_text("Remove startup entry")
del_btn.connect("clicked", lambda *_, i=idx: self._on_delete(i))
row.add_suffix(del_btn)
return row
def _on_add(self, *_):
self._show_dialog(None, -1)
def _on_edit(self, idx: int):
entries = self._get_entries()
if 0 <= idx < len(entries):
self._show_dialog(entries[idx], idx)
def _on_delete(self, idx: int):
entries = self._get_entries()
if 0 <= idx < len(entries):
self._nodes.remove(entries[idx])
self._commit("remove startup entry")
self._rebuild()
def _show_dialog(self, node: KdlNode | None, idx: int):
dialog = Adw.AlertDialog(
heading="Startup Program", body="Enter the command to launch at startup."
)
cmd_entry = Adw.EntryRow(title="Command")
sh_switch = Adw.SwitchRow(title="Use shell (spawn-sh-at-startup)")
if node:
cmd_entry.set_text(" ".join(str(a) for a in node.args))
sh_switch.set_active("sh" in node.name)
box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=4)
grp = Adw.PreferencesGroup()
grp.add(cmd_entry)
grp.add(sh_switch)
box.append(grp)
dialog.set_extra_child(box)
dialog.add_response("cancel", "Cancel")
dialog.add_response("save", "Save")
dialog.set_response_appearance("save", Adw.ResponseAppearance.SUGGESTED)
def _on_resp(d, r):
if r != "save":
return
cmd = cmd_entry.get_text().strip()
if not cmd:
return
is_sh = sh_switch.get_active()
node_name = "spawn-sh-at-startup" if is_sh else "spawn-at-startup"
if is_sh:
# sh -c expects a single string; store the whole command as one arg
args = [cmd]
else:
try:
args = shlex.split(cmd)
except ValueError:
args = cmd.split()
new_node = KdlNode(node_name, args=args)
entries = self._get_entries()
if idx >= 0 and 0 <= idx < len(entries):
i = self._nodes.index(entries[idx])
self._nodes[i] = new_node
else:
self._nodes.append(new_node)
self._commit("startup entry")
self._rebuild()
dialog.connect("response", _on_resp)
dialog.present(self._win)