aboutsummaryrefslogtreecommitdiff
path: root/scripts/theme-studio/Makefile
diff options
context:
space:
mode:
Diffstat (limited to 'scripts/theme-studio/Makefile')
-rw-r--r--scripts/theme-studio/Makefile121
1 files changed, 121 insertions, 0 deletions
diff --git a/scripts/theme-studio/Makefile b/scripts/theme-studio/Makefile
new file mode 100644
index 000000000..7b8430182
--- /dev/null
+++ b/scripts/theme-studio/Makefile
@@ -0,0 +1,121 @@
+# Makefile for the theme-studio tool — a self-contained Python + JS subproject.
+# Its toolchain (python3, node, uvx, headless Chrome) is independent of the repo
+# root's Elisp/ERT world, so the build logic lives here with the code. The root
+# Makefile delegates: `make theme-studio-test` and `make theme-studio-coverage`
+# call `make -C scripts/theme-studio ...`.
+#
+# Recipes run in this directory, so the relative paths below resolve whether you
+# `cd` here or invoke via the root's `-C` delegation.
+
+# Absolute path to this directory (for `open`, which hands Chrome a file path).
+HERE := $(dir $(abspath $(lastword $(MAKEFILE_LIST))))
+
+# Optional palette seed for `gen` / `open`: make gen SEED=dupre.json
+SEED ?=
+OUT ?= ../../themes
+EMACS ?= emacs
+EMACSCLIENT ?= emacsclient
+
+.PHONY: help test check check-generated coverage gen open theme theme-load theme-reload face-coverage-dump face-coverage face-coverage-diff
+
+# Scratch path for the face-coverage Emacs data dump.
+FACE_DUMP ?= /tmp/face-coverage-data.json
+
+.DEFAULT_GOAL := help
+
+help:
+ @echo "theme-studio targets:"
+ @echo " make test - Full suite: Python + Node + browser hash gates"
+ @echo " make check - Fast gate: regenerate + Python + Node (no browser)"
+ @echo " make check-generated - Verify committed theme-studio.html is current"
+ @echo " make coverage - JS (node) + generate.py (uvx coverage) numbers"
+ @echo " make gen [SEED=x.json] - Regenerate theme-studio.html (optionally from a seed)"
+ @echo " make open [SEED=x.json] - Regenerate and open the page in Chrome"
+ @echo " make theme JSON=x.json - Convert a Theme Studio JSON export to OUT/<name>-theme.el"
+ @echo " make theme-load THEME=x - Disable all custom themes, then load THEME in current Emacs"
+ @echo " make theme-reload JSON=x - Convert JSON, then cleanly reload its theme in current Emacs"
+ @echo " make face-coverage - Regenerate face-coverage.org from the live Emacs daemon"
+ @echo " make face-coverage-diff - Show the coverage delta vs the committed face-coverage.org"
+
+test:
+ @./run-tests.sh
+
+check:
+ @./run-tests.sh --no-browser
+
+check-generated:
+ @tmp="$$(mktemp)"; \
+ cp theme-studio.html "$$tmp"; \
+ restore() { cp "$$tmp" theme-studio.html; rm -f "$$tmp"; }; \
+ if ! python3 generate.py >/dev/null; then restore; exit 1; fi; \
+ if cmp -s theme-studio.html "$$tmp"; then rm -f "$$tmp"; echo "theme-studio.html is current"; \
+ else restore; echo "theme-studio.html is stale; run make gen and commit it" >&2; exit 1; fi
+
+coverage:
+ @echo "== JS coverage (node --experimental-test-coverage) =="
+ @node --test --experimental-test-coverage ./*.mjs 2>/dev/null \
+ | sed -n '/start of coverage report/,/end of coverage report/p'
+ @echo ""
+ @echo "== generate.py coverage =="
+ @if command -v uvx >/dev/null 2>&1; then \
+ uvx coverage run --include='generate.py' -m unittest test_generate >/dev/null 2>&1; \
+ uvx coverage report -m; \
+ uvx coverage erase >/dev/null 2>&1; \
+ else \
+ echo "uvx not found — skipping generate.py line coverage"; \
+ echo "($$(grep -c 'def test_' test_generate.py) test_generate.py tests exist)"; \
+ fi
+
+gen:
+ @THEME_STUDIO_SEED="$(SEED)" python3 generate.py
+
+open: gen
+ @c=""; for b in google-chrome-stable google-chrome chromium chromium-browser; do \
+ command -v $$b >/dev/null 2>&1 && { c=$$b; break; }; \
+ done; \
+ if [ -n "$$c" ]; then \
+ "$$c" "$(HERE)theme-studio.html" >/dev/null 2>&1 & \
+ echo "opened theme-studio.html in $$c"; \
+ else \
+ echo "no Chromium-family browser found"; exit 1; \
+ fi
+
+theme:
+ifndef JSON
+ @echo "Error: JSON parameter required"
+ @echo "Usage: make theme JSON=/path/to/theme.json [OUT=../../themes]"
+ @exit 1
+endif
+ @$(EMACS) --batch -l build-theme.el --eval '(princ (concat "wrote " (build-theme/convert-file "$(JSON)" "$(OUT)") "\n"))'
+
+theme-load:
+ifndef THEME
+ @echo "Error: THEME parameter required"
+ @echo "Usage: make theme-load THEME=theme [OUT=../../themes]"
+ @exit 1
+endif
+ @$(EMACSCLIENT) -e "(progn (add-to-list 'custom-theme-load-path \"$(abspath $(OUT))\") (mapc #'disable-theme (copy-sequence custom-enabled-themes)) (load-theme '$(THEME) t) custom-enabled-themes)"
+
+theme-reload:
+ifndef JSON
+ @echo "Error: JSON parameter required"
+ @echo "Usage: make theme-reload JSON=/path/to/theme.json [OUT=../../themes] [THEME=name]"
+ @exit 1
+endif
+ @$(MAKE) theme JSON='$(JSON)' OUT='$(OUT)' EMACS='$(EMACS)'
+ @theme_name='$(THEME)'; \
+ if [ -z "$$theme_name" ]; then theme_name="$$(basename '$(JSON)' .json)"; fi; \
+ $(MAKE) theme-load THEME="$$theme_name" OUT='$(OUT)' EMACSCLIENT='$(EMACSCLIENT)'
+
+# Dump face/group/package data from the running daemon (falls back to a batch
+# Emacs that loads the full init when no daemon is reachable).
+face-coverage-dump:
+ @$(EMACSCLIENT) -e '(progn (load "$(HERE)face-coverage-dump.el") (face-coverage-dump "$(FACE_DUMP)"))' >/dev/null 2>&1 \
+ || $(EMACS) --batch -l "$$HOME/.emacs.d/init.el" -l "$(HERE)face-coverage-dump.el" \
+ --eval '(face-coverage-dump "$(FACE_DUMP)")'
+
+face-coverage: face-coverage-dump
+ @python3 face_coverage.py --data "$(FACE_DUMP)"
+
+face-coverage-diff: face-coverage-dump
+ @python3 face_coverage.py --data "$(FACE_DUMP)" --compare face-coverage.org