-
-
Notifications
You must be signed in to change notification settings - Fork 97
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
13 changed files
with
215 additions
and
19 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,6 @@ | ||
*.map | ||
coverage | ||
lib | ||
_lib | ||
node_modules | ||
tsconfig.*.tsbuildinfo | ||
tsconfig.tsbuildinfo |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -20,6 +20,5 @@ | |
}, | ||
"[typescriptreact]": { | ||
"editor.defaultFormatter": "biomejs.biome" | ||
}, | ||
"css.validate": false | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
/** @jsx jsx */ | ||
/** @jsxImportSource hono/jsx */ | ||
/** @jsxFrag */ | ||
|
||
import { Button, Framework } from "@wevm/framework"; | ||
|
||
const app = new Framework(); | ||
|
||
app.frame("/", () => { | ||
return { | ||
image: <div>hello</div>, | ||
intents: ( | ||
<> | ||
<Button>Apples</Button> | ||
<Button>Oranges</Button> | ||
</> | ||
), | ||
}; | ||
}); | ||
|
||
export default { | ||
port: 3001, | ||
fetch: app.fetch, | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
import fs from "node:fs/promises"; | ||
import path from "node:path"; | ||
import { glob } from "glob"; | ||
|
||
// Symlinks package sources to dist for local development | ||
|
||
console.log("Setting up packages for development."); | ||
|
||
// Get all package.json files | ||
const packagePaths = await glob("**/package.json", { | ||
ignore: ["**/dist/**", "**/node_modules/**"], | ||
}); | ||
|
||
let count = 0; | ||
for (const packagePath of packagePaths) { | ||
type Package = { | ||
bin?: Record<string, string> | undefined; | ||
exports?: | ||
| Record<string, { types: string; default: string } | string> | ||
| undefined; | ||
name?: string | undefined; | ||
private?: boolean | undefined; | ||
}; | ||
const file = Bun.file(packagePath); | ||
const packageJson = (await file.json()) as Package; | ||
|
||
// Skip private packages | ||
if (packageJson.private) continue; | ||
if (!packageJson.exports) continue; | ||
|
||
count += 1; | ||
console.log(`${packageJson.name} — ${path.dirname(packagePath)}`); | ||
|
||
const dir = path.resolve(path.dirname(packagePath)); | ||
|
||
// Empty dist directory | ||
const distDirName = "_lib"; | ||
const dist = path.resolve(dir, distDirName); | ||
let files: string[] = []; | ||
try { | ||
files = await fs.readdir(dist); | ||
} catch { | ||
await fs.mkdir(dist); | ||
} | ||
|
||
const promises: Promise<void>[] = []; | ||
for (const file of files) { | ||
promises.push( | ||
fs.rm(path.join(dist, file), { recursive: true, force: true }), | ||
); | ||
} | ||
await Promise.all(promises); | ||
|
||
// Link exports to dist locations | ||
for (const [key, exports] of Object.entries(packageJson.exports)) { | ||
// Skip `package.json` exports | ||
if (/package\.json$/.test(key)) continue; | ||
|
||
let entries: string[][]; | ||
if (typeof exports === "string") | ||
entries = [ | ||
["default", exports], | ||
["types", exports.replace(".js", ".d.ts")], | ||
]; | ||
else entries = Object.entries(exports); | ||
|
||
// Link exports to dist locations | ||
for (const [, value] of entries as [ | ||
type: "types" | "default", | ||
value: string, | ||
][]) { | ||
const srcDir = path.resolve( | ||
dir, | ||
path.dirname(value).replace(distDirName, ""), | ||
); | ||
let srcFileName: string; | ||
if (key === ".") srcFileName = "index.tsx"; | ||
else srcFileName = path.basename(`${key}.tsx`); | ||
const srcFilePath = path.resolve(srcDir, srcFileName); | ||
|
||
const distDir = path.resolve(dir, path.dirname(value)); | ||
const distFileName = path.basename(value); | ||
const distFilePath = path.resolve(distDir, distFileName); | ||
|
||
await fs.mkdir(distDir, { recursive: true }); | ||
|
||
// Symlink src to dist file | ||
await fs.symlink(srcFilePath, distFilePath, "file"); | ||
} | ||
} | ||
} | ||
|
||
console.log(`Done. Set up ${count} ${count === 1 ? "package" : "packages"}.`); |
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,79 @@ | ||
import { type Context, Hono } from "hono"; | ||
import { ImageResponse } from "hono-og"; | ||
import { type JSXNode } from "hono/jsx"; | ||
|
||
type FrameReturnType = { | ||
image: JSX.Element; | ||
intents: JSX.Element; | ||
}; | ||
|
||
export class Framework extends Hono { | ||
frame( | ||
path: string, | ||
handler: (c: Context) => FrameReturnType | Promise<FrameReturnType>, | ||
) { | ||
this.get(path, async (c) => { | ||
const { intents } = await handler(c); | ||
return c.render( | ||
<html lang="en"> | ||
<head> | ||
<meta property="fc:frame" content="vNext" /> | ||
<meta property="fc:frame:image" content={`${c.req.url}_og`} /> | ||
<meta property="og:image" content={`${c.req.url}_og`} /> | ||
{parseIntents(intents)} | ||
</head> | ||
</html>, | ||
); | ||
}); | ||
|
||
// TODO: don't slice | ||
this.get(`${path.slice(1)}/_og`, async (c) => { | ||
const { image } = await handler(c); | ||
return new ImageResponse(image); | ||
}); | ||
} | ||
} | ||
|
||
//////////////////////////////////////////////////////////////////////// | ||
// Components | ||
|
||
export type ButtonProps = { | ||
children: string; | ||
}; | ||
|
||
export function Button({ children }: ButtonProps) { | ||
return <meta property="fc:frame:button" content={children} />; | ||
} | ||
|
||
//////////////////////////////////////////////////////////////////////// | ||
// Utilities | ||
|
||
type Counter = { button: number }; | ||
|
||
function parseIntents(intents_: JSX.Element) { | ||
const intents = intents_ as unknown as JSXNode; | ||
const counter: Counter = { | ||
button: 0, | ||
}; | ||
|
||
if (typeof intents.children[0] === "object") | ||
return Object.assign(intents, { | ||
children: intents.children.map((e) => parseIntent(e as JSXNode, counter)), | ||
}); | ||
return parseIntent(intents, counter); | ||
} | ||
|
||
function parseIntent(node: JSXNode, counter: Counter) { | ||
const intent = ( | ||
typeof node.tag === "function" ? node.tag({}) : node | ||
) as JSXNode; | ||
|
||
const props = intent.props || {}; | ||
|
||
if (props.property === "fc:frame:button") { | ||
props.property = `fc:frame:button:${counter.button++}`; | ||
props.content = node.children; | ||
} | ||
|
||
return Object.assign(intent, { props }); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters