skills$openclaw/trmnl
peetzweg5.2k

by peetzweg

trmnl – OpenClaw Skill

trmnl is an OpenClaw Skills integration for coding workflows. Generate content for TRMNL e-ink display devices using the TRMNL CSS framework. Use when the user wants to display information on their TRMNL device, send messages to an e-ink display, create dashboard content, show notifications, or update their terminal display. Supports rich layouts with the TRMNL framework (flexbox, grid, tables, progress bars, typography utilities) and sends content via webhook API.

5.2k stars6.8k forksSecurity L1
Updated Feb 7, 2026Created Feb 7, 2026coding

Skill Snapshot

nametrmnl
descriptionGenerate content for TRMNL e-ink display devices using the TRMNL CSS framework. Use when the user wants to display information on their TRMNL device, send messages to an e-ink display, create dashboard content, show notifications, or update their terminal display. Supports rich layouts with the TRMNL framework (flexbox, grid, tables, progress bars, typography utilities) and sends content via webhook API. OpenClaw Skills integration.
ownerpeetzweg
repositorypeetzweg/trmnl
languageMarkdown
licenseMIT
topics
securityL1
installopenclaw add @peetzweg/trmnl
last updatedFeb 7, 2026

Maintainer

peetzweg

peetzweg

Maintains trmnl in the OpenClaw Skills directory.

View GitHub profile
File Explorer
19 files
.
assets
good-examples
simple-message.html
391 B
stats-dashboard.html
1.2 KB
table-data.html
1.2 KB
task-list.html
1.1 KB
two-column-image.html
656 B
anti-patterns.md
6.1 KB
references
components.md
8.2 KB
css-utilities.md
6.5 KB
framework-overview.md
4.7 KB
layout-systems.md
7.9 KB
patterns.md
9.9 KB
webhook-api.md
5.0 KB
scripts
check_payload.py
2.4 KB
_meta.json
450 B
SKILL.md
7.8 KB
SKILL.md

name: trmnl description: Generate content for TRMNL e-ink display devices using the TRMNL CSS framework. Use when the user wants to display information on their TRMNL device, send messages to an e-ink display, create dashboard content, show notifications, or update their terminal display. Supports rich layouts with the TRMNL framework (flexbox, grid, tables, progress bars, typography utilities) and sends content via webhook API.

TRMNL Content Generator

Generate HTML content for TRMNL e-ink display devices.

Quick Start Workflow

  1. Check for $TRMNL_WEBHOOK environment variable
  2. If missing, prompt user for webhook URL
  3. Ask user to verify TRMNL display markup is set to: <div>{{content}}</div>
  4. Confirm device type (default: TRMNL OG, 2-bit, 800x480)
  5. Read relevant reference docs based on content needs
  6. Generate HTML using TRMNL framework classes
  7. Send via POST to webhook (use temp file method)
  8. Send minimal confirmation only - Do NOT echo content back to chat

Device & Setup

