A keyboard-driven markdown editor in C, SDL2, and Lua. Aims at the Obsidian / Lite XL ergonomics — vault sidebar, wiki links, full preview rendering, plugin host — without the Electron tax.
- Live preview of CommonMark via md4c — headings, lists, task lists, code fences, block quotes, tables, inline styles, links, soft and hard breaks.
- Edit mode with a real text buffer (undo/redo, multi-byte caret, selection, smart Enter for lists, auto-pairs, find/replace). Soft word wrap on by default; toggle off (Alt+Z) for a horizontal scrollbar with arrow buttons.
- Tabs — several files open at once, lossless switching (each tab keeps its text, cursor, scroll, and undo), reopened on next launch. A toggleable split live-preview (Ctrl+\) renders the active file source-left / preview-right.
- Vault sidebar with collapsible folders, drag-and-drop reorder, right-click context menu, recents tracking, and image files that open straight into the preview pane.
- Wiki links
[[note name]]resolve case-insensitively across the vault; backlinks panel shows every note that points at the current one. - Outline panel auto-generated from headings, pinned or modal.
- Find / Replace with regex, case-insensitive, whole-word, live match count, caret-aware editing, click-to-position. Searches the raw source in edit mode and the rendered text in preview so the highlights line up in either view.
- Vault-wide search across every note with a results overlay.
- Quick switcher (Ctrl+P) — recents-first, fuzzy match.
- Command palette (Ctrl+Shift+P) — every action plus every Lua-registered plugin action with a category chip and bound shortcut.
- Plugin host — drop
*.luaindata/plugins/, register actions withdescry.register_action("name", fn), and do real work: read and edit the open document (descry.buffer.*), list/open vault notes (descry.vault.list,descry.open), and subscribe toopen/save/text_changeevents (descry.on). Surface output withdescry.notify/descry.dialog. The Plugins overlay lists every loaded file and the actions each registered, with a hot-reload button. - LaTeX math — inline
$…$and display$$…$$spans render typographically: Greek letters, operators,\frac{a}{b}, roots, and simple super/subscripts become real Unicode glyphs ($\sum_{i=1}^{n} x_i^2$→∑ᵢ₌₁ⁿ xᵢ²). A typographic pass, not a full TeX engine. - Graph view (Ctrl+Shift+M) — a force-directed map of the vault's wiki-link graph. Node size scales with degree, the current note is highlighted; drag to pan, scroll to zoom, click a node to open it.
- Spell check — opt-in (
spellcheck = trueinsettings.lua), dictionary-driven red squiggles under unknown words in the editor. Points at a system word list or your owndictionary_path; "Add to dictionary" remembers words indata/.dictionary_user. - In-app modals for Save / Save As / Rename / New File — no native Win32 dialogs, full keyboard nav, sidebar-style file browser.
- Custom title bar with File / Edit / View / Help menus, aero
snap, drag-to-move, min/maximize/close, working under
SDL_WINDOW_BORDERLESSvia a WS_THICKFRAME + WM_NCCALCSIZE trick. - Live resize indicator — a centered
W x Hbadge plus a window outline render while the user drags an edge. The resize loop pumps through an SDL event watch so the indicator animates during the drag, not after. - Plugin actions, Ctrl+click external links, image preview, About dialog, settings persistence, theme picker, keybinding editor, daily-notes, HTML export, ASCII-art table aligner ...
| Layer | Library |
|---|---|
| Language | C11 |
| Window / input | SDL2 |
| Glyph cache | FreeType + HarfBuzz |
| Markdown parser | md4c (vendored) |
| Scripting | Lua 5.4 (vendored) |
| SVG icons / pills | nanosvg (vendored) |
| Image decode | libpng, libjpeg |
| Anti-aliasing | custom analytic signed-distance-field pill rasterizer |
| Build | CMake + Ninja |
No GTK, no Qt, no web view, no JS runtime. The compiled binary is a
single descry.exe plus its DLL dependencies (SDL2, FreeType,
HarfBuzz, libpng, libjpeg).
pacman -S mingw-w64-x86_64-{gcc,cmake,ninja,SDL2,freetype,harfbuzz,libpng,libjpeg-turbo}
cmake -G Ninja -B build
ninja -C build
./build/descry.exeApple Silicon and Intel both work; the app renders crisp on Retina and falls back to system fonts (Menlo) automatically.
brew install cmake ninja pkg-config sdl2 freetype harfbuzz libpng jpeg
# Apple Silicon Homebrew lives under /opt/homebrew; on Intel use /usr/local.
# libjpeg is keg-only, so point pkg-config at it explicitly:
export PKG_CONFIG_PATH="/opt/homebrew/opt/jpeg/lib/pkgconfig:/opt/homebrew/lib/pkgconfig"
cmake -G Ninja -B build
ninja -C build
./build/descryPackage as a double-clickable app. To get a self-contained Descry.app
(SDL/FreeType/HarfBuzz dylibs bundled inside, HiDPI-aware, ad-hoc signed so it
launches locally) instead of running the binary from a terminal:
brew install dylibbundler # bundles the dylibs into the app
./package_macos.sh # -> build/Descry.app
./package_macos.sh --dmg # also -> dist/Descry-<version>.dmg (drag to /Applications)For distribution to other Macs, sign with a Developer ID
(--sign "Developer ID Application: …") and notarize.
apt install build-essential cmake ninja-build libsdl2-dev libfreetype-dev libharfbuzz-dev libpng-dev libjpeg-dev
cmake -G Ninja -B build
ninja -C build
./build/descryOn Windows the build pulls every transitive MinGW DLL next to the exe
via file(GET_RUNTIME_DEPENDENCIES) so the build dir is portable —
zip it and run anywhere without an MSYS2 install. The data/ folder
is not copied; the exe loads data/ from its own directory at
runtime, so put your vault wherever you like and point Descry at it.
The repo ships thin wrappers around the commands above: build.ps1
for Windows (MSYS2 MinGW-w64) and build.sh for macOS/Linux. Both
configure + build into build/; pass -Run / --run to launch
afterwards.
- Retina / HiDPI renders at native pixel density — glyphs are rasterized at the display scale while layout stays in logical points, so text is sharp rather than upscaled. Crossing between a Retina and a non-Retina display re-rasterizes the fonts automatically.
- Fonts default to Menlo and fall back through Apple Symbols,
PingFang and Apple Color Emoji for symbols / CJK / emoji. A
settings.luacopied from another OS that points at a missing font is detected and swapped for the platform default instead of failing to start. - Reveal in Finder uses
open -R; external links and the log folder open viaopen. - File dialogs use AppleScript (
osascript); logs and settings live under~/Library/Logs/Descry/.
src/ - C sources (single-binary)
main.c - app loop, UI, every overlay
buffer.c/h - the gap-free text buffer + cursor/selection/undo
markdown.c/h - md4c wrapper, line/style/wiki/link extraction
font.c/h - FreeType+HarfBuzz glyph cache, fallback chain
icons.c/h - nanosvg icon raster + SDF pill rasterizer
lua_host.c/h - Lua state, plugin loader, action registry, buffer/
vault/event bridge for plugins
vault.c/h - recursive directory scan + native dialogs
image.c/h - PNG/JPG decode -> SDL_Texture cache
regex.c/h - in-house regex engine (find/replace)
mermaid.c/h - mermaid diagram parse + layout
graph.c/h - force-directed wiki-link graph layout
spell.c/h - hash-set spell dictionary
tabs.c/h - open-file tab list + park/restore
data/ - default vault: sample notes, init.lua, plugins/
vendor/ - lua 5.4, md4c, nanosvg
A plugin is any .lua file under data/plugins/. The host exposes a
tiny global table:
-- data/plugins/hello.lua
descry.register_action("say_hello", function()
descry.dialog("Hello", "Hello from the plugin system!")
end)
descry.notify("[hello plugin] loaded")After a reload (Ctrl+Alt+P > Reload, or restart), the action shows up
in the command palette with a Plugin category chip and can be
invoked by name or bound to a key.
Full reference — every available API call, lifecycle, debugging, and an honest list of what's not exposed yet — lives in docs/plugins.md.
A non-exhaustive list. Every binding is editable from Settings > Keybindings and persists to settings.lua next to the exe.
| Action | Shortcut |
|---|---|
| Toggle edit / preview | Ctrl+E |
| Save | Ctrl+S |
| Save As | Ctrl+Shift+S |
| New file | Ctrl+N |
| Rename | F2 |
| Quick switcher | Ctrl+P |
| Command palette | Ctrl+Shift+P |
| Plugins overlay | Ctrl+Alt+P |
| Find | Ctrl+F |
| Find / Replace | Ctrl+H |
| Vault search | Ctrl+Shift+F |
| Outline | Ctrl+Shift+O |
| Backlinks | Ctrl+Shift+B |
| Tags | Ctrl+Shift+G |
| Graph view | Ctrl+Shift+M |
| Daily note | Ctrl+D |
| Toggle sidebar | Ctrl+B |
| Toggle word wrap | Alt+Z |
| Help & Keybindings | F1 |
| Settings | Ctrl+, or F10 |
When word wrap is off, long edit-mode lines extend past the viewport; Shift+wheel pans, or use the horizontal scrollbar that appears at the bottom of the editor. In preview, a table wider than the pane gets its own horizontal scrollbar beneath it — drag the thumb, or hover the table and use Shift+wheel or a horizontal swipe.




