// previews.js -- the bespoke per-package preview renderers, extracted from
// app.js. Pure preview HTML builders (ofs/os/previewLines + renderXxxPreview);
// they reference shared globals (PKGMAP, MAP, faceCss, effFg, ...) and are
// inlined into the page's single script element via the PREVIEWS_J token in app.js.
function ofs(app,face){const f=PKGMAP[app][face]||{},fg=effFg(pkgEffFg(app,face)),bg=pkgEffBg(app,face);return faceCss(f,fg,bg,{fontSize:(f.height||1),boxBg:bg||MAP['bg']});}
// The CSS for a UI-owned face rendered off any preview surface: effective fg
// (floored to the default fg) and bg, following the built-in UI inherit chain so
// the rendered color matches what the registry reports. The @ui counterpart to ofs.
function ulocateCss(face){const o=UIMAP[face]||{},fg=effFg(resolveUiAttr(face,'fg',UIMAP)),bg=resolveUiAttr(face,'bg',UIMAP)||null;return faceCss(o,fg,bg,{boxBg:bg||MAP['bg']});}
// previewSpan -- the one stateful locate adapter (preview-locate spec). Reads the
// live globals (PKGMAP / UIMAP / MAP), dispatches by the owner's surface to the
// package (ofs / PKGMAP) or @ui (UIMAP) style path, and emits the shared locate
// attributes: data-owner-app (the internal owner key), data-face, and the
// locate-onpane class when the owner is the pane currently viewed. TEXT is trusted
// preview HTML -- callers pre-escape entities, matching the old os() contract, so
// previewSpan does not re-escape it (that would double-escape < etc.). os
// delegates here for package owners; an @ui or cross-package owner makes an
// off-pane, hover-only span.
function attresc(s){return esc(String(s)).replace(/"/g,'"');}
function previewSpan(owner,face,text){
const style=owner==='@ui'?ulocateCss(face):ofs(owner,face);
const cls=isLocateOnPane(owner,curApp())?' class="locate-onpane"':'';
const title=attresc(formatLocateTitle(locateFaceMeta(owner,face,LOCATE_REG)));
return `${text}`;
}
function os(app,face,txt){return previewSpan(app,face,txt);}
// Preview font stack: the embedded @font-face (family "ThemeStudioNerd",
// Symbols Nerd Font Mono inlined as a data: URI in styles.css) supplies the nerd
// glyphs; monospace supplies everything else. The family name is deliberately
// custom, NOT the real "Symbols Nerd Font Mono": when the @font-face name matches
// a font the user has installed system-wide, Chrome resolves the family to the
// local copy instead of our embedded one and the glyphs render as tofu (the
// embedded font only wins in environments without that system font, e.g. headless
// CI). A unique family name forces the embedded font. "ThemeStudioNerd" carries
// only icon glyphs, so plain text falls through to monospace and the layout is
// unchanged — only the nerd codepoints pull from the embedded font.
// NOTE: the family name is UNQUOTED here on purpose. PREVIEW_FONT is interpolated
// into inline style="..." attributes (previewLines, genericPreview, the mock
// frame), and a double-quoted family name inside a double-quoted attribute
// terminates the attribute early, silently dropping the font-family (the glyphs
// then fall back to monospace = tofu). A no-space identifier needs no quotes, so
// keep ThemeStudioNerd quote-free and never reintroduce a spaced/quoted name here.
const PREVIEW_FONT='ThemeStudioNerd,monospace';
// Shared wrapper for the line-based package previews: a nerd-font pre block.
// Each renderer builds its own L array of os(...) lines and returns previewLines(L).
function previewLines(L){return `
${L.join('\n')}
`;}
function renderOrgPreview(){const a='org-mode',L=[];
L.push(os(a,'org-document-info-keyword','#+TITLE:')+' '+os(a,'org-document-title','Project Notes'));
L.push(os(a,'org-document-info-keyword','#+AUTHOR:')+' '+os(a,'org-document-info','Craig Jennings'));
L.push(os(a,'org-meta-line','#+STARTUP: overview'));
L.push('');
L.push(os(a,'org-level-1','* Inbox')+' '+os(a,'org-tag',':work:')+os(a,'org-tag-group',':@office:'));
L.push(os(a,'org-level-2','** ')+os(a,'org-todo','TODO')+os(a,'org-level-2',' Draft the spec')+' '+os(a,'org-priority','[#A]')+' '+os(a,'org-tag',':spec:'));
L.push(' '+os(a,'org-special-keyword','SCHEDULED:')+' '+os(a,'org-date','<2026-06-08 Sun>')+' '+os(a,'org-special-keyword','DEADLINE:')+' '+os(a,'org-date','<2026-06-12 Thu>'));
L.push(' '+os(a,'org-drawer',':PROPERTIES:'));
L.push(' '+os(a,'org-special-keyword',':ID:')+' '+os(a,'org-property-value','abc-123-def'));
L.push(' '+os(a,'org-drawer',':END:'));
L.push(' '+os(a,'org-list-dt','- term ::')+' definition, with a '+os(a,'org-footnote','[fn:1]')+' note.');
L.push(' '+os(a,'org-checkbox','[X]')+' done item '+os(a,'org-checkbox-statistics-done','[2/2]'));
L.push(' '+os(a,'org-checkbox','[ ]')+' open item '+os(a,'org-checkbox-statistics-todo','[0/3]')+' '+os(a,'org-warning','(!)'));
L.push(os(a,'org-level-2','** ')+os(a,'org-done','DONE')+os(a,'org-headline-done',' Ship the tool'));
L.push(os(a,'org-level-3','*** ')+os(a,'org-todo','TODO')+os(a,'org-headline-todo',' Heading three'));
L.push(os(a,'org-level-4','**** four')+' / '+os(a,'org-level-5','***** five')+' / '+os(a,'org-level-6','****** six')+' / '+os(a,'org-level-7','******* seven')+' / '+os(a,'org-level-8','******** eight'));
L.push(' Inline '+os(a,'org-code','=code=')+', '+os(a,'org-verbatim','~verbatim~')+', '+os(a,'org-inline-src-block','src_py{1+1}')+',');
L.push(' a '+os(a,'org-link','[[https://gnu.org][link]]')+', a '+os(a,'org-target','<<target>>')+', a '+os(a,'org-macro','{{{macro}}}')+',');
L.push(' a '+os(a,'org-cite','[cite:')+os(a,'org-cite-key','@knuth1984')+os(a,'org-cite',']')+', a date '+os(a,'org-sexp-date','<%%(diary-float 6 5 2)>')+'.');
L.push(' '+os(a,'org-quote','#+begin_quote')+' a '+os(a,'org-verse','verse')+' line, latex '+os(a,'org-latex-and-related','$E = mc^2$')+'.');
L.push('');
L.push(' '+os(a,'org-block-begin-line','#+begin_src elisp'));
L.push(' '+os(a,'org-block',' (message "hi")'));
L.push(' '+os(a,'org-block-end-line','#+end_src'));
L.push('');
L.push(' '+os(a,'org-table-header','| name | hex |'));
L.push(' '+os(a,'org-table','|------+---------|'));
L.push(' '+os(a,'org-table-row','| blue | #67809c |')+' '+os(a,'org-formula',':=vsum(@2)'));
L.push(' '+os(a,'org-column-title','Effort')+' '+os(a,'org-column','| 0:30 |')+' '+os(a,'org-archived','* archived')+os(a,'org-ellipsis',' ...'));
L.push('');
L.push(os(a,'org-agenda-structure','Week-agenda (W23):'));
L.push(os(a,'org-agenda-date','Monday 8 June 2026'));
L.push(os(a,'org-agenda-date-today','Tuesday 9 June 2026')+' '+os(a,'org-agenda-current-time','10:24')+' '+os(a,'org-time-grid','----------'));
L.push(os(a,'org-agenda-date-weekend','Saturday 13 June')+' / '+os(a,'org-agenda-date-weekend-today','wknd-today'));
L.push(' '+os(a,'org-scheduled-previously','Sched.past:')+' overdue '+os(a,'org-agenda-done','x done item'));
L.push(' '+os(a,'org-scheduled','Scheduled:')+' a task '+os(a,'org-scheduled-today','due today'));
L.push(' '+os(a,'org-imminent-deadline','Deadline!')+' / '+os(a,'org-upcoming-deadline','upcoming')+' / '+os(a,'org-upcoming-distant-deadline','distant'));
L.push(' '+os(a,'org-agenda-dimmed-todo-face','dimmed todo')+' '+os(a,'org-agenda-diary','diary')+' '+os(a,'org-agenda-clocking','clocking'));
L.push(' '+os(a,'org-agenda-calendar-event','cal-event')+' / '+os(a,'org-agenda-calendar-sexp','cal-sexp')+' / '+os(a,'org-agenda-calendar-daterange','range'));
L.push(' '+os(a,'org-agenda-structure-secondary','secondary')+' '+os(a,'org-agenda-structure-filter','filter')+' '+os(a,'org-agenda-restriction-lock','lock')+' '+os(a,'org-agenda-column-dateline','col-date'));
L.push(' Filters: '+os(a,'org-agenda-filter-category','cat')+' '+os(a,'org-agenda-filter-tags','tags')+' '+os(a,'org-agenda-filter-effort','effort')+' '+os(a,'org-agenda-filter-regexp','re'));
L.push(' '+os(a,'org-mode-line-clock','[0:45]')+' / '+os(a,'org-mode-line-clock-overrun','[OVER]')+' '+os(a,'org-dispatcher-highlight','[d]ispatch'));
return previewLines(L);
}
function renderMagitPreview(){const a='magit',L=[];
L.push(os(a,'magit-header-line',' Magit: dotemacs ')+' '+os(a,'magit-header-line-key','g')+os(a,'magit-header-line-log-select',' refresh'));
L.push(os(a,'magit-head','Head:')+' '+os(a,'magit-branch-current','main')+' '+os(a,'magit-diff-revision-summary','Ship the tool'));
L.push(os(a,'magit-head','Merge:')+' '+os(a,'magit-branch-remote','origin/main')+' '+os(a,'magit-branch-local','main'));
L.push(os(a,'magit-head','Push:')+' '+os(a,'magit-branch-remote-head','origin/main'));
L.push(os(a,'magit-head','Upstream:')+' '+os(a,'magit-branch-upstream','origin/main')+' '+os(a,'magit-branch-warning','(diverged)'));
L.push('');
L.push(os(a,'magit-section-heading','Untracked files')+' '+os(a,'magit-section-child-count','(2)'));
L.push(' '+os(a,'magit-filename','notes.txt')+' '+os(a,'magit-dimmed','(ignored sibling)'));
L.push(os(a,'magit-section-highlight',' scratch.el (highlighted row)'));
L.push('');
L.push(os(a,'magit-section-heading','Unstaged changes')+' '+os(a,'magit-section-child-count','(1)'));
L.push(os(a,'magit-diff-file-heading','modified generate.py'));
L.push(os(a,'magit-diff-hunk-heading','@@ -1,4 +1,5 @@ def main'));
L.push(os(a,'magit-diff-context',' unchanged context'));
L.push(os(a,'magit-diff-removed','- old line')+os(a,'magit-diff-whitespace-warning',' '));
L.push(os(a,'magit-diff-added','+ new line'));
L.push('');
L.push(os(a,'magit-section-heading','Staged changes')+' '+os(a,'magit-diffstat-added','++++')+os(a,'magit-diffstat-removed','--'));
L.push(os(a,'magit-diff-file-heading-highlight','modified README.md (highlighted heading)'));
L.push(os(a,'magit-diff-hunk-heading-highlight','@@ hunk heading highlight @@'));
L.push(os(a,'magit-diff-added-highlight','+ added highlight')+' '+os(a,'magit-diff-removed-highlight','- removed highlight'));
L.push(os(a,'magit-diff-context-highlight',' context highlight'));
L.push('');
L.push(os(a,'magit-section-heading','Stashes'));
L.push(' '+os(a,'magit-refname-stash','stash@{0}')+' '+os(a,'magit-refname-wip','wip')+' '+os(a,'magit-refname-pullreq','pr/42')+' '+os(a,'magit-refname','refs/heads/x'));
L.push('');
L.push(os(a,'magit-section-heading','Recent commits'));
L.push(os(a,'magit-log-graph','* ')+os(a,'magit-hash','b5b1869f')+' '+os(a,'magit-log-date','06-08')+' '+os(a,'magit-log-author','Craig')+' enlarge the picker');
L.push(os(a,'magit-log-graph','* ')+os(a,'magit-hash','4fa5e995')+' '+os(a,'magit-log-date','06-07')+' '+os(a,'magit-log-author','Craig')+' '+os(a,'magit-keyword','[feat]')+' picker');
L.push(os(a,'magit-log-graph','* ')+os(a,'magit-hash','de07e01a')+' '+os(a,'magit-log-date','06-05')+' '+os(a,'magit-log-author','Craig')+' '+os(a,'magit-tag','v0.3')+' '+os(a,'magit-keyword-squash','!squash'));
L.push('');
L.push(os(a,'magit-section-secondary-heading','Merge conflict')+' '+os(a,'magit-diff-lines-heading','lines 10-14')+os(a,'magit-diff-lines-boundary','|'));
L.push(' '+os(a,'magit-diff-conflict-heading','=======')+' '+os(a,'magit-diff-conflict-heading-highlight','(hl)'));
L.push(' '+os(a,'magit-diff-base','base')+'/'+os(a,'magit-diff-base-highlight','base-hl')+' '+os(a,'magit-diff-our','ours')+'/'+os(a,'magit-diff-our-highlight','ours-hl')+' '+os(a,'magit-diff-their','theirs')+'/'+os(a,'magit-diff-their-highlight','theirs-hl'));
L.push(' '+os(a,'magit-diff-hunk-region','hunk-region')+' '+os(a,'magit-diff-file-heading-selection','file-sel')+' '+os(a,'magit-diff-hunk-heading-selection','hunk-sel')+' '+os(a,'magit-section-heading-selection','sec-sel')+' '+os(a,'magit-diff-revision-summary-highlight','rev-sum-hl'));
L.push('');
L.push(os(a,'magit-section-heading','Reflog'));
L.push(' '+os(a,'magit-reflog-commit','commit')+' '+os(a,'magit-reflog-amend','amend')+' '+os(a,'magit-reflog-merge','merge')+' '+os(a,'magit-reflog-checkout','checkout')+' '+os(a,'magit-reflog-reset','reset')+' '+os(a,'magit-reflog-rebase','rebase')+' '+os(a,'magit-reflog-cherry-pick','cherry-pick')+' '+os(a,'magit-reflog-remote','remote')+' '+os(a,'magit-reflog-other','other'));
L.push(os(a,'magit-section-heading','Rebase sequence'));
L.push(' '+os(a,'magit-sequence-pick','pick')+' '+os(a,'magit-sequence-stop','stop')+' '+os(a,'magit-sequence-part','part')+' '+os(a,'magit-sequence-head','head')+' '+os(a,'magit-sequence-drop','drop')+' '+os(a,'magit-sequence-done','done')+' '+os(a,'magit-sequence-onto','onto')+' '+os(a,'magit-sequence-exec','exec'));
L.push(os(a,'magit-section-heading','Bisect / Cherry / Process'));
L.push(' '+os(a,'magit-bisect-good','good')+' '+os(a,'magit-bisect-bad','bad')+' '+os(a,'magit-bisect-skip','skip')+' '+os(a,'magit-cherry-equivalent','equivalent')+' '+os(a,'magit-cherry-unmatched','unmatched'));
L.push(' '+os(a,'magit-process-ok','OK')+' '+os(a,'magit-process-ng','NG')+' '+os(a,'magit-mode-line-process','[fetch]')+' '+os(a,'magit-mode-line-process-error','[error]'));
L.push(os(a,'magit-section-heading','Blame'));
L.push(os(a,'magit-blame-margin','margin')+os(a,'magit-blame-heading',' b5b1869f '))
L.push(' '+os(a,'magit-blame-hash','b5b1869f')+' '+os(a,'magit-blame-name','Craig')+' '+os(a,'magit-blame-date','2026-06-08')+' '+os(a,'magit-blame-summary','enlarge picker')+' '+os(a,'magit-blame-highlight','hl')+' '+os(a,'magit-blame-dimmed','dim'));
L.push(os(a,'magit-section-heading','Signatures')+os(a,'magit-left-margin',' '));
L.push(' '+os(a,'magit-signature-good','good')+' '+os(a,'magit-signature-bad','bad')+' '+os(a,'magit-signature-untrusted','untrusted')+' '+os(a,'magit-signature-expired','expired')+' '+os(a,'magit-signature-expired-key','expired-key')+' '+os(a,'magit-signature-revoked','revoked')+' '+os(a,'magit-signature-error','error'));
return previewLines(L);}
function renderElfeedPreview(){const a='elfeed',L=[];
L.push(os(a,'elfeed-search-filter-face','@6-months-ago +unread')+' '+os(a,'elfeed-search-unread-count-face','3/120')+' '+os(a,'elfeed-search-last-update-face','updated 02:24'));
L.push('');
L.push(os(a,'elfeed-search-date-face','2026-06-08')+' '+os(a,'elfeed-search-feed-face','Planet Emacs')+' '+os(a,'elfeed-search-unread-title-face','New release of Magit')+' '+os(a,'elfeed-search-tag-face',':emacs:'));
L.push(os(a,'elfeed-search-date-face','2026-06-07')+' '+os(a,'elfeed-search-feed-face','LWN')+' '+os(a,'elfeed-search-unread-title-face','Kernel 6.18 lands')+' '+os(a,'elfeed-search-tag-face',':linux:'));
L.push(os(a,'elfeed-search-date-face','2026-06-05')+' '+os(a,'elfeed-search-feed-face','Hacker News')+' '+os(a,'elfeed-search-title-face','Show HN: a theme editor')+' '+os(a,'elfeed-search-tag-face',':show:'));
L.push('');
L.push(os(a,'elfeed-log-date-face','02:24:01')+' '+os(a,'elfeed-log-info-level-face','INFO ')+' updated 12 feeds');
L.push(os(a,'elfeed-log-date-face','02:24:02')+' '+os(a,'elfeed-log-warn-level-face','WARN ')+' slow feed: example.com');
L.push(os(a,'elfeed-log-date-face','02:24:03')+' '+os(a,'elfeed-log-error-level-face','ERROR')+' failed: bad.example');
L.push(os(a,'elfeed-log-date-face','02:24:04')+' '+os(a,'elfeed-log-debug-level-face','DEBUG')+' parsed 340 entries');
return previewLines(L);}
function renderDashboardPreview(){const a='dashboard',L=[];
L.push(os(a,'dashboard-text-banner',' [ dashboard banner image ]'));
L.push(os(a,'dashboard-banner-logo-title','Emacs: The Editor That Saves Your Soul'));
L.push('');
L.push(os(a,'dashboard-navigator',' Code Files Terminal Agenda'));
L.push(os(a,'dashboard-navigator',' Feeds Books Flashcards Music'));
L.push(os(a,'dashboard-navigator',' Email IRC Telegram'));
L.push(os(a,'dashboard-navigator',' Slack Linear'));
L.push('');
L.push('');
L.push(os(a,'dashboard-heading','Projects:'));
L.push(os(a,'dashboard-items-face',' ~/'));
L.push(os(a,'dashboard-items-face',' ~/.emacs.d/'));
L.push(os(a,'dashboard-items-face',' ~/projects/work/'));
L.push(os(a,'dashboard-items-face',' ~/org/roam/'));
L.push(os(a,'dashboard-items-face',' ~/projects/home/'));
L.push('');
L.push(os(a,'dashboard-heading','Bookmarks'));
L.push(os(a,'dashboard-items-face',' Cesar Aira, The Little Buddhist Monk & the Proof'));
L.push(os(a,'dashboard-items-face',' Edward Abbey, The Fool’s Progress: An Honest Novel'));
L.push(os(a,'dashboard-items-face',' Agatha Christie, The A.B.C. Murders'));
L.push('');
L.push(os(a,'dashboard-heading','Recent Files:'));
L.push(os(a,'dashboard-items-face',' theme-theme.el'));
L.push(os(a,'dashboard-items-face',' todo.org'));
L.push(os(a,'dashboard-items-face',' theme-studio-palette-generator-spec.org'));
return previewLines(L);}
function renderMu4ePreview(){const a='mu4e',L=[];
const pad=(s,n)=>{s=String(s);return s.length>=n?s.slice(0,n):s+' '.repeat(n-s.length);};
// One header line: the flags column in mu4e-header-marks-face, the rest of the
// row in the message's state face (unread, replied, flagged, ...).
const row=(flags,date,from,subj,face)=>
os(a,'mu4e-header-marks-face',pad(flags,4))+os(a,face,pad(date,12)+pad(from,17)+subj);
// status / context bar
L.push(os(a,'mu4e-title-face','mu4e')+' '+os(a,'mu4e-context-face','[Personal]')+' '+os(a,'mu4e-ok-face','online')+' '+os(a,'mu4e-warning-face','2 retrying')+' '+os(a,'mu4e-modeline-face','[12/340]'));
L.push('');
// column header + the message list, one row per state
L.push(os(a,'mu4e-header-title-face',pad('Flags',4)+pad('Date',12)+pad('From',17)+'Subject'));
L.push(row('N','2026-06-14','Christine Park','Re: quarterly numbers','mu4e-unread-face'));
L.push(row('','2026-06-13','Bob Lin','Lunch on Friday?','mu4e-header-face'));
// current line at point: the whole row gets the highlight background
L.push(os(a,'mu4e-header-highlight-face',row('R','2026-06-13','dev-list','merged the parser fix','mu4e-replied-face')));
L.push(row('F','2026-06-12','Carol Reyes','Fwd: the signed contract','mu4e-forwarded-face'));
L.push(row('D','2026-06-11','(draft)','Notes to finish later','mu4e-draft-face'));
L.push(row('T','2026-06-10','spam@nowhere','You have won a prize','mu4e-trashed-face'));
L.push(row('','2026-06-09','Erin (cc)','thread you follow','mu4e-related-face'));
L.push(row('!','2026-06-08','Frank Diaz','budget needs sign-off','mu4e-flagged-face'));
L.push('');
// a message view below the list
L.push(os(a,'mu4e-header-key-face','From:')+' '+os(a,'mu4e-contact-face','Christine Park <christine@example.com>'));
L.push(os(a,'mu4e-header-key-face','To:')+' '+os(a,'mu4e-special-header-value-face','craig, dev-list@gnu.org'));
L.push(os(a,'mu4e-header-key-face','Subject:')+' '+os(a,'mu4e-header-value-face','Re: quarterly numbers'));
L.push('');
L.push(' Body with a '+os(a,'mu4e-highlight-face','search hit')+', a link '+os(a,'mu4e-url-number-face','[1]')+' '+os(a,'mu4e-link-face','https://example.com')+', and a '+os(a,'mu4e-region-code','code region')+'.');
L.push(' '+os(a,'mu4e-system-face','*** mu: 340 messages indexed ***'));
L.push(' '+os(a,'mu4e-footer-face','-- Sent with mu4e'));
L.push('');
L.push(os(a,'mu4e-compose-separator-face','--text follows this line--'));
return previewLines(L);}
function renderGnusPreview(){const a='gnus',L=[];
// mu4e renders the open message with gnus, so this is the article view:
// a header block, a body with inline emphasis and a button, then a quoted
// reply chain (one cite face per nesting level) and the signature.
L.push(os(a,'gnus-header-name','From: ')+os(a,'gnus-header-from','Christine Park <christine@example.com>'));
L.push(os(a,'gnus-header-name','To: ')+os(a,'gnus-header-content','craig@cjennings.net'));
L.push(os(a,'gnus-header-name','Newsgroups: ')+os(a,'gnus-header-newsgroups','gnu.emacs.help'));
L.push(os(a,'gnus-header-name','Subject: ')+os(a,'gnus-header-subject','Re: quarterly numbers'));
L.push(os(a,'gnus-header-name','Date: ')+os(a,'gnus-header-content','Sat, 14 Jun 2026 09:12:04 -0500'));
L.push('');
L.push('Thanks for the draft. The '+os(a,'gnus-emphasis-bold','revenue line')+' is '+os(a,'gnus-emphasis-italic','close')+', but the '+os(a,'gnus-emphasis-underline','footnote')+' is '+os(a,'gnus-emphasis-strikethru','wrong')+' '+os(a,'gnus-emphasis-highlight-words','FIXME')+'.');
L.push('See the worksheet: '+os(a,'gnus-button','[https://example.com/q2]'));
L.push('');
L.push(os(a,'gnus-cite-attribution','On Fri, Bob Lin wrote:'));
L.push(os(a,'gnus-cite-1','> The Q2 totals are ready for review.'));
L.push(os(a,'gnus-cite-2','>> Did the Segpay refund post yet?'));
L.push(os(a,'gnus-cite-3','>>> Yes, it cleared on the 5th.'));
L.push(os(a,'gnus-cite-4','>>>> Good, then we are square.'));
L.push(os(a,'gnus-cite-5','>>>>> earlier reply, level 5'));
L.push(os(a,'gnus-cite-6','>>>>>> level 6'));
L.push(os(a,'gnus-cite-7','>>>>>>> level 7'));
L.push(os(a,'gnus-cite-8','>>>>>>>> level 8'));
L.push(os(a,'gnus-cite-9','>>>>>>>>> level 9'));
L.push(os(a,'gnus-cite-10','>>>>>>>>>> level 10'));
L.push(os(a,'gnus-cite-11','>>>>>>>>>>> level 11'));
L.push('');
L.push(os(a,'gnus-signature','-- '));
L.push(os(a,'gnus-signature','Christine Park, Finance'));
return previewLines(L);}
function renderOrgFacesPreview(){const a='org-faces',L=[];
L.push('Agenda header row -- one face per keyword and priority (this config, not built-in org):');
L.push('');
L.push(os(a,'org-faces-todo','TODO')+' Draft the spec '+os(a,'org-faces-priority-a','[#A]'));
L.push(os(a,'org-faces-project','PROJECT')+' Theme studio overhaul '+os(a,'org-faces-priority-b','[#B]'));
L.push(os(a,'org-faces-doing','DOING')+' Wire the faces '+os(a,'org-faces-priority-c','[#C]'));
L.push(os(a,'org-faces-waiting','WAITING')+' On review '+os(a,'org-faces-priority-d','[#D]'));
L.push(os(a,'org-faces-verify','VERIFY')+' Confirm the round-trip');
L.push(os(a,'org-faces-stalled','STALLED')+' Blocked on upstream');
L.push(os(a,'org-faces-delegated','DELEGATED')+' Handed to Kostya');
L.push(os(a,'org-faces-failed','FAILED')+' Could not reproduce');
L.push(os(a,'org-faces-done','DONE')+' Shipped the module');
L.push(os(a,'org-faces-cancelled','CANCELLED')+' Dropped the approach');
L.push('');
L.push('Unfocused (auto-dim) -- the -dim variants auto-dim remaps onto in non-selected windows:');
L.push('');
L.push(os(a,'org-faces-todo-dim','TODO')+' Draft the spec '+os(a,'org-faces-priority-a-dim','[#A]'));
L.push(os(a,'org-faces-project-dim','PROJECT')+' Theme studio overhaul '+os(a,'org-faces-priority-b-dim','[#B]'));
L.push(os(a,'org-faces-doing-dim','DOING')+' Wire the faces '+os(a,'org-faces-priority-c-dim','[#C]'));
L.push(os(a,'org-faces-waiting-dim','WAITING')+' On review '+os(a,'org-faces-priority-d-dim','[#D]'));
L.push(os(a,'org-faces-verify-dim','VERIFY')+' Confirm the round-trip');
L.push(os(a,'org-faces-stalled-dim','STALLED')+' Blocked on upstream');
L.push(os(a,'org-faces-delegated-dim','DELEGATED')+' Handed to Kostya');
L.push(os(a,'org-faces-failed-dim','FAILED')+' Could not reproduce');
L.push(os(a,'org-faces-done-dim','DONE')+' Shipped the module');
L.push(os(a,'org-faces-cancelled-dim','CANCELLED')+' Dropped the approach');
return previewLines(L);}
function renderLspPreview(){const a='lsp-mode',L=[];
L.push(os(a,'lsp-signature-face','process(')+os(a,'lsp-signature-highlight-function-argument','items: list')+os(a,'lsp-signature-face',') -> None'));
L.push(os(a,'lsp-signature-posframe',' docs: iterate over items and process each one '));
L.push('');
L.push('def process(items):');
L.push(' n = len(items)'+os(a,'lsp-inlay-hint-type-face',': int'));
L.push(' handle('+os(a,'lsp-inlay-hint-parameter-face','arg:')+'n)'+os(a,'lsp-inlay-hint-face',' # hint'));
L.push(' '+os(a,'lsp-face-highlight-read','value')+' = '+os(a,'lsp-face-highlight-write','value')+' + '+os(a,'lsp-face-highlight-textual','value'));
L.push(' rename '+os(a,'lsp-face-rename','oldName')+' to '+os(a,'lsp-rename-placeholder-face','newName'));
L.push(' getName() '+os(a,'lsp-details-face','str the cached getter'));
L.push('');
L.push(os(a,'lsp-installation-buffer-face','Installing pyright...')+' '+os(a,'lsp-installation-finished-buffer-face','done.'));
return previewLines(L);}
function renderGitGutterPreview(){const a='git-gutter',L=[];
L.push(os(a,'git-gutter:added','+')+os(a,'git-gutter:separator','|')+' added line of code');
L.push(os(a,'git-gutter:modified','~')+os(a,'git-gutter:separator','|')+' modified line of code');
L.push(os(a,'git-gutter:deleted','_')+os(a,'git-gutter:separator','|')+' (deleted lines marker)');
L.push(os(a,'git-gutter:unchanged',' ')+os(a,'git-gutter:separator','|')+' '+os(a,'git-gutter:unchanged','unchanged line of code'));
return previewLines(L);}
function renderEatPreview(){const a='eat',L=[],c=(f,t)=>os(a,'eat-term-color-'+f,t),x=(f,t)=>os(a,'eat-term-'+f,t),an=(g,t)=>os(a,'eat-shell-prompt-annotation-'+g,t);
const p=g=>an(g,g==='success'?'✔':g==='failure'?'✘':'…')+' ~/projects/app $ ';
// 1. directory listing -- the widest palette block (dircolors)
L.push(p('success')+'eza -la --color');
L.push('drwxr-xr-x - 14:02 '+c('blue','.git/'));
L.push('.rw-r--r-- 120 09:11 .gitignore');
L.push('drwxr-xr-x - 14:02 '+c('blue','src/'));
L.push('drwxr-xr-x - 13:48 '+c('blue','tests/'));
L.push('.rwxr-xr-x 2.1k 14:00 '+c('bright-green','run.sh'));
L.push('lrwxr-xr-x - 14:01 '+c('cyan','latest')+' -> '+c('blue','v2.1/'));
L.push('.rw-r--r-- 4.5M 22:30 '+c('red','backup.tar.gz'));
L.push('.rw-r--r-- 88k 18:05 '+c('magenta','logo.png'));
L.push('.rw-r--r-- 3.2k 14:02 README.md');
L.push('');
// 2. git status -- staged green, unstaged/untracked red
L.push(p('success')+'git status -sb');
L.push(c('bright-cyan','## main...origin/main [ahead 2]'));
L.push(c('green','A src/eat-preview.js'));
L.push(c('green','A src/cache.el'));
L.push(c('green','M README.md'));
L.push(c('red',' M init.el'));
L.push(c('red',' M modules/term-config.el'));
L.push(c('red',' D modules/old-vterm.el'));
L.push(c('red','?? docs/design/eat.org'));
L.push(c('red','?? scratch.txt'));
L.push('');
// 3. git log --decorate -- yellow hashes, colored refs, a merge
L.push(p('success')+'git log --oneline --graph --decorate');
L.push(c('bright-black','* ')+c('yellow','a1b2c3d')+' '+c('bright-cyan','(HEAD -> ')+c('bright-green','main')+c('bright-cyan',')')+' richer eat preview blocks');
L.push(c('bright-black','* ')+c('yellow','9f8e7d6')+' '+c('bright-yellow','(tag: v2.1, ')+c('bright-red','origin/main')+c('bright-yellow',')')+' lowercase the labels');
L.push(c('bright-black','* ')+c('yellow','3c4d5e6')+' Merge branch '+c('green',"'eat-faces'"));
L.push(c('bright-black','|\\ '));
L.push(c('bright-black','| * ')+c('yellow','7a8b9c0')+' expose eat faces to studio');
L.push(c('bright-black','| * ')+c('yellow','1d2e3f4')+' add eat-term-color docstrings');
L.push(c('bright-black','|/ '));
L.push(c('bright-black','* ')+c('yellow','5f6a7b8')+' default video player to mpv');
L.push(c('bright-black','* ')+c('yellow','2c3d4e5')+' calendar-sync robustness fixes');
L.push('');
// 4. test run -- pass green, skip yellow, fail red, bold summary, faint detail
L.push(p('failure')+'make test');
L.push(c('green','✔ PASS')+' term-toggle '+x('faint','(19 tests)'));
L.push(c('green','✔ PASS')+' ai-term '+x('faint','(158 tests)'));
L.push(c('green','✔ PASS')+' calendar-sync '+x('faint','(575 tests)'));
L.push(c('green','✔ PASS')+' dashboard '+x('faint','(18 tests)'));
L.push(c('yellow','⚠ SKIP')+' network-sync '+x('faint','(2 tests, offline)'));
L.push(c('green','✔ PASS')+' transcription '+x('faint','(44 tests)'));
L.push(c('red','✘ FAIL')+' org-roam-refile '+x('faint','(1 test)'));
L.push(' '+x('italic','expected 3 refile targets, got 0'));
L.push(x('bold','Ran 817 tests, 815 passed, ')+c('yellow','1 skipped, ')+c('red','1 failed')+' '+x('faint','0.84s'));
L.push('');
// swatch reference key, below the realistic blocks
L.push(x('faint','palette')+' '+c('black','■')+c('red','■')+c('green','■')+c('yellow','■')+c('blue','■')+c('magenta','■')+c('cyan','■')+c('white','■')+' '+c('bright-black','■')+c('bright-red','■')+c('bright-green','■')+c('bright-yellow','■')+c('bright-blue','■')+c('bright-magenta','■')+c('bright-cyan','■')+c('bright-white','■'));
L.push(x('faint','attrs')+' '+x('bold','bold')+' '+x('faint','faint')+' '+x('italic','italic')+' '+x('slow-blink','slow-blink')+' '+x('fast-blink','fast-blink'));
L.push(x('faint','prompt')+' '+an('success','✔ ok')+' '+an('running','… running')+' '+an('failure','✘ failed'));
return previewLines(L);}
function renderFlycheckPreview(){const a='flycheck',L=[];
L.push(os(a,'flycheck-fringe-error','E')+os(a,'flycheck-fringe-warning','W')+os(a,'flycheck-fringe-info','I')+' x = '+os(a,'flycheck-error','undefined_name')+'('+os(a,'flycheck-warning','unused_arg')+') '+os(a,'flycheck-info','# note'));
L.push(' '+os(a,'flycheck-error-delimiter','[')+os(a,'flycheck-delimited-error','err')+os(a,'flycheck-error-delimiter',']'));
L.push('');
L.push(os(a,'flycheck-error-list-checker-name','pyright')+' '+os(a,'flycheck-verify-select-checker','(selected checker)'));
L.push(os(a,'flycheck-error-list-filename','main.py')+':'+os(a,'flycheck-error-list-line-number','12')+':'+os(a,'flycheck-error-list-column-number','4')+' '+os(a,'flycheck-error-list-error','error')+' '+os(a,'flycheck-error-list-error-message','undefined name x')+' '+os(a,'flycheck-error-list-id','[E0602]'));
L.push(os(a,'flycheck-error-list-filename','main.py')+':'+os(a,'flycheck-error-list-line-number','18')+':'+os(a,'flycheck-error-list-column-number','1')+' '+os(a,'flycheck-error-list-warning','warning')+' '+os(a,'flycheck-error-list-error-message','unused import')+' '+os(a,'flycheck-error-list-id-with-explainer','[W0611?]'));
L.push(os(a,'flycheck-error-list-highlight','main.py:20 '+os(a,'flycheck-error-list-info','info')+' highlighted row'));
return previewLines(L);}
function renderDiredPreview(){const a='dired',L=[];
L.push(os(a,'dired-header','/home/craig/code:'));
L.push(' '+os(a,'dired-perm-write','drwxr-xr-x')+' craig 4096 '+os(a,'dired-directory','src/'));
L.push(' -rw-r--r-- craig 120 notes.org');
L.push(' '+os(a,'dired-perm-write','lrwxrwxrwx')+' craig 18 '+os(a,'dired-symlink','latest -> v2.1'));
L.push(' lrwxrwxrwx craig -- '+os(a,'dired-broken-symlink','dead -> gone'));
L.push(os(a,'dired-flagged','D')+' -rw-r--r-- craig 40 deleteme.tmp');
L.push(os(a,'dired-mark','*')+' '+os(a,'dired-marked','-rw-r--r-- craig 210 marked.txt'));
L.push(' -rw-r--r-- craig 0 '+os(a,'dired-ignored','.gitignore'));
L.push(' '+os(a,'dired-set-id','-rwsr-xr-x')+' root 900 setuid.bin');
L.push(' '+os(a,'dired-special','prw-r--r--')+' craig 0 fifo.pipe');
L.push(os(a,'dired-warning','! disk space low on /home'));
return previewLines(L);}
// A believable two-pane dirvish: an active directory listing on the left
// (nerd-icon per file type, dir-entry counts / file sizes, the hl-line on the
// selected row, a dimmed backup) and the selected dir's ls-l preview on the
// right. Faces that don't fit a calm listing (vc, git, subtree, media, proc,
// narrow, emerge) live in a labeled extras strip below so theme coverage stays
// complete. Glyphs/colors mirror what nerd-icons actually emits per type.
function renderDirvishPreview(){
const D='dirvish', N='nerd-icons', DR='dired';
// foreground-only span, so a row background (the hl-line) shows through it
const fg=(app,face,t)=>`${t}`;
const ic=(face,g)=>os(N,face,g);
const pad=(name,w)=>esc(name)+' '.repeat(Math.max(1,w-name.length));
const HL=pkgEffBg(D,'dirvish-hl-line')||MAP['bg'];
// ---- left pane: the active directory ----
const left=[os(DR,'dired-header','~/code/emacs-wttrin:')];
for(const [name,cnt,sel] of [['assets','4',true],['examples','1'],['githooks','1'],
['inbox','3'],['scripts','1'],['tests','71']]){
if(sel) left.push(``+fg(N,'nerd-icons-yellow','')+' '
+fg(DR,'dired-directory',pad(name,21))+fg(D,'dirvish-file-size',cnt)+``);
else left.push(ic('nerd-icons-yellow','')+' '+os(DR,'dired-directory',pad(name,21))
+os(D,'dirvish-file-size',cnt));
}
for(const [face,g,name,size] of [['nerd-icons-lblue','','CLAUDE.md','4.9k'],
['nerd-icons-blue','','Eask','518'],['nerd-icons-blue','','LICENSE','34k'],
['nerd-icons-dorange','','Makefile','12k'],['nerd-icons-lcyan','','README.org','24k'],
['nerd-icons-lgreen','','todo.org','23k'],['nerd-icons-purple','','wttrin.el','69k'],
['nerd-icons-dsilver','','wttrin.elc','4.3k']])
left.push(ic(face,g)+' '+pad(name,21)+os(D,'dirvish-file-size',size));
left.push(ic('nerd-icons-lgreen','')+' '+os(DR,'dired-ignored',pad('todo.org~',21))
+os(D,'dirvish-file-size','8.8k'));
// ---- right pane: ls -l preview of the selected dir ----
const ll=(size,time,name)=>os(D,'dirvish-file-modes','-rw-r--r--')+' '
+os(D,'dirvish-file-link-number','1')+' '+os(D,'dirvish-file-user-id','cjennings')+' '
+os(D,'dirvish-file-group-id','cjennings')+' '+os(D,'dirvish-file-size',size)+' '
+os(D,'dirvish-file-time',time)+' '+ic('nerd-icons-orange','\u{F0E2D}')+' '+esc(name);
const right=[os(DR,'dired-header','assets:'),
ll('54K','Jun 26 10:53','geolocation.png'),ll('52K','Jun 26 10:53','location-menu.png'),
ll('3.1K','Apr 10 12:03','made-for-emacs.svg'),ll('346K','Jun 26 10:53','wttrin.png')];
// ---- extras: remaining dirvish faces, kept for theme coverage ----
const ex=[
os(D,'dirvish-inactive','inactive pane')+' '+os(D,'dirvish-hl-line-inactive',' inactive current line ')+' '+os(D,'dirvish-free-space','[free 24G]'),
'vc '+os(D,'dirvish-vc-added-state','A')+os(D,'dirvish-vc-edited-state','M')+os(D,'dirvish-vc-removed-state','D')+os(D,'dirvish-vc-conflict-state','C')+os(D,'dirvish-vc-locked-state','L')+os(D,'dirvish-vc-missing-state','!')+os(D,'dirvish-vc-needs-merge-face','m')+os(D,'dirvish-vc-needs-update-state','u')+os(D,'dirvish-vc-unregistered-face','?')+' git '+os(D,'dirvish-git-commit-message-face','feat: enlarge the picker'),
'subtree '+os(D,'dirvish-collapse-dir-face','src')+os(D,'dirvish-subtree-state','+')+os(D,'dirvish-subtree-guide',' | ')+os(D,'dirvish-collapse-empty-dir-face','empty/')+' '+os(D,'dirvish-collapse-file-face','file.txt')+' inode '+os(D,'dirvish-file-inode-number','1048576')+' dev '+os(D,'dirvish-file-device-number','8,1'),
'media '+os(D,'dirvish-media-info-heading','Media')+' '+os(D,'dirvish-media-info-property-key','Dimensions:')+' 1920x1080 proc '+os(D,'dirvish-proc-running','running')+'/'+os(D,'dirvish-proc-finished','finished')+'/'+os(D,'dirvish-proc-failed','failed'),
'narrow '+os(D,'dirvish-narrow-match-face-0','m0')+' '+os(D,'dirvish-narrow-match-face-1','m1')+' '+os(D,'dirvish-narrow-match-face-2','m2')+' '+os(D,'dirvish-narrow-match-face-3','m3')+os(D,'dirvish-narrow-split',' | ')+os(D,'dirvish-emerge-group-title','Group: images')];
const col=(lines)=>`${lines.join('\n')}
`;
return ``
+`
${col(left)}${col(right)}
`
+`
${col(ex)}
`;}
function renderCalibredbPreview(){const a='calibredb',L=[];
L.push(os(a,'calibredb-search-header-library-name-face','Calibre')+' '+os(a,'calibredb-search-header-library-path-face','~/books')+' '+os(a,'calibredb-search-header-total-face','412 books')+' '+os(a,'calibredb-search-header-filter-face','tag:scifi')+' '+os(a,'calibredb-search-header-sort-face','sort:date')+' '+os(a,'calibredb-search-header-highlight-face','[*]'));
L.push('');
L.push(os(a,'calibredb-id-face','1')+' '+os(a,'calibredb-title-face','Dune')+' '+os(a,'calibredb-author-face','Herbert')+' '+os(a,'calibredb-format-face','EPUB')+' '+os(a,'calibredb-size-face','2.1M')+' '+os(a,'calibredb-tag-face',':scifi:')+' '+os(a,'calibredb-date-face','2026-06-08'));
L.push(os(a,'calibredb-mark-face','*')+os(a,'calibredb-id-face','2')+' '+os(a,'calibredb-title-face','Foundation')+' '+os(a,'calibredb-author-face','Asimov')+' '+os(a,'calibredb-series-face','[Foundation #1]')+' '+os(a,'calibredb-publisher-face','Bantam')+' '+os(a,'calibredb-pubdate-face','1951'));
L.push('');
L.push(os(a,'calibredb-title-detailed-view-face','Foundation (detailed)')+' '+os(a,'calibredb-language-face','eng')+' '+os(a,'calibredb-favorite-face','* fav')+' '+os(a,'calibredb-archive-face','archived'));
L.push(os(a,'calibredb-ids-face','isbn:0553293354')+' '+os(a,'calibredb-file-face','foundation.epub')+' '+os(a,'calibredb-comment-face','A classic of the genre.'));
L.push(os(a,'calibredb-edit-annotation-header-title-face','Annotations')+' '+os(a,'calibredb-highlight-face','highlighted passage')+' '+os(a,'calibredb-current-page-button-face','[page 42]')+' '+os(a,'calibredb-mouse-face','hover row'));
return previewLines(L);}
function renderErcPreview(){const a='erc',L=[];
L.push(os(a,'erc-header-line',' #emacs on Libera.Chat 18 users '));
L.push(os(a,'erc-timestamp-face','[10:24]')+' '+os(a,'erc-notice-face','*** alice has joined #emacs'));
L.push(os(a,'erc-timestamp-face','[10:25]')+' <'+os(a,'erc-my-nick-prefix-face','@')+os(a,'erc-my-nick-face','craig')+'> '+os(a,'erc-input-face','hello everyone'));
L.push(os(a,'erc-timestamp-face','[10:25]')+' <'+os(a,'erc-nick-prefix-face','+')+os(a,'erc-nick-default-face','bob')+'> '+os(a,'erc-default-face','hi craig, see ')+os(a,'erc-button','this link')+os(a,'erc-default-face',' cc ')+os(a,'erc-button-nick-default-face','@alice'));
L.push(os(a,'erc-timestamp-face','[10:26]')+' '+os(a,'erc-action-face','* craig waves')+' '+os(a,'erc-keyword-face','emacs')+' '+os(a,'erc-pal-face','<friend>')+' '+os(a,'erc-fool-face','<troll>')+' '+os(a,'erc-dangerous-host-face','<bad@host>'));
L.push(os(a,'erc-timestamp-face','[10:27]')+' '+os(a,'erc-direct-msg-face','(DM)')+' <'+os(a,'erc-nick-msg-face','bob')+'> psst '+os(a,'erc-current-nick-face','craig')+' '+os(a,'erc-information','-info-'));
L.push(os(a,'erc-error-face','*** ERROR: connection reset'));
L.push(os(a,'erc-command-indicator-face','/help')+' '+os(a,'erc-bold-face','bold')+' '+os(a,'erc-italic-face','italic')+' '+os(a,'erc-underline-face','underline')+' '+os(a,'erc-inverse-face','inverse')+' '+os(a,'erc-spoiler-face','spoiler'));
L.push(os(a,'erc-keep-place-indicator-arrow','>')+os(a,'erc-keep-place-indicator-line',' ---- last read ---- ')+os(a,'erc-fill-wrap-merge-indicator-face','+'));
L.push(os(a,'erc-prompt-face','craig>')+' '+os(a,'erc-input-face','type a message...'));
return previewLines(L);}
function renderOrgdrillPreview(){const a='org-drill',L=[];
L.push('Q: The capital of France is '+os(a,'org-drill-hidden-cloze-face','[...]')+'.');
L.push('A: The capital of France is '+os(a,'org-drill-visible-cloze-face','Paris')+'.');
L.push(' '+os(a,'org-drill-visible-cloze-hint-face','hint: P____'));
return previewLines(L);}
function renderOrgnoterPreview(){const a='org-noter',L=[];
L.push('org-noter paper.pdf');
L.push(' page 1 '+os(a,'org-noter-notes-exist-face','[notes]'));
L.push(' page 2 '+os(a,'org-noter-no-notes-exist-face','[no notes]'));
return previewLines(L);}
function renderSignelPreview(){const a='signel',L=[];
L.push(os(a,'signel-timestamp-face','[10:24]')+' '+os(a,'signel-my-msg-face','Me: hey, are we still on for tonight?'));
L.push(os(a,'signel-timestamp-face','[10:25]')+' '+os(a,'signel-other-msg-face','Christine: yes! see you at 7'));
L.push(os(a,'signel-error-face','(failed to send -- retrying)'));
return previewLines(L);}
function renderPearlPreview(){const a='pearl',L=[];
L.push(os(a,'pearl-preamble-summary','PEARL-42 Fix the broken picker'));
L.push('State: '+os(a,'pearl-modified-local','In Progress')+' Priority: '+os(a,'pearl-modified-highlight','High')+' Estimate: '+os(a,'pearl-modified-unknown','?'));
L.push(' '+os(a,'pearl-editable-comment','> add a comment (editable)'));
L.push(' '+os(a,'pearl-readonly-comment','> created by automation (read-only)'));
return previewLines(L);}
function renderShrPreview(){const a='shr',L=[];
L.push(os(a,'shr-text','shr renders nov (EPUB), eww (web), elfeed, and HTML mail.'));
L.push('');
L.push(os(a,'shr-h1','Chapter One: The Beginning'));
L.push(os(a,'shr-h2','A Section Heading'));
L.push(os(a,'shr-h3','A subsection')+' '+os(a,'shr-h4','h4')+' / '+os(a,'shr-h5','h5')+' / '+os(a,'shr-h6','h6'));
L.push(os(a,'shr-text','Body text flows in shr-text, with a ')+os(a,'shr-link','hyperlink')+os(a,'shr-text',' and a ')+os(a,'shr-selected-link','focused link')+os(a,'shr-text',','));
L.push(os(a,'shr-text','some ')+os(a,'shr-code','inline_code()')+os(a,'shr-text',', a ')+os(a,'shr-mark','highlighted mark')+os(a,'shr-text',', ')+os(a,'shr-strike-through','struck out')+os(a,'shr-text',', a footnote')+os(a,'shr-sup','[1]')+os(a,'shr-text',','));
L.push(os(a,'shr-text','an ')+os(a,'shr-abbreviation','HTML')+os(a,'shr-text',' abbreviation, and an ')+os(a,'shr-sliced-image','[image]')+os(a,'shr-text',' slice.'));
return previewLines(L);}
// nov-reading: a realistic book page per palette, not the line-based format.
// Each palette face supplies the page bg+fg (via ofs); the serif typography and
// hierarchy are CSS so the preview reads like an actual novel page. Tuning a
// palette face repaints its page. data-owner-app/-face keep face-locate working.
function novReadingPage(a,face,label){
const cls=isLocateOnPane(a,curApp())?' class="locate-onpane"':'';
const title=attresc(formatLocateTitle(locateFaceMeta(a,face,LOCATE_REG)));
const page=ofs(a,face)+";width:34em;max-width:100%;border-radius:6px;box-shadow:0 1px 8px #0007;padding:24px 30px 18px;font:13pt/1.62 Georgia,'Times New Roman',serif";
// Structural faces recolor the title (heading) and an inline link, derived by
// suffix from the palette face so tuning them in the studio repaints the page.
const hface=face+'-heading',lface=face+'-link';
const htitle=attresc(formatLocateTitle(locateFaceMeta(a,hface,LOCATE_REG)));
const hfg=effFg(pkgEffFg(a,hface));
return ``
+`
Nathaniel Hawthorne · Twice-Told Tales
`
+`
Dr. Heidegger’s Experiment
`
+`
`
+`T`
+`hat very singular man, old Dr. Heidegger, once invited four venerable friends to meet him in his study. There were three white-bearded gentlemen, Mr. Medbourne, Colonel Killigrew, and Mr. Gascoigne, and a withered gentlewoman whose name was the Widow Wycherly.
`
+`
They were all melancholy old creatures, who had been unfortunate in life. If the powder be genuine, said the doctor, `+os(a,lface,'the rose of half a century')+` should bloom again.
`
+`
${esc(label)} · 12
`
+`
`;}
function renderNovReadingPreview(){
const a='nov-reading',faces=(APPS[a]&&APPS[a].faces)||[];
if(!faces.length)return genericPreview(a);
// One book page per base palette (the bg/fg faces); the per-palette heading
// and link faces color the title and inline link within each page rather than
// getting a page of their own.
const base=faces.filter(r=>!/-heading$|-link$/.test(r[0]));
let h='';
for(const row of base)h+=novReadingPage(a,row[0],row[1]);
return h+'
';}
function renderCompanyPreview(){const a='company',L=[],o=(f,t)=>os(a,'company-'+f,t);
// A code buffer mid-completion: the inline ghost preview at point, the
// tooltip popup with every row variant, and the echo-area fallback.
const pad=(s,n)=>s+' '.repeat(Math.max(0,n-s.length));
L.push('(defun cj/play-track (file)');
L.push(' (cj/music-'+o('preview-common','pl')+o('preview','aylist-load')+' ; inline preview at point');
L.push('');
const row=(qa,common,rest,ann,rowFace,commonFace,annFace,qaFace,bar)=>
o(qaFace,' '+qa+' ')+o(commonFace,common)+o(rowFace,pad(rest,26-common.length))+o(annFace,pad(ann,10))+bar;
L.push(' '+row('1','cj/music-pl','aylist-load','Command','tooltip-selection','tooltip-common-selection','tooltip-annotation-selection','tooltip-quick-access-selection',o('tooltip-scrollbar-thumb',' ')));
L.push(' '+row('2','cj/music-pl','aylist-save','Command','tooltip','tooltip-common','tooltip-annotation','tooltip-quick-access',o('tooltip-scrollbar-thumb',' ')));
L.push(' '+row('3','cj/music-pl','aylist-reload','Command','tooltip','tooltip-common','tooltip-annotation','tooltip-quick-access',o('tooltip-scrollbar-track',' ')));
L.push(' '+o('tooltip-quick-access',' 4 ')+o('tooltip-mouse',pad('cj/music-playing-p (mouse is here)',36))+o('tooltip-annotation','Function ')+o('tooltip-scrollbar-track',' '));
L.push(' '+o('tooltip-quick-access',' 5 ')+o('tooltip-deprecated','cj/music-play-file-old')+o('tooltip',pad('',14))+o('tooltip-annotation','Command ')+o('tooltip-scrollbar-track',' '));
L.push('');
L.push(' During C-s "'+o('tooltip-search','loa')+'" in the tooltip:');
L.push(' '+o('tooltip-quick-access-selection',' 1 ')+o('tooltip-search-selection',pad('cj/music-playlist-load',36))+o('tooltip-annotation-selection','Command ')+o('tooltip-scrollbar-thumb',' '));
L.push(' '+o('tooltip-quick-access',' 2 ')+o('tooltip','cj/music-down')+o('tooltip-search','loa')+o('tooltip',pad('der-fn',20))+o('tooltip-annotation','Function ')+o('tooltip-scrollbar-track',' '));
L.push(' and the ghost text becomes '+o('preview-search','cj/music-playlist-load'));
L.push('');
L.push(' M-x display: '+o('echo-common','cj/music-pl')+o('echo','aylist-load')+' '+o('echo','cj/music-playlist-save')+' (company-echo frontend)');
return previewLines(L);}
function renderCompanyBoxPreview(){const a='company-box',L=[],o=(f,t)=>os(a,'company-box-'+f,t);
// company-box is the icons frontend: same popup, plus an icon column and
// its own background/scrollbar/numbers faces.
const pad=(s,n)=>s+' '.repeat(Math.max(0,n-s.length));
L.push('(setq cj/theme-ac|');
L.push('');
L.push(' '+o('background',' ƒ ')+o('selection',pad(' cj/theme-accent-color',30))+o('numbers',' 1 ')+o('scrollbar','▐'));
L.push(' '+o('background',' ƒ ')+o('candidate',pad(' cj/theme-accent-face',30))+o('numbers',' 2 ')+o('scrollbar','▐'));
L.push(' '+o('background',' □ ')+o('candidate',pad(' cj/theme-accents',24))+o('annotation','(list)')+o('numbers',' 3 ')+o('background','▕'));
L.push(' '+o('background',' ⚙ ')+o('candidate',pad(' cj/theme-active-p',24))+o('annotation','(fn) ')+o('numbers',' 4 ')+o('background','▕'));
return previewLines(L);}
function renderTransientPreview(){const a='transient',L=[],o=(f,t)=>os(a,'transient-'+f,t);
// A magit-commit-style transient: argument infixes with every key class,
// action columns, and the exotic key faces in plausible bindings.
L.push(o('heading','Arguments'));
L.push(' '+o('key','-a')+' Stage all modified and deleted files ('+o('argument','--all')+')');
L.push(' '+o('key','-e')+' Allow empty commit ('+o('inactive-argument','--allow-empty')+')');
L.push(' '+o('key','-v')+' Show diff of changes ('+o('argument','--verbose')+')');
L.push(' '+o('key','=A')+' Override the author '+o('active-infix','--author=')+o('value','Craig Jennings'));
L.push(' '+o('key','-S')+' Sign using gpg '+o('argument','--gpg-sign=')+o('inactive-value','unset'));
L.push('');
L.push(o('heading','Create')+' '+o('heading','Edit HEAD')+' '+o('heading','Edit'));
L.push(' '+o('key','c')+' '+o('enabled-suffix','Commit')+' '+o('key','e')+' Extend '+o('key','f')+' Fixup');
L.push(' '+o('key-exit','q')+' Quit '+o('key','w')+' Reword '+o('key','F')+' Instant fixup');
L.push(' '+o('key-return','C-g')+' Back '+o('key','a')+' Amend '+o('disabled-suffix','s Squash (disabled)'));
L.push('');
L.push(o('heading','Key classes')+' '+o('delimiter','|')+' = transient-delimiter');
L.push(' '+o('key-stay','-t')+' toggles stay open '+o('key-recurse','o')+' opens a sub-transient');
L.push(' '+o('key-stack','C-z')+' suspends to stack '+o('key-noop','x')+' bound to nothing here');
L.push(' '+o('nonstandard-key','')+' nonstandard binding '+o('mismatched-key','d!')+' mismatched with suffix');
L.push(' '+o('unreachable-key','M-x')+' '+o('unreachable','unreachable at this level')+' '+o('higher-level','shown at a higher level'));
L.push(' '+o('inapt-suffix','r Rebase (inapt while merging)')+' '+o('inapt-argument','--interactive (inapt)'));
return previewLines(L);}
function renderMagitSectionPreview(){const a='magit-section',L=[],o=(f,t)=>os(a,'magit-section-'+f,t);
const pad=(s,n)=>s+' '.repeat(Math.max(0,n-s.length));
// The section library under magit/forge/etc: headings with child counts,
// the highlighted current section, and a region-selected heading.
L.push(os(a,'magit-left-margin','▶ ')+o('heading','Unstaged changes')+' '+o('child-count','(2)'));
L.push(o('highlight',pad(' modified modules/custom-buffer-file.el',52)));
L.push(o('highlight',pad(' modified tests/test-custom-buffer-file.el',52)));
L.push('');
L.push(os(a,'magit-left-margin','▷ ')+o('heading-selection','Staged changes')+' '+o('child-count','(1)')+' (heading inside an active region)');
L.push(' new file scripts/theme-studio/screenshot-previews.sh');
L.push('');
L.push(o('secondary-heading','Recent commits'));
L.push(' 8a93f68c pin retired packages');
L.push(' 67a609dd screenshot harness');
return previewLines(L);}
function renderRainbowDelimitersPreview(){const a='rainbow-delimiters',L=[];
const DEPTH=[null,'rainbow-delimiters-depth-1-face','rainbow-delimiters-depth-2-face',
'rainbow-delimiters-depth-3-face','rainbow-delimiters-depth-4-face',
'rainbow-delimiters-depth-5-face','rainbow-delimiters-depth-6-face',
'rainbow-delimiters-depth-7-face','rainbow-delimiters-depth-8-face',
'rainbow-delimiters-depth-9-face'];
const d=(n,t)=>os(a,DEPTH[n],t);
// Nested elisp colored by depth 1-9, plus the two error faces and the
// base faces the depth faces inherit from.
L.push(d(1,'(')+'defun demo '+d(2,'(')+'xs'+d(2,')')+' ; parens colored by nesting depth');
L.push(' '+d(2,'(')+'let '+d(3,'(')+d(4,'(')+'acc '+d(5,'(')+'mapcar '+d(6,'(')+'lambda '+d(7,'(')+'x'+d(7,')'));
L.push(' '+d(7,'(')+'round '+d(8,'(')+'+ 1 '+d(9,'(')+'* x '+d(1,'(')+'abs x'+d(1,')')+d(9,')')+d(8,')')+d(7,')')+d(6,')')+d(5,')')+' ; depth 10 cycles back to depth-1');
L.push(' xs'+d(4,')')+d(3,')'));
L.push(' acc'+d(2,')'));
L.push(d(1,')'));
L.push('');
L.push('(car xs'+os(a,'rainbow-delimiters-unmatched-face',')')+') ; the extra closer is unmatched');
L.push('(nth 0 xs'+os(a,'rainbow-delimiters-mismatched-face',']')+' ; a ] closing a ( is mismatched');
L.push('');
L.push(os(a,'rainbow-delimiters-base-face','base-face')+' underlies every depth; '
+os(a,'rainbow-delimiters-base-error-face','base-error-face')+' underlies both error faces');
return previewLines(L);}
function renderWebModePreview(){const a='web-mode',L=[],o=(f,t)=>os(a,'web-mode-'+f,t);
// One mixed HTML document exercising all 81 faces: markup, an inline CSS
// part, a JS part with a JSON island / JSX / template literals / SQL-in-a-
// string, a generic {{...}}/{%...%} template block (engine-agnostic on
// purpose), and a JSDoc-style annotation comment.
const B=(t)=>o('html-tag-bracket-face',t);
L.push(o('doctype-face',''));
L.push(o('comment-face',''));
L.push(B('<')+o('html-tag-face','html')+' '+o('html-attr-name-face','lang')+o('html-attr-equal-face','=')+o('html-attr-value-face','"en"')+B('>'));
L.push('');
L.push(B('<')+o('html-tag-face','style')+B('>')+o('style-face',' /* the css part rides web-mode-style-face */'));
L.push(' '+o('css-comment-face','/* layout */'));
L.push(' '+o('css-at-rule-face','@media')+' (min-width: 40rem) {');
L.push(' '+o('css-selector-tag-face','body')+' '+o('css-selector-face','#page')+' '+o('css-selector-class-face','.card')+o('css-pseudo-class-face',':hover')+' {');
L.push(' '+o('css-property-name-face','color')+': '+o('css-color-face','#67809c')+' '+o('css-priority-face','!important')+';');
L.push(' '+o('css-property-name-face','width')+': '+o('css-function-face','calc')+'(100% - '+o('css-variable-face','--gutter')+');');
L.push(' '+o('css-property-name-face','font-family')+': '+o('css-string-face','"Berkeley Mono"')+'; } }');
L.push(B('')+o('html-tag-face','style')+B('>'));
L.push('');
L.push(B('<')+o('html-tag-face','body')+' '+o('html-attr-custom-face','data-theme')+o('html-attr-equal-face','=')+o('html-attr-value-face','"dupre"')+B('>'));
L.push(' '+o('current-element-highlight-face',B('<')+o('html-tag-face','main')+B('>'))+' '+o('comment-face',''));
L.push(' '+B('<')+o('html-tag-custom-face','cart-widget')+' '+o('html-attr-engine-face',':items')+o('html-attr-equal-face','=')+o('html-attr-value-face','"cart.lines"')+B('/>'));
L.push(' '+B('<')+o('html-tag-namespaced-face','svg:rect')+' '+o('html-attr-name-face','width')+o('html-attr-equal-face','=')+o('html-attr-value-face','"12"')+B('/>'));
L.push(' '+B('<')+o('html-tag-face','p')+B('>')+'Fish '+o('html-entity-face','&')+' chips '+B('<')+o('html-tag-face','b')+B('>')+o('bold-face','bold')+B('')+o('html-tag-face','b')+B('>')+' '+B('<')+o('html-tag-face','i')+B('>')+o('italic-face','italic')+B('')+o('html-tag-face','i')+B('>')+' '+B('<')+o('html-tag-face','u')+B('>')+o('underline-face','underlined')+B('')+o('html-tag-face','u')+B('>')+B('')+o('html-tag-face','p')+B('>'));
L.push(' '+B('<')+o('html-tag-unclosed-face','li')+B('>')+'this li never closes '+o('error-face','')+' '+o('warning-face','duplicate id')+' '+o('whitespace-face','···')+' '+o('folded-face','[details folded…]')+' '+o('inlay-face',' 3 items ')+' '+o('current-column-highlight-face','│'));
L.push('');
L.push(' '+o('block-delimiter-face','{%')+' '+o('block-control-face','if')+' '+o('variable-name-face','user')+o('block-face','.signed_in')+' '+o('block-delimiter-face','%}')+' '+o('block-comment-face','{# template block, any engine #}'));
L.push(' '+o('block-delimiter-face','{{')+' '+o('variable-name-face','user')+o('block-face','.name')+' '+o('filter-face','| capitalize')+' '+o('block-delimiter-face','}}'));
L.push(' '+o('block-delimiter-face','{%')+' '+o('block-control-face','include')+' '+o('block-string-face','"badge.html"')+' '+o('block-attr-name-face','with')+' '+o('block-attr-value-face','tier=gold')+' '+o('block-delimiter-face','%}'));
L.push(' '+o('block-delimiter-face','{%')+' '+o('block-control-face','endif')+' '+o('block-delimiter-face','%}'));
L.push('');
L.push(B('<')+o('html-tag-face','script')+B('>')+o('script-face',' // the js part rides web-mode-script-face')+o('part-face',' (embedded parts share web-mode-part-face)'));
L.push(' '+o('comment-face','/**'));
L.push(' '+o('annotation-face',' * Render one cart row.'));
L.push(' '+o('annotation-face',' * ')+o('annotation-tag-face','@param')+' '+o('annotation-type-face','{Line}')+' '+o('annotation-value-face','line')+' '+o('annotation-face','the row, e.g. ')+o('annotation-html-face','line.sku'));
L.push(' '+o('annotation-face',' */'));
L.push(' '+o('keyword-face','const')+' '+o('constant-face','VAT')+' = '+o('variable-name-face','rate')+' ?? 0.21; '+o('javascript-comment-face','// js comment')+' '+o('part-comment-face','/* part comment */'));
L.push(' '+o('keyword-face','function')+' '+o('function-name-face','renderLine')+'('+o('param-name-face','line')+', '+o('param-name-face','fmt')+') {');
L.push(' '+o('keyword-face','const')+' '+o('type-face','Money')+' '+o('variable-name-face','total')+' = '+o('function-call-face','round')+'('+o('variable-name-face','line')+'.'+o('symbol-face','qty')+' * '+o('builtin-face','Number')+'('+o('variable-name-face','line')+'.price));');
L.push(' '+o('keyword-face','return')+' '+o('javascript-string-face','`')+o('interpolate-color1-face','${')+o('interpolate-color2-face','fmt(')+o('interpolate-color3-face','${')+o('interpolate-color4-face','total')+o('interpolate-color3-face','}')+o('interpolate-color2-face',')')+o('interpolate-color1-face','}')+o('javascript-string-face',' EUR`')+'; }');
L.push(' '+o('keyword-face','const')+' '+o('variable-name-face','q')+' = '+o('part-string-face','"')+o('sql-keyword-face','SELECT')+o('part-string-face',' sku ')+o('sql-keyword-face','FROM')+o('part-string-face',' lines ')+o('sql-keyword-face','WHERE')+o('part-string-face',' qty > 0"')+';');
L.push(' '+o('keyword-face','const')+' '+o('variable-name-face','cfg')+' = '+o('json-context-face','{ ')+o('json-key-face','"currency"')+': '+o('json-string-face','"EUR"')+o('json-context-face',', ')+o('json-comment-face','/* json island */')+o('json-context-face',' }')+';');
L.push(' '+o('keyword-face','return')+' ('+o('jsx-depth-1-face','')+o('jsx-depth-2-face','')+o('jsx-depth-3-face','| ')+o('jsx-depth-4-face','')+o('jsx-depth-5-face','{n}')+o('jsx-depth-4-face','')+o('jsx-depth-3-face',' | ')+o('jsx-depth-2-face','
')+o('jsx-depth-1-face','')+');');
L.push(' '+o('preprocessor-face',''));
L.push(B('')+o('html-tag-face','script')+B('>'));
L.push(B('')+o('html-tag-face','html')+B('>'));
return previewLines(L);}
// -- The minibuffer stack: vertico / marginalia / consult / embark / orderless --
// ONE shared scene (a single C-x b consult-buffer session) that all five
// packages draw together; each renderer shows that same session with its own
// faces at work, then extends it with the states only it owns. Cross-package
// spans are the point: hovering any run shows which package paints it.
function mbStackScene(){
const v=(f,t)=>os('vertico',f,t),m=(f,t)=>os('marginalia',f,t),c=(f,t)=>os('consult',f,t);
const OM=['orderless-match-face-0','orderless-match-face-1',
'orderless-match-face-2','orderless-match-face-3'];
const om=(n,t)=>os('orderless',OM[n],t);
const L=[];
L.push('One C-x b session — the whole stack draws it:');
L.push('');
L.push('Switch to buffer ('+c('consult-narrow-indicator','[b]')+'): the stu▌');
L.push(v('vertico-group-title','Buffer')+' '+v('vertico-group-separator','────────────────────────────────────────'));
L.push(v('vertico-current',' '+c('consult-buffer',om(0,'the')+'me-'+om(1,'stu')+'dio/app.js')+' '+m('marginalia-mode','js-mode')+' '+m('marginalia-size','48k')));
L.push(' '+c('consult-buffer',om(0,'the')+'me-'+om(1,'stu')+'dio/previews.js')+' '+m('marginalia-mode','js-mode')+' '+m('marginalia-size','36k'));
L.push(v('vertico-group-title','File')+' '+v('vertico-group-separator','──────────────────────────────────────────'));
L.push(' '+c('consult-file',om(0,'the')+'me-notes/'+om(1,'stu')+'dio-plan.org'));
L.push(v('vertico-group-title','Bookmark')+' '+v('vertico-group-separator','──────────────────────────────────────'));
L.push(' '+c('consult-bookmark',om(0,'the')+'me-'+om(1,'stu')+'dio gallery'));
return L;}
function renderVerticoPreview(){const a='vertico',L=mbStackScene();
L.push('');
L.push('vertico owns the frame: '+os(a,'vertico-current','the highlighted candidate row')+', the');
L.push(os(a,'vertico-group-title','group titles')+' and '+os(a,'vertico-group-separator','──── separators')+' above.');
L.push('');
L.push('M-y (yank-pop) — a two-line kill folds onto one row,');
L.push('its newline drawn via vertico-multiline:');
L.push(' (setq cj/theme \'dupre)'+os(a,'vertico-multiline','\\n')+'(load-theme cj/theme t)');
return previewLines(L);}
function renderMarginaliaPreview(){const a='marginalia',L=mbStackScene(),m=(f,t)=>os(a,f,t);
L.push('');
L.push('marginalia annotates every command in the stack:');
L.push('');
L.push('C-x b app.js '+m('marginalia-file-name','~/…/theme-studio/app.js'));
L.push('M-x consult-line '+m('marginalia-key','M-s l')+' '+m('marginalia-documentation','Search for a matching line.'));
L.push('C-h f cj/switch-theme '+m('marginalia-function','(&optional THEME)')+' '+m('marginalia-documentation','Load THEME.'));
L.push('C-h v fill-column '+m('marginalia-type','integer')+' '+m('marginalia-number','72'));
L.push(' user-full-name '+m('marginalia-type','string')+' '+m('marginalia-string','"Craig Jennings"'));
L.push(' debug-on-error '+m('marginalia-type','boolean')+' '+m('marginalia-true','t')+' / '+m('marginalia-null','nil'));
L.push(' major-mode '+m('marginalia-type','symbol')+' '+m('marginalia-symbol','js-mode'));
L.push(' load-path '+m('marginalia-type','list')+' '+m('marginalia-list','("~/.emacs.d/modules" ...)'));
L.push(' enabled-themes '+m('marginalia-type','value')+' '+m('marginalia-value','(dupre)'));
L.push('C-x C-f app.js '+m('marginalia-file-priv-no','-')+m('marginalia-file-priv-read','r')+m('marginalia-file-priv-write','w')+m('marginalia-file-priv-no','-')+m('marginalia-file-priv-read','r')+m('marginalia-file-priv-no','--r--')+' '+m('marginalia-file-owner','craig:users')+' '+m('marginalia-size','48k')+' '+m('marginalia-date','2026-07-02')+' '+m('marginalia-modified','*'));
L.push(' theme-studio/ '+m('marginalia-file-priv-dir','d')+m('marginalia-file-priv-read','r')+m('marginalia-file-priv-write','w')+m('marginalia-file-priv-exec','x')+m('marginalia-file-priv-read','r')+m('marginalia-file-priv-rare','s')+m('marginalia-file-priv-no','----')+' '+m('marginalia-file-owner','craig:users')+' '+m('marginalia-size','4.0k')+' '+m('marginalia-date','2026-06-28'));
L.push(' current.js '+m('marginalia-file-priv-other','→')+' app.js '+m('marginalia-file-priv-link','l')+m('marginalia-file-priv-read','r')+m('marginalia-file-priv-write','w')+m('marginalia-file-priv-exec','x')+m('marginalia-file-priv-no','------')+' '+m('marginalia-date','2026-07-01'));
L.push('C-h P vertico '+m('marginalia-version','1.9')+' '+m('marginalia-archive','gnu')+' '+m('marginalia-installed','installed'));
L.push('minor which-key-mode '+m('marginalia-on','on')+' lighter '+m('marginalia-lighter','" WK"'));
L.push('modes flymake-mode '+m('marginalia-off','off'));
L.push('C-x 8 RET BULLET • '+m('marginalia-char','#x2022'));
return previewLines(L);}
function renderConsultPreview(){const a='consult',L=mbStackScene(),c=(f,t)=>os(a,f,t);
L.push('');
L.push('consult owns the '+c('consult-buffer','buffer')+' / '+c('consult-file','file')+' / '+c('consult-bookmark','bookmark')+' candidates');
L.push('above, and the '+c('consult-narrow-indicator','[b]')+' narrow indicator.');
L.push('Narrow help (?): '+c('consult-key','b')+' '+c('consult-help','Buffer')+' '+c('consult-key','f')+' '+c('consult-help','File')+' '+c('consult-key','m')+' '+c('consult-help','Bookmark'));
L.push('');
L.push('M-s l (consult-line) — the same window, searched:');
L.push(' '+c('consult-line-number-prefix',' 12 ')+'(load-theme \''+c('consult-highlight-match','dupre')+' t)');
L.push(' '+c('consult-line-number-prefix',' 118 ')+'(setq '+c('consult-highlight-match','dupre')+'-accent "#67809c")');
L.push(' '+c('consult-line-number-wrapped',' 340 ')+'(provide \'dupre-theme) · wrapped past point');
L.push(' jump lands at line '+c('consult-line-number','12')+'; the preview gives the target');
L.push(' '+c('consult-preview-line','a preview-line band')+' with the match '+c('consult-preview-match','dupre')+', mark '+c('consult-highlight-mark','▮')+';');
L.push(' consult-yank previews the insertion '+c('consult-preview-insertion','(load-theme ...)'));
L.push('');
L.push('M-s r (consult-ripgrep) — async input splits at '+c('consult-async-split','#')+'dupre'+c('consult-async-split','#'));
L.push(' '+c('consult-async-running','●')+' running '+c('consult-async-finished','✓')+' finished '+c('consult-async-failed','✗')+' failed: rg exited 2');
L.push(' '+c('consult-grep-context','app.js:10- // registry of bespoke renderers'));
L.push(' app.js:12: registered the dupre palette');
L.push(' '+c('consult-grep-context','app.js:14- // gallery wiring below'));
L.push(' '+c('consult-separator','│')+' separator between async input and candidates');
return previewLines(L);}
function renderEmbarkPreview(){const a='embark',L=mbStackScene(),e=(f,t)=>os(a,f,t);
L.push('');
L.push('C-. on the current candidate — embark-act:');
L.push(' '+e('embark-verbose-indicator-title','Act on buffer theme-studio/app.js'));
L.push(' target: '+e('embark-target','theme-studio/app.js'));
L.push(' '+e('embark-keybinding','k')+' kill-buffer '+e('embark-verbose-indicator-documentation','Kill it, offering to save.'));
L.push(' '+e('embark-keybinding','r')+' rename-buffer '+e('embark-verbose-indicator-documentation','Change the buffer name.'));
L.push(' '+e('embark-keybinding-repeat','n')+' next-buffer · repeat key: act stays live');
L.push(' '+e('embark-keybinding','C-h')+' '+e('embark-keymap','embark-buffer-map')+' · a bound sub-keymap');
L.push(' '+e('embark-verbose-indicator-shadowed','s save-buffer — shadowed by a later binding'));
L.push('');
L.push('S-SPC marks, E exports — embark-collect:');
L.push(' '+e('embark-collect-group-title','Buffer')+' '+e('embark-collect-group-separator','────────────────────────────────────'));
L.push(' '+e('embark-collect-candidate','theme-studio/app.js')+' '+e('embark-collect-annotation','js-mode 48k'));
L.push(' '+e('embark-selected','theme-studio/previews.js')+' '+e('embark-collect-annotation','36k')+' · selected');
return previewLines(L);}
function renderOrderlessPreview(){const L=mbStackScene();
const OM=['orderless-match-face-0','orderless-match-face-1',
'orderless-match-face-2','orderless-match-face-3'];
const om=(n,t)=>os('orderless',OM[n],t);
L.push('');
L.push('orderless splits the input on spaces; component N highlights its matches');
L.push('with match-face-N, cycling every four:');
L.push('');
L.push(' input: the stu ap js → '+om(0,'the')+'me-'+om(1,'stu')+'dio/'+om(2,'ap')+'p.'+om(3,'js'));
L.push(' component 1 '+om(0,'the')+' · component 2 '+om(1,'stu')+' · component 3 '+om(2,'ap')+' · component 4 '+om(3,'js'));
L.push(' a fifth component cycles back to '+om(0,'match-face-0'));
return previewLines(L);}
function renderEmmsPreview(){const a='emms',L=[],o=(f,t)=>os(a,f,t);
// The three EMMS UIs: browser tree, active playlist, metaplaylist list.
L.push('M-x emms-browser — the library, by artist:');
L.push(' '+o('emms-browser-artist-face','Bill Evans'));
L.push(' '+o('emms-browser-year/genre-face','1959')+' '+o('emms-browser-album-face','Portrait in Jazz'));
L.push(' '+o('emms-browser-track-face','01. Come Rain or Come Shine'));
L.push(' '+o('emms-browser-albumartist-face','Various — Jazz Piano'));
L.push(' '+o('emms-browser-composer-face','Cole Porter (composer)'));
L.push(' '+o('emms-browser-performer-face','Oscar Peterson (performer)'));
L.push('');
L.push('*EMMS Playlist*:');
L.push(' '+o('emms-playlist-track-face','Peace Piece'));
L.push(' '+o('emms-playlist-selected-face','▶ Waltz for Debby')+' · playing now');
L.push(' '+o('emms-playlist-track-face','Blue in Green'));
L.push('');
L.push('M-x emms-metaplaylist-mode — every playlist buffer:');
L.push(' '+o('emms-metaplaylist-mode-current-face','*EMMS Playlist*')+' · the active one');
L.push(' '+o('emms-metaplaylist-mode-face','jazz-standards'));
return previewLines(L);}
function renderOrgRoamPreview(){const a='org-roam',L=[],o=(f,t)=>os(a,f,t);
// The org-roam backlinks buffer plus the dailies calendar mark.
L.push(o('org-roam-header-line','org-roam buffer, node: Theme Design'));
L.push('');
L.push(o('org-roam-title','Theme Design'));
L.push('Backlinks (3):');
L.push(' '+o('org-roam-olp','Dupre Palette › Colors › Accent')+' · outline path');
L.push(' '+o('org-roam-preview-heading','* Accent selection'));
L.push(' '+o('org-roam-preview-region','Picked dupre blue #67809c as the default agent'));
L.push(' '+o('org-roam-preview-region','accent; /color names remap to dupre hues.'));
L.push(' '+o('org-roam-preview-heading-highlight','* Banner recolor')+' · heading under the mouse');
L.push(' '+o('org-roam-preview-heading-selection','* Palette research')+' · selected heading');
L.push(' '+o('org-roam-dim','12 links from 3 nodes')+' · dimmed chrome');
L.push('');
L.push('org-roam-dailies — calendar days that carry a note:');
L.push(' Jun 28 29 30 Jul 1 '+o('org-roam-dailies-calendar-note','2')+' 3 · the 2nd has a daily');
return previewLines(L);}
function renderHlTodoPreview(){const a='hl-todo',L=[],o=(f,t)=>os(a,f,t);
L.push(';; hl-todo highlights keywords inside comments:');
L.push('(defun cj/next-track ()');
L.push(' ;; '+o('hl-todo','TODO')+': handle the empty-playlist case');
L.push(' ;; '+o('hl-todo','FIXME')+' the shuffle seed is constant');
L.push(' (emms-next))');
L.push('');
L.push('flymake integration — the keyword becomes a diagnostic:');
L.push(' 12:5 '+o('hl-todo-flymake-type','TODO')+' handle the empty-playlist case (hl-todo)');
return previewLines(L);}
function renderSymbolOverlayPreview(){const a='symbol-overlay',L=[];
const F=['symbol-overlay-face-1','symbol-overlay-face-2','symbol-overlay-face-3',
'symbol-overlay-face-4','symbol-overlay-face-5','symbol-overlay-face-6',
'symbol-overlay-face-7','symbol-overlay-face-8'];
const s=(n,t)=>os(a,F[n-1],t);
// Pinned symbols cycle through the eight overlay faces; the default face
// is the transient at-point highlight.
L.push(';; symbol-overlay-put pins a face per symbol:');
L.push('(defun '+s(1,'play-track')+' ('+s(2,'file')+' '+s(3,'playlist')+')');
L.push(' (let (('+s(4,'track')+' (probe '+s(2,'file')+')))');
L.push(' (push '+s(4,'track')+' '+s(3,'playlist')+')');
L.push(' ('+s(5,'emit')+' :'+s(6,'event')+' '+s(7,'level')+' '+s(8,'origin')+')');
L.push(' ('+s(1,'play-track')+'-start '+s(4,'track')+')))');
L.push('');
L.push('transient highlight at point (no pin):');
L.push(' (setq '+os(a,'symbol-overlay-default-face','current-symbol')+' 42)');
return previewLines(L);}
function renderAiTermPreview(){const a='ai-term',L=[];
// What these faces actually paint: the Claude Code TUI inside an agent
// terminal. The banner is the fixed accent (every session); each /color
// session color draws the input-box rules and prompt.
L.push(os(a,'cj/ai-term-accent','⏵⏵ bypass permissions on (shift+tab to cycle)')+' · the fixed banner, every agent');
const names=['red','blue','green','yellow','purple','orange','pink','cyan'];
for(const n of names){
L.push('');
L.push(os(a,'cj/ai-term-color-'+n,'──────────────────────────────────────────')+' /color '+n);
L.push(os(a,'cj/ai-term-color-'+n,'❯')+' implement the feature, then run the tests');
L.push(os(a,'cj/ai-term-color-'+n,'──────────────────────────────────────────'));
}
return previewLines(L);}
function renderSlackPreview(){const a='slack',L=[];
L.push(os(a,'slack-room-info-title-room-name-face','#general')+' '+os(a,'slack-room-info-title-face','Acme Workspace'));
L.push(os(a,'slack-room-info-section-title-face','Topic')+' '+os(a,'slack-room-info-section-label-face','daily standup')+' '+os(a,'slack-room-unread-face','3 unread'));
L.push(os(a,'slack-new-message-marker-face','---------------- new messages ----------------'));
L.push(os(a,'slack-message-output-header','craig 10:24'));
L.push(' '+os(a,'slack-message-output-text','hey ')+os(a,'slack-message-mention-me-face','@craig')+os(a,'slack-message-output-text',', see ')+os(a,'slack-message-mention-face','@alice')+os(a,'slack-message-output-text',' in ')+os(a,'slack-channel-button-face','#general')+' '+os(a,'slack-message-mention-keyword-face','urgent'));
L.push(' '+os(a,'slack-mrkdwn-bold-face','*bold*')+' '+os(a,'slack-mrkdwn-italic-face','_italic_')+' '+os(a,'slack-mrkdwn-code-face','`code`')+' '+os(a,'slack-mrkdwn-strike-face','~strike~'));
L.push(' '+os(a,'slack-mrkdwn-blockquote-face','> quoted')+' '+os(a,'slack-mrkdwn-list-face','- item'));
L.push(' '+os(a,'slack-mrkdwn-code-block-face','``` code block ```'));
L.push(' '+os(a,'slack-message-output-reaction',':thumbsup: 3')+' '+os(a,'slack-message-output-reaction-pressed',':heart: 1')+' '+os(a,'slack-message-deleted-face','(message deleted)'));
L.push(' '+os(a,'slack-all-thread-buffer-thread-header-face','Thread: 2 replies'));
L.push(os(a,'slack-attachment-header','Attachment')+' '+os(a,'slack-attachment-field-title','Field:')+' val '+os(a,'slack-message-attachment-preview-header-face','Preview')+' '+os(a,'slack-preview-face','snippet')+os(a,'slack-attachment-pad',' | ')+os(a,'slack-attachment-footer','footer'));
L.push(os(a,'slack-block-highlight-source-overlay-face',' highlighted source block '));
L.push('Actions: '+os(a,'slack-message-action-face','Edit')+' '+os(a,'slack-message-action-primary-face','Approve')+' '+os(a,'slack-message-action-danger-face','Delete'));
L.push('Blocks: '+os(a,'slack-button-block-element-face','[Button]')+os(a,'slack-button-primary-block-element-face','[Primary]')+os(a,'slack-button-danger-block-element-face','[Danger]')+os(a,'slack-select-block-element-face','[Select v]')+os(a,'slack-overflow-block-element-face','[...]')+os(a,'slack-date-picker-block-element-face','[Date]'));
L.push('Dialog: '+os(a,'slack-dialog-title-face','Title')+' '+os(a,'slack-dialog-element-label-face','Label')+' '+os(a,'slack-dialog-element-hint-face','(hint)')+' '+os(a,'slack-dialog-element-placeholder-face','placeholder')+' '+os(a,'slack-dialog-element-error-face','error')+' '+os(a,'slack-dialog-select-element-input-face','[input v]')+' '+os(a,'slack-dialog-submit-button-face','[Submit]')+os(a,'slack-dialog-cancel-button-face','[Cancel]'));
L.push('Users: '+os(a,'slack-user-active-face','alice (active)')+' '+os(a,'slack-user-dnd-face','bob (dnd)')+' '+os(a,'slack-profile-image-face','[img]')+' '+os(a,'slack-user-profile-header-face','Profile')+' '+os(a,'slack-user-profile-property-name-face','Title:')+' Dev');
L.push('Search: '+os(a,'slack-search-result-message-header-face','#general')+' '+os(a,'slack-search-result-message-username-face','craig'));
L.push('Modeline: '+os(a,'slack-modeline-has-unreads-face','* unreads')+' '+os(a,'slack-modeline-channel-has-unreads-face','#ch')+' '+os(a,'slack-modeline-thread-has-unreads-face','thread'));
return previewLines(L);}
function renderTelegaPreview(){const a='telega',L=[];
L.push(os(a,'telega-root-heading','Telegram')+' '+os(a,'telega-tracking','[tracking]')+' '+os(a,'telega-unread-unmuted-modeline','5 unread'));
L.push(os(a,'telega-has-chatbuf-brackets','[')+os(a,'telega-username','Christine')+os(a,'telega-has-chatbuf-brackets',']')+' '+os(a,'telega-user-online-status','online')+' '+os(a,'telega-unmuted-count','3')+' '+os(a,'telega-mention-count','@2')+os(a,'telega-delim-face',' | ')+os(a,'telega-secret-title','Secret')+' '+os(a,'telega-muted-count','muted'));
L.push(os(a,'telega-username','Bob')+' '+os(a,'telega-user-non-online-status','last seen recently')+' '+os(a,'telega-contact-birthdays-today','birthday today')+' '+os(a,'telega-shadow','shadow')+' '+os(a,'telega-link','link')+' '+os(a,'telega-blue','blue')+' '+os(a,'telega-red','red'));
L.push('');
L.push(os(a,'telega-msg-heading','Today'));
L.push(os(a,'telega-msg-user-title','Christine')+' '+os(a,'telega-msg-inline-reply','| reply to Bob')+' '+os(a,'telega-msg-inline-forward','fwd from Carol')+' '+os(a,'telega-msg-inline-other','via bot'));
L.push(' '+os(a,'telega-entity-type-bold','bold')+' '+os(a,'telega-entity-type-italic','italic')+' '+os(a,'telega-entity-type-underline','underline')+' '+os(a,'telega-entity-type-strikethrough','strike')+' '+os(a,'telega-entity-type-code','code')+' '+os(a,'telega-entity-type-spoiler','spoiler'));
L.push(' '+os(a,'telega-entity-type-pre','pre block')+' '+os(a,'telega-entity-type-blockquote','> quote')+' '+os(a,'telega-entity-type-mention','@user')+' '+os(a,'telega-entity-type-hashtag','#tag')+' '+os(a,'telega-entity-type-cashtag','$USD')+' '+os(a,'telega-entity-type-botcommand','/start')+' '+os(a,'telega-entity-type-texturl','link'));
L.push(os(a,'telega-msg-self-title','Me')+' '+os(a,'telega-reaction',':+1: 2')+' '+os(a,'telega-reaction-chosen',':heart: 1')+' '+os(a,'telega-reaction-paid',':star: 5')+' '+os(a,'telega-reaction-paid-chosen',':star: paid')+' '+os(a,'telega-msg-deleted','(deleted)')+' '+os(a,'telega-msg-sponsored','Sponsored'));
L.push(' checklist '+os(a,'telega-checklist-stats-done','2 done')+' / '+os(a,'telega-checklist-stats-todo','3 todo')+' '+os(a,'telega-highlight-text-face','search hit')+' '+os(a,'telega-button-highlight','[active btn]'));
L.push(os(a,'telega-chat-prompt','>')+' '+os(a,'telega-chat-prompt-aux','reply')+' '+os(a,'telega-chat-input-attachment','[photo.jpg]')+' '+os(a,'telega-topic-button','# Topic')+' '+os(a,'telega-filter-active','Main')+' '+os(a,'telega-filter-button-active','[Unread]')+os(a,'telega-filter-button-inactive','[All]'));
L.push('Buttons '+os(a,'telega-box-button','[box]')+os(a,'telega-box-button-active','[on]')+os(a,'telega-box-button-default-active','[def]')+os(a,'telega-box-button-default-passive','[def-]')+os(a,'telega-box-button-primary-active','[pri]')+os(a,'telega-box-button-primary-passive','[pri-]')+os(a,'telega-box-button-success-active','[ok]')+os(a,'telega-box-button-success-passive','[ok-]'));
L.push(' '+os(a,'telega-box-button-danger-active','[del]')+os(a,'telega-box-button-danger-passive','[del-]')+os(a,'telega-box-button-ui-active','[ui]')+os(a,'telega-box-button-ui-passive','[ui-]')+os(a,'telega-box-button2-active','[b2]')+os(a,'telega-box-button2-passive','[b2-]')+os(a,'telega-box-button2-white-foreground','[b2w]'));
L.push('Describe '+os(a,'telega-describe-section-title','Section')+' '+os(a,'telega-describe-subsection-title','Sub')+' '+os(a,'telega-describe-item-title','Item:')+' enckey '+os(a,'telega-enckey-00','00')+os(a,'telega-enckey-01','01')+os(a,'telega-enckey-10','10')+os(a,'telega-enckey-11','11'));
L.push('Palette '+os(a,'telega-palette-builtin-blue','blue')+' '+os(a,'telega-palette-builtin-green','green')+' '+os(a,'telega-palette-builtin-orange','orange')+' '+os(a,'telega-palette-builtin-purple','purple'));
L.push(os(a,'telega-link-preview-sitename','example.com')+' '+os(a,'telega-link-preview-title','Link preview title'));
L.push('Webpage '+os(a,'telega-webpage-title','Title')+' '+os(a,'telega-webpage-subtitle','Subtitle')+' '+os(a,'telega-webpage-header','Header')+' '+os(a,'telega-webpage-subheader','Subheader')+' '+os(a,'telega-webpage-outline','outline')+' '+os(a,'telega-webpage-fixed','fixed')+' '+os(a,'telega-webpage-preformatted','pre')+' '+os(a,'telega-webpage-marked','marked')+' '+os(a,'telega-webpage-strike-through','strike')+' '+os(a,'telega-webpage-chat-link','chat-link'));
return previewLines(L);}
function genericPreview(app){let h='';for(const [face,label] of APPS[app].faces)h+=`
${esc(label)}
`;return h+'
';}
// Bespoke split preview: a focused window beside its auto-dimmed twin, both
// showing the language selected at the top of the page (kept in sync via the
// langsel onchange, which re-runs buildPkgPreview). The left pane carries the
// real per-token syntax colors; the right pane shows what auto-dim does -- every
// default/font-lock face remaps to the single `auto-dim-other-buffers' face, so
// the same code collapses to one faded foreground on the dim background. The
// trailing row demonstrates `auto-dim-other-buffers-hide' (org hidden text whose
// foreground matches the background, so it vanishes in a dimmed window).
function renderAutodimPreview(){
const a='auto-dim-other-buffers';
const langsel=document.getElementById('langsel');
const lang=(langsel&&langsel.value)||Object.keys(SAMPLES)[0];
const lines=(SAMPLES[lang]||[]).slice(0,9);
let lit='';
for(const line of lines){
if(!line.length){lit+='\n';continue;}
for(const [k,t] of line)lit+=`${esc(t)}`;
lit+='\n';}
const dimFg=effFg(pkgEffFg(a,'auto-dim-other-buffers')),dimBg=pkgEffBg(a,'auto-dim-other-buffers')||'#000000';
let dim='';
for(const line of lines){
if(!line.length){dim+='\n';continue;}
for(const [,t] of line)dim+=esc(t);
dim+='\n';}
const hideFg=effFg(pkgEffFg(a,'auto-dim-other-buffers-hide')),hideBg=pkgEffBg(a,'auto-dim-other-buffers-hide')||dimBg;
const foldText='··· folded body (hidden when dimmed) ···';
const accent=uf('cursor').bg||'#67809c';
const pane=(label,body,bg,focused)=>
``;
const litBody=lit+'\n'+`${esc(foldText)}`;
const dimBody=`${dim}\n`
+`${esc(foldText)}`;
return ``
+pane('normal',litBody,MAP['bg'],true)
+pane('auto-dim',dimBody,dimBg,false)
+`
`;
}
function renderMarkdownPreview(){const a='markdown-mode',L=[];
const dl='markdown-header-delimiter-face',mk='markdown-markup-face';
L.push(os(a,mk,'---'));
L.push(os(a,'markdown-metadata-key-face','title:')+' '+os(a,'markdown-metadata-value-face','Project Name'));
L.push(os(a,'markdown-metadata-key-face','version:')+' '+os(a,'markdown-metadata-value-face','1.2.0'));
L.push(os(a,mk,'---'));
L.push('');
L.push(os(a,dl,'#')+' '+os(a,'markdown-header-face-1','Project Name'));
L.push('');
L.push(os(a,'markdown-comment-face','<!-- a one-line tagline -->'));
L.push('A library for '+os(a,'markdown-bold-face','**doing things**')+' and '+os(a,'markdown-italic-face','*other things*')+'.');
L.push('');
L.push(os(a,dl,'##')+' '+os(a,'markdown-header-face-2','Installation'));
L.push('');
L.push('Run '+os(a,'markdown-inline-code-face','`npm install project`')+' to get started.');
L.push('');
L.push(os(a,mk,'```')+os(a,'markdown-language-keyword-face','sh'));
L.push(os(a,'markdown-pre-face',' git clone https://example.com/project.git'));
L.push(os(a,'markdown-pre-face',' cd project; make'));
L.push(os(a,mk,'```'));
L.push('');
L.push(os(a,dl,'###')+' '+os(a,'markdown-header-face-3','Usage'));
L.push('');
L.push(os(a,'markdown-list-face','- ')+'See the '+os(a,'markdown-link-face','[docs]')+os(a,'markdown-url-face','(https://example.com/docs)')+' for details.');
L.push(os(a,'markdown-list-face','- ')+'Or browse '+os(a,'markdown-plain-url-face','https://example.com')+' directly.');
L.push(os(a,'markdown-gfm-checkbox-face','- [x]')+' shipped '+os(a,'markdown-gfm-checkbox-face','- [ ]')+' planned');
L.push('');
L.push(os(a,'markdown-blockquote-face','> A note worth quoting, with a footnote')+os(a,'markdown-footnote-marker-face','[^1]')+os(a,'markdown-blockquote-face','.'));
L.push('');
L.push(os(a,'markdown-table-face','| Option | Default |'));
L.push(os(a,'markdown-table-face','|--------|---------|'));
L.push(os(a,'markdown-table-face','| debug | false |'));
L.push('');
L.push(os(a,'markdown-hr-face','---'));
L.push('');
L.push(os(a,'markdown-strike-through-face','~~deprecated~~')+' '+os(a,'markdown-highlight-face','==important==')+' '+os(a,'markdown-math-face','$E = mc^2$'));
L.push(os(a,'markdown-html-tag-delimiter-face','<')+os(a,'markdown-html-tag-name-face','kbd')+os(a,'markdown-html-tag-delimiter-face','>')+'Ctrl-C'+os(a,'markdown-html-tag-delimiter-face','</')+os(a,'markdown-html-tag-name-face','kbd')+os(a,'markdown-html-tag-delimiter-face','>'));
L.push(os(a,'markdown-footnote-marker-face','[^1]:')+' '+os(a,'markdown-footnote-text-face','the footnote text.'));
return previewLines(L);}
// nerd-icons gallery grid: the full colored catalog. Every distinct face-bearing
// nerd-icons glyph (APPS['nerd-icons'].gallery, captured by build-nerd-icons-legend.el),
// one row per color face, the rows ordered by hue so families cluster (blues
// together, reds together). Each cell draws the glyph in its face color with the
// icon's nerd-font name beneath. SIZEPT (points, default 14) sizes the glyphs so
// the designer can view the grid at different buffer sizes via the preview-pane
// dropdown; the cell width scales with it. Recoloring a face repaints its swatch
// and every glyph in its row because os() reads the live registry. Falls back to
// the generic preview if the gallery is missing (the bespoke app registers with a
// valid legend, so that path is defensive).
function renderNerdIconsPreview(sizePt){
const a='nerd-icons',groups=(APPS[a]&&APPS[a].gallery)||[];
if(!groups.length)return genericPreview(a);
const pt=sizePt||14,cellW=Math.round(pt*4.6+24);
let h=``;
for(const g of groups){
h+='
'
+`
`
+os(a,g.face,'■')+' '+esc(g.face)+' ('+g.glyphs.length+')
'
+'
';
for(const e of g.glyphs)
h+=``
+``+os(a,g.face,e.glyph)+'
'
+``+esc(e.name)+''
+'';
h+='
';
}
return h+'
';}