feat: colour palettes
All checks were successful
Lint / shellcheck (push) Successful in 17s

This commit is contained in:
2026-04-01 14:23:12 +02:00
parent 69bd5832e7
commit 009877ae76
30 changed files with 446 additions and 85 deletions

View File

@@ -46,14 +46,9 @@ function mask(s, t) {
while (match(substr(line, p), /`+/)) { while (match(substr(line, p), /`+/)) {
pstart = p + RSTART - 1 pstart = p + RSTART - 1
plen = RLENGTH plen = RLENGTH
if (plen >= 3) {
out = out substr(line, p, pstart - p + plen)
p = pstart + plen
continue
}
# Found 1 or 2 backticks at pstart # Found backtick sequence at pstart
# Search for closing marker # Search for closing marker of same length
marker = substr(line, pstart, plen) marker = substr(line, pstart, plen)
tail = substr(line, pstart + plen) tail = substr(line, pstart + plen)
mpos = index(tail, marker) mpos = index(tail, marker)
@@ -69,7 +64,7 @@ function mask(s, t) {
# Found match! # Found match!
content = substr(tail, 1, mpos - 1) content = substr(tail, 1, mpos - 1)
out = out substr(line, p, pstart - p) out = out substr(line, p, pstart - p)
if (plen == 2 && substr(content, 1, 1) == " " && substr(content, length(content), 1) == " ") { if (plen >= 2 && substr(content, 1, 1) == " " && substr(content, length(content), 1) == " ") {
content = substr(content, 2, length(content) - 2) content = substr(content, 2, length(content) - 2)
} }
out = out "<code>" mask(content) "</code>" out = out "<code>" mask(content) "</code>"

30
kewt.sh
View File

@@ -324,6 +324,36 @@ if [ "$watch_mode" = "true" ]; then
if [ -n "$changed" ]; then if [ -n "$changed" ]; then
echo "" echo ""
echo "Change detected, rebuilding..." echo "Change detected, rebuilding..."
if [ "$clean_mode" = "true" ]; then
find "$out" -mindepth 1 -delete 2>/dev/null
fi
load_config "./site.conf"
load_config "$src/site.conf"
asset_version=""
if [ "$versioning" = "true" ]; then
asset_version="?v=$(date +%s)"
fi
template="$src/template.html"
[ -f "$template" ] || template="./template.html"
if [ ! -f "$template" ]; then
template="$KEWT_TMPDIR/default_template.html"
printf '%s\n' "$DEFAULT_TMPL" > "$template"
fi
nav=$(generate_nav "$src")
extra_links=$(nav_links_html)
if [ -n "$extra_links" ]; then
nav="$nav
$extra_links"
fi
if [ -n "$nav_extra" ]; then
nav="$nav
$nav_extra"
fi
build_site build_site
touch "$KEWT_TMPDIR/watch_mark" touch "$KEWT_TMPDIR/watch_mark"
fi fi

View File

@@ -7,6 +7,8 @@ needs_rebuild() {
[ -f "$src/site.conf" ] && [ "$src/site.conf" -nt "$out_file" ] && return 0 [ -f "$src/site.conf" ] && [ "$src/site.conf" -nt "$out_file" ] && return 0
[ -f "$template" ] && [ "$template" -nt "$out_file" ] && return 0 [ -f "$template" ] && [ "$template" -nt "$out_file" ] && return 0
[ -f "$script_dir/styles/$style.css" ] && [ "$script_dir/styles/$style.css" -nt "$out_file" ] && return 0 [ -f "$script_dir/styles/$style.css" ] && [ "$script_dir/styles/$style.css" -nt "$out_file" ] && return 0
[ -f "$script_dir/styles/$style.root.css" ] && [ "$script_dir/styles/$style.root.css" -nt "$out_file" ] && return 0
[ -f "$src/styles.root.css" ] && [ "$src/styles.root.css" -nt "$out_file" ] && return 0
return 1 return 1
} }
@@ -97,7 +99,7 @@ eval "find \"$src\" \( $IGNORE_ARGS \) -prune -o -type d -print" | sort | while
find "$dir" ! -name "$(basename "$dir")" -prune ! -name ".*" -print | while read -r entry; do find "$dir" ! -name "$(basename "$dir")" -prune ! -name ".*" -print | while read -r entry; do
name="${entry##*/}" name="${entry##*/}"
case "$name" in case "$name" in
template.html|site.conf|style.css|index.md) continue ;; template.html|site.conf|style.css|styles.root.css|index.md) continue ;;
esac esac
if [ -d "$entry" ]; then if [ -d "$entry" ]; then
echo "${name}|- [${name}/](${name}/index.html)" >> "$temp_entries" echo "${name}|- [${name}/](${name}/index.html)" >> "$temp_entries"
@@ -332,8 +334,23 @@ eval "find \"$src\" \( $IGNORE_ARGS \) -prune -o -type d -print" | sort | while
fi fi
done done
if [ -f "$script_dir/styles/$style.css" ] && needs_rebuild "$script_dir/styles/$style.css" "$out/styles.css"; then if [ ! -f "$src/styles.css" ] && [ ! -f "$src/style.css" ]; then
copy_style_with_resolved_vars "$script_dir/styles/$style.css" "$out/styles.css" if [ -f "$src/styles.root.css" ]; then
_base_css="$script_dir/styles/$style.css"
[ ! -f "$_base_css" ] && _base_css="$script_dir/styles/kewt.css"
if [ ! -f "$out/styles.css" ] || [ "$src/styles.root.css" -nt "$out/styles.css" ] || [ "$_base_css" -nt "$out/styles.css" ]; then
merge_root_style "$src/styles.root.css" "$_base_css" "$out/styles.css"
fi
elif [ -f "$script_dir/styles/$style.css" ]; then
if needs_rebuild "$script_dir/styles/$style.css" "$out/styles.css"; then
copy_style_with_resolved_vars "$script_dir/styles/$style.css" "$out/styles.css"
fi
elif [ -f "$script_dir/styles/$style.root.css" ]; then
_base_css="$script_dir/styles/kewt.css"
if [ ! -f "$out/styles.css" ] || [ "$script_dir/styles/$style.root.css" -nt "$out/styles.css" ] || [ "$_base_css" -nt "$out/styles.css" ]; then
merge_root_style "$script_dir/styles/$style.root.css" "$_base_css" "$out/styles.css"
fi
fi
fi fi
eval "find \"$src\" \( $IGNORE_ARGS \) -prune -o -type f -print" | sort | while IFS= read -r file; do eval "find \"$src\" \( $IGNORE_ARGS \) -prune -o -type f -print" | sort | while IFS= read -r file; do
@@ -343,7 +360,7 @@ eval "find \"$src\" \( $IGNORE_ARGS \) -prune -o -type f -print" | sort | while
out_dir="$out/$dir_rel" out_dir="$out/$dir_rel"
case "${file##*/}" in case "${file##*/}" in
template.html|site.conf|style.css|styles.css) continue ;; template.html|site.conf|style.css|styles.css|styles.root.css) continue ;;
esac esac
if [ "${file##*/}" = "index.md" ] && grep -q '^[[:space:]]*{{LIST}}[[:space:]]*$' "$file" 2>/dev/null; then if [ "${file##*/}" = "index.md" ] && grep -q '^[[:space:]]*{{LIST}}[[:space:]]*$' "$file" 2>/dev/null; then