Default target: TRMNL OG (7.5" e-ink, 800x480px, 2-bit)

Display markup required:

<div>{{content}}</div>

Webhook:

export TRMNL_WEBHOOK="https://trmnl.com/api/custom_plugins/{uuid}"

Sending content (temp file method):

cat > /tmp/trmnl.json << 'EOF'
{"merge_variables":{"content":"<div class=\"layout\">HTML</div>"}}
EOF
curl "$TRMNL_WEBHOOK" -H "Content-Type: application/json" -d @/tmp/trmnl.json -X POST

Webhook Limits

TierPayload SizeRate Limit
Free2 KB (2,048 bytes)12 requests/hour
TRMNL+5 KB (5,120 bytes)30 requests/hour

Check payload size before sending:

python3 scripts/check_payload.py /tmp/trmnl.json

Tips to reduce payload size:

  • Minify HTML (remove unnecessary whitespace)
  • Use framework classes instead of inline styles
  • Use short class names the framework provides
  • Remove comments from HTML
  • Use deep_merge strategy for incremental updates

Reference Documentation

Read these files as needed:

FileWhen to Read
references/patterns.mdStart here - Common plugin patterns from official examples
references/framework-overview.mdDevice specs, e-ink constraints, responsive prefixes
references/css-utilities.mdColors, typography, sizing, spacing utilities
references/layout-systems.mdFlexbox, grid, overflow/clamp engines
references/components.mdTitle bar, dividers, items, tables, charts
references/webhook-api.mdPayload format, rate limits, troubleshooting
assets/anti-patterns.mdCommon mistakes to avoid
assets/good-examples/HTML reference implementations

Scripts:

ScriptPurpose
scripts/check_payload.pyVerify payload size before sending (run on /tmp/trmnl.json)

Standard Plugin Structure

Every plugin follows this pattern:

<div class="layout layout--col gap--space-between">
  <!-- Content sections separated by dividers -->
</div>
<div class="title_bar">
  <img class="image" src="icon.svg">
  <span class="title">Plugin Name</span>
  <span class="instance">Context</span>
</div>
  • layout + layout--col = vertical flex container
  • gap--space-between = push sections to edges
  • title_bar = always at bottom, outside layout
  • divider = separate major sections
  • CRITICAL: Only ONE .layout element per view (no nesting)

Grid System (10-Column)

Column spans should sum to 10:

<div class="grid">
  <div class="col--span-3">30%</div>
  <div class="col--span-7">70%</div>
</div>

Simple equal columns: grid--cols-2, grid--cols-3, etc.

Item Component

Standard data display pattern:

<div class="item">
  <div class="meta"><span class="index">1</span></div>
  <div class="content">
    <span class="value value--xlarge value--tnums">$159,022</span>
    <span class="label">Total Sales</span>
  </div>
</div>

Value Typography

Always use value--tnums for numbers.

ClassUsage
value--xxxlargeHero KPIs
value--xxlargeLarge prices
value--xlargeSecondary metrics
value--smallTertiary data
value--tnumsAlways for numbers

Auto-fit: <span class="value" data-fit-value="true">...</span>

Columns (for Lists)

<div class="columns">
  <div class="column" data-overflow="true" data-overflow-counter="true">
    <span class="label label--medium group-header">Section</span>
    <div class="item">...</div>
  </div>
</div>

Grayscale Dithering

Use dithered classes, not inline gray colors:

  • bg--black, bg--gray-60, bg--gray-30, bg--gray-10, bg--white
  • text--black, text--gray-50

Data Attributes

AttributePurpose
data-fit-value="true"Auto-resize text to fit
data-value-format="true"Auto-format numbers (locale-aware)
data-clamp="N"Limit to N lines
data-overflow="true"Enable overflow management
data-overflow-counter="true"Show "and X more"
data-overflow-max-cols="N"Max columns for overflow
data-content-limiter="true"Auto-adjust text size
data-pixel-perfect="true"Crisp text rendering
data-table-limit="true"Table overflow with "and X more"

Label & Title Variants

<span class="label label--small">Small</span>
<span class="label label--medium">Medium</span>
<span class="label label--underline">Underlined</span>
<span class="label label--gray">Muted/completed</span>
<span class="title title--small">Compact title</span>

Gap Utilities

gap--space-between | gap--xxlarge | gap--xlarge | gap--large | gap--medium | gap | gap--small | gap--xsmall | gap--none

Typography Guidelines

Recommended: Georgia serif font for e-ink readability.

Content-aware sizing:

  • Short content = bigger fonts (24-28px body)
  • Long content = smaller fonts (20px body)
  • Headings: 36-48px

User Experience

Critical: Do NOT echo content back to chat. Just confirm "Sent to TRMNL".

Anti-Patterns

  1. Tiny fonts for short content
  2. Center-aligning columns with different lengths (use layout--start)
  3. Spoiling content in chat confirmation
  4. Missing value--tnums on numbers
  5. Missing title_bar
  6. Not using data-fit-value on primary metrics
  7. Skipping data-overflow on variable lists
  8. Using inline gray colors instead of bg--gray-*
  9. Forgetting dividers between sections

Best Practices

  1. Verify <div>{{content}}</div> display markup
  2. Use layout + title_bar structure
  3. Always value--tnums for numbers
  4. Use data-fit-value on primary metrics
  5. Use data-overflow on variable lists
  6. Use item component pattern
  7. Use divider between sections
  8. Use bg--gray-* dithered classes
  9. Content-aware font sizing
  10. Top-align columns (layout--start)
  11. Temp file method for curl
  12. Minimal confirmations

Mashup Layouts (Multi-Plugin)

For dashboard views with multiple plugins:

LayoutDescription
mashup--1Lx1R2 columns (50/50)
mashup--1Tx1B2 rows (50/50)
mashup--2x24 quadrants

See references/layout-systems.md for all 7 layouts.

Troubleshooting

ProblemSolution
Webhook failsVerify URL, check rate limits (12/hour free)
Content missingCheck display markup is <div>{{content}}</div>
Payload too largeRun scripts/check_payload.py, keep under 2KB (free) or 5KB (TRMNL+)
Numbers misalignedAdd value--tnums
Text overflowUse data-clamp or data-overflow
Columns misalignedUse layout--start not layout--center
Multiple layouts errorKeep only ONE .layout element per view
Nested content failsUse .richtext for nested/formatted content
README.md

No README available.

Permissions & Security

Security level L1: Low-risk skills with minimal permissions. Review inputs and outputs before running in production.

Requirements

  • OpenClaw CLI installed and configured.
  • Language: Markdown
  • License: MIT
  • Topics:

FAQ

How do I install trmnl?

Run openclaw add @peetzweg/trmnl in your terminal. This installs trmnl into your OpenClaw Skills catalog.

Does this skill run locally or in the cloud?

OpenClaw Skills execute locally by default. Review the SKILL.md and permissions before running any skill.

Where can I verify the source code?

The source repository is available at https://github.com/openclaw/skills/tree/main/skills/peetzweg/trmnl. Review commits and README documentation before installing.