Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/octamap/mesa
Build-time HTML component engine
https://github.com/octamap/mesa
compiler html npm vite
Last synced: about 4 hours ago
JSON representation
Build-time HTML component engine
- Host: GitHub
- URL: https://github.com/octamap/mesa
- Owner: octamap
- License: mit
- Created: 2025-01-06T20:50:00.000Z (17 days ago)
- Default Branch: main
- Last Pushed: 2025-01-17T15:07:32.000Z (7 days ago)
- Last Synced: 2025-01-17T15:44:23.027Z (7 days ago)
- Topics: compiler, html, npm, vite
- Language: TypeScript
- Homepage: https://www.npmjs.com/package/@octamap/mesa
- Size: 144 KB
- Stars: 22
- Watchers: 0
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# π **Mesa β Build-Time HTML Components**
**@octamap/mesa**---
Mesa is a **build-time HTML component engine** that allows you to write reusable, declarative HTML components with scoped CSS and dynamic attributes β **without requiring runtime JavaScript to render them**.
Whether you're building a **static website**, a **micro-frontend architecture**, or **HTML-first** website development, Mesa keeps your pages **lightweight, SEO-friendly, and blazing fast**.
## π οΈ **What is Mesa?**
Mesa allows you to:
1. **Write reusable HTML components**.
2. **Use them in other HTML files**.
3. **Automatically handle props, dynamic attributes, and scoped styles at build time**.## β‘ Super quick reactivity
No need for large runtimes for reactivity (such as Vue, React & Svelte)Mesa is framework agnostic. Achive the same level of reactivity by pairing it up with somthing quick and lightweight like alpine.js or HTMX for **β‘οΈsuper quick hydrationβ‘οΈ**
## π VS Code Extension!! (Beta)
Type completions & syntax highlighting is now supported through a brand new Mesa Visual Studio Code extension!!#### **Early Beta**:
[Install Mesa VS Code Extension](https://marketplace.visualstudio.com/items?itemName=octamap.mesa)#### **Report issues**:
This extension is really new. Please send any issues with the extension to `[email protected]`.## π€― Javascript with no runtime Javascript (NOT AVAILABLE YET, COMING SOON)
Mesa not only lets you create components without requiring runtime Javascript. Mesa also lets you **write Javascript without requiring runtime Javascript** π€―. (new in since version 1.2.0). Read more about this [here](#-compile-time-javascript).This feature will be expanded in the future (Make requests through tickets on github)
## π **1. Create a Component**
Define a reusable component in its own file:
### π **my-custom-button.html**
```html...
.close-button {
background: red;
color: white;
border: none;
padding: 8px 12px;
cursor: pointer;
}```
- The `` contains your markup.
- The `` block includes the scoped CSS.---
## π **2. Use the Component in Another File**
In your main HTML file, simply use the `<my-custom-button>` tag:
### π **index.html**
```html
<!DOCTYPE html>
<html lang="en">
<head>
<link rel="stylesheet" href="component-styles.css">
</head>
<body>
<h1>Welcome to Mesa</h1>
<my-custom-button @click="exampleEvent"></my-custom-button>
</body>
</html>
```---
## πͺ **3. Mesa Works Its Magic at Build Time**
Mesa compiles the above into:
### π **index.html**
```html
<!DOCTYPE html>
<html lang="en">
<head>
<link rel="stylesheet" href="component-styles.css">
</head>
<body>
<h1>Welcome to Mesa</h1>
<button class="close-button" @click="exampleEvent">
<svg>...</svg>
</button>
</body>
</html>
```### π **component-styles.css**
```css
.close-button {
background: red;
color: white;
border: none;
padding: 8px 12px;
cursor: pointer;
}
```β **Props and attributes are passed automatically.**
β **Styles are extracted and scoped correctly.**
β **No runtime rendering β everything happens at build time.**# βοΈ **Features**
### π¦ **1. Static Components, Dynamic Props**
Pass attributes and dynamic bindings directly on the root element:**index.html**
```html
<my-custom-button @click="handleClick"></my-custom-button>
```**Build Output:**
```html
<button class="close-button" @click="handleClick">
<svg>...</svg>
</button>
```β Events (`@click`) and bindings (`x-model`) are preserved and passed correctly.
---
### π¦ **2. Default Attribute Mapping**
If your component has an element marked as `#default`, props are mapped automatically:**my-input.html**
```html
<div>
<input #default class="input-field" />
</div>
```**index.html**
```html
<my-input placeholder="Enter your email" x-model="email"></my-input>
```**Build Output:**
```html
<div>
<input class="input-field" placeholder="Enter your email" x-model="email" />
</div>
```β Clean, predictable prop mapping.
**Details**
- The root element of the component is automatically the `#default` if no other element within the component has a `#default` attribute---
### π¦ **3. Named Slots with Target Mapping**
Map specific child elements to targets in your components:**my-card.html**
```html
<div class="card">
<h1 #title></h1>
<p #content></p>
</div>
```**index.html**
```html
<my-card>
<title>This is the Title</title>
<content>This is the Content</content>
</my-card>
```**Build Output:**
```html
<div class="card">
<h1>This is the Title</h1>
<p>This is the Content</p>
</div>
```β Clear and intuitive named slot-like behavior.
---
### π¦ **4. Scoped Styles**
Each componentβs styles are:
- **Scoped to their usage** (avoiding global CSS pollution).
- **Automatically extracted into `component-styles.css`.**β Styles only load when theyβre needed.
---
### π¦ **5. Nested Components**
Mesa supports **nested components** seamlessly:**nested-component.html**
```html
<div class="nested">
<inner-component></inner-component>
</div>
```**index.html**
```html
<nested-component></nested-component>
```Mesa recursively processes inner components during the build phase.
β Fully recursive parsing and transformation.
# π **Getting Started**
## π New Project
Setup a new mesa project by running the command below, this sets you up with a sample website:```
npx @octamap/create-mesa@latest project-name
```## π€ Add to existing project
### 1οΈβ£ **Install Mesa**
```bash
npm install @octamap/mesa --save-dev
```### 2οΈβ£ **Configure Vite**
Update your `vite.config.ts`:```typescript
import { defineConfig } from 'vite';
import { Mesa } from '@octamap/mesa';export default defineConfig({
plugins: [
Mesa({
'my-custom-button': './src/components/my-custom-button.html',
'my-card': './src/components/my-card.html'
})
]
});
```### 3οΈβ£ **Run Your Project**
```bash
npm run dev
```β Enjoy live-reloading and build-time transformations.
### 4οΈβ£ **Build for Production**
```bash
npm run build
```β Your components are compiled, styles extracted, and everything is optimized.
# β‘οΈ Ultra quick hydration
Svelte SSG has been the wholy grail of quick websites. With Mesa you achive this but even quicker.
- SvelteKit SSG hydration involves loading the Svelte runtime and then essentially "rebuilding" the pageβs structure by mapping pre-rendered HTML to its components.
- Mesa achieves the same level of interactivity by allowing you to use lightweight frameworks like Alpine.js or HTMX, which operate directly on the DOM. These frameworks donβt need to rebuild the pageβs structureβthey simply bind functionality to the already-rendered HTML.
### Comparison:
| Feature | SvelteKit (SSG) | Mesa + alpine.js |
| ------------------------- | ---------------------------- | --------------------- |
| **Initial HTML Load** | Instant | Instant |
| **JavaScript Payload** | Medium (slightly larger) | Small (lightweight) |
| **Hydration Speed** | Slightly Slower | Faster (no hydration) |
| **Reactivity Setup Time** | Fast, but hydration required | Very Fast |# π€― Compile Time JavaScript
Mesa allows you to use **JavaScript at build time** to dynamically generate static HTML. This means you can use logic like loops, conditions, and bindings during the build process, but the final output is fully-rendered, SEO-friendly, and lightning-fast HTML.
### π **Getting Started with Compile-Time JavaScript**
#### **1οΈβ£ Define Compile-Time Logic**
Use the `<script #mesa>` tag to write your JavaScript logic. This script runs only during the build process and is not included in the final output.
```html
<script #mesa>
const items = ["Home", "About", "Contact"];
</script>
```#### **2οΈβ£ Use the Variables in Your HTML**
Mesa's declarative syntax makes it easy to loop through data and dynamically generate HTML.
```html
<ul>
<li :for="item in items">{{ item }}</li>
</ul>
```#### **3οΈβ£ Mesa Compiles the HTML**
During the build, Mesa executes the `#mesa` script and replaces the dynamic logic with fully-rendered static HTML.
---
### β **Simple Example: Dynamic List**
#### **Input:**
```html
<script #mesa>
const items = ["Home", "About", "Contact"];
</script>
<ul>
<li :for="item in items">{{ item }}</li>
</ul>
```#### **Output:**
```html
<ul>
<li>Home</li>
<li>About</li>
<li>Contact</li>
</ul>
```This produces clean, static HTML without requiring any JavaScript at runtime.
### β **Dynamic Attributes and Event Bindings**
You can also bind attributes and events dynamically at build time.
#### **Input:**
```html
<script #mesa>
const buttons = [
{ label: "Save", action: "saveData()" },
{ label: "Delete", action: "deleteData()" },
];
</script>
<div>
<button :for="button in buttons" @click="button.action">
{{ button.label }}
</button>
</div>
```#### **Output:**
```html
<div>
<button @click="saveData()">Save</button>
<button @click="deleteData()">Delete</button>
</div>
```---
### π **More Advanced Examples**
Once you're comfortable with the basics, you can use Mesa's compile-time JavaScript for more complex scenarios.
#### **Dynamic Segments**
Use `:for` loops and bindings to create interactive elements of a component called `segment-control`
##### **Input:**
```html
<script #mesa>
const segments = [
{ name: "Document Tools", url: "/videos/doc-tools.mov" },
{ name: "3D Configurators", url: "/videos/configurators.mov" },
];
</script>
<segment-control
:for="(segment, index) in segments"
@click="play(segment.url)"
:class="{selected: index === selectedIndex}">
{{ segment.name }}
</segment-control>
```##### **Output:**
```html
<div class="segment-control">
<button @click="play('/videos/doc-tools.mov')" :class="{selected: 0 === selectedIndex}">
Document Tools
</button>
<button @click="play('/videos/configurators.mov')" :class="{selected: 1 === selectedIndex}">
3D Configurators
</button>
</div>
```# π¦ **Examples & Additional Features**
## π¦ **Defining Components Outside of `vite.config.ts`**
In larger projects or when working with shared component libraries, managing all your component mappings directly inside `vite.config.ts` can become cumbersome. **Mesa** allows you to define components in an external JavaScript or TypeScript file using the `components` function.
This approach ensures better **separation of concerns**, easier **reusability**, and improved **readability**.
---
### π οΈ **Step 1: Create a Components Configuration File**
Create a file to define your components, for example:
### π **components.ts**
```typescript
import { components } from "@octamap/mesa";// Use import.meta.url to resolve absolute paths to component files
const OctamapHtmlComponents = components(import.meta.url, {
"icon-field": "./src/icon-field.html",
"close-button": "./src/close-button.html",
"back-button": "./src/back-button.html",
"head-defaults": "./src/head-defaults.html",
});export { OctamapHtmlComponents };
```### β **Why `import.meta.url` is Necessary?**
The `import.meta.url` ensures that **Mesa can resolve absolute paths** to your HTML component files. Without it, the plugin might struggle to correctly locate your component files, especially when working across different environments or build pipelines.---
### π οΈ **Step 2: Use the Configuration in `vite.config.ts`**
Import your component configuration into `vite.config.ts`:
### π **vite.config.ts**
```typescript
import { defineConfig } from 'vite';
import { Mesa } from '@octamap/mesa';
import { OctamapHtmlComponents } from './components';export default defineConfig({
plugins: [
Mesa(OctamapHtmlComponents)
]
});
```---
### π§ **How it Works**
1. The `components` function maps component names (`icon-field`, `close-button`) to their respective HTML files (`icon-field.html`, `close-button.html`).
2. `import.meta.url` ensures **absolute paths** are resolved correctly at runtime.
3. `vite.config.ts` references these mappings via `OctamapHtmlComponents`.This approach keeps your `vite.config.ts` **clean and focused** while centralizing component definitions in a separate file.
---
### π **Advantages of This Approach**
β **Improved Maintainability:** Component mappings are easier to update and manage in one dedicated file.
β **Reusability:** The configuration can be reused across different projects or environments.
β **Scalability:** Large projects with hundreds of components are easier to organize.
β **Clear Separation:** `vite.config.ts` remains focused on build configuration.---
### π¦ **Example Folder Structure**
```
/src
/components
icon-field.html
close-button.html
back-button.html
head-defaults.html
/scripts
components.ts <-- Define component mappings here
vite.config.ts <-- Reference the mappings
```## π **Register Entire Folders of Components with `folder`**
### π **Simplify Component Registration with `folder`**
Mesa now supports **bulk registration** of `.html` or `.svg` components from an entire folder using the `folder` function. This makes it incredibly easy to manage and include multiple components without manually defining each one in your configuration.
---
### π οΈ **1. Folder-Based Component Registration**
You can register all `.html` and `.svg` components in a folder using the `folder` utility:
**vite.config.ts**
```typescript
import { defineConfig } from 'vite';
import { Mesa, folder } from '@octamap/mesa';
import { OctamapHtmlComponents } from './components';export default defineConfig({
plugins: [
Mesa({
...OctamapHtmlComponents,
...folder("./icons") // Registers all components in the ./icons folder
}),
],
});
```β The `folder` function scans the `./icons` directory and registers all `.html` and `.svg` files as components.
β Nested folders are supported, and components are named based on their folder structure.---
### π **2. Folder Structure Example**
Your project might have a folder structure like this:
```
/icons
letter-notification.svg
/another-folder
profile-icon.svg
```---
### π **3. Use Registered Components in HTML**
After registering the folder, you can use the components directly in your HTML:
**index.html**
```html
<div>
<div class="check-inbox-page">
<letter-notification height="60px"></letter-notification>
<another-folder-profile-icon></another-folder-profile-icon>
</div>
</div>
```β **Automatic Naming:**
- `letter-notification.svg` β `<letter-notification>`
- `another-folder/profile-icon.svg` β `<another-folder-profile-icon>`β **Scoped Naming:** Nested folder structures are preserved in the component names to avoid conflicts.
---
### π§ **How it Works**
1. The `folder` function scans the specified directory.
2. Each `.html` and `.svg` file is registered as a Mesa component.
3. Nested files are automatically prefixed with their folder names.**Example Naming Convention:**
- `icons/alert.svg` β `<alert>`
- `icons/notifications/alert.svg` β `<notifications-alert>`This ensures unique and collision-free component names.
With `folder`, managing and scaling your components becomes effortless. Say goodbye to repetitive mappings and enjoy clean, intuitive configurations. πβ¨
---
Here's an illustrative section explaining how Mesa handles **scoped styles for components** and ensures they are appropriately applied regardless of where the component is used:
---
## π¦ **Scoped Styles Across Different Usage Scenarios**
Mesa ensures **scoped styles** are applied consistently across your components, whether they're used in a **single page** or **multiple pages**.
### π **1. Component Definition**
First, define your reusable component with its styles.
**π letter-notification.svg**
```html
<svg>
..
</svg>
```**π check-inbox.html**
```html
<div>
<div class="check-inbox-page">
<letter-notification height="60px"></letter-notification>
</div>
</div><style>
.check-inbox-page {
height: 100px;
width: 200px;
}```
---
### π **2. Using the Component in `index.html`**
When the component is used in a dedicated page, Mesa extracts and consolidates its styles into a global stylesheet (`component-styles.css`).
**π index.html**
```html
```
**π Build Output:**
```html
..
```
**π component-styles.css**
```css
.check-inbox-page {
height: 100px;
width: 200px;
}
```β **Global styles ensure optimized loading** for dedicated pages.
---
### π **3. Using the Component in Another Page (`some-page.html`)**
When the same component is used **outside of its primary context** (e.g., `some-page.html`), Mesa intelligently inlines the required styles to prevent missing styles or dependency on global CSS.
**π some-page.html**
```html
```**π Build Output:**
```html
.check-inbox-page {
height: 100px;
width: 200px;
}
..
```β **Inline styles guarantee isolation** and ensure the component renders consistently even without global styles.
---
# π **Advanced Example: Multi-Package Integration with Mesa**
In this example, we'll explore how **Mesa** enables seamless integration between separate packages for **SVG icons** and **web components**, allowing clean, reusable, and optimized component architecture.
---
## π **Scenario Overview**
We have two npm packages:
1. **oicon** β A package containing SVG icons.
2. **components** β A package containing reusable UI components, including one that uses icons from **oicon**.### π **1. oicon Package**
```plaintext
oicon/
βββ package.json
βββ index.ts
βββ src/
βββ chevron-left.svg
βββ checkmark.svg
```#### π **index.ts**
```typescript
import { folder } from '@octamap/mesa';const OIconComponents = folder("./src/", {
importMetaUrl: import.meta.url, // Ensures Mesa resolves the correct paths
prefix: "oicon" // Enables usage as
});export { OIconComponents };
```With this setup:
- Icons are accessible via `` and ``.---
### π **2. Components Package**
```plaintext
components/
βββ package.json
βββ index.ts
βββ src/
βββ back-button.html
```#### π **back-button.html**
```html
.top-left-back {
position: fixed;
left: 30px;
top: 30px;
--diameter: 55px;
height: var(--diameter);
width: var(--diameter);
display: flex;
align-items: center;
justify-content: center;
border-radius: 100em;
border: none;
outline: solid 1px var(--line-color);
background-color: white;
cursor: pointer;
}```
Here:
- The **Back Button** component references ``.#### π **index.ts**
```typescript
import { folder } from '@octamap/mesa';const Components = folder("./src/", {
importMetaUrl: import.meta.url, // Correct path resolution
});export { Components };
```---
### π **3. Main Project Integration**
Now, let's integrate these packages into our **Vite** configuration.
#### π **vite.config.ts**
```typescript
import { defineConfig } from 'vite';
import { Mesa } from '@octamap/mesa';
import { OIconComponents } from '@your-package/oicon';
import { Components } from '@your-package/components';export default defineConfig({
plugins: [
Mesa({
...OIconComponents,
...Components,
}),
],
});
```---
### π **4. Using Components in Your Project**
Let's create a `check-inbox.html` page.
#### π **check-inbox.html**
```html
.check-inbox-page {
display: flex;
flex-direction: column;
align-items: center;
position: absolute;
left: 0;
top: 0;
height: 100vh;
width: 100vw;
z-index: 100;
}```
**Mesa Handles:**
1. **Scoped Styles** β `check-inbox-page` and `top-left-back` styles are scoped and applied correctly.
2. **Lazy Loading** β `check-inbox.html` styles are inlined only when the page is loaded.
3. **Optimized Rendering** β No unnecessary CSS in your `index.html`.---
## πͺ **How Mesa Optimizes Your Build**
When a user navigates to `check-inbox.html`:
1. The router dynamically loads `check-inbox.html`.
2. Mesa inlines the required CSS for `.check-inbox-page` and `.top-left-back`.
3. SVG content is embedded in the `` component.**Resulting HTML:**
```html
.check-inbox-page {
display: flex;
flex-direction: column;
align-items: center;
position: absolute;
left: 0;
top: 0;
height: 100vh;
width: 100vw;
z-index: 100;
}.top-left-back {
position: fixed;
left: 30px;
top: 30px;
--diameter: 55px;
height: var(--diameter);
width: var(--diameter);
display: flex;
align-items: center;
justify-content: center;
border-radius: 100em;
border: none;
outline: solid 1px var(--line-color);
background-color: white;
cursor: pointer;
}```
---
### π§ **How Mesa Manages Scoped Styles:**
1. **Global Styles:** For components used consistently across an entire page, styles are extracted into `component-styles.css`.
2. **Inline Styles:** When components are used in isolated contexts or ad-hoc pages, Mesa inlines the necessary styles directly.β **No CSS leakage across components.**
β **Optimal performance with minimal style duplication.**
β **Scoped styles ensure predictable rendering.**# π **Documentation & Examples**
- **Full Documentation:** [Coming Soon]
- **Starter Templates:** [Coming Soon]
- **Live Demo Projects:** [Coming Soon]---
# π§ **Why Mesa?**
β **Static-first:** Everything is rendered at build time.
β **SEO-friendly:** Full markup available to crawlers.
β **No runtime overhead:** Zero JavaScript parsing for rendering components.
β **Scalable:** Perfect for micro-frontends and component-based architectures.
β **Framework-agnostic:** Use with Alpine.js, HTMX, or vanilla JS.---
# β€οΈ **Join the Community**
- GitHub: https://github.com/octamap/mesa**Letβs redefine the way we build HTML components.** π