View File

@@ -109,6 +109,27 @@ copy_style_with_resolved_vars() {
out_style="$2" out_style="$2"
awk -f "$awk_dir/replace_variables.awk" "$src_style" > "$out_style" awk -f "$awk_dir/replace_variables.awk" "$src_style" > "$out_style"
} }
merge_root_style() {
root_file="$1"
base_css="$2"
out_file="$3"
{
cat "$root_file"
awk '
BEGIN { in_root = 0; brace_depth = 0 }
/^:root[[:space:]]*\{/ { in_root = 1; brace_depth = 1; next }
in_root {
for (i = 1; i <= length($0); i++) {
c = substr($0, i, 1)
if (c == "{") brace_depth++
if (c == "}") { brace_depth--; if (brace_depth == 0) { in_root = 0; next } }
}
next
}
{ print }
' "$base_css"
} | awk -f "$awk_dir/replace_variables.awk" > "$out_file"
}
render_markdown() { render_markdown() {
file="$1" file="$1"
is_home="$2" is_home="$2"

View File

@@ -1,44 +1,54 @@
document.addEventListener('DOMContentLoaded', function() { document.addEventListener("DOMContentLoaded", function () {
var params = new URLSearchParams(window.location.search); var params = new URLSearchParams(window.location.search);
var query = params.get('q'); var query = params.get("q");
var box = document.getElementById('search-box'); var box = document.getElementById("search-box");
var resultsContainer = document.getElementById('search-results-list'); var resultsContainer = document.getElementById("search-results-list");
if (box && query) box.value = query; if (box && query) box.value = query;
if (!query) { if (!query) {
resultsContainer.innerHTML = '<p>Enter a search term above.</p>'; resultsContainer.innerHTML = "<p>Enter a search term above.</p>";
return; return;
} }
fetch('/search.json') fetch("/search.json")
.then(function(response) { return response.json(); }) .then(function (response) {
.then(function(data) { return response.json();
})
.then(function (data) {
var q = query.toLowerCase(); var q = query.toLowerCase();
var results = data.filter(function(item) { var results = data.filter(function (item) {
return item.title.toLowerCase().indexOf(q) !== -1 || return (
item.content.toLowerCase().indexOf(q) !== -1; item.title.toLowerCase().indexOf(q) !== -1 ||
item.content.toLowerCase().indexOf(q) !== -1
);
}); });
var esc = query.replace(/</g, '&lt;').replace(/>/g, '&gt;'); var esc = query.replace(/</g, "&lt;").replace(/>/g, "&gt;");
if (results.length === 0) { if (results.length === 0) {
resultsContainer.innerHTML = '<p>No results found for "<strong>' + esc + '</strong>".</p>'; resultsContainer.innerHTML =
'<p>No results found for "<strong>' + esc + '</strong>".</p>';
return; return;
} }
var html = '<p>Found ' + results.length + ' result(s) for "<strong>' + esc + '</strong>":</p>'; var html =
results.forEach(function(result) { "<p>Found " +
results.length +
' result(s) for "<strong>' +
esc +
'</strong>":</p>';
results.forEach(function (result) {
var snippet = result.content.substring(0, 200); var snippet = result.content.substring(0, 200);
if (result.content.length > 200) snippet += '...'; if (result.content.length > 200) snippet += "...";
html += '<div class="search-result">'; html += '<div class="search-result">';
html += '<a href="' + result.url + '">' + result.title + '</a>'; html += '<a href="' + result.url + '">' + result.title + "</a>";
if (snippet) html += '<p>' + snippet + '</p>'; if (snippet) html += "<p>" + snippet + "</p>";
html += '</div>'; html += "</div>";
}); });
resultsContainer.innerHTML = html; resultsContainer.innerHTML = html;
}) })
.catch(function() { .catch(function () {
resultsContainer.innerHTML = '<p>Error loading search index.</p>'; resultsContainer.innerHTML = "<p>Error loading search index.</p>";
}); });
}); });

View File

@@ -3,5 +3,7 @@
"description": "A minimalist static site generator inspired by werc", "description": "A minimalist static site generator inspired by werc",
"global": "true", "global": "true",
"install": "make install", "install": "make install",
"scripts": ["kewt"] "scripts": [
"kewt"
]
} }

View File

@@ -0,0 +1,26 @@
:root {
--bg: #f5f0ff;
--bg-deep: #e8dffa;
--fg: #2d1b4e;
--fg-muted: #7a6898;
--fg-link: #7b3fba;
--fg-heading: #3d2466;
--code-bg: #e8dffa;
--code-border: #d0c0e8;
--code-fg: #2d1b4e;
--code-sel: #b8860b;
--code-prop: #6f42c1;
--code-val: #0366d6;
--code-var: #22863a;
--code-com: #7a6898;
--adm-note-bg: #e0e0f8;
--adm-note-border: #5b6fc4;
--adm-tip-bg: #ddf0dd;
--adm-tip-border: #46a758;
--adm-important-bg: #eeddf5;
--adm-important-border: #8b5fc7;
--adm-warning-bg: #f5f0dd;
--adm-warning-border: #b8860b;
--adm-caution-bg: #f5dddd;
--adm-caution-border: #c44569;
}

View File

@@ -0,0 +1,26 @@
:root {
--bg: #f5f5f5;
--bg-deep: #e8e8e8;
--fg: #1a1a1a;
--fg-muted: #808080;
--fg-link: #333333;
--fg-heading: #000000;
--code-bg: #e8e8e8;
--code-border: #c0c0c0;
--code-fg: #1a1a1a;
--code-sel: #555555;
--code-prop: #333333;
--code-val: #666666;
--code-var: #444444;
--code-com: #999999;
--adm-note-bg: #e0e0ee;
--adm-note-border: #8888aa;
--adm-tip-bg: #e0eee0;
--adm-tip-border: #88aa88;
--adm-important-bg: #eee0ee;
--adm-important-border: #aa88aa;
--adm-warning-bg: #eeeed0;
--adm-warning-border: #aaaa78;
--adm-caution-bg: #eee0e0;
--adm-caution-border: #aa8888;
}

26
styles/mono.root.css Normal file
View File

@@ -0,0 +1,26 @@
:root {
--bg: #1a1a1a;
--bg-deep: #111111;
--fg: #d4d4d4;
--fg-muted: #808080;
--fg-link: #e0e0e0;
--fg-heading: #ffffff;
--code-bg: #0d0d0d;
--code-border: #404040;
--code-fg: #d4d4d4;
--code-sel: #b8b8b8;
--code-prop: #e0e0e0;
--code-val: #a0a0a0;
--code-var: #c8c8c8;
--code-com: #585858;
--adm-note-bg: #1a1a2e;
--adm-note-border: #5a5a8a;
--adm-tip-bg: #1a2e1a;
--adm-tip-border: #5a8a5a;
--adm-important-bg: #2e1a2e;
--adm-important-border: #8a5a8a;
--adm-warning-bg: #2e2e1a;
--adm-warning-border: #8a8a5a;
--adm-caution-bg: #2e1a1a;
--adm-caution-border: #8a5a5a;
}

View File

@@ -0,0 +1,26 @@
:root {
--bg: #eceff4;
--bg-deep: #d8dee9;
--fg: #2e3440;
--fg-muted: #4c566a;
--fg-link: #5e81ac;
--fg-heading: #3b4252;
--code-bg: #d8dee9;
--code-border: #c5cdd9;
--code-fg: #2e3440;
--code-sel: #d08770;
--code-prop: #5e81ac;
--code-val: #8fbcbb;
--code-var: #a3be8c;
--code-com: #4c566a;
--adm-note-bg: #d8dee9;
--adm-note-border: #5e81ac;
--adm-tip-bg: #e0ebd8;
--adm-tip-border: #a3be8c;
--adm-important-bg: #e5dbe8;
--adm-important-border: #b48ead;
--adm-warning-bg: #ede5d6;
--adm-warning-border: #ebcb8b;
--adm-caution-bg: #eddcdc;
--adm-caution-border: #bf616a;
}

26
styles/nord.root.css Normal file
View File

@@ -0,0 +1,26 @@
:root {
--bg: #2e3440;
--bg-deep: #242933;
--fg: #d8dee9;
--fg-muted: #a5b0c1;
--fg-link: #88c0d0;
--fg-heading: #eceff4;
--code-bg: #3b4252;
--code-border: #4c566a;
--code-fg: #d8dee9;
--code-sel: #ebcb8b;
--code-prop: #8fbcbb;
--code-val: #81a1c1;
--code-var: #a3be8c;
--code-com: #616e88;
--adm-note-bg: #3b4252;
--adm-note-border: #88c0d0;
--adm-tip-bg: #3b4340;
--adm-tip-border: #a3be8c;
--adm-important-bg: #3b4044;
--adm-important-border: #b48ead;
--adm-warning-bg: #3b4038;
--adm-warning-border: #ebcb8b;
--adm-caution-bg: #3b3840;
--adm-caution-border: #bf616a;
}

26
styles/onedark.root.css Normal file
View File

@@ -0,0 +1,26 @@
:root {
--bg: #282c34;
--bg-deep: #21252b;
--fg: #abb2bf;
--fg-muted: #636d83;
--fg-link: #61afef;
--fg-heading: #c8ccd4;
--code-bg: #21252b;
--code-border: #3e4451;
--code-fg: #abb2bf;
--code-sel: #e5c07b;
--code-prop: #e06c75;
--code-val: #56b6c2;
--code-var: #98c379;
--code-com: #5c6370;
--adm-note-bg: #2c313c;
--adm-note-border: #61afef;
--adm-tip-bg: #2c3a30;
--adm-tip-border: #98c379;
--adm-important-bg: #33303c;
--adm-important-border: #c678dd;
--adm-warning-bg: #3a352c;
--adm-warning-border: #e5c07b;
--adm-caution-bg: #3a2c2e;
--adm-caution-border: #e06c75;
}

26
styles/onelight.root.css Normal file
View File

@@ -0,0 +1,26 @@
:root {
--bg: #fafafa;
--bg-deep: #eaeaeb;
--fg: #383a42;
--fg-muted: #a0a1a7;
--fg-link: #4078f2;
--fg-heading: #383a42;
--code-bg: #eaeaeb;
--code-border: #d4d4d5;
--code-fg: #383a42;
--code-sel: #986801;
--code-prop: #e45649;
--code-val: #0184bc;
--code-var: #50a14f;
--code-com: #a0a1a7;
--adm-note-bg: #e8eefa;
--adm-note-border: #4078f2;
--adm-tip-bg: #e8f5e8;
--adm-tip-border: #50a14f;
--adm-important-bg: #f2e8f5;
--adm-important-border: #a626a4;
--adm-warning-bg: #f5f0e0;
--adm-warning-border: #986801;
--adm-caution-bg: #fae8e8;
--adm-caution-border: #e45649;
}

View File

@@ -0,0 +1,26 @@
:root {
--bg: #faf4ed;
--bg-deep: #f2e9e1;
--fg: #575279;
--fg-muted: #9893a5;
--fg-link: #907aa9;
--fg-heading: #286983;
--code-bg: #f2e9e1;
--code-border: #dfdad9;
--code-fg: #575279;
--code-sel: #ea9d34;
--code-prop: #b4637a;
--code-val: #56949f;
--code-var: #286983;
--code-com: #9893a5;
--adm-note-bg: #f0e8f5;
--adm-note-border: #907aa9;
--adm-tip-bg: #e8f0ee;
--adm-tip-border: #56949f;
--adm-important-bg: #f0e8f0;
--adm-important-border: #907aa9;
--adm-warning-bg: #f5f0e0;
--adm-warning-border: #ea9d34;
--adm-caution-bg: #f5e5e8;
--adm-caution-border: #b4637a;
}

26
styles/rosepine.root.css Normal file
View File

@@ -0,0 +1,26 @@
:root {
--bg: #191724;
--bg-deep: #13111e;
--fg: #e0def4;
--fg-muted: #908caa;
--fg-link: #c4a7e7;
--fg-heading: #ebbcba;
--code-bg: #1f1d2e;
--code-border: #26233a;
--code-fg: #e0def4;
--code-sel: #f6c177;
--code-prop: #eb6f92;
--code-val: #9ccfd8;
--code-var: #31748f;
--code-com: #6e6a86;
--adm-note-bg: #1f1d2e;
--adm-note-border: #c4a7e7;
--adm-tip-bg: #1a2332;
--adm-tip-border: #9ccfd8;
--adm-important-bg: #2a1f2e;
--adm-important-border: #c4a7e7;
--adm-warning-bg: #2a251f;
--adm-warning-border: #f6c177;
--adm-caution-bg: #2a1f22;
--adm-caution-border: #eb6f92;
}

View File

@@ -0,0 +1,26 @@
:root {
--bg: #002b36;
--bg-deep: #001e26;
--fg: #839496;
--fg-muted: #586e75;
--fg-link: #268bd2;
--fg-heading: #93a1a1;
--code-bg: #073642;
--code-border: #094959;
--code-fg: #839496;
--code-sel: #d33682;
--code-prop: #268bd2;
--code-val: #2aa198;
--code-var: #859900;
--code-com: #586e75;
--adm-note-bg: #073642;
--adm-note-border: #268bd2;
--adm-tip-bg: #07382e;
--adm-tip-border: #2aa198;
--adm-important-bg: #2a0736;
--adm-important-border: #d33682;
--adm-warning-bg: #363007;
--adm-warning-border: #b58900;
--adm-caution-bg: #360a07;
--adm-caution-border: #cb4b16;
}

26
styles/solarized.root.css Normal file
View File

@@ -0,0 +1,26 @@
:root {
--bg: #fdf6e3;
--bg-deep: #eee8d5;
--fg: #657b83;
--fg-muted: #93a1a1;
--fg-link: #268bd2;
--fg-heading: #586e75;
--code-bg: #eee8d5;
--code-border: #d3cbb7;
--code-fg: #657b83;
--code-sel: #d33682;
--code-prop: #268bd2;
--code-val: #2aa198;
--code-var: #859900;
--code-com: #93a1a1;
--adm-note-bg: #eee8d5;
--adm-note-border: #268bd2;
--adm-tip-bg: #e8f5e0;
--adm-tip-border: #859900;
--adm-important-bg: #f0e8f5;
--adm-important-border: #6c71c4;
--adm-warning-bg: #fdf0e3;
--adm-warning-border: #b58900;
--adm-caution-bg: #fde8e8;
--adm-caution-border: #dc322f;
}