53 Commits

Author SHA1 Message Date
aac9198878 feat: draft_by_default and lang
All checks were successful
Lint / shellcheck (push) Successful in 36s
Release Standalone Builder / build (release) Successful in 31s
Release Standalone Builder / publish-aur (release) Successful in 34s
Release Standalone Builder / publish-homebrew (release) Successful in 7s
2026-03-30 09:55:26 +02:00
99e1f5dd24 fix: markdown embedding 2026-03-30 09:54:44 +02:00
3970c6eb47 fix: inline html 2026-03-30 09:54:23 +02:00
3075719963 feat: might be actually feature-complete by now
All checks were successful
Lint / shellcheck (push) Successful in 21s
Release Standalone Builder / build (release) Successful in 33s
Release Standalone Builder / publish-aur (release) Successful in 36s
Release Standalone Builder / publish-homebrew (release) Successful in 7s
2026-03-26 13:16:34 +01:00
0379d38234 docs: new docs
All checks were successful
Lint / shellcheck (push) Successful in 22s
Release Standalone Builder / build (release) Successful in 33s
Release Standalone Builder / publish-aur (release) Successful in 34s
Release Standalone Builder / publish-homebrew (release) Successful in 6s
2026-03-25 10:42:24 +01:00
b22897135e feat: --generate-template 2026-03-25 10:20:19 +01:00
185cf769c3 fix: shellcheck style 2026-03-25 10:07:04 +01:00
b2acd26660 feat: custom directory indexes 2026-03-25 10:02:22 +01:00
4069bafd52 fix: typo in codeblock handling that made it so headers appear
All checks were successful
Lint / shellcheck (push) Successful in 19s
Release Standalone Builder / build (release) Successful in 33s
Release Standalone Builder / publish-aur (release) Successful in 36s
Release Standalone Builder / publish-homebrew (release) Successful in 7s
2026-03-24 08:18:11 +01:00
9dbd41392e docs: update installation instructions link in readme
All checks were successful
Lint / shellcheck (push) Successful in 22s
2026-03-24 08:12:48 +01:00
35eac48dcd fix: v1.4.0 hotfix
All checks were successful
Lint / shellcheck (push) Successful in 22s
Release Standalone Builder / build (release) Successful in 32s
Release Standalone Builder / publish-aur (release) Successful in 35s
Release Standalone Builder / publish-homebrew (release) Successful in 6s
2026-03-23 12:09:54 +01:00
ef16ed4c88 feat: frontmatter
Some checks failed
Lint / shellcheck (push) Successful in 53s
Release Standalone Builder / publish-aur (release) Successful in 36s
Release Standalone Builder / publish-homebrew (release) Failing after 6s
Release Standalone Builder / build (release) Successful in 34s
2026-03-23 11:39:05 +01:00
30b7681234 Update index.md
All checks were successful
Lint / shellcheck (push) Successful in 19s
2026-03-22 07:24:36 +01:00
13b6106efd docs: bpkg
All checks were successful
Lint / shellcheck (push) Successful in 20s
2026-03-22 07:23:24 +01:00
831b081fc7 docs: new contribution instructions
All checks were successful
Lint / shellcheck (push) Successful in 19s
2026-03-21 16:10:33 +01:00
fde423a32b docs: brew
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-20 09:36:53 +01:00
55a82f75a9 fix: link in homebrew
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-20 09:33:45 +01:00
f85abd43c4 fix: brew
Some checks failed
Lint / shellcheck (push) Has been cancelled
Release Standalone Builder / build (release) Successful in 30s
Release Standalone Builder / publish-aur (release) Successful in 32s
Release Standalone Builder / publish-homebrew (release) Successful in 6s
2026-03-20 09:26:20 +01:00
0f66ebf52a dist: brew
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-20 09:23:43 +01:00
55a515ccd5 dist: bpkg preparation
All checks were successful
Lint / shellcheck (push) Successful in 18s
Release Standalone Builder / build (release) Successful in 31s
Release Standalone Builder / publish-aur (release) Successful in 34s
2026-03-20 08:46:13 +01:00
de8cbefb8e feat: task lists
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-20 08:32:28 +01:00
cc7fee573f feat: custom admonitions 2026-03-20 08:31:58 +01:00
137be9579a feat: incremental rebuilds 2026-03-20 08:30:25 +01:00
5afd0170e5 fix: less tmp file spam 2026-03-20 08:29:05 +01:00
5a2053cfb4 fix: publish-aur-git shoudln't run so often now
All checks were successful
Lint / shellcheck (push) Successful in 19s
2026-03-19 22:34:31 +01:00
2fc3d6fc6f fix: AUR links
All checks were successful
Lint / shellcheck (push) Successful in 18s
Publish kewt-git to AUR / publish-aur-git (push) Successful in 29s
Release Standalone Builder / build (release) Successful in 30s
Release Standalone Builder / publish-aur (release) Successful in 44s
2026-03-19 21:50:34 +01:00
c5a9355871 fix: config slashes
All checks were successful
Lint / shellcheck (push) Successful in 18s
Publish kewt-git to AUR / publish-aur-git (push) Successful in 30s
Release Standalone Builder / build (release) Successful in 31s
Release Standalone Builder / publish-aur (release) Successful in 34s
2026-03-19 21:45:38 +01:00
b8cd129c47 fix: rename to later in the alphabet, just in case
All checks were successful
Lint / shellcheck (push) Successful in 17s
Publish kewt-git to AUR / publish-aur-git (push) Successful in 33s
2026-03-19 21:29:22 +01:00
5e033a65e7 dist: AUR updates and upgrades
All checks were successful
Publish kewt-git to AUR / publish-aur-git (push) Successful in 34s
Lint / shellcheck (push) Successful in 20s
2026-03-19 21:26:17 +01:00
5bf2e2abe5 Update LICENSE
All checks were successful
Lint / shellcheck (push) Successful in 20s
2026-03-19 16:27:56 +01:00
3a2056ff8f Update site/site.conf
All checks were successful
Lint / shellcheck (push) Successful in 20s
2026-03-19 16:20:52 +01:00
fd829a3f22 branding: icon
All checks were successful
Lint / shellcheck (push) Successful in 19s
2026-03-19 16:12:21 +01:00
bad02decba docs: add contributing instructions
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-19 15:58:41 +01:00
2aef6ec4a1 Update site/site.conf
All checks were successful
Lint / shellcheck (push) Successful in 19s
2026-03-19 15:40:39 +01:00
78eac182dc dist: github release descriptions
All checks were successful
Lint / shellcheck (push) Successful in 19s
Release Standalone Builder / build (release) Successful in 30s
Release Standalone Builder / publish-aur (release) Successful in 33s
2026-03-19 15:37:55 +01:00
b7382a20ab feat: new default style and more
All checks were successful
Lint / shellcheck (push) Successful in 20s
2026-03-19 15:35:14 +01:00
d8cf07ee2a docs: move to only be on the website
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-19 15:25:01 +01:00
76e2ae0117 make automatic filesnames not use : for better windows support ig
All checks were successful
Lint / shellcheck (push) Successful in 20s
2026-03-19 09:35:14 +01:00
2e331b5d9a fix: to the fix, clap your hands together, caramelldanse o-o-or whatever
All checks were successful
Lint / shellcheck (push) Successful in 17s
Release Standalone Builder / build (release) Successful in 31s
Release Standalone Builder / publish-aur (release) Successful in 35s
2026-03-18 22:26:32 +01:00
9f5d1089a2 feat: --version, fix: --post fixes
All checks were successful
Lint / shellcheck (push) Successful in 19s
2026-03-18 22:18:13 +01:00
9ccba8fd4e feat/fix: Site.conf now uses escaped " so it's more conventional
All checks were successful
Lint / shellcheck (push) Successful in 19s
Release Standalone Builder / build (release) Successful in 31s
Release Standalone Builder / publish-aur (release) Successful in 36s
2026-03-18 08:49:41 +01:00
95679abd85 docs: update bianry download url
All checks were successful
Lint / shellcheck (push) Successful in 17s
2026-03-17 13:54:13 +01:00
8b1e793510 docs: new features
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-17 12:56:15 +01:00
19f96553d9 feat: RSS!!!!!!!!!!
All checks were successful
Lint / shellcheck (push) Successful in 17s
Release Standalone Builder / build (release) Successful in 32s
Release Standalone Builder / publish-aur (release) Successful in 33s
2026-03-17 12:46:29 +01:00
1a7525a857 feat: config updating
All checks were successful
Lint / shellcheck (push) Successful in 17s
2026-03-17 11:53:15 +01:00
e7d90d18e8 feat: sitemap 2026-03-17 11:37:17 +01:00
4019d2721d Header links 2026-03-17 11:28:47 +01:00
b58604a4cf feat/docs: LICENSE
All checks were successful
Lint / shellcheck (push) Successful in 17s
2026-03-17 11:17:26 +01:00
99e805b180 fix: Title formatting parsing fix
All checks were successful
Lint / shellcheck (push) Successful in 18s
2026-03-17 02:18:42 +01:00
62075dea4a docs: forgot
All checks were successful
Lint / shellcheck (push) Successful in 17s
2026-03-17 02:07:52 +01:00
7afd041e53 functionality: Dynamic page titles, versioning, automatic 404 page
All checks were successful
Lint / shellcheck (push) Successful in 17s
Release Standalone Builder / build (release) Successful in 32s
Release Standalone Builder / publish-aur (release) Successful in 33s
2026-03-17 01:55:32 +01:00
64d08a0de3 functionality: BetterHelp
All checks were successful
Lint / shellcheck (push) Successful in 19s
Release Standalone Builder / build (release) Successful in 32s
Release Standalone Builder / publish-aur (release) Successful in 35s
2026-03-16 10:33:21 +01:00
dd18bc3367 docs: update readme
All checks were successful
Lint / shellcheck (push) Successful in 39s
2026-03-16 09:33:45 +01:00
46 changed files with 2694 additions and 384 deletions

View File

@@ -0,0 +1,38 @@
name: Publish kewt-git to AUR
on:
push:
branches:
- main
paths:
- 'packaging/AUR/PKGBUILD.git'
- 'packaging/AUR/.SRCINFO.git'
workflow_dispatch:
jobs:
publish-aur-git:
runs-on: local
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Setup Arch Linux environment
run: |
sudo apt-get update
sudo apt-get install -y pacman-package-manager curl jq || true
- name: Prepare AUR files
run: |
mkdir -p aur-work
cp packaging/AUR/PKGBUILD.git aur-work/PKGBUILD
cp packaging/AUR/.SRCINFO.git aur-work/.SRCINFO
- name: Publish to AUR
uses: KSXGitHub/github-actions-deploy-aur@v3.0.1
with:
pkgname: kewt-git
pkgbuild: ./aur-work/PKGBUILD
commit_username: ${{ github.actor }}
commit_email: ${{ github.actor }}@users.noreply.github.com
ssh_private_key: ${{ secrets.AUR_SSH_PRIVATE_KEY }}
commit_message: "Update kewt-git to ${{ github.sha }}"

View File

