THE BOOK OF LIFE
How Embeddings and Footnotes Work in The Secret Book of Walt
Read this before modifying ANY text rendering in the SBoW codebase. This file is for Lee Sharks and for any AI instance that works on this code.
Last updated: 2026-04-28 by TACHYON
THE THREE LAYERS
The site renders text through three layers stacked on top of each other. Every piece of body text passes through all three, in this order:
RAW TEXT (from JSON data file)
↓
LAYER 1: FOOTNOTES (footnotes.js + footnotes.jsx)
Scans for superscript markers (¹²³)
Makes them clickable (veil mode) or styled-but-passive (pierce mode)
↓
LAYER 2: EMPHASIS (inside LinkedText, in App.jsx)
Converts *italic* → <em> and **bold** → <strong>
↓
LAYER 3: GLOSSARY LINKS (inside LinkedText, in App.jsx)
Scans for terms from the TERMS dictionary
Wraps them in blue <a> tags that link to Google/Wikipedia/custom URLs
↓
RENDERED HTML (what the reader sees)
THE CRITICAL RULE: Every layer must pass text through to the next layer. If any layer returns raw text instead of calling the next layer's function, everything downstream breaks silently — no error, just missing features.
This is how the glossary links disappeared: Layer 2 (emphasis) had an early return that skipped Layer 3 (glossary). The footnotes still worked because they're Layer 1. The emphasis still worked because it's Layer 2. But all blue links vanished because Layer 3 was never called.
WHERE THINGS LIVE
The Data Files (what gets rendered)
| File | What's in it | How it's built |
|---|---|---|
public/walt_full_data.json |
All of Walt — every section, paragraph, footnote. 158 footnotes numbered 1-158 in reading order. | Built by scripts/build_walt_data.py from scripts/walt_source.md |
public/walt_gospel_versed.json |
The gospel in verse-by-verse format (numbered sayings with verse references). Used for the reading spine's verse view. | Pre-existing; not rebuilt by the script |
public/antioch_gospel_data.json |
All of Antioch — 3 front matter + 8 chapters + 13 back matter. 19 footnotes numbered 1-19 in reading order. | Built by scripts/build_antioch_data.py from scripts/antioch_source.md |
To change what text appears: Edit the source markdown in scripts/,
then re-run the build script. Or edit the JSON directly (for small fixes).
To change how text renders: Edit src/App.jsx (Walt) or src/Antioch.jsx.
The Rendering Code
| File | What it does | What it contains |
|---|---|---|
src/App.jsx |
Walt's entire reading interface | LinkedText (Layer 2+3), Leaf, SectionContent, Verse, GospelSection, TERMS dictionary, TERM_REGEX |
src/Antioch.jsx |
Antioch's entire reading interface | SectionRenderer, ChapterSection, Verse, uses LinkedText from App.jsx |
src/footnotes.js |
Pure JS: footnote scanning + disambiguation | buildGlobalFnMap, splitTextWithFootnotes, hasFootnoteMarkers |
src/footnotes.jsx |
React: footnote rendering components | FootnotedText (Layer 1), InlineFootnote (popup body) |
The Architecture Document
| File | What it is |
|---|---|
docs/FOOTNOTES.md |
Load-bearing spec for the footnote system. Read before touching footnotes. |
This file (docs/BOOK_OF_LIFE.md) |
How all three layers connect. Read before touching ANY rendering. |
LAYER 1: FOOTNOTES
What it does
Scans body text for Unicode superscript characters (¹²³⁴⁵⁶⁷⁸⁹⁰) and either makes them clickable (veil mode) or styles them as passive blue markers (pierce mode).
The disambiguation rule
NOT every superscript is a footnote. In codicological tables, G⁴⁶
means "Golden Ticket 46" — that's a label, not a footnote reference.
The rule: a superscript run is a footnote only if it is NOT immediately preceded by a letter (A-Z, a-z). So:
catalogue.³→ footnote ✓ (preceded by.)G⁴⁶→ NOT a footnote ✗ (preceded byG)text"¹³⁸→ footnote ✓ (preceded by")
How it flows
Body text: "The description is heavier than gold.¹³⁸ Individual copies vary."
splitTextWithFootnotes() breaks this into:
[
{ type: 'text', content: 'The description is heavier than gold.' },
{ type: 'fn', id: '¹³⁸' },
{ type: 'text', content: ' Individual copies vary.' }
]
FootnotedText component renders each piece:
- text pieces → passed to Layer 2+3 via linkText prop
- fn pieces → rendered as blue superscript spans (clickable in veil)
The global footnote map
Built once at page load by buildGlobalFnMap(data). Walks every section
and collects all footnote-type paragraphs into a lookup table:
{
'¹': { id: '¹', body: 'Sharks, L. (2015)…', sectionKey: 'manuscripts' },
'²': { id: '²', body: 'The parallel to…', sectionKey: 'manuscripts' },
...
'¹³⁸': { id: '¹³⁸', body: 'The description…', sectionKey: 'introduction' },
...
'¹⁵⁸': { id: '¹⁵⁸', body: '…', sectionKey: 'appendix_k' }
}
When a reader clicks ¹³⁸, the popup looks up globalFnMap['¹³⁸'] and
shows the body text below the paragraph.
Files involved
src/footnotes.js— the scanner and map builder (pure JS, no React)src/footnotes.jsx— the React renderer (FootnotedText + InlineFootnote)
LAYER 2: EMPHASIS
What it does
Converts markdown emphasis markers to HTML:
*text*→ text (italic,<em>)**text**→ text (bold,<strong>)
Where it lives
Inside LinkedText in src/App.jsx, in the processEmphasis() function.
How it flows
Input: "*August 2015 (translated) / 2037 (discovered)*"
Output: <em>August 2015 (translated) / 2037 (discovered)</em>
Input: "**The Editors, Pergamon Press**"
Output: <strong>The Editors, Pergamon Press</strong>
Input: "No asterisks in this text at all."
Output: "No asterisks in this text at all." (passed straight to Layer 3)
THE CRITICAL EARLY RETURN
if (!t || !t.includes("*")) return [linkifyText(t)];
// ^^^^^^^^^^^^
// MUST call linkifyText, not just return [t]
// If you return [t], Layer 3 is skipped and
// all glossary links disappear.
This is the line that broke the glossary links. It was return [t]
(raw string, no linking). Fixed to return [linkifyText(t)] (glossary-linked).
LAYER 3: GLOSSARY LINKS (EMBEDDINGS)
What it does
Scans text for terms from the TERMS dictionary and wraps matches in
blue <a> tags. These are the "embeddings" — the blue links that turn
body text into a navigable index of AI Overview and search nodes.
Where it lives
Inside LinkedText in src/App.jsx, in the linkifyText() function.
The TERMS dictionary and TERM_REGEX are defined earlier in the same file.
The TERMS dictionary
Located at approximately line 1249 of src/App.jsx. It looks like this:
const TERMS = {
"Crimson Hexagonal Archive": { q: "crimson hexagonal archive" },
"Deep Web": { w: "Deep_web" },
"Jack Feist": { q: "jack feist secret book of walt" },
"Lee Sharks": { q: "lee sharks" },
"Secret Book of Walt": { q: "secret book of walt" },
"Nag Hammadi": { w: "Nag_Hammadi_library" },
"Apocryphon of John": { w: "Apocryphon_of_John" },
"hologrammatic": { q: "\"Holographic Kernel\"" },
...
};
Each entry maps a term to a link target:
{ w: "Page_Name" }→ links toen.wikipedia.org/wiki/Page_Name{ q: "search terms" }→ links togoogle.com/search?q=search+terms{ u: "https://..." }→ links to a custom URL
How to ADD a new embedding
Add a line to the TERMS dictionary:
"My New Term": { q: "my new term" },
That's it. Every occurrence of "My New Term" in the body text will automatically become a blue link.
How to CHANGE where a link goes
Find the entry in TERMS and change the target:
// Before (links to Google search with quotes):
"Secret Book of Walt": { q: "\"secret book of walt\"" },
// After (links to Google search without quotes):
"Secret Book of Walt": { q: "secret book of walt" },
How to REMOVE an embedding
Delete the line from TERMS. The term will render as plain text.
How matching works
The TERM_REGEX is built from all the keys in TERMS, joined with |.
It's a global regex that scans the text left to right. When it finds a
match, it wraps the matched text in an <a> tag. It also extends the
match to include trailing word characters and possessives ('s).
TERMS count
Currently 109 entries. To verify:
grep -c '":\s*{' src/App.jsx # approximate count
HOW THE LAYERS CONNECT
In SectionContent (Walt front/back matter — src/App.jsx)
Data paragraph → FootnotedText component
↓
[splits into text + fn parts]
↓
text parts → linkText prop → LinkedText
↓
processEmphasis
↓
linkifyText (TERMS)
↓
rendered <a> tags
fn parts → blue superscript (clickable in veil)
The linkText prop is the bridge between Layer 1 and Layers 2+3.
It's defined as: const linkText = (s) => <LinkedText text={s} />
If FootnotedText doesn't call linkText for a text part, Layers 2+3 are skipped for that text. This is the second place links can break.
In Verse (Walt gospel — src/App.jsx)
The Verse component has its OWN inline rendering that doesn't use FootnotedText. It scans for superscripts directly and renders them. Glossary linking in verses goes through a separate path.
In Leaf (Walt — used in a few places)
Leaf wraps text in <LinkedText>. It does NOT use FootnotedText.
This means Leaf has glossary links (Layer 3) and emphasis (Layer 2)
but NO clickable footnotes (Layer 1). This is correct — Leaf is used
for hardcoded prose that doesn't have footnotes.
In SectionRenderer (Antioch — src/Antioch.jsx)
Same pattern as Walt's SectionContent:
Data paragraph → FootnotedText
↓
text parts → linkText → LinkedText (from App.jsx)
fn parts → blue superscript
Antioch imports LinkedText from App.jsx, so it uses the same TERMS dictionary and the same emphasis + linking logic.
THE FRAGILE POINTS (where things break)
1. The LinkedText early return
File: src/App.jsx, inside processEmphasis()
Line: if (!t || !t.includes("*")) return [linkifyText(t)];
Risk: If someone changes this to return [t], all glossary links
disappear from text without asterisks. This has happened once already.
2. The FootnotedText linkText delegation
File: src/footnotes.jsx, inside FootnotedText
Risk: If someone adds a conditional that skips linkText for certain
text (like text with emphasis markers), glossary links disappear for those
paragraphs. This has also happened.
Rule: When linkText is provided, ALWAYS use it. No conditions.
3. The TERMS dictionary key ordering
File: src/App.jsx, the TERMS object
Risk: If a shorter term is a substring of a longer term (e.g., "Walt"
is inside "Secret Book of Walt"), the regex may match the shorter one first.
The regex builder sorts by length (longest first) to prevent this, but
adding new terms that are substrings of existing terms needs care.
4. The build scripts overwriting the JSON
Files: scripts/build_walt_data.py, scripts/build_antioch_data.py
Risk: Re-running the build script regenerates the entire JSON data file
from the source markdown. Any manual edits to the JSON will be lost.
Rule: If you edit the JSON directly, note what you changed. If you
re-run the build script later, you'll need to re-apply manual edits.
5. The Verse component's separate footnote path
File: src/App.jsx, function Verse
Risk: The Verse component has its own footnote scanning that's separate
from FootnotedText. Changes to FootnotedText don't affect Verse, and vice
versa. This is intentional (Verse renders differently) but means footnote
changes need to be applied in both places.
HOW TO INSPECT WHAT'S DEPLOYED
Check the current TERMS dictionary
# From the repo root:
grep -A1 '"[A-Z]' src/App.jsx | grep -B1 '{' | head -40
This shows the first ~20 term entries and their link targets.
Check the footnote count
python3 -c "
import json
with open('public/walt_full_data.json') as f:
d = json.load(f)
count = 0
for k, v in d.items():
if isinstance(v, dict):
for p in v.get('paragraphs', []):
if p.get('type') == 'footnote': count += 1
print(f'Total footnote definitions: {count}')
"
Verify glossary links will render
# Build the site and check that LinkedText produces <a> tags:
npm run build
# The built JS file is in dist/assets/index-*.js
# Search for a known TERMS entry to confirm it's embedded:
grep -c "Nag Hammadi" dist/assets/index-*.js
# Should return > 0
Check what the reader actually sees
Open the deployed site, right-click any body text paragraph, and
"Inspect Element." Blue-linked terms should be <a> tags with
style="color: rgb(106, 159, 216)". If they're plain text, the
glossary layer is broken.
INSTRUCTIONS FOR FUTURE CLAUDE INSTANCES
If Lee asks you to modify text rendering in The Secret Book of Walt:
- Read
docs/FOOTNOTES.mdfirst. It explains the footnote system. - Read this file (
docs/BOOK_OF_LIFE.md) first. It explains how the three layers connect. - Do NOT rewrite LinkedText. It has been rewritten twice already and both times broke the glossary links. If you need to change emphasis handling, change it INSIDE the existing processEmphasis function without altering the early return or the linkifyText calls.
- Do NOT add conditionals to FootnotedText that skip the linkText prop. When linkText is provided, always use it.
- Test all three layers after any change:
- Are footnotes still clickable in veil mode? (Layer 1)
- Do italic and bold render correctly? (Layer 2)
- Are glossary terms still blue and linked? (Layer 3)
- Build before committing.
npm run build— if it fails, don't push.
QUICK REFERENCE: "I WANT TO..."
| Task | What to edit | File |
|---|---|---|
| Add a new glossary term | Add entry to TERMS dict | src/App.jsx ~line 1249 |
| Change where a glossary link goes | Edit the entry in TERMS | src/App.jsx ~line 1249 |
| Remove a glossary term | Delete the entry from TERMS | src/App.jsx ~line 1249 |
| Fix a typo in Walt body text | Edit the paragraph in the JSON | public/walt_full_data.json |
| Fix a typo permanently | Edit source MD + re-run script | scripts/walt_source.md + build_walt_data.py |
| Add a new footnote to Walt | Add to source MD + re-run | scripts/walt_source.md + build_walt_data.py |
| Change footnote numbering | Re-run the build script | scripts/build_walt_data.py (handles numbering) |
| Fix Antioch body text | Edit the JSON or source | public/antioch_gospel_data.json or scripts/ |
| Change footnote popup styling | Edit InlineFootnote | src/footnotes.jsx |
| Change footnote marker color | Edit FN_BLUE constant | src/footnotes.jsx line 19 |
| Change veil/pierce behavior | Edit FootnotedText | src/footnotes.jsx |
MAINTENANCE LOG
| Date | What changed | Who | What broke (if anything) |
|---|---|---|---|
| 2026-04-28 | Universal footnote system (Layer 1) | TACHYON | Nothing — new code |
| 2026-04-28 | Data rebuild (renumbering, quotes, asterisks) | TACHYON | Nothing — data layer |
| 2026-04-28 | Emphasis added to LinkedText (Layer 2) | TACHYON | Broke Layer 3 — processEmphasis returned raw text instead of calling linkifyText |
| 2026-04-28 | Fix: return [t] → return [linkifyText(t)] |
TACHYON | Nothing — one-line fix |
| 2026-04-28 | Fix: FootnotedText always delegates to linkText | TACHYON | Nothing — restores Layer 3 |
When you change rendering code, add a row to this table.
∮ = 1
No comments:
Post a Comment