@@ -3,6 +3,7 @@ name: Release Standalone Builder
on: on:
release: release:
types: [published] types: [published]
workflow_dispatch:
jobs: jobs:
build: build:
@@ -32,12 +33,24 @@ jobs:
run: | run: |
TAG="${GITHUB_REF#refs/tags/}" TAG="${GITHUB_REF#refs/tags/}"
# Fetch release body from Gitea
RELEASE_BODY=$(curl -sL \
"https://git.krzak.org/api/v1/repos/N0VA/kewt/releases/tags/${TAG}" \
| jq -r '.body // ""')
# Build JSON payload
PAYLOAD=$(jq -n \
--arg tag "$TAG" \
--arg name "Release $TAG" \
--arg body "$RELEASE_BODY" \
'{tag_name: $tag, name: $name, body: $body, draft: false, prerelease: false}')
# Create the release on GitHub # Create the release on GitHub
curl -sL -X POST \ curl -sL -X POST \
-H "Authorization: token ${{ secrets.GH_RELEASE_TOKEN }}" \ -H "Authorization: token ${{ secrets.GH_RELEASE_TOKEN }}" \
-H "Accept: application/vnd.github+json" \ -H "Accept: application/vnd.github+json" \
"https://api.github.com/repos/n0va-bot/kewt/releases" \ "https://api.github.com/repos/n0va-bot/kewt/releases" \
-d "{\"tag_name\":\"${TAG}\",\"name\":\"${TAG}\",\"draft\":false,\"prerelease\":false}" || true -d "$PAYLOAD" || true
# Get the release ID # Get the release ID
RELEASE_ID=$(curl -sL \ RELEASE_ID=$(curl -sL \
@@ -78,25 +91,9 @@ jobs:
-e "s/SHA256SUM_PLACEHOLDER/${CHECKSUM}/g" \ -e "s/SHA256SUM_PLACEHOLDER/${CHECKSUM}/g" \
packaging/AUR/PKGBUILD.template > aur-work/PKGBUILD packaging/AUR/PKGBUILD.template > aur-work/PKGBUILD
cat > aur-work/.SRCINFO << SRCEOF sed -e "s/VERSION_PLACEHOLDER/${VERSION}/g" \
pkgbase = kewt-bin -e "s/SHA256SUM_PLACEHOLDER/${CHECKSUM}/g" \
pkgdesc = A minimalist, 100% POSIX, static site generator inspired by werc and kew packaging/AUR/.SRCINFO.template > aur-work/.SRCINFO
pkgver = ${VERSION}
pkgrel = 1
url = https://git.krzak.org/N0VA/kewt
arch = any
license = MIT
depends = sh
provides = kewt
conflicts = kewt
conflicts = kewt-git
source = kewt-bin-${VERSION}.sh::https://git.krzak.org/N0VA/kewt/releases/download/v${VERSION}/kewt
sha256sums = ${CHECKSUM}
pkgname = kewt-bin
SRCEOF
# Remove leading whitespace from heredoc
sed -i 's/^ //' aur-work/.SRCINFO
- name: Publish to AUR - name: Publish to AUR
uses: KSXGitHub/github-actions-deploy-aur@v3.0.1 uses: KSXGitHub/github-actions-deploy-aur@v3.0.1
@@ -107,3 +104,34 @@ jobs:
commit_email: ${{ github.actor }}@users.noreply.github.com commit_email: ${{ github.actor }}@users.noreply.github.com
ssh_private_key: ${{ secrets.AUR_SSH_PRIVATE_KEY }} ssh_private_key: ${{ secrets.AUR_SSH_PRIVATE_KEY }}
commit_message: "Update kewt-bin to ${{ github.ref_name }}" commit_message: "Update kewt-bin to ${{ github.ref_name }}"
publish-homebrew:
runs-on: local
needs: build
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Render Formula and push to GitHub
run: |
TAG="${GITHUB_REF#refs/tags/}"
VERSION="${TAG#v}"
curl -sL -o kewt-binary \
"https://git.krzak.org/N0VA/kewt/releases/download/${TAG}/kewt"
CHECKSUM=$(sha256sum kewt-binary | awk '{print $1}')
rm -f kewt-binary
git clone https://x-access-token:${{ secrets.GH_RELEASE_TOKEN }}@github.com/n0va-bot/homebrew-tap.git brew-work || true
mkdir -p brew-work/Formula
sed -e "s/VERSION_PLACEHOLDER/${VERSION}/g" \
-e "s/SHA256SUM_PLACEHOLDER/${CHECKSUM}/g" \
packaging/homebrew/kewt.rb.template > brew-work/Formula/kewt.rb
cd brew-work
[ -d .git ] || { git init && git checkout --orphan main && git remote add origin https://x-access-token:${{ secrets.GH_RELEASE_TOKEN }}@github.com/n0va-bot/homebrew-tap.git; }
git add Formula/kewt.rb
git config user.name "${{ github.actor }}"
git config user.email "${{ github.actor }}@users.noreply.github.com"
git commit -m "Update kewt to ${TAG}" || echo "No changes to commit"
git push -u origin main

34
LICENSE Normal file
View File

@@ -0,0 +1,34 @@
ISC License
Copyright 2026 N0\A
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA
OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
PERFORMANCE OF THIS SOFTWARE.
---
This project incorporates code (CSS style) from the 'kew' project,
which is also licensed under the ISC License: Copyright (c) 2026 uint23
Permission to use, copy, modify, and/or distribute this software for any
purpose with or without fee is hereby granted, provided that the above
copyright notice and this permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.

19
Makefile Normal file
View File

@@ -0,0 +1,19 @@
PREFIX ?= /usr/local
BINDIR = $(PREFIX)/bin
all: kewt
kewt:
./tools/build-standalone.sh
install: kewt
install -d $(DESTDIR)$(BINDIR)
install -m 755 kewt $(DESTDIR)$(BINDIR)/kewt
uninstall:
rm -f $(DESTDIR)$(BINDIR)/kewt
clean:
rm -f kewt
.PHONY: all install uninstall clean

106
README.md
View File

@@ -1,107 +1,21 @@
# _kewt_ # ![kewt](/icon.svg)
### Pronounced "cute" ### Pronounced "cute"
***
# [Go to the website](https://kewt.krzak.org) # [Go to the website](https://kewt.krzak.org)
***
_kewt_ is a minimalist ssg inspired by _[werc](http://werc.cat-v.org/)_ and _[kew](https://github.com/uint23/kew)_ _kewt_ is a minimalist ssg inspired by _[werc](http://werc.cat-v.org/)_ and _[kew](https://github.com/uint23/kew)_
It's meant to be a static site generator, like _[kew](https://github.com/uint23/kew)_ but use only default (POSIX) tooling, like _[werc](http://werc.cat-v.org/)_ (and definitely unlike _[kew](https://github.com/uint23/kew)_)
## Features ## [Installation](https://kewt.krzak.org/#installation)
- No dependencies ## Contributing
- Supports many embed types
- Automatic css variable replacement for older browsers
- Automatic inlining and embedding of many filetypes with `\![link]` or `\![alt](link)`
- Inline html support
- MFM `$font` and `\<plain>` tags
- Admonition support (that's what the blocks like the warning block below are called)
If you want to **force** a file to be inlined, use `\!![]` instead of `\![]` Either open an issue or pull request on the **home** repository ([N0VA/kewt](https://git.krzak.org/N0VA/kewt)) or message me on my email address ([n0va@krzak.org](mailto:n0va@krzak.org?subject=%5Bkewt%5D%20something)) with the subjectline being `[kewt] something`.
## Installation ## License
You can clone the repository to use `kewt.sh` directly, or you can download the standalone executable, which bundles all dependencies into a single file: ISC
```sh
curl -L -o kewt https://git.krzak.org/N0VA/kewt/releases/latest/download/kewt
chmod +x kewt
```
On Arch Linux, _kewt_ is available on the AUR:
- [kewt-bin](https://aur.archlinux.org/packages/kewt-bin) — prebuilt standalone binary from the latest release
- [kewt-git](https://aur.archlinux.org/packages/kewt-git) — built from the latest git source
## Usage
```sh
./kewt.sh --help
./kewt.sh --new [title]
./kewt.sh --from <src> --to <out>
./kewt.sh [src] [out]
```
`--new [title]` creates a new site directory with a copied `site.conf` and a default `index.md`.
## site.conf
```conf
title = "kewt"
style = "kewt"
dir_indexes = true
single_file_index = true
flatten = false
order = ""
home_name = "Home"
show_home_in_nav = true
nav_links = ""
nav_extra = ""
footer = "made with <a href="https://kewt.krzak.org">kewt</a>"
logo = ""
display_logo = false
display_title = true
logo_as_favicon = true
favicon = ""
```
- `title` site title
- `style` style file name from `./styles` (without `.css`)
- `dir_indexes` generate directory index pages when missing `index.md`
- `single_file_index` if a directory has one markdown file and no `index.md`, use that file as `index.html`
- `flatten` flatten sidebar directory levels
- `order` comma separated file/directory name list to order the sidebar (alphabetical by default)
- `home_name` text for the home link in navigation (default: "Home")
- `show_home_in_nav` show home link in navigation (default: true)
- `nav_links` comma separated extra nav links, as bare URLs or Markdown links like `[Label](https://example.com)`
- `nav_extra` raw HTML appended inside the `<nav>` after the generated link list
- `footer` footer html/text shown at the bottom of pages
- `logo` logo image path (used in header if enabled)
- `display_logo` show logo in header
- `display_title` show title text in header
- `logo_as_favicon` use `logo` as favicon
- `favicon` explicit favicon path (used when `logo_as_favicon` is false or no logo is set)
## Ignores
- `.kewtignore`: Files/directories to ignore. If empty, the whole directory gets ignored
- `.kewthide`: Files/directories to hide from navigation but still process. Same empty rules as with ignore
- `.kewtpreserve`: Files/directories to copy but not convert markdown to html. Same empty rules again
## Embeds
- `\![link]`:
- local image/audio/video files are embedded as media tags
- local text/code files are inlined directly
- global image/audio/video links are embedded as media tags
- other global links are embedded as `<iframe>`
- `\![alt](link)` works the same, with `alt` used for images
- `\!![]` and `\!![alt](link)` force inline local file contents
## Credits
- Markdown to html conversion based on [markdown.bash](https://github.com/chadbraunduin/markdown.bash) by [chadbraunduin](https://github.com/chadbraunduin)
- Default css style and html template based on _[kew](https://github.com/uint23/kew)_ by [uint23](https://github.com/uint23)
>[!WARNING]
>Most of this was coded at night, while sleepy and a bit sick, and after walking for about 4 hours around a forest, so...

View File

@@ -22,7 +22,19 @@ END {
sub(/^\[!/, "", kind) sub(/^\[!/, "", kind)
sub(/\]$/, "", kind) sub(/\]$/, "", kind)
lkind = tolower(kind) lkind = tolower(kind)
if (lkind == "note" || lkind == "tip" || lkind == "important" || lkind == "warning" || lkind == "caution") { is_valid = 0
if (custom_admonitions != "") {
n = split(tolower(custom_admonitions), adms, ",")
for (idx = 1; idx <= n; idx++) {
adm = adms[idx]
sub(/^[ \t]+/, "", adm)
sub(/[ \t]+$/, "", adm)
if (lkind == adm) { is_valid = 1; break }
}
} else if (lkind == "note" || lkind == "tip" || lkind == "important" || lkind == "warning" || lkind == "caution") {
is_valid = 1
}
if (is_valid) {
print "<div class=\"admonition admonition-" lkind "\">" print "<div class=\"admonition admonition-" lkind "\">"
print "<p class=\"admonition-title\">" cap(lkind) "</p>" print "<p class=\"admonition-title\">" cap(lkind) "</p>"
has_body = 0 has_body = 0

View File

@@ -1,4 +1,5 @@
BEGIN { BEGIN {
src = ENVIRON["AWK_SRC"]
slen = length(src) slen = length(src)
} }

62
awk/definition_lists.awk Normal file
View File

@@ -0,0 +1,62 @@
BEGIN {
in_dl = 0
has_prev = 0
prev_line = ""
in_pre = 0
}
{
if ($0 ~ /^<pre>/) {
if (!in_pre) in_pre = 1
}
if (!in_pre && $0 ~ /^:[ \t]+[^ \t]/) {
if (!in_dl) {
in_dl = 1
print "<dl>"
print "<dt>" prev_line "</dt>"
has_prev = 0
} else {
if (has_prev && prev_line != "") {
print "<dt>" prev_line "</dt>"
has_prev = 0
}
}
def_text = $0
sub(/^:[ \t]+/, "", def_text)
print "<dd>" def_text "</dd>"
if ($0 ~ /<\/pre>/) {
if (in_pre) in_pre = 0
}
next
} else {
if (in_dl) {
if ($0 == "") {
# End of definition list
print "</dl>"
in_dl = 0
print ""
has_prev = 0
next
}
}
if (has_prev) {
print prev_line
}
prev_line = $0
has_prev = 1
}
if ($0 ~ /<\/pre>/) {
if (in_pre) in_pre = 0
}
}
END {
if (in_dl) {
print "</dl>"
} else {
if (has_prev) {
print prev_line
}
}
}

857
awk/emoji.awk Normal file
View File

@@ -0,0 +1,857 @@
BEGIN {
map[":+1:"] = "👍"
map[":100:"] = "💯"
map[":1234:"] = "🔢"
map[":8ball:"] = "🎱"
map[":a:"] = "🅰️"
map[":ab:"] = "🆎"
map[":abc:"] = "🔤"
map[":abcd:"] = "🔡"
map[":accept:"] = "🉑"
map[":aerial_tramway:"] = "🚡"
map[":airplane:"] = "✈️"
map[":alarm_clock:"] = "⏰"
map[":alien:"] = "👽"
map[":ambulance:"] = "🚑"
map[":anchor:"] = "⚓"
map[":angel:"] = "👼"
map[":anger:"] = "💢"
map[":angry:"] = "😠"
map[":anguished:"] = "😧"
map[":ant:"] = "🐜"
map[":apple:"] = "🍎"
map[":aquarius:"] = "♒"
map[":aries:"] = "♈"
map[":arrow_backward:"] = "◀️"
map[":arrow_double_down:"] = "⏬"
map[":arrow_double_up:"] = "⏫"
map[":arrow_down:"] = "⬇️"
map[":arrow_down_small:"] = "🔽"
map[":arrow_forward:"] = "▶️"
map[":arrow_heading_down:"] = "⤵️"
map[":arrow_heading_up:"] = "⤴️"
map[":arrow_left:"] = "⬅️"
map[":arrow_lower_left:"] = "↙️"
map[":arrow_lower_right:"] = "↘️"
map[":arrow_right:"] = "➡️"
map[":arrow_right_hook:"] = "↪️"
map[":arrow_up:"] = "⬆️"
map[":arrow_up_down:"] = "↕️"
map[":arrow_up_small:"] = "🔼"
map[":arrow_upper_left:"] = "↖️"
map[":arrow_upper_right:"] = "↗️"
map[":arrows_clockwise:"] = "🔃"
map[":arrows_counterclockwise:"] = "🔄"
map[":art:"] = "🎨"
map[":articulated_lorry:"] = "🚛"
map[":astonished:"] = "😲"
map[":atm:"] = "🏧"
map[":b:"] = "🅱️"
map[":baby:"] = "👶"
map[":baby_bottle:"] = "🍼"
map[":baby_chick:"] = "🐤"
map[":baby_symbol:"] = "🚼"
map[":baggage_claim:"] = "🛄"
map[":balloon:"] = "🎈"
map[":ballot_box_with_check:"] = "☑️"
map[":bamboo:"] = "🎍"
map[":banana:"] = "🍌"
map[":bangbang:"] = "‼️"
map[":bank:"] = "🏦"
map[":bar_chart:"] = "📊"
map[":barber:"] = "💈"
map[":baseball:"] = "⚾"
map[":basketball:"] = "🏀"
map[":bath:"] = "🛀"
map[":bathtub:"] = "🛁"
map[":battery:"] = "🔋"
map[":bear:"] = "🐻"
map[":beer:"] = "🍺"
map[":beers:"] = "🍻"
map[":beetle:"] = "🪲"
map[":beginner:"] = "🔰"
map[":bell:"] = "🔔"
map[":bento:"] = "🍱"
map[":bicyclist:"] = "🚴"
map[":bike:"] = "🚲"
map[":bikini:"] = "👙"
map[":bird:"] = "🐦"
map[":birthday:"] = "🎂"
map[":black_circle:"] = "⚫"
map[":black_joker:"] = "🃏"
map[":black_nib:"] = "✒️"
map[":black_square_button:"] = "🔲"
map[":blossom:"] = "🌼"
map[":blowfish:"] = "🐡"
map[":blue_book:"] = "📘"
map[":blue_car:"] = "🚙"
map[":blue_heart:"] = "💙"
map[":blush:"] = "😊"
map[":boar:"] = "🐗"
map[":boat:"] = "⛵"
map[":bomb:"] = "💣"
map[":book:"] = "📖"
map[":bookmark:"] = "🔖"
map[":bookmark_tabs:"] = "📑"
map[":books:"] = "📚"
map[":boom:"] = "💥"
map[":boot:"] = "👢"
map[":bouquet:"] = "💐"
map[":bow:"] = "🙇"
map[":bowling:"] = "🎳"
map[":boy:"] = "👦"
map[":bread:"] = "🍞"
map[":bride_with_veil:"] = "👰‍♀️"
map[":bridge_at_night:"] = "🌉"
map[":briefcase:"] = "💼"
map[":broken_heart:"] = "💔"
map[":bug:"] = "🐛"
map[":bulb:"] = "💡"
map[":bullettrain_front:"] = "🚅"
map[":bullettrain_side:"] = "🚄"
map[":bus:"] = "🚌"
map[":busstop:"] = "🚏"
map[":bust_in_silhouette:"] = "👤"
map[":busts_in_silhouette:"] = "👥"
map[":cactus:"] = "🌵"
map[":cake:"] = "🍰"
map[":calendar:"] = "📆"
map[":calling:"] = "📲"
map[":camel:"] = "🐫"
map[":camera:"] = "📷"
map[":cancer:"] = "♋"
map[":candy:"] = "🍬"
map[":capital_abcd:"] = "🔠"
map[":capricorn:"] = "♑"
map[":car:"] = "🚗"
map[":card_index:"] = "📇"
map[":carousel_horse:"] = "🎠"
map[":cat2:"] = "🐈"
map[":cat:"] = "🐱"
map[":cd:"] = "💿"
map[":chart:"] = "💹"
map[":chart_with_downwards_trend:"] = "📉"
map[":chart_with_upwards_trend:"] = "📈"
map[":checkered_flag:"] = "🏁"
map[":cherries:"] = "🍒"
map[":cherry_blossom:"] = "🌸"
map[":chestnut:"] = "🌰"
map[":chicken:"] = "🐔"
map[":children_crossing:"] = "🚸"
map[":chocolate_bar:"] = "🍫"
map[":christmas_tree:"] = "🎄"
map[":church:"] = "⛪"
map[":cinema:"] = "🎦"
map[":circus_tent:"] = "🎪"
map[":city_sunrise:"] = "🌇"
map[":city_sunset:"] = "🌆"
map[":cl:"] = "🆑"
map[":clap:"] = "👏"
map[":clapper:"] = "🎬"
map[":clipboard:"] = "📋"
map[":clock1030:"] = "🕥"
map[":clock10:"] = "🕙"
map[":clock1130:"] = "🕦"
map[":clock11:"] = "🕚"
map[":clock1230:"] = "🕧"
map[":clock12:"] = "🕛"
map[":clock130:"] = "🕜"
map[":clock1:"] = "🕐"
map[":clock230:"] = "🕝"
map[":clock2:"] = "🕑"
map[":clock330:"] = "🕞"
map[":clock3:"] = "🕒"
map[":clock430:"] = "🕟"
map[":clock4:"] = "🕓"
map[":clock530:"] = "🕠"
map[":clock5:"] = "🕔"
map[":clock630:"] = "🕡"
map[":clock6:"] = "🕕"
map[":clock730:"] = "🕢"
map[":clock7:"] = "🕖"
map[":clock830:"] = "🕣"
map[":clock8:"] = "🕗"
map[":clock930:"] = "🕤"
map[":clock9:"] = "🕘"
map[":closed_book:"] = "📕"
map[":closed_lock_with_key:"] = "🔐"
map[":closed_umbrella:"] = "🌂"
map[":cloud:"] = "☁️"
map[":clubs:"] = "♣️"
map[":cn:"] = "🇨🇳"
map[":cocktail:"] = "🍸"
map[":coffee:"] = "☕"
map[":cold_sweat:"] = "😰"
map[":computer:"] = "💻"
map[":confetti_ball:"] = "🎊"
map[":confounded:"] = "😖"
map[":confused:"] = "😕"
map[":congratulations:"] = "㊗️"
map[":construction:"] = "🚧"
map[":construction_worker:"] = "👷"
map[":convenience_store:"] = "🏪"
map[":cookie:"] = "🍪"
map[":cool:"] = "🆒"
map[":cop:"] = "👮"
map[":copyright:"] = "©️"
map[":corn:"] = "🌽"
map[":couple:"] = "👫"
map[":couple_with_heart:"] = "💑"
map[":couplekiss:"] = "💏"
map[":cow2:"] = "🐄"
map[":cow:"] = "🐮"
map[":credit_card:"] = "💳"
map[":crocodile:"] = "🐊"
map[":crossed_flags:"] = "🎌"
map[":crown:"] = "👑"
map[":cry:"] = "😢"
map[":crying_cat_face:"] = "😿"
map[":crystal_ball:"] = "🔮"
map[":cupid:"] = "💘"
map[":curly_loop:"] = "➰"
map[":currency_exchange:"] = "💱"
map[":curry:"] = "🍛"
map[":custard:"] = "🍮"
map[":customs:"] = "🛃"
map[":cyclone:"] = "🌀"
map[":dancer:"] = "💃"
map[":dancers:"] = "👯"
map[":dango:"] = "🍡"
map[":dart:"] = "🎯"
map[":dash:"] = "💨"
map[":date:"] = "📅"
map[":de:"] = "🇩🇪"
map[":deciduous_tree:"] = "🌳"
map[":department_store:"] = "🏬"
map[":diamond_shape_with_a_dot_inside:"] = "💠"
map[":diamonds:"] = "♦️"
map[":disappointed:"] = "😞"
map[":disappointed_relieved:"] = "😥"
map[":dizzy:"] = "💫"
map[":dizzy_face:"] = "😵"
map[":do_not_litter:"] = "🚯"
map[":dog2:"] = "🐕"
map[":dog:"] = "🐶"
map[":dollar:"] = "💵"
map[":dolls:"] = "🎎"
map[":dolphin:"] = "🐬"
map[":door:"] = "🚪"
map[":doughnut:"] = "🍩"
map[":dragon:"] = "🐉"
map[":dragon_face:"] = "🐲"
map[":dress:"] = "👗"
map[":dromedary_camel:"] = "🐪"
map[":droplet:"] = "💧"
map[":dvd:"] = "📀"
map[":ear:"] = "👂"
map[":ear_of_rice:"] = "🌾"
map[":earth_africa:"] = "🌍"
map[":earth_americas:"] = "🌎"
map[":earth_asia:"] = "🌏"
map[":egg:"] = "🥚"
map[":eggplant:"] = "🍆"
map[":eight:"] = "8⃣"
map[":eight_pointed_black_star:"] = "✴️"
map[":eight_spoked_asterisk:"] = "✳️"
map[":electric_plug:"] = "🔌"
map[":elephant:"] = "🐘"
map[":email:"] = "📧"
map[":end:"] = "🔚"
map[":envelope:"] = "✉️"
map[":es:"] = "🇪🇸"
map[":euro:"] = "💶"
map[":european_castle:"] = "🏰"
map[":european_post_office:"] = "🏤"
map[":evergreen_tree:"] = "🌲"
map[":exclamation:"] = "❗"
map[":expressionless:"] = "😑"
map[":eyeglasses:"] = "👓"
map[":eyes:"] = "👀"
map[":factory:"] = "🏭"
map[":fallen_leaf:"] = "🍂"
map[":family:"] = "👪"
map[":fast_forward:"] = "⏩"
map[":fax:"] = "📠"
map[":fearful:"] = "😨"
map[":feet:"] = "🐾"
map[":ferris_wheel:"] = "🎡"
map[":file_folder:"] = "📁"
map[":fire:"] = "🔥"
map[":fire_engine:"] = "🚒"
map[":fireworks:"] = "🎆"
map[":first_quarter_moon:"] = "🌓"
map[":first_quarter_moon_with_face:"] = "🌛"
map[":fish:"] = "🐟"
map[":fish_cake:"] = "🍥"
map[":fishing_pole_and_fish:"] = "🎣"
map[":fist:"] = "✊"
map[":five:"] = "5⃣"
map[":flags:"] = "🎏"
map[":flashlight:"] = "🔦"
map[":floppy_disk:"] = "💾"
map[":flower_playing_cards:"] = "🎴"
map[":flushed:"] = "😳"
map[":foggy:"] = "🌁"
map[":football:"] = "🏈"
map[":fork_and_knife:"] = "🍴"
map[":fountain:"] = "⛲"
map[":four:"] = "4⃣"
map[":four_leaf_clover:"] = "🍀"
map[":fr:"] = "🇫🇷"
map[":free:"] = "🆓"
map[":fried_shrimp:"] = "🍤"
map[":fries:"] = "🍟"
map[":frog:"] = "🐸"
map[":frowning:"] = "😦"
map[":fuelpump:"] = "⛽"
map[":full_moon:"] = "🌕"
map[":full_moon_with_face:"] = "🌝"
map[":game_die:"] = "🎲"
map[":gem:"] = "💎"
map[":gemini:"] = "♊"
map[":ghost:"] = "👻"
map[":gift:"] = "🎁"
map[":gift_heart:"] = "💝"
map[":girl:"] = "👧"
map[":globe_with_meridians:"] = "🌐"
map[":goat:"] = "🐐"
map[":golf:"] = "⛳"
map[":grapes:"] = "🍇"
map[":green_apple:"] = "🍏"
map[":green_book:"] = "📗"
map[":green_heart:"] = "💚"
map[":grey_exclamation:"] = "❕"
map[":grey_question:"] = "❔"
map[":grimacing:"] = "😬"
map[":grin:"] = "😁"
map[":grinning:"] = "😀"
map[":guardsman:"] = "💂‍♂️"
map[":guitar:"] = "🎸"
map[":gun:"] = "🔫"
map[":haircut:"] = "💇"
map[":hamburger:"] = "🍔"
map[":hammer:"] = "🔨"
map[":hamster:"] = "🐹"
map[":handbag:"] = "👜"
map[":hankey:"] = "💩"
map[":hash:"] = "#️⃣"
map[":hatched_chick:"] = "🐥"
map[":hatching_chick:"] = "🐣"
map[":headphones:"] = "🎧"
map[":hear_no_evil:"] = "🙉"
map[":heart:"] = "❤️"
map[":heart_decoration:"] = "💟"
map[":heart_eyes:"] = "😍"
map[":heart_eyes_cat:"] = "😻"
map[":heartbeat:"] = "💓"
map[":hearts:"] = "♥️"
map[":heavy_check_mark:"] = "✔️"
map[":heavy_division_sign:"] = "➗"
map[":heavy_dollar_sign:"] = "💲"
map[":heavy_minus_sign:"] = ""
map[":heavy_multiplication_x:"] = "✖️"
map[":heavy_plus_sign:"] = ""
map[":helicopter:"] = "🚁"
map[":herb:"] = "🌿"
map[":hibiscus:"] = "🌺"
map[":high_brightness:"] = "🔆"
map[":high_heel:"] = "👠"
map[":honey_pot:"] = "🍯"
map[":horse:"] = "🐴"
map[":horse_racing:"] = "🏇"
map[":hospital:"] = "🏥"
map[":hotel:"] = "🏨"
map[":hotsprings:"] = "♨️"
map[":hourglass:"] = "⌛"
map[":hourglass_flowing_sand:"] = "⏳"
map[":house:"] = "🏠"
map[":house_with_garden:"] = "🏡"
map[":hushed:"] = "😯"
map[":ice_cream:"] = "🍨"
map[":icecream:"] = "🍦"
map[":id:"] = "🆔"
map[":ideograph_advantage:"] = "🉐"
map[":imp:"] = "👿"
map[":inbox_tray:"] = "📥"
map[":incoming_envelope:"] = "📨"
map[":information_desk_person:"] = "💁"
map[":information_source:"] = ""
map[":innocent:"] = "😇"
map[":interrobang:"] = "⁉️"
map[":iphone:"] = "📱"
map[":it:"] = "🇮🇹"
map[":izakaya_lantern:"] = "🏮"
map[":jack_o_lantern:"] = "🎃"
map[":japan:"] = "🗾"
map[":japanese_castle:"] = "🏯"
map[":japanese_goblin:"] = "👺"
map[":japanese_ogre:"] = "👹"
map[":jeans:"] = "👖"
map[":joy:"] = "😂"
map[":joy_cat:"] = "😹"
map[":jp:"] = "🇯🇵"
map[":key:"] = "🔑"
map[":keycap_ten:"] = "🔟"
map[":kimono:"] = "👘"
map[":kiss:"] = "💋"
map[":kissing:"] = "😗"
map[":kissing_cat:"] = "😽"
map[":kissing_closed_eyes:"] = "😚"
map[":kissing_heart:"] = "😘"
map[":kissing_smiling_eyes:"] = "😙"
map[":koala:"] = "🐨"
map[":koko:"] = "🈁"
map[":kr:"] = "🇰🇷"
map[":large_blue_diamond:"] = "🔷"
map[":large_orange_diamond:"] = "🔶"
map[":last_quarter_moon:"] = "🌗"
map[":last_quarter_moon_with_face:"] = "🌜"
map[":laughing:"] = "😆"
map[":leaves:"] = "🍃"
map[":ledger:"] = "📒"
map[":left_luggage:"] = "🛅"
map[":left_right_arrow:"] = "↔️"
map[":leftwards_arrow_with_hook:"] = "↩️"
map[":lemon:"] = "🍋"
map[":leo:"] = "♌"
map[":leopard:"] = "🐆"
map[":libra:"] = "♎"
map[":light_rail:"] = "🚈"
map[":link:"] = "🔗"
map[":lips:"] = "👄"
map[":lipstick:"] = "💄"
map[":lock:"] = "🔒"
map[":lock_with_ink_pen:"] = "🔏"
map[":lollipop:"] = "🍭"
map[":loop:"] = "➿"
map[":loudspeaker:"] = "📢"
map[":love_hotel:"] = "🏩"
map[":love_letter:"] = "💌"
map[":low_brightness:"] = "🔅"
map[":m:"] = "Ⓜ️"
map[":mag:"] = "🔍"
map[":mag_right:"] = "🔎"
map[":mahjong:"] = "🀄"
map[":mailbox:"] = "📫"
map[":mailbox_closed:"] = "📪"
map[":mailbox_with_mail:"] = "📬"
map[":mailbox_with_no_mail:"] = "📭"
map[":man:"] = "👨"
map[":man_with_gua_pi_mao:"] = "👲"
map[":man_with_turban:"] = "👳‍♂️"
map[":mans_shoe:"] = "👞"
map[":maple_leaf:"] = "🍁"
map[":mask:"] = "😷"
map[":massage:"] = "💆"
map[":meat_on_bone:"] = "🍖"
map[":mega:"] = "📣"
map[":melon:"] = "🍈"
map[":memo:"] = "📝"
map[":mens:"] = "🚹"
map[":metal:"] = "🤘"
map[":metro:"] = "🚇"
map[":microphone:"] = "🎤"
map[":microscope:"] = "🔬"
map[":milky_way:"] = "🌌"
map[":minibus:"] = "🚐"
map[":minidisc:"] = "💽"
map[":mobile_phone_off:"] = "📴"
map[":money_with_wings:"] = "💸"
map[":moneybag:"] = "💰"
map[":monkey:"] = "🐒"
map[":monkey_face:"] = "🐵"
map[":monorail:"] = "🚝"
map[":mortar_board:"] = "🎓"
map[":mount_fuji:"] = "🗻"
map[":mountain_bicyclist:"] = "🚵"
map[":mountain_cableway:"] = "🚠"
map[":mountain_railway:"] = "🚞"
map[":mouse2:"] = "🐁"
map[":mouse:"] = "🐭"
map[":movie_camera:"] = "🎥"
map[":moyai:"] = "🗿"
map[":muscle:"] = "💪"
map[":mushroom:"] = "🍄"
map[":musical_keyboard:"] = "🎹"
map[":musical_note:"] = "🎵"
map[":musical_score:"] = "🎼"
map[":mute:"] = "🔇"
map[":nail_care:"] = "💅"
map[":name_badge:"] = "📛"
map[":necktie:"] = "👔"
map[":negative_squared_cross_mark:"] = "❎"
map[":neutral_face:"] = "😐"
map[":new:"] = "🆕"
map[":new_moon:"] = "🌑"
map[":new_moon_with_face:"] = "🌚"
map[":newspaper:"] = "📰"
map[":ng:"] = "🆖"
map[":nine:"] = "9⃣"
map[":no_bell:"] = "🔕"
map[":no_bicycles:"] = "🚳"
map[":no_entry:"] = "⛔"
map[":no_entry_sign:"] = "🚫"
map[":no_good:"] = "🙅"
map[":no_mobile_phones:"] = "📵"
map[":no_mouth:"] = "😶"
map[":no_pedestrians:"] = "🚷"
map[":no_smoking:"] = "🚭"
map[":nose:"] = "👃"
map[":notebook:"] = "📓"
map[":notebook_with_decorative_cover:"] = "📔"
map[":notes:"] = "🎶"
map[":nut_and_bolt:"] = "🔩"
map[":o2:"] = "🅾️"
map[":o:"] = "⭕"
map[":ocean:"] = "🌊"
map[":octopus:"] = "🐙"
map[":oden:"] = "🍢"
map[":office:"] = "🏢"
map[":ok:"] = "🆗"
map[":ok_hand:"] = "👌"
map[":ok_woman:"] = "🙆‍♀️"
map[":older_man:"] = "👴"
map[":older_woman:"] = "👵"
map[":on:"] = "🔛"
map[":oncoming_automobile:"] = "🚘"
map[":oncoming_bus:"] = "🚍"
map[":oncoming_police_car:"] = "🚔"
map[":oncoming_taxi:"] = "🚖"
map[":one:"] = "1⃣"
map[":open_file_folder:"] = "📂"
map[":open_hands:"] = "👐"
map[":open_mouth:"] = "😮"
map[":ophiuchus:"] = "⛎"
map[":orange_book:"] = "📙"
map[":outbox_tray:"] = "📤"
map[":ox:"] = "🐂"
map[":page_facing_up:"] = "📄"
map[":page_with_curl:"] = "📃"
map[":pager:"] = "📟"
map[":palm_tree:"] = "🌴"
map[":panda_face:"] = "🐼"
map[":paperclip:"] = "📎"
map[":parking:"] = "🅿️"
map[":part_alternation_mark:"] = "〽️"
map[":partly_sunny:"] = "⛅"
map[":passport_control:"] = "🛂"
map[":paw_prints:"] = "🐾"
map[":peach:"] = "🍑"
map[":pear:"] = "🍐"
map[":pencil2:"] = "✏️"
map[":pencil:"] = "📝"
map[":penguin:"] = "🐧"
map[":pensive:"] = "😔"
map[":performing_arts:"] = "🎭"
map[":persevere:"] = "😣"
map[":pig2:"] = "🐖"
map[":pig:"] = "🐷"
map[":pig_nose:"] = "🐽"
map[":pill:"] = "💊"
map[":pineapple:"] = "🍍"
map[":pisces:"] = "♓"
map[":pizza:"] = "🍕"
map[":point_down:"] = "👇"
map[":point_left:"] = "👈"
map[":point_right:"] = "👉"
map[":point_up:"] = "☝️"
map[":point_up_2:"] = "👆"
map[":police_car:"] = "🚓"
map[":poodle:"] = "🐩"
map[":poop:"] = "💩"
map[":post_office:"] = "🏣"
map[":postal_horn:"] = "📯"
map[":postbox:"] = "📮"
map[":potable_water:"] = "🚰"
map[":pouch:"] = "👝"
map[":poultry_leg:"] = "🍗"
map[":pound:"] = "💷"
map[":pouting_cat:"] = "😾"
map[":pray:"] = "🙏"
map[":princess:"] = "👸"
map[":punch:"] = "👊"
map[":purple_heart:"] = "💜"
map[":purse:"] = "👛"
map[":pushpin:"] = "📌"
map[":put_litter_in_its_place:"] = "🚮"
map[":question:"] = "❓"
map[":rabbit2:"] = "🐇"
map[":rabbit:"] = "🐰"
map[":racehorse:"] = "🐎"
map[":radio:"] = "📻"
map[":radio_button:"] = "🔘"
map[":rage:"] = "😡"
map[":railway_car:"] = "🚃"
map[":rainbow:"] = "🌈"
map[":raised_hand:"] = "✋"
map[":raised_hands:"] = "🙌"
map[":raising_hand:"] = "🙋"
map[":ram:"] = "🐏"
map[":ramen:"] = "🍜"
map[":rat:"] = "🐀"
map[":recycle:"] = "♻️"
map[":red_car:"] = "🚗"
map[":red_circle:"] = "🔴"
map[":registered:"] = "®️"
map[":relaxed:"] = "☺️"
map[":relieved:"] = "😌"
map[":repeat:"] = "🔁"
map[":repeat_one:"] = "🔂"
map[":restroom:"] = "🚻"
map[":revolving_hearts:"] = "💞"
map[":rewind:"] = "⏪"
map[":ribbon:"] = "🎀"
map[":rice:"] = "🍚"
map[":rice_ball:"] = "🍙"
map[":rice_cracker:"] = "🍘"
map[":rice_scene:"] = "🎑"
map[":ring:"] = "💍"
map[":rocket:"] = "🚀"
map[":roller_coaster:"] = "🎢"
map[":rooster:"] = "🐓"
map[":rose:"] = "🌹"
map[":rotating_light:"] = "🚨"
map[":round_pushpin:"] = "📍"
map[":rowboat:"] = "🚣"
map[":ru:"] = "🇷🇺"
map[":rugby_football:"] = "🏉"
map[":runner:"] = "🏃"
map[":running_shirt_with_sash:"] = "🎽"
map[":sa:"] = "🈂️"
map[":sagittarius:"] = "♐"
map[":sailboat:"] = "⛵"
map[":sake:"] = "🍶"
map[":sandal:"] = "👡"
map[":santa:"] = "🎅"
map[":satellite:"] = "📡"
map[":satisfied:"] = "😆"
map[":saxophone:"] = "🎷"
map[":school:"] = "🏫"
map[":school_satchel:"] = "🎒"
map[":scissors:"] = "✂️"
map[":scorpius:"] = "♏"
map[":scream:"] = "😱"
map[":scream_cat:"] = "🙀"
map[":scroll:"] = "📜"
map[":seat:"] = "💺"
map[":secret:"] = "㊙️"
map[":see_no_evil:"] = "🙈"
map[":seedling:"] = "🌱"
map[":seven:"] = "7⃣"
map[":shaved_ice:"] = "🍧"
map[":sheep:"] = "🐑"
map[":shell:"] = "🐚"
map[":ship:"] = "🚢"
map[":shirt:"] = "👕"
map[":shit:"] = "💩"
map[":shower:"] = "🚿"
map[":signal_strength:"] = "📶"
map[":six:"] = "6⃣"
map[":six_pointed_star:"] = "🔯"
map[":ski:"] = "🎿"
map[":skull:"] = "💀"
map[":sleeping:"] = "😴"
map[":sleepy:"] = "😪"
map[":slot_machine:"] = "🎰"
map[":small_blue_diamond:"] = "🔹"
map[":small_orange_diamond:"] = "🔸"
map[":small_red_triangle:"] = "🔺"
map[":small_red_triangle_down:"] = "🔻"
map[":smile:"] = "😄"
map[":smile_cat:"] = "😸"
map[":smiley:"] = "😃"
map[":smiley_cat:"] = "😺"
map[":smiling_imp:"] = "😈"
map[":smirk:"] = "😏"
map[":smirk_cat:"] = "😼"
map[":smoking:"] = "🚬"
map[":snail:"] = "🐌"
map[":snake:"] = "🐍"
map[":snowboarder:"] = "🏂"
map[":snowflake:"] = "❄️"
map[":snowman:"] = "⛄"
map[":sob:"] = "😭"
map[":soccer:"] = "⚽"
map[":soon:"] = "🔜"
map[":sos:"] = "🆘"
map[":sound:"] = "🔉"
map[":space_invader:"] = "👾"
map[":spades:"] = "♠️"
map[":spaghetti:"] = "🍝"
map[":sparkler:"] = "🎇"
map[":sparkles:"] = "✨"
map[":sparkling_heart:"] = "💖"
map[":speak_no_evil:"] = "🙊"
map[":speaker:"] = "🔈"
map[":speech_balloon:"] = "💬"
map[":speedboat:"] = "🚤"
map[":star2:"] = "🌟"
map[":star:"] = "⭐"
map[":stars:"] = "🌠"
map[":station:"] = "🚉"
map[":statue_of_liberty:"] = "🗽"
map[":steam_locomotive:"] = "🚂"
map[":stew:"] = "🍲"
map[":straight_ruler:"] = "📏"
map[":strawberry:"] = "🍓"
map[":stuck_out_tongue:"] = "😛"
map[":stuck_out_tongue_closed_eyes:"] = "😝"
map[":stuck_out_tongue_winking_eye:"] = "😜"
map[":sun_with_face:"] = "🌞"
map[":sunflower:"] = "🌻"
map[":sunglasses:"] = "😎"
map[":sunny:"] = "☀️"
map[":sunrise:"] = "🌅"
map[":sunrise_over_mountains:"] = "🌄"
map[":surfer:"] = "🏄"
map[":sushi:"] = "🍣"
map[":suspension_railway:"] = "🚟"
map[":sweat:"] = "😓"
map[":sweat_drops:"] = "💦"
map[":sweat_smile:"] = "😅"
map[":sweet_potato:"] = "🍠"
map[":swimmer:"] = "🏊"
map[":symbols:"] = "🔣"
map[":syringe:"] = "💉"
map[":tada:"] = "🎉"
map[":tanabata_tree:"] = "🎋"
map[":tangerine:"] = "🍊"
map[":taurus:"] = "♉"
map[":taxi:"] = "🚕"
map[":tea:"] = "🍵"
map[":telephone:"] = "☎️"
map[":telephone_receiver:"] = "📞"
map[":telescope:"] = "🔭"
map[":tennis:"] = "🎾"
map[":tent:"] = "⛺"
map[":thought_balloon:"] = "💭"
map[":three:"] = "3⃣"
map[":thumbsdown:"] = "👎"
map[":thumbsup:"] = "👍"
map[":ticket:"] = "🎫"
map[":tiger2:"] = "🐅"
map[":tiger:"] = "🐯"
map[":tired_face:"] = "😫"
map[":tm:"] = "™️"
map[":toilet:"] = "🚽"
map[":tokyo_tower:"] = "🗼"
map[":tomato:"] = "🍅"
map[":tongue:"] = "👅"
map[":top:"] = "🔝"
map[":tophat:"] = "🎩"
map[":tractor:"] = "🚜"
map[":traffic_light:"] = "🚥"
map[":train2:"] = "🚆"
map[":train:"] = "🚋"
map[":tram:"] = "🚊"
map[":triangular_flag_on_post:"] = "🚩"
map[":triangular_ruler:"] = "📐"
map[":trident:"] = "🔱"
map[":triumph:"] = "😤"
map[":trolleybus:"] = "🚎"
map[":trophy:"] = "🏆"
map[":tropical_drink:"] = "🍹"
map[":tropical_fish:"] = "🐠"
map[":truck:"] = "🚚"
map[":trumpet:"] = "🎺"
map[":tulip:"] = "🌷"
map[":turtle:"] = "🐢"
map[":tv:"] = "📺"
map[":twisted_rightwards_arrows:"] = "🔀"
map[":two:"] = "2⃣"
map[":two_men_holding_hands:"] = "👬"
map[":two_women_holding_hands:"] = "👭"
map[":u5272:"] = "🈹"
map[":u5408:"] = "🈴"
map[":u55b6:"] = "🈺"
map[":u6307:"] = "🈯"
map[":u6708:"] = "🈷️"
map[":u6709:"] = "🈶"
map[":u6e80:"] = "🈵"
map[":u7121:"] = "🈚"
map[":u7533:"] = "🈸"
map[":u7981:"] = "🈲"
map[":u7a7a:"] = "🈳"
map[":umbrella:"] = "☔"
map[":unamused:"] = "😒"
map[":underage:"] = "🔞"
map[":unlock:"] = "🔓"
map[":up:"] = "🆙"
map[":us:"] = "🇺🇸"
map[":v:"] = "✌️"
map[":vertical_traffic_light:"] = "🚦"
map[":vhs:"] = "📼"
map[":vibration_mode:"] = "📳"
map[":video_camera:"] = "📹"
map[":video_game:"] = "🎮"
map[":violin:"] = "🎻"
map[":virgo:"] = "♍"
map[":volcano:"] = "🌋"
map[":vs:"] = "🆚"
map[":walking:"] = "🚶"
map[":waning_crescent_moon:"] = "🌘"
map[":waning_gibbous_moon:"] = "🌖"
map[":warning:"] = "⚠️"
map[":watch:"] = "⌚"
map[":water_buffalo:"] = "🐃"
map[":watermelon:"] = "🍉"
map[":wave:"] = "👋"
map[":wavy_dash:"] = "〰️"
map[":waxing_crescent_moon:"] = "🌒"
map[":waxing_gibbous_moon:"] = "🌔"
map[":wc:"] = "🚾"
map[":weary:"] = "😩"
map[":wedding:"] = "💒"
map[":whale2:"] = "🐋"
map[":whale:"] = "🐳"
map[":wheelchair:"] = "♿"
map[":white_check_mark:"] = "✅"
map[":white_circle:"] = "⚪"
map[":white_flower:"] = "💮"
map[":white_square_button:"] = "🔳"
map[":wind_chime:"] = "🎐"
map[":wine_glass:"] = "🍷"
map[":wink:"] = "😉"
map[":wolf:"] = "🐺"
map[":woman:"] = "👩"
map[":womans_clothes:"] = "👚"
map[":womans_hat:"] = "👒"
map[":womens:"] = "🚺"
map[":worried:"] = "😟"
map[":wrench:"] = "🔧"
map[":x:"] = "❌"
map[":yellow_heart:"] = "💛"
map[":yen:"] = "💴"
map[":yum:"] = "😋"
map[":zap:"] = "⚡"
map[":zero:"] = "0⃣"
map[":zzz:"] = "💤"
}
{
if ($0 ~ /<pre>/) in_pre = 1
if (!in_pre) {
code_count = 0
line = $0
out = ""
while (match(line, /<code>[^<]*<\/code>/)) {
code_count++
code_store[code_count] = substr(line, RSTART, RLENGTH)
out = out substr(line, 1, RSTART - 1) "\034EC" code_count "\034"
line = substr(line, RSTART + RLENGTH)
}
out = out line
for (sc in map) {
if (index(out, sc)) {
gsub(sc, map[sc], out)
}
}
for (i = 1; i <= code_count; i++) {
gsub("\034EC" i "\034", code_store[i], out)
}
$0 = out
}
if ($0 ~ /<\/pre>/) in_pre = 0
print
}

View File

@@ -1,11 +1,20 @@
BEGIN { in_fence = 0; first_line = 0 } BEGIN { in_fence = 0; first_line = 0; code_tag = "<code>" }
{ {
if (!in_fence && $0 ~ /^```/) { if (!in_fence && $0 ~ /^```/) {
in_fence = 1 in_fence = 1
first_line = 1 first_line = 1
lang = $0
sub(/^```[[:space:]]*/, "", lang)
sub(/[[:space:]]*$/, "", lang)
if (lang != "") {
code_tag = "<code class=\"language-" lang "\">"
} else {
code_tag = "<code>"
}
next next
} }
if (in_fence && $0 ~ /^```[[:space:]]*$/) { if (in_fence && $0 ~ /^```[[:space:]]*$/) {
if (first_line) printf "%s", "<pre>" code_tag
print "</code></pre>" print "</code></pre>"
in_fence = 0 in_fence = 0
next next
@@ -14,8 +23,12 @@ BEGIN { in_fence = 0; first_line = 0 }
gsub(/&/, "\\&amp;"); gsub(/</, "\\&lt;"); gsub(/>/, "\\&gt;") gsub(/&/, "\\&amp;"); gsub(/</, "\\&lt;"); gsub(/>/, "\\&gt;")
if (first_line) { if (first_line) {
first_line = 0 first_line = 0
if ($0 == "") next printf "%s", "<pre>" code_tag
print "<pre><code>" $0 if ($0 == "") {
print ""
next
}
print $0
} else { } else {
print print
} }
@@ -24,5 +37,8 @@ BEGIN { in_fence = 0; first_line = 0 }
} }
} }
END { END {
if (in_fence) print "</code></pre>" if (in_fence) {
if (first_line) printf "%s", "<pre>" code_tag
print "</code></pre>"
}
} }

51
awk/footnotes.awk Normal file
View File

@@ -0,0 +1,51 @@
BEGIN { fn_count = 0 }
# Match [^id]: text
/^\[\^[a-zA-Z0-9_-]+\]:/ {
id_start = index($0, "[^") + 2
id_end = index($0, "]:")
id = substr($0, id_start, id_end - id_start)
text = substr($0, id_end + 2)
# Trim leading space
sub(/^[ \t]+/, "", text)
fn_ids[++fn_count] = id
fn_texts[id] = text
next
}
{
lines[++line_count] = $0
}
END {
for (i = 1; i <= line_count; i++) {
line = lines[i]
for (j = 1; j <= fn_count; j++) {
id = fn_ids[j]
marker = "[^" id "]"
repl = "<sup><a href=\"#fn:" id "\" id=\"fnref:" id "\">" id "</a></sup>"
while ((pos = index(line, marker)) > 0) {
line = substr(line, 1, pos - 1) repl substr(line, pos + length(marker))
}
}
print line
}
if (fn_count > 0) {
print "<hr />"
print "<section class=\"footnotes\">"
print "<ol>"
for (j = 1; j <= fn_count; j++) {
id = fn_ids[j]
text = fn_texts[id]
print "<li id=\"fn:" id "\">"
print "<p>" text " <a href=\"#fnref:" id "\" class=\"reversefootnote\">&#8617;</a></p>"
print "</li>"
}
print "</ol>"
print "</section>"
}
}

46
awk/frontmatter.awk Normal file
View File

@@ -0,0 +1,46 @@
BEGIN {
state = "start"
}
{
if (state == "start") {
if ($0 == "---") {
state = "in_fm"
next
} else {
state = "body"
print
next
}
}
if (state == "in_fm") {
if ($0 == "---") {
state = "body"
next
}
line = $0
if (line ~ /^[[:space:]]*$/ || line ~ /^[[:space:]]*#/) next
if (line !~ /=/) next
key = line
val = line
sub(/=.*/, "", key)
sub(/[^=]*=/, "", val)
gsub(/^[[:space:]]+|[[:space:]]+$/, "", key)
gsub(/^[[:space:]]+|[[:space:]]+$/, "", val)
if (val ~ /^".*"$/) {
val = substr(val, 2, length(val) - 2)
gsub(/\\"/, "\"", val)
} else if (val ~ /^'.*'$/) {
val = substr(val, 2, length(val) - 2)
gsub(/\\'/, "'", val)
}
if (fm_out != "") {
print key "=" val >> fm_out
}
next
}
print
}

View File

@@ -33,6 +33,13 @@ function compare_paths(p1, p2, parts1, parts2, n1, n2, i, name1, name2, lname
} }
BEGIN { BEGIN {
src = ENVIRON["AWK_SRC"]
single_file_index = ENVIRON["AWK_SINGLE_FILE_INDEX"]
flatten = ENVIRON["AWK_FLATTEN"]
order = ENVIRON["AWK_ORDER"]
home_name = ENVIRON["AWK_HOME_NAME"]
show_home_in_nav = ENVIRON["AWK_SHOW_HOME_IN_NAV"]
dinfo = ENVIRON["AWK_DINFO"]
n_dlines = split(dinfo, dlines, "\n") n_dlines = split(dinfo, dlines, "\n")
for (i = 1; i <= n_dlines; i++) { for (i = 1; i <= n_dlines; i++) {
if (split(dlines[i], dparts, "|") == 3) { if (split(dlines[i], dparts, "|") == 3) {

View File

@@ -3,23 +3,32 @@ function strip_markdown(s) {
gsub(/[*_`~]/, "", s) gsub(/[*_`~]/, "", s)
gsub(/[\[\]]/, "", s) gsub(/[\[\]]/, "", s)
gsub(/\([^\)]*\)/, "", s) gsub(/\([^\)]*\)/, "", s)
s = tolower(s)
gsub(/[^a-z0-9 -]/, "", s)
gsub(/^[[:space:]]+|[[:space:]]+$/, "", s) gsub(/^[[:space:]]+|[[:space:]]+$/, "", s)
gsub(/[[:space:]]+/, "-", s) gsub(/[[:space:]]+/, "-", s)
gsub(/-{2,}/, "-", s)
gsub(/^-+|-+$/, "", s)
if (length(s) > 80) s = substr(s, 1, 80)
gsub(/-+$/, "", s)
return s return s
} }
function print_header(line) { function print_header(line) {
if (line ~ /^# /) { tag = ""
sub(/^# /, "", line); print "<h1 id=\"" strip_markdown(line) "\">" line "</h1>" if (line ~ /^# /) { tag = "h1"; sub(/^# /, "", line) }
} else if (line ~ /^## /) { else if (line ~ /^## /) { tag = "h2"; sub(/^## /, "", line) }
sub(/^## /, "", line); print "<h2 id=\"" strip_markdown(line) "\">" line "</h2>" else if (line ~ /^### /) { tag = "h3"; sub(/^### /, "", line) }
} else if (line ~ /^### /) { else if (line ~ /^#### /) { tag = "h4"; sub(/^#### /, "", line) }
sub(/^### /, "", line); print "<h3 id=\"" strip_markdown(line) "\">" line "</h3>" else if (line ~ /^##### /) { tag = "h5"; sub(/^##### /, "", line) }
} else if (line ~ /^#### /) { else if (line ~ /^###### /) { tag = "h6"; sub(/^###### /, "", line) }
sub(/^#### /, "", line); print "<h4 id=\"" strip_markdown(line) "\">" line "</h4>"
} else if (line ~ /^##### /) { if (tag != "") {
sub(/^##### /, "", line); print "<h5 id=\"" strip_markdown(line) "\">" line "</h5>" id = strip_markdown(line)
} else if (line ~ /^###### /) { if (enable_header_links == "true") {
sub(/^###### /, "", line); print "<h6 id=\"" strip_markdown(line) "\">" line "</h6>" print "<" tag " id=\"" id "\"><a href=\"#" id "\" class=\"header-anchor\">" line "</a></" tag ">"
} else {
print "<" tag " id=\"" id "\">" line "</" tag ">"
}
} else { } else {
print line print line
} }
@@ -29,7 +38,7 @@ BEGIN {
in_pre = 0 in_pre = 0
} }
{ {
if ($0 ~ /^<pre><code>/) { if ($0 ~ /^<pre><code/) {
in_pre = 1 in_pre = 1
if (has_prev && prev != "") { print_header(prev); has_prev = 0 } if (has_prev && prev != "") { print_header(prev); has_prev = 0 }
print print

View File

@@ -1,9 +1,20 @@
BEGIN { in_code = 0 } BEGIN { in_code = 0; in_html_pre = 0 }
/^ | / { {
if (!in_code) { print "<pre><code>"; in_code = 1 } if ($0 ~ /<pre>/) in_html_pre = 1
sub(/^ | /, "", $0) if ($0 ~ /<\/pre>/) { in_html_pre = 0; if (in_code) { print "</code></pre>"; in_code = 0 }; print; next }
if (!in_html_pre && $0 ~ /^(\t| )/) {
if (!in_code) { printf "%s", "<pre><code>"; in_code = 1 }
sub(/^(\t| )/, "", $0)
gsub(/&/, "\\&amp;"); gsub(/</, "\\&lt;"); gsub(/>/, "\\&gt;") gsub(/&/, "\\&amp;"); gsub(/</, "\\&lt;"); gsub(/>/, "\\&gt;")
print; next print
next
}
if (in_code) {
print "</code></pre>"
in_code = 0
}
print
} }
{ if (in_code) { print "</code></pre>"; in_code = 0 } print }
END { if (in_code) print "</code></pre>" } END { if (in_code) print "</code></pre>" }

View File

@@ -48,7 +48,26 @@ BEGIN {
} }
} }
has_checkbox = 0
if (content ~ /^\[[ \t]\] /) {
has_checkbox = 1
is_checked = 0
sub(/^\[[ \t]\] /, "", content)
} else if (content ~ /^\[[xX]\] /) {
has_checkbox = 1
is_checked = 1
sub(/^\[[xX]\] /, "", content)
}
if (has_checkbox) {
if (is_checked) {
print "<li class=\"task-list-item\"><input type=\"checkbox\" class=\"task-list-item-checkbox\" checked disabled> " content "</li>"
} else {
print "<li class=\"task-list-item\"><input type=\"checkbox\" class=\"task-list-item-checkbox\" disabled> " content "</li>"
}
} else {
print "<li>" content "</li>" print "<li>" content "</li>"
}
} else { } else {
while (depth > 0) { while (depth > 0) {
print "</" cur_type[depth] ">" print "</" cur_type[depth] ">"

View File

@@ -97,6 +97,23 @@ function read_file(path, out, line, rc) {
return out return out
} }
function read_file_or_render_md(path, ext, cmd, content, line, rc) {
content = ""
if (ext == "md") {
cmd = "sh \"" script_dir "/markdown.sh\" \"" path "\""
while ((cmd | getline line) > 0) {
content = content line "\n"
}
close(cmd)
} else {
while ((rc = getline line < path) > 0) {
content = content line "\n"
}
close(path)
}
return content
}
function escape_html(s, t) { function escape_html(s, t) {
t = s t = s
gsub(/&/, "\\&amp;", t) gsub(/&/, "\\&amp;", t)
@@ -189,7 +206,8 @@ function render_embed(src, alt, has_alt, force_inline, ext, local_path, conte
if (force_inline && !is_global_url(src)) { if (force_inline && !is_global_url(src)) {
local_path = resolve_local_path(src) local_path = resolve_local_path(src)
if (local_path != "") { if (local_path != "") {
content = read_file(local_path) ext = ext_of(src)
content = read_file_or_render_md(local_path, ext)
if (content ~ /\n$/) sub(/\n$/, "", content) if (content ~ /\n$/) sub(/\n$/, "", content)
return content return content
} }
@@ -204,7 +222,7 @@ function render_embed(src, alt, has_alt, force_inline, ext, local_path, conte
} }
if (is_audio_ext(ext)) return "<audio controls src=\"" src "\"></audio>" if (is_audio_ext(ext)) return "<audio controls src=\"" src "\"></audio>"
if (is_video_ext(ext)) return "<video controls src=\"" src "\"></video>" if (is_video_ext(ext)) return "<video controls src=\"" src "\"></video>"
return "<iframe src=\"" src "\"></iframe>" return "<iframe src=\"" src "\" allowfullscreen></iframe>"
} }
if (is_image_ext(ext)) { if (is_image_ext(ext)) {
@@ -217,13 +235,35 @@ function render_embed(src, alt, has_alt, force_inline, ext, local_path, conte
if (is_inline_text_ext(ext)) { if (is_inline_text_ext(ext)) {
local_path = resolve_local_path(src) local_path = resolve_local_path(src)
if (local_path != "") { if (local_path != "") {
content = read_file(local_path) content = read_file_or_render_md(local_path, ext)
if (content ~ /\n$/) sub(/\n$/, "", content) if (content ~ /\n$/) sub(/\n$/, "", content)
return content return content
} }
} }
return "<iframe src=\"" src "\"></iframe>" return "<iframe src=\"" src "\" allowfullscreen></iframe>"
}
function render_typed_embed(etype, src, alt, has_alt, local_path, content) {
if (etype == "i") {
if (has_alt) return "<img alt=\"" alt "\" src=\"" src "\" />"
return "<img src=\"" src "\" />"
}
if (etype == "v") return "<video controls src=\"" src "\"></video>"
if (etype == "a") return "<audio controls src=\"" src "\"></audio>"
if (etype == "f") return "<iframe src=\"" src "\" allowfullscreen></iframe>"
if (etype == "e") {
if (!is_global_url(src)) {
local_path = resolve_local_path(src)
if (local_path != "") {
content = read_file_or_render_md(local_path, ext_of(src))
if (content ~ /\n$/) sub(/\n$/, "", content)
return content
}
}
return render_embed(src, alt, has_alt, 1)
}
return render_embed(src, alt, has_alt, 0)
} }
function extract_attr(tag, attr, pat, m, token) { function extract_attr(tag, attr, pat, m, token) {
@@ -319,7 +359,7 @@ function apply_td_vertical_align(line, out, rest, seg, td_tag, img_tag, after
return out rest return out rest
} }
function rewrite_img_tags(line, out, rest, tag, src, alt, force_inline_tag, pre, post, repl) { function rewrite_img_tags(line, out, rest, tag, src, alt, force_inline_tag, embed_type, pre, post, repl) {
out = "" out = ""
rest = line rest = line
while (match(rest, /<img[^>]*\/?>/)) { while (match(rest, /<img[^>]*\/?>/)) {
@@ -329,7 +369,10 @@ function rewrite_img_tags(line, out, rest, tag, src, alt, force_inline_tag, p
src = extract_attr(tag, "src") src = extract_attr(tag, "src")
alt = extract_attr(tag, "alt") alt = extract_attr(tag, "alt")
force_inline_tag = extract_attr(tag, "data-force-inline") force_inline_tag = extract_attr(tag, "data-force-inline")
if (is_image_ext(ext_of(src)) && force_inline_tag == "") { embed_type = extract_attr(tag, "data-embed-type")
if (embed_type != "") {
repl = render_typed_embed(embed_type, src, alt, (alt != ""))
} else if (is_image_ext(ext_of(src)) && force_inline_tag == "") {
# Preserve hand-written <img> attributes (style/class/etc) for normal images. # Preserve hand-written <img> attributes (style/class/etc) for normal images.
repl = tag repl = tag
} else { } else {

View File

@@ -20,9 +20,11 @@ function mask_html_tags(s, out, rest, start, len, tag, token) {
return out rest return out rest
} }
function restore_html_tags(s, i) { function restore_html_tags(s, i, val) {
for (i = 1; i <= html_tag_count; i++) { for (i = 1; i <= html_tag_count; i++) {
gsub(html_tag_token[i], html_tag_value[i], s) val = html_tag_value[i]
gsub(/&/, "\\\\&", val)
gsub(html_tag_token[i], val, s)
} }
return s return s
} }
@@ -58,6 +60,36 @@ function restore_html_tags(s, i) {
line = substr(line, 1, start - 1) repl substr(line, start + len) line = substr(line, 1, start - 1) repl substr(line, start + len)
} }
# typed embeds: !i, !v, !a, !f, !e
while (match(line, /![ivafe]\[[^\]]*\]\([^\)]+ "[^"]*"\)/)) {
start = RSTART; len = RLENGTH
token = substr(line, start, len)
etype = substr(token, 2, 1)
match(token, /\[[^\]]*\]/); alt = substr(token, RSTART + 1, RLENGTH - 2)
match(token, /"[^"]*"/); etitle = substr(token, RSTART + 1, RLENGTH - 2)
match(token, /\([^\)]+/); inner = substr(token, RSTART + 1, RLENGTH - 1)
sub(/[[:space:]]*"[^"]*"/, "", inner); src = inner
repl = "<img data-embed-type=\"" etype "\" alt=\"" alt "\" src=\"" src "\" title=\"" etitle "\" />"
line = substr(line, 1, start - 1) repl substr(line, start + len)
}
while (match(line, /![ivafe]\[[^\]]*\]\([^\)]+\)/)) {
start = RSTART; len = RLENGTH
token = substr(line, start, len)
etype = substr(token, 2, 1)
match(token, /\[[^\]]*\]/); alt = substr(token, RSTART + 1, RLENGTH - 2)
match(token, /\([^\)]+/); src = substr(token, RSTART + 1, RLENGTH - 1)
repl = "<img data-embed-type=\"" etype "\" alt=\"" alt "\" src=\"" src "\" />"
line = substr(line, 1, start - 1) repl substr(line, start + len)
}
while (match(line, /![ivafe]\[[^\]]+\]/)) {
start = RSTART; len = RLENGTH
token = substr(line, start, len)
etype = substr(token, 2, 1)
src = substr(token, 4, len - 4)
repl = "<img data-embed-type=\"" etype "\" src=\"" src "\" />"
line = substr(line, 1, start - 1) repl substr(line, start + len)
}
# force-inline image syntax (double bang) # force-inline image syntax (double bang)
while (match(line, /!!\[[^\]]*\]\([^\)]+ "[^"]*"\)/)) { while (match(line, /!!\[[^\]]*\]\([^\)]+ "[^"]*"\)/)) {
start = RSTART; len = RLENGTH start = RSTART; len = RLENGTH
@@ -198,5 +230,7 @@ function restore_html_tags(s, i) {
} }
} }
gsub(/<a href="https?:\/\/[^"]*"/, "& rel=\"noopener noreferrer\"", line)
print line print line
} }

View File

@@ -22,7 +22,7 @@ function mask_plain(s, t) {
gsub(/\$/, "\034P8\034", t) gsub(/\$/, "\034P8\034", t)
return t return t
} }
BEGIN { in_plain = 0 } BEGIN { in_plain = 0; in_script_style = 0 }
{ {
line = $0 line = $0
out = "" out = ""
@@ -48,5 +48,41 @@ BEGIN { in_plain = 0 }
in_plain = 0 in_plain = 0
} }
} }
print out tmp_line = out
out2 = ""
while (1) {
if (!in_script_style) {
pos_script = match(tolower(tmp_line), /<script([ >]|$)/)
script_start = RSTART; script_len = RLENGTH
pos_style = match(tolower(tmp_line), /<style([ >]|$)/)
style_start = RSTART; style_len = RLENGTH
if (pos_script == 0 && pos_style == 0) {
out2 = out2 tmp_line
break
}
if (pos_script > 0 && (pos_style == 0 || pos_script < pos_style)) {
out2 = out2 substr(tmp_line, 1, script_start + script_len - 1)
tmp_line = substr(tmp_line, script_start + script_len)
in_script_style = 1
end_tag = "</script>"
} else {
out2 = out2 substr(tmp_line, 1, style_start + style_len - 1)
tmp_line = substr(tmp_line, style_start + style_len)
in_script_style = 1
end_tag = "</style>"
}
} else {
pos_end = match(tolower(tmp_line), end_tag)
if (pos_end == 0) {
out2 = out2 mask_plain(tmp_line)
tmp_line = ""
break
}
out2 = out2 mask_plain(substr(tmp_line, 1, RSTART - 1)) substr(tmp_line, RSTART, RLENGTH)
tmp_line = substr(tmp_line, RSTART + RLENGTH)
in_script_style = 0
}
}
print out2
} }

View File

@@ -4,7 +4,7 @@ BEGIN {
} }
{ {
if ($0 ~ /^<pre>/) in_pre = 1 if ($0 ~ /<pre>/) in_pre = 1
if (in_pre) { if (in_pre) {
if (in_p) { print "</p>"; in_p = 0 } if (in_p) { print "</p>"; in_p = 0 }
@@ -13,7 +13,16 @@ BEGIN {
next next
} }
if ($0 ~ /^<\/?(div|table|p|[ou]l|h[1-6]|[bh]r|blockquote|li|hr)/) { if ($0 ~ /^<\/?(div|table|p|[ou]l|h[1-6]|[bh]r|blockquote|li|hr|section|article|nav|aside|header|footer|dl|dt|dd|script|style|iframe|details|summary|figure|figcaption|audio|video|picture)/) {
if (in_p) {
print "</p>"
in_p = 0
}
print
next
}
if ($0 ~ /^[\t ]*!([a-zA-Z])?\[[^\]]*\](\([^)]*\))?[\t ]*$/ || $0 ~ /^[\t ]*!!?\[[^\]]*\](\([^)]*\))?[\t ]*$/) {
if (in_p) { if (in_p) {
print "</p>" print "</p>"
in_p = 0 in_p = 0

View File

@@ -1,14 +1,31 @@
function replace_all(text, token, value, pos, token_len) { function replace_all(text, token, value, pos, token_len, res) {
token_len = length(token) token_len = length(token)
res = ""
while ((pos = index(text, token)) > 0) { while ((pos = index(text, token)) > 0) {
text = substr(text, 1, pos - 1) value substr(text, pos + token_len) res = res substr(text, 1, pos - 1) value
text = substr(text, pos + token_len)
}
return res text
}
BEGIN {
current_url = ENVIRON["AWK_CURRENT_URL"]
nav = ENVIRON["AWK_NAV"]
title = ENVIRON["AWK_TITLE"]
footer = ENVIRON["AWK_FOOTER"]
style_path = ENVIRON["AWK_STYLE_PATH"]
head_extra = ENVIRON["AWK_HEAD_EXTRA"]
header_brand = ENVIRON["AWK_HEADER_BRAND"]
lang = ENVIRON["AWK_LANG"]
if (current_url != "") {
nav = replace_all(nav, "href=\"" current_url "\"", "href=\"" current_url "\" class=\"current-page\"")
} }
return text
} }
{ {
line = $0 line = $0
line = replace_all(line, "{{TITLE}}", title) line = replace_all(line, "{{TITLE}}", title)
line = replace_all(line, "{{LANG}}", lang)
line = replace_all(line, "{{NAV}}", nav) line = replace_all(line, "{{NAV}}", nav)
line = replace_all(line, "{{FOOTER}}", footer) line = replace_all(line, "{{FOOTER}}", footer)
line = replace_all(line, "{{CSS}}", style_path) line = replace_all(line, "{{CSS}}", style_path)

50
awk/toc.awk Normal file
View File

@@ -0,0 +1,50 @@
BEGIN {
toc_str = "<ol class=\"toc\">\n"
has_toc = 0
}
{
lines[++n] = $0
if ($0 ~ /<pre>/) in_pre = 1
if (!in_pre && $0 ~ /\{\{TOC\}\}/) {
has_toc = 1
toc_lines[n] = 1
}
if ($0 ~ /<\/pre>/) in_pre = 0
if (match($0, /<h[23][^>]*>/)) {
tag_len = RLENGTH
title_start = RSTART + tag_len
title_str = substr($0, title_start)
title_end = index(title_str, "</h")
if (title_end > 0) {
title = substr(title_str, 1, title_end - 1)
gsub(/<[^>]+>/, "", title)
# extract id
id_start = match($0, /id="[^"]*"/)
if (id_start > 0) {
id_str = substr($0, id_start + 4)
id_end = index(id_str, "\"")
id = substr(id_str, 1, id_end - 1)
# what tag? level
level = substr($0, match($0, /<h[23]/) + 2, 1)
if (level == "2") {
toc_str = toc_str "<li class=\"toc-h2\"><a href=\"#" id "\">" title "</a></li>\n"
} else if (level == "3") {
toc_str = toc_str "<li class=\"toc-h3\"><a href=\"#" id "\">" title "</a></li>\n"
}
}
}
}
}
END {
toc_str = toc_str "</ol>"
for (i = 1; i <= n; i++) {
if (has_toc && toc_lines[i] && lines[i] ~ /^[[:space:]]*\{\{TOC\}\}[[:space:]]*$/) {
toc_lines[i] = 0 # Mark as processed if we want, but not strictly needed
sub(/\{\{TOC\}\}/, toc_str, lines[i])
}
print lines[i]
}
}

View File

@@ -1,4 +1,7 @@
BEGIN { done = 0 } BEGIN {
new_title = ENVIRON["AWK_NEW_TITLE"]
done = 0
}
/^title[[:space:]]*=/ { /^title[[:space:]]*=/ {
print "title = \"" new_title "\"" print "title = \"" new_title "\""
done = 1 done = 1

BIN
icon.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 28 KiB

21
icon.svg Normal file
View File

@@ -0,0 +1,21 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512" width="512" height="512">
<defs>
<linearGradient id="bg-grad" x1="0" y1="0" x2="1" y2="1">
<stop offset="0%" stop-color="#4a3b69"/>
<stop offset="100%" stop-color="#352654"/>
</linearGradient>
</defs>
<rect width="512" height="512" rx="80" ry="80" fill="url(#bg-grad)"/>
<text
x="256"
y="296"
text-anchor="middle"
dominant-baseline="central"
font-family="Georgia, 'Times New Roman', Times, serif"
font-size="220"
font-weight="bold"
font-style="italic"
fill="#debfff"
letter-spacing="-8"
>kewt</text>
</svg>

After

Width:  |  Height:  |  Size: 639 B

779
kewt.sh

File diff suppressed because it is too large Load Diff

View File

@@ -15,10 +15,15 @@ sed_inplace() {
fi fi
} }
temp_file="/tmp/markdown.$$.md" temp_file="${KEWT_TMPDIR:-/tmp}/markdown.$$.md"
cat "$@" > "$temp_file" cat "$@" > "$temp_file"
trap 'rm -f "$temp_file" "$temp_file.tmp"' EXIT INT TERM trap 'rm -f "$temp_file" "$temp_file.tmp" "$temp_file.fm"' EXIT INT TERM
# Frontmatter
fm_file="$temp_file.fm"
: > "$fm_file"
awk -v fm_out="$fm_file" -f "$awk_dir/frontmatter.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
# Mask # Mask
awk -f "$awk_dir/mask_inline_code.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/mask_inline_code.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
@@ -40,25 +45,37 @@ done
sed_inplace "/^\[[^\]]*\]: */d" "$temp_file" sed_inplace "/^\[[^\]]*\]: */d" "$temp_file"
# Blocks # Blocks
sed_inplace "s/^>!\[/> [!/g" "$temp_file"
sed_inplace "s/^>\[!/> [!/g" "$temp_file"
loop_count=0
max_iterations=100
while grep '^>' "$temp_file" >/dev/null; do while grep '^>' "$temp_file" >/dev/null; do
awk -f "$awk_dir/blockquote.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/blockquote.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
loop_count=$((loop_count + 1))
if [ "$loop_count" -gt "$max_iterations" ]; then
echo "Warning: Blockquote processing exceeded $max_iterations iterations on $1. Breaking to prevent infinite loop." >&2
break
fi
done done
awk -f "$awk_dir/blockquote_to_admonition.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -v custom_admonitions="$CUSTOM_ADMONITIONS" -f "$awk_dir/blockquote_to_admonition.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/fenced_code.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/fenced_code.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/indented_code.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/indented_code.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/pipe_tables.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/pipe_tables.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/headers.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -v enable_header_links="$ENABLE_HEADER_LINKS" -f "$awk_dir/headers.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/definition_lists.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/lists.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/lists.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
# TOC
awk -f "$awk_dir/toc.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
# Footnotes
awk -f "$awk_dir/footnotes.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
# Spacing # Spacing
awk -f "$awk_dir/breaks.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/breaks.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/paragraphs.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/paragraphs.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
# Inline styles # Inline styles
awk -f "$awk_dir/emoji.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -f "$awk_dir/markdown_inline.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file" awk -f "$awk_dir/markdown_inline.awk" "$temp_file" > "$temp_file.tmp" && mv "$temp_file.tmp" "$temp_file"
awk -v input_file="$1" -v site_root="$MARKDOWN_SITE_ROOT" -v fallback_file="$MARKDOWN_FALLBACK_FILE" -f "$awk_dir/markdown_embed.awk" "$temp_file" awk -v input_file="$1" -v site_root="$MARKDOWN_SITE_ROOT" -v fallback_file="$MARKDOWN_FALLBACK_FILE" -v script_dir="$script_dir" -f "$awk_dir/markdown_embed.awk" "$temp_file"
rm "$temp_file" rm "$temp_file"

7
package.json Normal file
View File

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

View File

@@ -0,0 +1,16 @@
pkgbase = kewt-git
pkgdesc = A minimalist, 100% POSIX, static site generator inspired by werc and kew
pkgver = r0.0000000
pkgrel = 3
url = https://kewt.krzak.org
arch = any
license = ISC
makedepends = git
depends = sh
provides = kewt
conflicts = kewt
conflicts = kewt-bin
source = kewt-git::git+https://git.krzak.org/N0VA/kewt.git
sha256sums = SKIP
pkgname = kewt-git

View File

@@ -0,0 +1,15 @@
pkgbase = kewt-bin
pkgdesc = A minimalist, 100% POSIX, static site generator inspired by werc and kew
pkgver = VERSION_PLACEHOLDER
pkgrel = 2
url = https://kewt.krzak.org
arch = any
license = ISC
depends = sh
provides = kewt
conflicts = kewt
conflicts = kewt-git
source = kewt-bin-VERSION_PLACEHOLDER.sh::https://git.krzak.org/N0VA/kewt/releases/download/vVERSION_PLACEHOLDER/kewt
sha256sums = SHA256SUM_PLACEHOLDER
pkgname = kewt-bin

View File

@@ -1,16 +1,16 @@
# Maintainer: n0va <n0va@krzak.org> # Maintainer: n0va <n0va@krzak.org>
pkgname=kewt-git pkgname=kewt-git
pkgver=r0.0000000 pkgver=r0.0000000
pkgrel=1 pkgrel=2
pkgdesc="A minimalist, 100% POSIX, static site generator inspired by werc and kew" pkgdesc="A minimalist, 100% POSIX, static site generator inspired by werc and kew"
arch=('any') arch=('any')
url="https://git.krzak.org/N0VA/kewt" url="https://kewt.krzak.org"
license=('MIT') license=('ISC')
makedepends=('git') makedepends=('git')
depends=('sh') depends=('sh')
provides=('kewt') provides=('kewt')
conflicts=('kewt' 'kewt-bin') conflicts=('kewt' 'kewt-bin')
source=("${pkgname}::git+${url}.git") source=("${pkgname}::git+https://git.krzak.org/N0VA/kewt.git")
sha256sums=('SKIP') sha256sums=('SKIP')
pkgver() { pkgver() {

View File

@@ -4,12 +4,12 @@ pkgver=VERSION_PLACEHOLDER
pkgrel=1 pkgrel=1
pkgdesc="A minimalist, 100% POSIX, static site generator inspired by werc and kew" pkgdesc="A minimalist, 100% POSIX, static site generator inspired by werc and kew"
arch=('any') arch=('any')
url="https://git.krzak.org/N0VA/kewt" url="https://kewt.krzak.org"
license=('MIT') license=('ISC')
depends=('sh') depends=('sh')
provides=('kewt') provides=('kewt')
conflicts=('kewt' 'kewt-git') conflicts=('kewt' 'kewt-git')
source=("${pkgname}-${pkgver}.sh::${url}/releases/download/v${pkgver}/kewt") source=("${pkgname}-${pkgver}.sh::https://git.krzak.org/N0VA/kewt/releases/download/v${pkgver}/kewt")
sha256sums=('SHA256SUM_PLACEHOLDER') sha256sums=('SHA256SUM_PLACEHOLDER')
build() { build() {

View File

@@ -0,0 +1,16 @@
class Kewt < Formula
desc "Minimalist static site generator inspired by werc"
homepage "https://kewt.krzak.org"
url "https://github.com/n0va-bot/kewt/releases/download/vVERSION_PLACEHOLDER/kewt"
sha256 "SHA256SUM_PLACEHOLDER"
license "ISC"
version "VERSION_PLACEHOLDER"
def install
bin.install "kewt"
end
test do
system "#{bin}/kewt", "--version"
end
end

View File

@@ -1,16 +0,0 @@
title = "kewt"
style = "kewt"
dir_indexes = true
single_file_index = true
flatten = false
footer = "made with <a href="https://kewt.krzak.org">kewt</a>"
logo = ""
display_logo = false
display_title = true
logo_as_favicon = true
favicon = ""
order = ""
home_name = "Home"
show_home_in_nav = true
nav_links = ""
nav_extra = ""

View File

@@ -0,0 +1,61 @@
# Configuration
## Dot Files
- `.kewtignore` - files/directories to ignore completely. If the file is empty, the whole directory gets ignored.
- `.kewthide` - files/directories to hide from navigation but still process. Same empty-file rules as `.kewtignore`.
- `.kewtpreserve` - files/directories to copy as-is without converting markdown to HTML. Same empty-file rules again.
## Frontmatter
You can set metadata for a page using a `site.conf`-style frontmatter block at the very top of `.md` files:
```conf
---
title = "Custom Page Title"
date = "2026-03-23 11:32"
draft = false
description = "A short page summary"
---
```
- `title` - overrides the page title, post name in index links, and RSS `<title>`.
- `date` - overrides the post date and time. Supports `YYYY-MM-DD` and `YYYY-MM-DD HH:MM` (or `HH-MM`).
- `draft` - if `true`, the file is excluded from HTML generation.
- `description` - page description, used for Open Graph `og:description` meta tag.
## Directory Index Customisation
By default, directories without an `index.md` get an auto-generated index page listing their contents.
If you create your own `index.md` in a directory, you can still include the auto-generated file list by using the `{{LIST}}` placeholder:
```md
# Blog
This is my blog. The posts are below. The top-most one is the most recent.
{{LIST}}
```
The `{{LIST}}` tag will be replaced with the generated list of links to child pages and files, exactly as in case the custom index didn't exist.
## Table of Contents
You can auto-generate a Table of Contents by placing `{{TOC}}` anywhere in your markdown file. It collects all `h2` and `h3` headings and generates an ordered list with anchor links.
## Footnotes
Footnotes use the `[^id]` syntax inline and `[^id]: text` for definitions at the bottom of the file. They are rendered as a numbered `<section>` at the end of the page.
## Definition Lists
Definition lists use the standard syntax:
```md
Term
: Definition
```
This renders as `<dl><dt>Term</dt><dd>Definition</dd></dl>`. Multiple definitions per term are supported.
## Emoji Shortcodes
Standard GitHub/MkDocs emoji shortcodes like `:smile:`, `:fire:`, `:rocket:` are automatically replaced with their Unicode emoji equivalents. Shortcodes inside code blocks are left as-is.

21
site/Docs/embeds.md Normal file
View File

@@ -0,0 +1,21 @@
# Embeds
- `\![link]`:
- local image/audio/video files are embedded as media tags
- local text/code files are inlined directly
- global image/audio/video links are embedded as media tags
- other global links are embedded as `<iframe>`
- `\![alt](link)` works the same, with `alt` used for images
- `\!![link]` and `\!![alt](link)` force inline local file contents
If you want to **force** a file to be inlined, use `\!![]` instead of `\![]`
## Typed Embeds
Force specific output regardless of extension:
- `\!i[link]` or `\!i[alt](link)` - **I**mage
- `\!v[link]` - **V**ideo
- `\!a[link]` - **A**udio
- `\!f[link]` - I**f**rame
- `\!e[link]` - Inline/**e**mbed text/code file directly

3
site/Docs/index.md Normal file
View File

@@ -0,0 +1,3 @@
# Documentation
{{LIST}}

42
site/Docs/installation.md Normal file
View File

@@ -0,0 +1,42 @@
# Installation
## Standalone
```sh
curl -L -o kewt https://git.krzak.org/N0VA/kewt/releases/download/latest/kewt
chmod +x kewt
```
## From source
```sh
git clone https://git.krzak.org/N0VA/kewt.git
cd kewt
```
### Building
```sh
make
```
### Installing
```sh
sudo make install
```
## Package Managers
### AUR
- [kewt-bin](https://aur.archlinux.org/packages/kewt-bin) - prebuilt standalone binary from the latest release
- [kewt-git](https://aur.archlinux.org/packages/kewt-git) - built from the latest git source
### Homebrew
```sh
brew tap n0va-bot/tap
brew install kewt
```
### bpkg
```sh
bpkg install n0va-bot/kewt
```

77
site/Docs/usage.md Normal file
View File

@@ -0,0 +1,77 @@
# Usage
```sh
kewt --help
kewt --version
kewt --new [title]
kewt --post [title]
kewt --generate-template [path]
kewt --update [dir]
kewt --from <src> --to <out>
kewt [src] [out]
kewt --watch
kewt --serve [port]
```
- `--new [title]` creates a new site directory with a default `site.conf`, `template.html`, and `index.md`.
- `--post [title]` creates a new markdown file in the configured `posts_dir` with the current date/time as the filename and default frontmatter.
- `--generate-template [path]` writes the default `template.html` to the given path (defaults to `template.html` in the current directory).
- `--update [dir]` adds any missing keys to `site.conf` and checks `template.html` against the latest default.
- `--watch` (`-w`) watches for file changes in the source directory and rebuilds automatically.
- `--serve` (`-s`) starts a local HTTP server (python3 or busybox) in the output directory after building. Use with the port number to specify the port. Composable with `--watch`.
## site.conf
```conf
title = "kewt"
style = "kewt"
dir_indexes = true
single_file_index = true
flatten = false
order = ""
home_name = "Home"
show_home_in_nav = true
nav_links = ""
nav_extra = ""
footer = "made with <a href=\"https://kewt.krzak.org\">kewt</a>"
logo = ""
display_logo = false
display_title = true
logo_as_favicon = true
favicon = ""
generate_page_title = true
error_page = "not_found.html"
versioning = false
enable_header_links = true
base_url = ""
generate_feed = false
feed_file = "rss.xml"
posts_dir = ""
posts_per_page = 12
custom_admonitions = ""
```
- `title` - site title
- `style` - style file name from `./styles` (without `.css`)
- `dir_indexes` - generate directory index pages when missing `index.md`
- `single_file_index` - if a directory has one markdown file and no `index.md`, use that file as `index.html`
- `flatten` - flatten sidebar directory levels
- `order` - comma separated file/directory name list to order the sidebar (alphabetical by default)
- `home_name` - text for the home link in navigation (default: "Home")
- `show_home_in_nav` - show home link in navigation (default: true)
- `nav_links` - comma separated extra nav links, as bare URLs or Markdown links like `[Label](https://example.com)`
- `nav_extra` - raw HTML appended inside the `<nav>` after the generated link list
- `footer` - footer html/text shown at the bottom of pages
- `logo` - logo image path (used in header if enabled)
- `display_logo` - show logo in header
- `display_title` - show title text in header
- `logo_as_favicon` - use `logo` as favicon
- `favicon` - explicit favicon path (used when `logo_as_favicon` is false or no logo is set)
- `generate_page_title` - automatically generate title text from the first markdown heading or filename (default: true)
- `error_page` - filename for the generated 404 error page (default: "not_found.html", empty to disable)
- `versioning` - append a version query parameter (`?v=timestamp`) to css asset urls to bypass cache (default: false)
- `base_url` - absolute URL of the site, used for sitemap and RSS feed generation
- `generate_feed` - enable RSS feed generation (requires `base_url`)
- `feed_file` - filename for the generated RSS feed (default: "rss.xml")
- `posts_dir` - directory name containing posts (e.g., "posts"). Enables reverse-chronological sorting, title headings in indexes, and automatic backlinks.
- `posts_per_page` - number of posts per page in paginated post indexes (default: 12). Set to 0 to disable pagination.
- `enable_header_links` - turns markdown section headings into clickable anchor links (default: true)
- `custom_admonitions` - comma separated list of custom admonitions

5
site/depths/index.md Normal file
View File

@@ -0,0 +1,5 @@
# Depths
This is a custom index for a directory
{{LIST}}

BIN
site/favicon.ico Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 361 KiB

View File

@@ -1,8 +1,12 @@
# _kewt_ # _kewt_
### Pronounced "cute" ### Pronounced "cute"
***
# [Go to the repo](https://git.krzak.org/N0VA/kewt) # [Go to the repo](https://git.krzak.org/N0VA/kewt)
***
_kewt_ is a minimalist ssg inspired by _[werc](http://werc.cat-v.org/)_ and _[kew](https://github.com/uint23/kew)_ _kewt_ is a minimalist ssg inspired by _[werc](http://werc.cat-v.org/)_ and _[kew](https://github.com/uint23/kew)_
It's meant to be a static site generator, like _[kew](https://github.com/uint23/kew)_ but use only default (POSIX) tooling, like _[werc](http://werc.cat-v.org/)_ (and definitely unlike _[kew](https://github.com/uint23/kew)_) It's meant to be a static site generator, like _[kew](https://github.com/uint23/kew)_ but use only default (POSIX) tooling, like _[werc](http://werc.cat-v.org/)_ (and definitely unlike _[kew](https://github.com/uint23/kew)_)
@@ -10,98 +14,32 @@ It's meant to be a static site generator, like _[kew](https://github.com/uint23/
## Features ## Features
- No dependencies - No dependencies
- Frontmatter support (title, date, draft, description)
- Supports many embed types - Supports many embed types
- Automatic css variable replacement for older browsers - Automatic css variable replacement for older browsers
- Automatic inlining and embedding of many filetypes with `\![link]` or `\![alt](link)` - Automatic inlining and embedding of many filetypes with `\![link]` or `\![alt](link)`
- Typed embeds: `\!i`, `\!v`, `\!a`, `\!f`, `\!e`
- Inline html support - Inline html support
- MFM `$font` and `\<plain>` tags - MFM `$font` and `\<plain>` tags
- Admonition support (that's what the blocks like the warning block below are called) - GFM Admonition support (that's what the blocks like the warning block below are called)
- Task list support (`- [ ]`, `- [x]`)
- RSS/Feed generation and Sitemap support
- Post creation via `--post`
- Automatic 404 page generation
- `?v=n` support for cache busting
- Code block classes for use with external libraries like highlight.js or prism.js (both tested)
- Clickable markdown header anchors
- Mobile responsive layout
- Customisable directory index pages with `{{LIST}}`
- Open Graph meta tags from frontmatter
- Auto-generated Table of Contents via `{{TOC}}`
- Footnotes (`[^id]`)
- Definition lists
- Emoji shortcodes (`:smile:`, `:fire:`, etc.)
- Post pagination
- `--watch` and `--serve` modes for development
If you want to **force** a file to be inlined, use `\!![]` instead of `\![]` ***
## Installation > [!WARNING]
> The base that all of this is built upon was coded at night, while sleepy and a bit sick, and after walking for about 4 hours around a forest, so...
You can clone the repository to use `kewt.sh` directly, or you can download the standalone executable, which bundles all dependencies into a single file:
```sh
curl -L -o kewt https://git.krzak.org/N0VA/kewt/releases/latest/download/kewt
chmod +x kewt
```
On Arch Linux, _kewt_ is available on the AUR:
- [kewt-bin](https://aur.archlinux.org/packages/kewt-bin) — prebuilt standalone binary from the latest release
- [kewt-git](https://aur.archlinux.org/packages/kewt-git) — built from the latest git source
## Usage
```sh
./kewt.sh --help
./kewt.sh --new [title]
./kewt.sh --from <src> --to <out>
./kewt.sh [src] [out]
```
`--new [title]` creates a new site directory with a copied `site.conf` and a default `index.md`.
## site.conf
```conf
title = "kewt"
style = "kewt"
dir_indexes = true
single_file_index = true
flatten = false
order = ""
home_name = "Home"
show_home_in_nav = true
nav_links = ""
nav_extra = ""
footer = "made with <a href="https://kewt.krzak.org">kewt</a>"
logo = ""
display_logo = false
display_title = true
logo_as_favicon = true
favicon = ""
```
- `title` site title
- `style` style file name from `./styles` (without `.css`)
- `dir_indexes` generate directory index pages when missing `index.md`
- `single_file_index` if a directory has one markdown file and no `index.md`, use that file as `index.html`
- `flatten` flatten sidebar directory levels
- `order` comma separated file/directory name list to order the sidebar (alphabetical by default)
- `home_name` text for the home link in navigation (default: "Home")
- `show_home_in_nav` show home link in navigation (default: true)
- `nav_links` comma separated extra nav links, as bare URLs or Markdown links like `[Label](https://example.com)`
- `nav_extra` raw HTML appended inside the `<nav>` after the generated link list
- `footer` footer html/text shown at the bottom of pages
- `logo` logo image path (used in header if enabled)
- `display_logo` show logo in header
- `display_title` show title text in header
- `logo_as_favicon` use `logo` as favicon
- `favicon` explicit favicon path (used when `logo_as_favicon` is false or no logo is set)
## Ignores
- `.kewtignore`: Files/directories to ignore. If empty, the whole directory gets ignored
- `.kewthide`: Files/directories to hide from navigation but still process. Same empty rules as with ignore
- `.kewtpreserve`: Files/directories to copy but not convert markdown to html. Same empty rules again
## Embeds
- `\![link]`:
- local image/audio/video files are embedded as media tags
- local text/code files are inlined directly
- global image/audio/video links are embedded as media tags
- other global links are embedded as `<iframe>`
- `\![alt](link)` works the same, with `alt` used for images
- `\!![]` and `\!![alt](link)` force inline local file contents
## Credits
- Markdown to html conversion based on [markdown.bash](https://github.com/chadbraunduin/markdown.bash) by [chadbraunduin](https://github.com/chadbraunduin)
- Default css style and html template based on _[kew](https://github.com/uint23/kew)_ by [uint23](https://github.com/uint23)
>![WARNING]
>Most of this was coded at night, while sleepy and a bit sick, and after walking for about 4 hours around a forest, so...

View File

@@ -3,10 +3,23 @@ style = "kewt"
dir_indexes = true dir_indexes = true
single_file_index = true single_file_index = true
flatten = false flatten = false
footer = "<a href="https://kewt.krzak.org"><img src="/button.gif" /></a>" footer = "<a href=\"https://kewt.krzak.org\"><img src=\"/button.gif\" /></a>"
logo = "" logo = ""
display_logo = false display_logo = false
display_title = true display_title = true
logo_as_favicon = true logo_as_favicon = false
favicon = "" favicon = "favicon.ico"
order = "" order = "Home, Docs, depths, Heaven"
home_name = "Home"
show_home_in_nav = true
nav_links = ""
nav_extra = ""
generate_page_title = true
error_page = "not_found.html"
versioning = true
enable_header_links = true
base_url = "https://kewt.krzak.org"
custom_admonitions = ""
generate_feed = false
feed_file = "rss.xml"
posts_dir = ""

View File

@@ -1,25 +1,28 @@
:root { :root {
--bg: #646c7f; --bg: #4a3b69;
--fg: #fffde0; --bg-deep: #352654;
--fg-link: #fff18f; --fg: #fbf5ff;
--code-bg: #32394a; --fg-muted: #c8b9df;
--code-border: #8f95a4; --fg-link: #dfaeff;
--code-fg: #fffde0; --fg-heading: #debfff;
--code-sel: #fff18f; --code-bg: #31234c;
--code-prop: #ffd27f; --code-border: #8060af;
--code-val: #cde7ff; --code-fg: #fbf5ff;
--code-var: #b9ffbe; --code-sel: #ffef99;
--code-com: #d0d0d0; --code-prop: #ffdfba;
--adm-note-bg: #3f5666; --code-val: #cae2ff;
--adm-note-border: #a8d8ff; --code-var: #caffc2;
--adm-tip-bg: #3f664c; --code-com: #b8aac8;
--adm-tip-border: #b9ffbe; --adm-note-bg: #353866;
--adm-important-bg: #5a4a6c; --adm-note-border: #b8c5ff;
--adm-important-border: #e4c7ff; --adm-tip-bg: #295246;
--adm-warning-bg: #6b5539; --adm-tip-border: #aeffda;
--adm-warning-border: #ffe0a8; --adm-important-bg: #533076;
--adm-caution-bg: #6f3f3f; --adm-important-border: #f4d9ff;
--adm-caution-border: #ffb4b4; --adm-warning-bg: #634631;
--adm-warning-border: #ffe2bd;
--adm-caution-bg: #662d43;
--adm-caution-border: #ffc4d5;
} }
body { body {
@@ -29,18 +32,22 @@ body {
color: var(--fg); color: var(--fg);
font-family: serif; font-family: serif;
font-size: 16px; font-size: 16px;
line-height: 1.2; line-height: 1.5;
} }
header { header {
padding: 20px; padding: 20px;
padding-bottom: 0;
border-bottom: 1px solid var(--code-border);
margin-bottom: 20px;
} }
header h1 { header h1 {
margin: 0; margin: 0;
font-size: 35px; font-size: 35px;
font-weight: normal; font-weight: bold;
font-style: italic; font-style: italic;
color: var(--fg-heading);
} }
.site-logo { .site-logo {
@@ -57,18 +64,26 @@ header a {
text-decoration: none; text-decoration: none;
} }
header a:hover {
color: var(--bg-deep);
background: var(--fg);
}
#side-bar { #side-bar {
position: absolute; position: absolute;
top: 80px; top: 80px;
left: 0; left: 0;
width: 200px; width: 200px;
padding-left: 20px; padding-left: 20px;
margin-right: 14px;
border-right: 1px solid var(--code-border);
padding-right: 7px;
} }
.side-title { .side-title {
font-size: 25px; font-size: 25px;
margin: 20px 0 8px 0; margin: 20px 0 8px 0;
color: var(--fg); color: var(--fg-heading);
} }
#side-bar ul { #side-bar ul {
@@ -87,6 +102,14 @@ a {
padding: 1px 2px; padding: 1px 2px;
} }
#side-bar a.current-page {
font-weight: bold;
color: var(--fg);
border-left: 3px solid var(--fg-link);
padding-left: 7px;
margin-left: -10px;
}
a:hover { a:hover {
background: var(--fg); background: var(--fg);
color: var(--bg); color: var(--bg);
@@ -100,7 +123,7 @@ article {
h3 { h3 {
margin-top: 30px; margin-top: 30px;
font-size: 25px; font-size: 25px;
color: var(--fg); color: var(--fg-heading);
font-weight: normal; font-weight: normal;
} }
@@ -197,10 +220,11 @@ pre code {
} }
footer { footer {
padding-top: 80px; padding-top: 60px;
font-style: italic; font-style: italic;
font-size: 17px; font-size: 17px;
margin-bottom: 20px; margin-bottom: 20px;
color: var(--fg-muted);
} }
article, article,
@@ -220,3 +244,65 @@ footer img {
display: inline-block; display: inline-block;
vertical-align: top; vertical-align: top;
} }
hr {
height: 0;
margin: 24px 0;
border: 0;
border-top: 1px solid var(--code-border);
}
.nav-toggle, .nav-toggle-label {
display: none;
}
@media screen and (max-width: 600px) {
header {
display: flex;
justify-content: space-between;
align-items: center;
}
.nav-toggle-label {
display: block;
font-size: 30px;
cursor: pointer;
color: var(--fg-heading);
user-select: none;
}
#side-bar {
display: none;
position: relative;
top: auto;
left: auto;
width: auto;
border-right: none;
border-bottom: 1px solid var(--code-border);
padding: 0 0 20px 0;
margin: 0 20px 20px 20px;
}
.nav-toggle:checked ~ #side-bar {
display: block;
}
article {
margin: 0 20px 0 20px;
}
footer {
margin-left: 20px;
margin-right: 20px;
padding-top: 30px;
}
}
.task-list-item {
list-style-type: none;
}
.task-list-item-checkbox {
margin: 0 0.2em 0.25em -1.6em;
vertical-align: middle;
}

View File

@@ -1,21 +0,0 @@
<!doctype html>
<html>
<head>
<meta charset="UTF-8" />
<title>{{TITLE}}</title>
<link rel="stylesheet" href="{{CSS}}" type="text/css" />
{{HEAD_EXTRA}}
</head>
<body>
<header>
<h1>{{HEADER_BRAND}}</h1>
</header>
<nav id="side-bar">{{NAV}}</nav>
<article>{{CONTENT}}</article>
<footer>{{FOOTER}}</footer>
</body>
</html>

View File

@@ -26,7 +26,13 @@ exit $?
#==PAYLOAD== #==PAYLOAD==
EOF EOF
tar -cz -C "$REPO_ROOT" kewt.sh markdown.sh awk styles >> "$OUT_FILE" VERSION=$(git describe --tags 2>/dev/null || echo "standalone")
tmpbuild=$(mktemp -d)
cp -r "$REPO_ROOT/kewt.sh" "$REPO_ROOT/markdown.sh" "$REPO_ROOT/awk" "$REPO_ROOT/styles" "$tmpbuild/"
sed -e "s/kewt version git/kewt version $VERSION/" "$tmpbuild/kewt.sh" > "$tmpbuild/kewt.sh.tmp" && mv "$tmpbuild/kewt.sh.tmp" "$tmpbuild/kewt.sh"
chmod +x "$tmpbuild/kewt.sh" "$tmpbuild/markdown.sh"
tar -cz -C "$tmpbuild" kewt.sh markdown.sh awk styles >> "$OUT_FILE"
rm -rf "$tmpbuild"
chmod +x "$OUT_FILE" chmod +x "$OUT_FILE"