howl
This commit is contained in:
commit
67fae70c83
34
.gitignore
vendored
Normal file
34
.gitignore
vendored
Normal file
|
|
@ -0,0 +1,34 @@
|
||||||
|
# dependencies (bun install)
|
||||||
|
node_modules
|
||||||
|
|
||||||
|
# output
|
||||||
|
out
|
||||||
|
dist
|
||||||
|
*.tgz
|
||||||
|
|
||||||
|
# code coverage
|
||||||
|
coverage
|
||||||
|
*.lcov
|
||||||
|
|
||||||
|
# logs
|
||||||
|
logs
|
||||||
|
_.log
|
||||||
|
report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
|
||||||
|
|
||||||
|
# dotenv environment variable files
|
||||||
|
.env
|
||||||
|
.env.development.local
|
||||||
|
.env.test.local
|
||||||
|
.env.production.local
|
||||||
|
.env.local
|
||||||
|
|
||||||
|
# caches
|
||||||
|
.eslintcache
|
||||||
|
.cache
|
||||||
|
*.tsbuildinfo
|
||||||
|
|
||||||
|
# IntelliJ based IDEs
|
||||||
|
.idea
|
||||||
|
|
||||||
|
# Finder (MacOS) folder config
|
||||||
|
.DS_Store
|
||||||
111
CLAUDE.md
Normal file
111
CLAUDE.md
Normal file
|
|
@ -0,0 +1,111 @@
|
||||||
|
---
|
||||||
|
description: Use Bun instead of Node.js, npm, pnpm, or vite.
|
||||||
|
globs: "*.ts, *.tsx, *.html, *.css, *.js, *.jsx, package.json"
|
||||||
|
alwaysApply: false
|
||||||
|
---
|
||||||
|
|
||||||
|
Default to using Bun instead of Node.js.
|
||||||
|
|
||||||
|
- Use `bun <file>` instead of `node <file>` or `ts-node <file>`
|
||||||
|
- Use `bun test` instead of `jest` or `vitest`
|
||||||
|
- Use `bun build <file.html|file.ts|file.css>` instead of `webpack` or `esbuild`
|
||||||
|
- Use `bun install` instead of `npm install` or `yarn install` or `pnpm install`
|
||||||
|
- Use `bun run <script>` instead of `npm run <script>` or `yarn run <script>` or `pnpm run <script>`
|
||||||
|
- Bun automatically loads .env, so don't use dotenv.
|
||||||
|
|
||||||
|
## APIs
|
||||||
|
|
||||||
|
- `Bun.serve()` supports WebSockets, HTTPS, and routes. Don't use `express`.
|
||||||
|
- `bun:sqlite` for SQLite. Don't use `better-sqlite3`.
|
||||||
|
- `Bun.redis` for Redis. Don't use `ioredis`.
|
||||||
|
- `Bun.sql` for Postgres. Don't use `pg` or `postgres.js`.
|
||||||
|
- `WebSocket` is built-in. Don't use `ws`.
|
||||||
|
- Prefer `Bun.file` over `node:fs`'s readFile/writeFile
|
||||||
|
- Bun.$`ls` instead of execa.
|
||||||
|
|
||||||
|
## Testing
|
||||||
|
|
||||||
|
Use `bun test` to run tests.
|
||||||
|
|
||||||
|
```ts#index.test.ts
|
||||||
|
import { test, expect } from "bun:test";
|
||||||
|
|
||||||
|
test("hello world", () => {
|
||||||
|
expect(1).toBe(1);
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
## Frontend
|
||||||
|
|
||||||
|
Use HTML imports with `Bun.serve()`. Don't use `vite`. HTML imports fully support React, CSS, Tailwind.
|
||||||
|
|
||||||
|
Server:
|
||||||
|
|
||||||
|
```ts#index.ts
|
||||||
|
import index from "./index.html"
|
||||||
|
|
||||||
|
Bun.serve({
|
||||||
|
routes: {
|
||||||
|
"/": index,
|
||||||
|
"/api/users/:id": {
|
||||||
|
GET: (req) => {
|
||||||
|
return new Response(JSON.stringify({ id: req.params.id }));
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
// optional websocket support
|
||||||
|
websocket: {
|
||||||
|
open: (ws) => {
|
||||||
|
ws.send("Hello, world!");
|
||||||
|
},
|
||||||
|
message: (ws, message) => {
|
||||||
|
ws.send(message);
|
||||||
|
},
|
||||||
|
close: (ws) => {
|
||||||
|
// handle close
|
||||||
|
}
|
||||||
|
},
|
||||||
|
development: {
|
||||||
|
hmr: true,
|
||||||
|
console: true,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
```
|
||||||
|
|
||||||
|
HTML files can import .tsx, .jsx or .js files directly and Bun's bundler will transpile & bundle automatically. `<link>` tags can point to stylesheets and Bun's CSS bundler will bundle.
|
||||||
|
|
||||||
|
```html#index.html
|
||||||
|
<html>
|
||||||
|
<body>
|
||||||
|
<h1>Hello, world!</h1>
|
||||||
|
<script type="module" src="./frontend.tsx"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
```
|
||||||
|
|
||||||
|
With the following `frontend.tsx`:
|
||||||
|
|
||||||
|
```tsx#frontend.tsx
|
||||||
|
import React from "react";
|
||||||
|
|
||||||
|
// import .css files directly and it works
|
||||||
|
import './index.css';
|
||||||
|
|
||||||
|
import { createRoot } from "react-dom/client";
|
||||||
|
|
||||||
|
const root = createRoot(document.body);
|
||||||
|
|
||||||
|
export default function Frontend() {
|
||||||
|
return <h1>Hello, world!</h1>;
|
||||||
|
}
|
||||||
|
|
||||||
|
root.render(<Frontend />);
|
||||||
|
```
|
||||||
|
|
||||||
|
Then, run index.ts
|
||||||
|
|
||||||
|
```sh
|
||||||
|
bun --hot ./index.ts
|
||||||
|
```
|
||||||
|
|
||||||
|
For more information, read the Bun API docs in `node_modules/bun-types/docs/**.md`.
|
||||||
8
README.md
Normal file
8
README.md
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
# 🐺 howl
|
||||||
|
|
||||||
|
Howl is a fork of `werewolf-ui`, without any Tailwind.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
bun install
|
||||||
|
bun dev
|
||||||
|
```
|
||||||
33
bun.lock
Normal file
33
bun.lock
Normal file
|
|
@ -0,0 +1,33 @@
|
||||||
|
{
|
||||||
|
"lockfileVersion": 1,
|
||||||
|
"workspaces": {
|
||||||
|
"": {
|
||||||
|
"name": "howl",
|
||||||
|
"dependencies": {
|
||||||
|
"hono": "^4.10.7",
|
||||||
|
"lucide-static": "^0.555.0",
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/bun": "latest",
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"typescript": "^5",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"packages": {
|
||||||
|
"@types/bun": ["@types/bun@1.3.3", "", { "dependencies": { "bun-types": "1.3.3" } }, "sha512-ogrKbJ2X5N0kWLLFKeytG0eHDleBYtngtlbu9cyBKFtNL3cnpDZkNdQj8flVf6WTZUX5ulI9AY1oa7ljhSrp+g=="],
|
||||||
|
|
||||||
|
"@types/node": ["@types/node@24.10.1", "", { "dependencies": { "undici-types": "~7.16.0" } }, "sha512-GNWcUTRBgIRJD5zj+Tq0fKOJ5XZajIiBroOF0yvj2bSU1WvNdYS/dn9UxwsujGW4JX06dnHyjV2y9rRaybH0iQ=="],
|
||||||
|
|
||||||
|
"bun-types": ["bun-types@1.3.3", "", { "dependencies": { "@types/node": "*" } }, "sha512-z3Xwlg7j2l9JY27x5Qn3Wlyos8YAp0kKRlrePAOjgjMGS5IG6E7Jnlx736vH9UVI4wUICwwhC9anYL++XeOgTQ=="],
|
||||||
|
|
||||||
|
"hono": ["hono@4.10.7", "", {}, "sha512-icXIITfw/07Q88nLSkB9aiUrd8rYzSweK681Kjo/TSggaGbOX4RRyxxm71v+3PC8C/j+4rlxGeoTRxQDkaJkUw=="],
|
||||||
|
|
||||||
|
"lucide-static": ["lucide-static@0.555.0", "", {}, "sha512-FMMaYYsEYsUA6xlEzIMoKEV3oGnxIIvAN+AtLmYXvlTJptJTveJjVBQwvtA/zZLrD6KLEu89G95dQYlhivw5jQ=="],
|
||||||
|
|
||||||
|
"typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="],
|
||||||
|
|
||||||
|
"undici-types": ["undici-types@7.16.0", "", {}, "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw=="],
|
||||||
|
}
|
||||||
|
}
|
||||||
18
package.json
Normal file
18
package.json
Normal file
|
|
@ -0,0 +1,18 @@
|
||||||
|
{
|
||||||
|
"name": "howl",
|
||||||
|
"module": "index.tsx",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "bun run --hot test/server.tsx"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/bun": "latest"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"typescript": "^5"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"hono": "^4.10.7",
|
||||||
|
"lucide-static": "^0.555.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
117
src/avatar.tsx
Normal file
117
src/avatar.tsx
Normal file
|
|
@ -0,0 +1,117 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { FC, JSX } from "hono/jsx"
|
||||||
|
import { VStack, HStack } from "./stack"
|
||||||
|
|
||||||
|
export type AvatarProps = {
|
||||||
|
src: string
|
||||||
|
alt?: string
|
||||||
|
size?: number
|
||||||
|
rounded?: boolean
|
||||||
|
style?: JSX.CSSProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Avatar: FC<AvatarProps> = (props) => {
|
||||||
|
const { src, size = 32, rounded, style, alt = "" } = props
|
||||||
|
|
||||||
|
const avatarStyle: JSX.CSSProperties = {
|
||||||
|
width: `${size}px`,
|
||||||
|
height: `${size}px`,
|
||||||
|
borderRadius: rounded ? "9999px" : undefined,
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
return <img src={src} alt={alt} style={avatarStyle} />
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
const sampleImages = [
|
||||||
|
"https://picsum.photos/seed/3/200/200",
|
||||||
|
"https://picsum.photos/seed/2/200/200",
|
||||||
|
"https://picsum.photos/seed/8/200/200",
|
||||||
|
"https://picsum.photos/seed/9/200/200",
|
||||||
|
]
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VStack gap={8} style={{ padding: "24px" }}>
|
||||||
|
{/* Size variations */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Size Variations</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
{[24, 32, 48, 64, 96].map((size) => (
|
||||||
|
<VStack key={size} h="center" gap={2}>
|
||||||
|
<Avatar src={sampleImages[0]!} size={size} alt="Sample" />
|
||||||
|
<p style={{ fontSize: "14px" }}>
|
||||||
|
{size}x{size}
|
||||||
|
</p>
|
||||||
|
</VStack>
|
||||||
|
))}
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Rounded vs Square */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Rounded vs Square</h2>
|
||||||
|
<HStack gap={6}>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Avatar src={sampleImages[0]!} size={64} alt="Sample" />
|
||||||
|
<p style={{ fontSize: "14px" }}>Square</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Avatar src={sampleImages[0]!} size={64} rounded alt="Sample" />
|
||||||
|
<p style={{ fontSize: "14px" }}>Rounded</p>
|
||||||
|
</VStack>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Different images */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Different Images</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
{sampleImages.map((src, index) => (
|
||||||
|
<VStack key={src} h="center" gap={2}>
|
||||||
|
<Avatar src={src} size={64} rounded alt={`Sample ${index + 1}`} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Image {index + 1}</p>
|
||||||
|
</VStack>
|
||||||
|
))}
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Custom styles */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>With Custom Styles</h2>
|
||||||
|
<HStack gap={6}>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Avatar
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
size={64}
|
||||||
|
rounded
|
||||||
|
style={{ border: "4px solid #3b82f6" }}
|
||||||
|
alt="With border"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>With Border</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Avatar
|
||||||
|
src={sampleImages[1]!}
|
||||||
|
size={64}
|
||||||
|
rounded
|
||||||
|
style={{ boxShadow: "0 10px 15px rgba(0, 0, 0, 0.3)" }}
|
||||||
|
alt="With shadow"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>With Shadow</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Avatar
|
||||||
|
src={sampleImages[2]!}
|
||||||
|
size={64}
|
||||||
|
rounded
|
||||||
|
style={{ border: "4px solid #22c55e", boxShadow: "0 10px 15px rgba(0, 0, 0, 0.3)" }}
|
||||||
|
alt="Border + shadow"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>Border + Shadow</p>
|
||||||
|
</VStack>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
134
src/button.tsx
Normal file
134
src/button.tsx
Normal file
|
|
@ -0,0 +1,134 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { JSX, FC } from "hono/jsx"
|
||||||
|
import { VStack, HStack } from "./stack"
|
||||||
|
|
||||||
|
export type ButtonProps = JSX.IntrinsicElements["button"] & {
|
||||||
|
variant?: "primary" | "secondary" | "outline" | "ghost" | "destructive"
|
||||||
|
size?: "sm" | "md" | "lg"
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Button: FC<ButtonProps> = (props) => {
|
||||||
|
const { variant = "primary", size = "md", style, ...buttonProps } = props
|
||||||
|
|
||||||
|
const baseStyles: JSX.CSSProperties = {
|
||||||
|
display: "inline-flex",
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
fontWeight: "500",
|
||||||
|
transition: "all 0.2s",
|
||||||
|
outline: "none",
|
||||||
|
cursor: "pointer",
|
||||||
|
borderRadius: "4px",
|
||||||
|
border: "1px solid transparent",
|
||||||
|
}
|
||||||
|
|
||||||
|
const variantStyles: Record<string, JSX.CSSProperties> = {
|
||||||
|
primary: {
|
||||||
|
backgroundColor: "#3b82f6",
|
||||||
|
color: "#ffffff",
|
||||||
|
},
|
||||||
|
secondary: {
|
||||||
|
backgroundColor: "#64748b",
|
||||||
|
color: "#ffffff",
|
||||||
|
},
|
||||||
|
outline: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
color: "#000000",
|
||||||
|
borderColor: "#d1d5db",
|
||||||
|
},
|
||||||
|
ghost: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
color: "#000000",
|
||||||
|
},
|
||||||
|
destructive: {
|
||||||
|
backgroundColor: "#ef4444",
|
||||||
|
color: "#ffffff",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const sizeStyles: Record<string, JSX.CSSProperties> = {
|
||||||
|
sm: {
|
||||||
|
height: "32px",
|
||||||
|
padding: "0 12px",
|
||||||
|
fontSize: "14px",
|
||||||
|
},
|
||||||
|
md: {
|
||||||
|
height: "40px",
|
||||||
|
padding: "0 16px",
|
||||||
|
fontSize: "14px",
|
||||||
|
},
|
||||||
|
lg: {
|
||||||
|
height: "48px",
|
||||||
|
padding: "0 24px",
|
||||||
|
fontSize: "16px",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const combinedStyles: JSX.CSSProperties = {
|
||||||
|
...baseStyles,
|
||||||
|
...variantStyles[variant],
|
||||||
|
...sizeStyles[size],
|
||||||
|
...(style || {}),
|
||||||
|
}
|
||||||
|
|
||||||
|
return <button {...buttonProps} style={combinedStyles} />
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
return (
|
||||||
|
<VStack gap={8} style={{ padding: "24px" }}>
|
||||||
|
{/* Variants */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Button Variants</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
<Button variant="primary">Primary</Button>
|
||||||
|
<Button variant="secondary">Secondary</Button>
|
||||||
|
<Button variant="outline">Outline</Button>
|
||||||
|
<Button variant="ghost">Ghost</Button>
|
||||||
|
<Button variant="destructive">Destructive</Button>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Sizes */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Button Sizes</h2>
|
||||||
|
<HStack gap={4} v="end">
|
||||||
|
<Button size="sm">Small</Button>
|
||||||
|
<Button size="md">Medium</Button>
|
||||||
|
<Button size="lg">Large</Button>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* With custom content */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Custom Content</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
<Button variant="primary">
|
||||||
|
<span>🚀</span>
|
||||||
|
<span style={{ marginLeft: "8px" }}>Launch</span>
|
||||||
|
</Button>
|
||||||
|
<Button variant="outline" style={{ flexDirection: "column", height: "80px", width: "96px" }}>
|
||||||
|
<span style={{ fontSize: "24px" }}>💳</span>
|
||||||
|
<span style={{ fontSize: "12px", marginTop: "4px" }}>Card</span>
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Native attributes work */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Native Attributes</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
<Button onClick={() => alert("Clicked!")} variant="primary">
|
||||||
|
Click Me
|
||||||
|
</Button>
|
||||||
|
<Button disabled variant="secondary" style={{ opacity: 0.5, pointerEvents: "none" }}>
|
||||||
|
Disabled
|
||||||
|
</Button>
|
||||||
|
<Button type="submit" variant="outline">
|
||||||
|
Submit
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
61
src/divider.tsx
Normal file
61
src/divider.tsx
Normal file
|
|
@ -0,0 +1,61 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { FC, PropsWithChildren, JSX } from "hono/jsx"
|
||||||
|
import { VStack } from "./stack"
|
||||||
|
|
||||||
|
type DividerProps = PropsWithChildren & {
|
||||||
|
style?: JSX.CSSProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Divider: FC<DividerProps> = ({ children, style }) => {
|
||||||
|
const containerStyle: JSX.CSSProperties = {
|
||||||
|
display: "flex",
|
||||||
|
alignItems: "center",
|
||||||
|
margin: "16px 0",
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
const lineStyle: JSX.CSSProperties = {
|
||||||
|
flex: 1,
|
||||||
|
borderTop: "1px solid #d1d5db",
|
||||||
|
}
|
||||||
|
|
||||||
|
const textStyle: JSX.CSSProperties = {
|
||||||
|
padding: "0 12px",
|
||||||
|
fontSize: "14px",
|
||||||
|
color: "#6b7280",
|
||||||
|
backgroundColor: "white",
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div style={containerStyle}>
|
||||||
|
<div style={lineStyle}></div>
|
||||||
|
{children && (
|
||||||
|
<>
|
||||||
|
<span style={textStyle}>{children}</span>
|
||||||
|
<div style={lineStyle}></div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
return (
|
||||||
|
<VStack gap={4} style={{ padding: "16px", maxWidth: "448px" }}>
|
||||||
|
<h2 style={{ fontSize: "18px", fontWeight: "bold" }}>Divider Examples</h2>
|
||||||
|
|
||||||
|
<VStack gap={0}>
|
||||||
|
<p>Would you like to continue</p>
|
||||||
|
<Divider>OR WOULD YOU LIKE TO</Divider>
|
||||||
|
<p>Submit to certain death</p>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Just a line */}
|
||||||
|
<VStack gap={0}>
|
||||||
|
<p>Look a line 👇</p>
|
||||||
|
<Divider />
|
||||||
|
<p>So cool, so straight!</p>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
158
src/grid.tsx
Normal file
158
src/grid.tsx
Normal file
|
|
@ -0,0 +1,158 @@
|
||||||
|
import type { TailwindSize } from "./types"
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { FC, PropsWithChildren, JSX } from "hono/jsx"
|
||||||
|
import { VStack } from "./stack"
|
||||||
|
import { Button } from "./button"
|
||||||
|
|
||||||
|
type GridProps = PropsWithChildren & {
|
||||||
|
cols?: GridCols
|
||||||
|
gap?: TailwindSize
|
||||||
|
v?: keyof typeof alignItemsMap
|
||||||
|
h?: keyof typeof justifyItemsMap
|
||||||
|
style?: JSX.CSSProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
type GridCols = number | { sm?: number; md?: number; lg?: number; xl?: number }
|
||||||
|
|
||||||
|
export const Grid: FC<GridProps> = (props) => {
|
||||||
|
const { cols = 2, gap = 4, v, h, style, children } = props
|
||||||
|
|
||||||
|
const gapPx = gap * 4
|
||||||
|
|
||||||
|
const baseStyles: JSX.CSSProperties = {
|
||||||
|
display: "grid",
|
||||||
|
gridTemplateColumns: getColumnsValue(cols),
|
||||||
|
gap: `${gapPx}px`,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (v) {
|
||||||
|
baseStyles.alignItems = alignItemsMap[v]
|
||||||
|
}
|
||||||
|
|
||||||
|
if (h) {
|
||||||
|
baseStyles.justifyItems = justifyItemsMap[h]
|
||||||
|
}
|
||||||
|
|
||||||
|
const combinedStyles = {
|
||||||
|
...baseStyles,
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
return <div style={combinedStyles}>{children}</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
function getColumnsValue(cols: GridCols): string {
|
||||||
|
if (typeof cols === "number") {
|
||||||
|
return `repeat(${cols}, minmax(0, 1fr))`
|
||||||
|
}
|
||||||
|
|
||||||
|
// For responsive grids, we'll use the largest value
|
||||||
|
// In a real implementation, you'd want media queries which require CSS
|
||||||
|
// For now, let's use the largest value specified
|
||||||
|
const largestCols = cols.xl || cols.lg || cols.md || cols.sm || 1
|
||||||
|
return `repeat(${largestCols}, minmax(0, 1fr))`
|
||||||
|
}
|
||||||
|
|
||||||
|
const alignItemsMap = {
|
||||||
|
start: "start",
|
||||||
|
center: "center",
|
||||||
|
end: "end",
|
||||||
|
stretch: "stretch",
|
||||||
|
} as const
|
||||||
|
|
||||||
|
const justifyItemsMap = {
|
||||||
|
start: "start",
|
||||||
|
center: "center",
|
||||||
|
end: "end",
|
||||||
|
stretch: "stretch",
|
||||||
|
} as const
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
return (
|
||||||
|
<VStack gap={4} style={{ padding: "16px" }}>
|
||||||
|
<VStack gap={6}>
|
||||||
|
<h2 style={{ fontSize: "18px", fontWeight: "bold" }}>Grid Examples</h2>
|
||||||
|
|
||||||
|
{/* Simple 3-column grid */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h3 style={{ fontSize: "16px", fontWeight: "600" }}>Simple 3 columns: cols=3</h3>
|
||||||
|
<Grid cols={3} gap={4}>
|
||||||
|
<div style={{ backgroundColor: "#fecaca", padding: "16px", textAlign: "center" }}>Item 1</div>
|
||||||
|
<div style={{ backgroundColor: "#bbf7d0", padding: "16px", textAlign: "center" }}>Item 2</div>
|
||||||
|
<div style={{ backgroundColor: "#bfdbfe", padding: "16px", textAlign: "center" }}>Item 3</div>
|
||||||
|
<div style={{ backgroundColor: "#fef08a", padding: "16px", textAlign: "center" }}>Item 4</div>
|
||||||
|
<div style={{ backgroundColor: "#e9d5ff", padding: "16px", textAlign: "center" }}>Item 5</div>
|
||||||
|
<div style={{ backgroundColor: "#fbcfe8", padding: "16px", textAlign: "center" }}>Item 6</div>
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Responsive grid */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h3 style={{ fontSize: "16px", fontWeight: "600" }}>
|
||||||
|
Responsive: cols={sm: 1, md: 2, lg: 3}
|
||||||
|
</h3>
|
||||||
|
<Grid cols={{ sm: 1, md: 2, lg: 3 }} gap={4}>
|
||||||
|
<div style={{ backgroundColor: "#fecaca", padding: "16px", textAlign: "center" }}>Card 1</div>
|
||||||
|
<div style={{ backgroundColor: "#bbf7d0", padding: "16px", textAlign: "center" }}>Card 2</div>
|
||||||
|
<div style={{ backgroundColor: "#bfdbfe", padding: "16px", textAlign: "center" }}>Card 3</div>
|
||||||
|
<div style={{ backgroundColor: "#fef08a", padding: "16px", textAlign: "center" }}>Card 4</div>
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* More responsive examples */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h3 style={{ fontSize: "16px", fontWeight: "600" }}>
|
||||||
|
More responsive: cols={sm: 2, lg: 4, xl: 6}
|
||||||
|
</h3>
|
||||||
|
<Grid cols={{ sm: 2, lg: 4, xl: 6 }} gap={4}>
|
||||||
|
<div style={{ backgroundColor: "#fecaca", padding: "16px", textAlign: "center" }}>Item A</div>
|
||||||
|
<div style={{ backgroundColor: "#bbf7d0", padding: "16px", textAlign: "center" }}>Item B</div>
|
||||||
|
<div style={{ backgroundColor: "#bfdbfe", padding: "16px", textAlign: "center" }}>Item C</div>
|
||||||
|
<div style={{ backgroundColor: "#fef08a", padding: "16px", textAlign: "center" }}>Item D</div>
|
||||||
|
<div style={{ backgroundColor: "#e9d5ff", padding: "16px", textAlign: "center" }}>Item E</div>
|
||||||
|
<div style={{ backgroundColor: "#fbcfe8", padding: "16px", textAlign: "center" }}>Item F</div>
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Payment method example */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h3 style={{ fontSize: "16px", fontWeight: "600" }}>Payment buttons example</h3>
|
||||||
|
<Grid cols={3} gap={4}>
|
||||||
|
<Button variant="outline" style={{ height: "80px", flexDirection: "column" }}>
|
||||||
|
<div style={{ fontSize: "24px" }}>💳</div>
|
||||||
|
<span style={{ fontSize: "12px" }}>Card</span>
|
||||||
|
</Button>
|
||||||
|
<Button variant="outline" style={{ height: "80px", flexDirection: "column" }}>
|
||||||
|
<div style={{ fontSize: "24px" }}>🍎</div>
|
||||||
|
<span style={{ fontSize: "12px" }}>Apple</span>
|
||||||
|
</Button>
|
||||||
|
<Button variant="outline" style={{ height: "80px", flexDirection: "column" }}>
|
||||||
|
<div style={{ fontSize: "24px" }}>💰</div>
|
||||||
|
<span style={{ fontSize: "12px" }}>PayPal</span>
|
||||||
|
</Button>
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Alignment examples */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h3 style={{ fontSize: "16px", fontWeight: "600" }}>
|
||||||
|
Alignment: v="center" h="center"
|
||||||
|
</h3>
|
||||||
|
<Grid cols={3} gap={4} v="center" h="center" style={{ height: "128px", backgroundColor: "#f3f4f6" }}>
|
||||||
|
<div style={{ backgroundColor: "#fecaca", padding: "8px" }}>Item 1</div>
|
||||||
|
<div style={{ backgroundColor: "#bbf7d0", padding: "8px" }}>Item 2</div>
|
||||||
|
<div style={{ backgroundColor: "#bfdbfe", padding: "8px" }}>Item 3</div>
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h3 style={{ fontSize: "16px", fontWeight: "600" }}>Alignment: v="start" h="end"</h3>
|
||||||
|
<Grid cols={2} gap={4} v="start" h="end" style={{ height: "96px", backgroundColor: "#f3f4f6" }}>
|
||||||
|
<div style={{ backgroundColor: "#e9d5ff", padding: "8px" }}>Left</div>
|
||||||
|
<div style={{ backgroundColor: "#fed7aa", padding: "8px" }}>Right</div>
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
210
src/icon.tsx
Normal file
210
src/icon.tsx
Normal file
|
|
@ -0,0 +1,210 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { FC, JSX } from "hono/jsx"
|
||||||
|
import * as icons from "lucide-static"
|
||||||
|
import { Grid } from "./grid"
|
||||||
|
import { VStack, HStack } from "./stack"
|
||||||
|
|
||||||
|
export type IconName = keyof typeof icons
|
||||||
|
|
||||||
|
type IconProps = {
|
||||||
|
name: IconName
|
||||||
|
size?: number
|
||||||
|
class?: string
|
||||||
|
style?: JSX.CSSProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
type IconLinkProps = IconProps & {
|
||||||
|
href?: string
|
||||||
|
target?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Icon: FC<IconProps> = (props) => {
|
||||||
|
const { name, size = 6, class: className, style } = props
|
||||||
|
|
||||||
|
const iconSvg = icons[name]
|
||||||
|
|
||||||
|
if (!iconSvg) {
|
||||||
|
throw new Error(`Icon "${name}" not found in Lucide icons`)
|
||||||
|
}
|
||||||
|
|
||||||
|
const pixelSize = sizeToPixels(size)
|
||||||
|
const iconStyle: JSX.CSSProperties = {
|
||||||
|
display: "block",
|
||||||
|
flexShrink: "0",
|
||||||
|
width: `${pixelSize}px`,
|
||||||
|
height: `${pixelSize}px`,
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Modify the SVG string to include our custom attributes
|
||||||
|
const modifiedSvg = iconSvg
|
||||||
|
.replace(/width="[^"]*"/, "")
|
||||||
|
.replace(/height="[^"]*"/, "")
|
||||||
|
.replace(/class="[^"]*"/, "")
|
||||||
|
.replace(
|
||||||
|
/<svg([^>]*)>/,
|
||||||
|
`<svg$1 style="display: block; flex-shrink: 0; width: ${pixelSize}px; height: ${pixelSize}px;" class="${className || ""}">`
|
||||||
|
)
|
||||||
|
|
||||||
|
return <div dangerouslySetInnerHTML={{ __html: modifiedSvg }} style={iconStyle} />
|
||||||
|
}
|
||||||
|
|
||||||
|
export const IconLink: FC<IconLinkProps> = (props) => {
|
||||||
|
const { href = "#", target, class: className, style, ...iconProps } = props
|
||||||
|
|
||||||
|
const linkStyle: JSX.CSSProperties = {
|
||||||
|
display: "inline-flex",
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
transition: "opacity 0.2s",
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<a href={href} target={target} class={className} style={linkStyle}>
|
||||||
|
<Icon {...iconProps} />
|
||||||
|
</a>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
return (
|
||||||
|
<div style={{ padding: "24px" }}>
|
||||||
|
{/* === ICON TESTS === */}
|
||||||
|
|
||||||
|
{/* Size variations */}
|
||||||
|
<div style={{ marginBottom: "32px" }}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold", marginBottom: "16px" }}>Icon Size Variations</h2>
|
||||||
|
<div style={{ display: "flex", alignItems: "center", gap: "16px" }}>
|
||||||
|
{([3, 4, 5, 6, 8, 10, 12, 16] as const).map((size) => (
|
||||||
|
<div key={size} style={{ display: "flex", flexDirection: "column", alignItems: "center", gap: "8px" }}>
|
||||||
|
<Icon name="Heart" size={size} />
|
||||||
|
<p style={{ fontSize: "14px" }}>{size}</p>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Styling with CSS classes */}
|
||||||
|
<div style={{ marginBottom: "32px" }}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold", marginBottom: "16px" }}>Styling with CSS Classes</h2>
|
||||||
|
<Grid cols={5} gap={6}>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Star" size={12} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Default</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Star" size={12} style={{ color: "#3b82f6" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Blue Color</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Star" size={12} class="stroke-1" />
|
||||||
|
<p style={{ fontSize: "14px" }}>Thin Stroke</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Star" size={12} style={{ color: "#fbbf24", fill: "currentColor", stroke: "none" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Filled</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Star" size={12} style={{ color: "#a855f7", transition: "color 0.2s" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Hover Effect</p>
|
||||||
|
</VStack>
|
||||||
|
</Grid>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Advanced styling */}
|
||||||
|
<div style={{ marginBottom: "32px" }}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold", marginBottom: "16px" }}>Advanced Styling</h2>
|
||||||
|
<Grid cols={4} gap={6}>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Heart" size={12} style={{ color: "#ef4444", fill: "currentColor", stroke: "none" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Filled Heart</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Shield" size={12} style={{ color: "#16a34a", strokeWidth: "2" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Thick Stroke</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Sun" size={12} style={{ color: "#eab308" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Sun Icon</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Icon name="Zap" size={12} style={{ color: "#60a5fa", filter: "drop-shadow(0 4px 6px rgba(0,0,0,0.3))" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Drop Shadow</p>
|
||||||
|
</VStack>
|
||||||
|
</Grid>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* === ICON LINK TESTS === */}
|
||||||
|
|
||||||
|
{/* Basic icon links */}
|
||||||
|
<div style={{ marginBottom: "32px" }}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold", marginBottom: "16px" }}>Icon Links</h2>
|
||||||
|
<div style={{ display: "flex", gap: "24px" }}>
|
||||||
|
<div style={{ display: "flex", flexDirection: "column", alignItems: "center", gap: "8px" }}>
|
||||||
|
<IconLink name="Home" size={8} href="/" />
|
||||||
|
<p style={{ fontSize: "14px" }}>Home Link</p>
|
||||||
|
</div>
|
||||||
|
<div style={{ display: "flex", flexDirection: "column", alignItems: "center", gap: "8px" }}>
|
||||||
|
<IconLink name="ExternalLink" size={8} href="https://example.com" target="_blank" />
|
||||||
|
<p style={{ fontSize: "14px" }}>External Link</p>
|
||||||
|
</div>
|
||||||
|
<div style={{ display: "flex", flexDirection: "column", alignItems: "center", gap: "8px" }}>
|
||||||
|
<IconLink name="Mail" size={8} href="mailto:hello@example.com" />
|
||||||
|
<p style={{ fontSize: "14px" }}>Email Link</p>
|
||||||
|
</div>
|
||||||
|
<div style={{ display: "flex", flexDirection: "column", alignItems: "center", gap: "8px" }}>
|
||||||
|
<IconLink name="Phone" size={8} href="tel:+1234567890" />
|
||||||
|
<p style={{ fontSize: "14px" }}>Phone Link</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Styled icon links */}
|
||||||
|
<div>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold", marginBottom: "16px" }}>Styled Icon Links</h2>
|
||||||
|
<Grid cols={4} gap={6}>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<IconLink
|
||||||
|
name="Download"
|
||||||
|
size={8}
|
||||||
|
href="#"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "#3b82f6",
|
||||||
|
color: "white",
|
||||||
|
padding: "8px",
|
||||||
|
borderRadius: "8px",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>Button Style</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<IconLink
|
||||||
|
name="Settings"
|
||||||
|
size={8}
|
||||||
|
href="#"
|
||||||
|
style={{
|
||||||
|
border: "2px solid #d1d5db",
|
||||||
|
padding: "8px",
|
||||||
|
borderRadius: "9999px",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>Circle Border</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<IconLink name="Heart" size={8} href="#" style={{ color: "#ef4444" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Red Heart</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<IconLink name="Star" size={8} href="#" style={{ color: "#fbbf24", fill: "currentColor" }} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Filled Star</p>
|
||||||
|
</VStack>
|
||||||
|
</Grid>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function sizeToPixels(size: number): number {
|
||||||
|
return size * 4
|
||||||
|
}
|
||||||
219
src/image.tsx
Normal file
219
src/image.tsx
Normal file
|
|
@ -0,0 +1,219 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { FC, JSX } from "hono/jsx"
|
||||||
|
import { VStack, HStack } from "./stack"
|
||||||
|
import { Grid } from "./grid"
|
||||||
|
|
||||||
|
export type ImageProps = {
|
||||||
|
src: string
|
||||||
|
alt?: string
|
||||||
|
width?: number
|
||||||
|
height?: number
|
||||||
|
objectFit?: "cover" | "contain" | "fill" | "none" | "scale-down"
|
||||||
|
style?: JSX.CSSProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Image: FC<ImageProps> = ({ src, alt = "", width, height, objectFit, style }) => {
|
||||||
|
const imageStyle: JSX.CSSProperties = {
|
||||||
|
width: width ? `${width}px` : undefined,
|
||||||
|
height: height ? `${height}px` : undefined,
|
||||||
|
objectFit: objectFit,
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
return <img src={src} alt={alt} style={imageStyle} />
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
const sampleImages = [
|
||||||
|
"https://picsum.photos/seed/1/400/600", // Portrait
|
||||||
|
"https://picsum.photos/seed/2/600/400", // Landscape
|
||||||
|
"https://picsum.photos/seed/3/300/300", // Square
|
||||||
|
"https://picsum.photos/seed/4/200/100", // Small image
|
||||||
|
]
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VStack gap={8} style={{ padding: "24px" }}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Image Examples</h2>
|
||||||
|
|
||||||
|
{/* Size variations */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h3 style={{ fontSize: "18px", fontWeight: "600" }}>Size Variations</h3>
|
||||||
|
<HStack gap={4} wrap>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image src={sampleImages[0]!} width={64} height={64} objectFit="cover" alt="64x64" />
|
||||||
|
<p style={{ fontSize: "14px" }}>64x64</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image src={sampleImages[0]!} width={96} height={96} objectFit="cover" alt="96x96" />
|
||||||
|
<p style={{ fontSize: "14px" }}>96x96</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image src={sampleImages[0]!} width={128} height={128} objectFit="cover" alt="128x128" />
|
||||||
|
<p style={{ fontSize: "14px" }}>128x128</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image src={sampleImages[0]!} width={192} height={128} objectFit="cover" alt="192x128" />
|
||||||
|
<p style={{ fontSize: "14px" }}>192x128</p>
|
||||||
|
</VStack>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Object fit variations */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h3 style={{ fontSize: "18px", fontWeight: "600" }}>Object Fit Variations</h3>
|
||||||
|
<p style={{ fontSize: "14px", color: "#6b7280" }}>
|
||||||
|
Same image with different object-fit values
|
||||||
|
</p>
|
||||||
|
<HStack gap={6} wrap>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{ border: "1px solid black" }}
|
||||||
|
alt="Object cover"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>object-fit: cover</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="contain"
|
||||||
|
style={{ border: "1px solid black", backgroundColor: "#f3f4f6" }}
|
||||||
|
alt="Object contain"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>object-fit: contain</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="fill"
|
||||||
|
style={{ border: "1px solid black" }}
|
||||||
|
alt="Object fill"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>object-fit: fill</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="scale-down"
|
||||||
|
style={{ border: "1px solid black", backgroundColor: "#f3f4f6" }}
|
||||||
|
alt="Object scale-down"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>object-fit: scale-down</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="none"
|
||||||
|
style={{ border: "1px solid black", backgroundColor: "#f3f4f6" }}
|
||||||
|
alt="Object none"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>object-fit: none</p>
|
||||||
|
</VStack>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Styling examples */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h3 style={{ fontSize: "18px", fontWeight: "600" }}>Styling Examples</h3>
|
||||||
|
<HStack gap={6} wrap>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{ borderRadius: "8px", border: "4px solid #3b82f6" }}
|
||||||
|
alt="Rounded with border"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>Rounded + Border</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[1]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{ boxShadow: "0 10px 15px rgba(0, 0, 0, 0.3)" }}
|
||||||
|
alt="With shadow"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>With Shadow</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[2]!}
|
||||||
|
width={128}
|
||||||
|
height={128}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{
|
||||||
|
borderRadius: "9999px",
|
||||||
|
border: "4px solid #22c55e",
|
||||||
|
boxShadow: "0 10px 15px rgba(0, 0, 0, 0.3)",
|
||||||
|
}}
|
||||||
|
alt="Circular with effects"
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>Circular + Effects</p>
|
||||||
|
</VStack>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Common use cases */}
|
||||||
|
<VStack gap={6}>
|
||||||
|
<h3 style={{ fontSize: "18px", fontWeight: "600" }}>Common Use Cases</h3>
|
||||||
|
|
||||||
|
{/* Avatar */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h4 style={{ fontWeight: "500" }}>Avatar</h4>
|
||||||
|
<Image
|
||||||
|
src={sampleImages[0]!}
|
||||||
|
width={48}
|
||||||
|
height={48}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{ borderRadius: "9999px" }}
|
||||||
|
alt="Avatar"
|
||||||
|
/>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Card image */}
|
||||||
|
<VStack gap={2} style={{ maxWidth: "384px" }}>
|
||||||
|
<h4 style={{ fontWeight: "500" }}>Card Image</h4>
|
||||||
|
<VStack gap={0} style={{ border: "1px solid #d1d5db", borderRadius: "8px", overflow: "hidden" }}>
|
||||||
|
<Image src={sampleImages[1]!} width={384} height={192} objectFit="cover" alt="Card image" />
|
||||||
|
<VStack gap={1} style={{ padding: "16px" }}>
|
||||||
|
<h5 style={{ fontWeight: "500" }}>Card Title</h5>
|
||||||
|
<p style={{ fontSize: "14px", color: "#6b7280" }}>Card description goes here</p>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Gallery grid */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h4 style={{ fontWeight: "500" }}>Gallery Grid</h4>
|
||||||
|
<Grid cols={3} gap={2}>
|
||||||
|
{sampleImages.map((src, i) => (
|
||||||
|
<Image
|
||||||
|
key={i}
|
||||||
|
src={src}
|
||||||
|
width={120}
|
||||||
|
height={120}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{ borderRadius: "4px" }}
|
||||||
|
alt={`Gallery ${i}`}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
28
src/index.tsx
Normal file
28
src/index.tsx
Normal file
|
|
@ -0,0 +1,28 @@
|
||||||
|
export { Button } from "./button"
|
||||||
|
export type { ButtonProps } from "./button"
|
||||||
|
|
||||||
|
export { Icon, IconLink } from "./icon"
|
||||||
|
export type { IconName } from "./icon"
|
||||||
|
|
||||||
|
export { VStack, HStack } from "./stack"
|
||||||
|
|
||||||
|
export { Grid } from "./grid"
|
||||||
|
|
||||||
|
export { Divider } from "./divider"
|
||||||
|
|
||||||
|
export { Avatar } from "./avatar"
|
||||||
|
export type { AvatarProps } from "./avatar"
|
||||||
|
|
||||||
|
export { Image } from "./image"
|
||||||
|
export type { ImageProps } from "./image"
|
||||||
|
|
||||||
|
export { Input } from "./input"
|
||||||
|
export type { InputProps } from "./input"
|
||||||
|
|
||||||
|
export { Select } from "./select"
|
||||||
|
export type { SelectProps, SelectOption } from "./select"
|
||||||
|
|
||||||
|
export { Placeholder } from "./placeholder"
|
||||||
|
export { default as PlaceholderDefault } from "./placeholder"
|
||||||
|
|
||||||
|
export type { TailwindSize } from "./types"
|
||||||
181
src/input.tsx
Normal file
181
src/input.tsx
Normal file
|
|
@ -0,0 +1,181 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { JSX, FC } from "hono/jsx"
|
||||||
|
import { VStack, HStack } from "./stack"
|
||||||
|
|
||||||
|
export type InputProps = JSX.IntrinsicElements["input"] & {
|
||||||
|
labelPosition?: "above" | "left" | "right"
|
||||||
|
children?: any
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Input: FC<InputProps> = (props) => {
|
||||||
|
const { labelPosition = "above", children, style, ...inputProps } = props
|
||||||
|
|
||||||
|
const inputStyle: JSX.CSSProperties = {
|
||||||
|
height: "40px",
|
||||||
|
padding: "8px 12px",
|
||||||
|
borderRadius: "6px",
|
||||||
|
border: "1px solid #d1d5db",
|
||||||
|
backgroundColor: "white",
|
||||||
|
fontSize: "14px",
|
||||||
|
outline: "none",
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!children) {
|
||||||
|
return <input style={inputStyle} {...inputProps} />
|
||||||
|
}
|
||||||
|
|
||||||
|
const labelStyle: JSX.CSSProperties = {
|
||||||
|
fontSize: "14px",
|
||||||
|
fontWeight: "500",
|
||||||
|
color: "#111827",
|
||||||
|
}
|
||||||
|
|
||||||
|
const labelElement = (
|
||||||
|
<label for={inputProps.id} style={labelStyle}>
|
||||||
|
{children}
|
||||||
|
</label>
|
||||||
|
)
|
||||||
|
|
||||||
|
if (labelPosition === "above") {
|
||||||
|
return (
|
||||||
|
<div style={{ display: "flex", flexDirection: "column", gap: "4px", flex: 1, minWidth: 0 }}>
|
||||||
|
{labelElement}
|
||||||
|
<input style={inputStyle} {...inputProps} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (labelPosition === "left") {
|
||||||
|
return (
|
||||||
|
<div style={{ display: "flex", alignItems: "center", gap: "4px", flex: 1 }}>
|
||||||
|
{labelElement}
|
||||||
|
<input style={{ ...inputStyle, flex: 1 }} {...inputProps} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (labelPosition === "right") {
|
||||||
|
return (
|
||||||
|
<div style={{ display: "flex", alignItems: "center", gap: "4px", flex: 1 }}>
|
||||||
|
<input style={{ ...inputStyle, flex: 1 }} {...inputProps} />
|
||||||
|
{labelElement}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
return (
|
||||||
|
<VStack gap={8} style={{ padding: "24px", maxWidth: "448px" }}>
|
||||||
|
{/* Basic inputs */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Basic Inputs</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input placeholder="Enter your name" />
|
||||||
|
<Input type="email" placeholder="Enter your email" />
|
||||||
|
<Input type="password" placeholder="Enter your password" />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Custom styling */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Custom Styling</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input style={{ height: "32px", fontSize: "12px" }} placeholder="Small input" />
|
||||||
|
<Input placeholder="Default input" />
|
||||||
|
<Input style={{ height: "48px", fontSize: "16px" }} placeholder="Large input" />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* With values */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>With Values</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input value="John Doe" placeholder="Name" />
|
||||||
|
<Input type="email" value="john@example.com" placeholder="Email" />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Disabled state */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Disabled State</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input disabled placeholder="Disabled input" style={{ opacity: 0.5, cursor: "not-allowed" }} />
|
||||||
|
<Input disabled value="Disabled with value" style={{ opacity: 0.5, cursor: "not-allowed" }} />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Label above */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Label Above</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input placeholder="Enter your name">Name</Input>
|
||||||
|
<Input type="email" placeholder="Enter your email">
|
||||||
|
Email
|
||||||
|
</Input>
|
||||||
|
<Input type="password" placeholder="Enter your password">
|
||||||
|
Password
|
||||||
|
</Input>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Label to the left */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Label Left</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input labelPosition="left" placeholder="Enter your name">
|
||||||
|
Name
|
||||||
|
</Input>
|
||||||
|
<Input labelPosition="left" type="email" placeholder="Enter your email">
|
||||||
|
Email
|
||||||
|
</Input>
|
||||||
|
<Input labelPosition="left" type="password" placeholder="Enter your password">
|
||||||
|
Password
|
||||||
|
</Input>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Label to the right */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Label Right</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input labelPosition="right" placeholder="Enter your name">
|
||||||
|
Name
|
||||||
|
</Input>
|
||||||
|
<Input labelPosition="right" type="email" placeholder="Enter your email">
|
||||||
|
Email
|
||||||
|
</Input>
|
||||||
|
<Input labelPosition="right" type="password" placeholder="Enter your password">
|
||||||
|
Password
|
||||||
|
</Input>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Horizontal layout */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Horizontal Layout</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
<Input placeholder="First name">First</Input>
|
||||||
|
<Input placeholder="Last name">Last</Input>
|
||||||
|
<Input placeholder="Age">Age</Input>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Custom styling */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Custom Input Styling</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Input style={{ borderColor: "#93c5fd" }} placeholder="Custom styled input">
|
||||||
|
<span style={{ color: "#2563eb", fontWeight: "bold" }}>Custom Label</span>
|
||||||
|
</Input>
|
||||||
|
<Input labelPosition="left" placeholder="Required input">
|
||||||
|
<span style={{ color: "#dc2626", minWidth: "96px" }}>Required Field</span>
|
||||||
|
</Input>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
245
src/placeholder.tsx
Normal file
245
src/placeholder.tsx
Normal file
|
|
@ -0,0 +1,245 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import { Avatar } from "./avatar"
|
||||||
|
import type { AvatarProps } from "./avatar"
|
||||||
|
import { Image } from "./image"
|
||||||
|
import type { ImageProps } from "./image"
|
||||||
|
import { VStack, HStack } from "./stack"
|
||||||
|
import { Grid } from "./grid"
|
||||||
|
|
||||||
|
export const Placeholder = {
|
||||||
|
Avatar(props: PlaceholderAvatarProps) {
|
||||||
|
const { size = 32, seed = "seed", type = "dylan", transparent, alt, style, rounded } = props
|
||||||
|
|
||||||
|
// Generate DiceBear avatar URL
|
||||||
|
const url = new URL(`https://api.dicebear.com/9.x/${type}/svg`)
|
||||||
|
url.searchParams.set("seed", seed)
|
||||||
|
url.searchParams.set("size", size.toString())
|
||||||
|
|
||||||
|
if (transparent) {
|
||||||
|
url.searchParams.set("backgroundColor", "transparent")
|
||||||
|
}
|
||||||
|
|
||||||
|
return <Avatar src={url.toString()} alt={alt} style={style} size={size} rounded={rounded} />
|
||||||
|
},
|
||||||
|
|
||||||
|
Image(props: PlaceholderImageProps) {
|
||||||
|
const { width = 200, height = 200, seed = 1, alt = "Placeholder image", objectFit, style } = props
|
||||||
|
|
||||||
|
// Generate Picsum Photos URL with seed for consistent images
|
||||||
|
const src = `https://picsum.photos/${width}/${height}?random=${seed}`
|
||||||
|
|
||||||
|
return <Image src={src} alt={alt} width={width} height={height} objectFit={objectFit} style={style} />
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
return (
|
||||||
|
<VStack gap={8} style={{ padding: "24px" }}>
|
||||||
|
{/* === AVATAR TESTS === */}
|
||||||
|
|
||||||
|
{/* Show all available avatar styles */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>
|
||||||
|
All Avatar Styles ({allStyles.length} total)
|
||||||
|
</h2>
|
||||||
|
<Grid cols={10} gap={3}>
|
||||||
|
{allStyles.map((style) => (
|
||||||
|
<VStack h="center" gap={1} key={style}>
|
||||||
|
<Placeholder.Avatar type={style} size={48} />
|
||||||
|
<p style={{ fontSize: "12px", fontWeight: "500" }}>{style}</p>
|
||||||
|
</VStack>
|
||||||
|
))}
|
||||||
|
</Grid>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Avatar size variations */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Avatar Size Variations</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
{[24, 32, 48, 64].map((size) => (
|
||||||
|
<VStack h="center" gap={2} key={size}>
|
||||||
|
<Placeholder.Avatar size={size} />
|
||||||
|
<p style={{ fontSize: "14px" }}>{size}px</p>
|
||||||
|
</VStack>
|
||||||
|
))}
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Avatar styling combinations */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Avatar Styling Options</h2>
|
||||||
|
<HStack gap={6}>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Placeholder.Avatar rounded size={64} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Rounded + Background</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<div style={{ backgroundColor: "#e5e7eb", padding: "8px" }}>
|
||||||
|
<Placeholder.Avatar rounded transparent size={64} />
|
||||||
|
</div>
|
||||||
|
<p style={{ fontSize: "14px" }}>Rounded + Transparent</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Placeholder.Avatar size={64} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Square + Background</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<div style={{ backgroundColor: "#e5e7eb", padding: "8px" }}>
|
||||||
|
<Placeholder.Avatar transparent size={64} />
|
||||||
|
</div>
|
||||||
|
<p style={{ fontSize: "14px" }}>Square + Transparent</p>
|
||||||
|
</VStack>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Avatar seed variations */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>
|
||||||
|
Avatar Seeds (Same Style, Different People)
|
||||||
|
</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
{["alice", "bob", "charlie", "diana"].map((seed) => (
|
||||||
|
<VStack h="center" gap={2} key={seed}>
|
||||||
|
<Placeholder.Avatar seed={seed} size={64} />
|
||||||
|
<p style={{ fontSize: "14px" }}>"{seed}"</p>
|
||||||
|
</VStack>
|
||||||
|
))}
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* === IMAGE TESTS === */}
|
||||||
|
<VStack gap={6}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Placeholder Images</h2>
|
||||||
|
|
||||||
|
{/* Size variations */}
|
||||||
|
<VStack gap={3}>
|
||||||
|
<h3 style={{ fontSize: "18px", fontWeight: "600" }}>Size Variations</h3>
|
||||||
|
<HStack gap={4}>
|
||||||
|
{[
|
||||||
|
{ width: 100, height: 100 },
|
||||||
|
{ width: 150, height: 100 },
|
||||||
|
{ width: 200, height: 150 },
|
||||||
|
{ width: 250, height: 200 },
|
||||||
|
].map(({ width, height }) => (
|
||||||
|
<VStack h="center" gap={2} key={`${width}x${height}`}>
|
||||||
|
<Placeholder.Image width={width} height={height} seed={1} />
|
||||||
|
<p style={{ fontSize: "14px" }}>
|
||||||
|
{width}×{height}
|
||||||
|
</p>
|
||||||
|
</VStack>
|
||||||
|
))}
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Different seeds - show variety */}
|
||||||
|
<VStack gap={3}>
|
||||||
|
<h3 style={{ fontSize: "18px", fontWeight: "600" }}>
|
||||||
|
Different Images (Different Seeds)
|
||||||
|
</h3>
|
||||||
|
<HStack gap={4}>
|
||||||
|
{[1, 2, 3, 4, 5].map((seed) => (
|
||||||
|
<VStack h="center" gap={2} key={seed}>
|
||||||
|
<Placeholder.Image width={150} height={150} seed={seed} />
|
||||||
|
<p style={{ fontSize: "14px" }}>Seed {seed}</p>
|
||||||
|
</VStack>
|
||||||
|
))}
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* With custom styles */}
|
||||||
|
<VStack gap={3}>
|
||||||
|
<h3 style={{ fontSize: "18px", fontWeight: "600" }}>With Custom Styles</h3>
|
||||||
|
<HStack gap={6}>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Placeholder.Image
|
||||||
|
width={150}
|
||||||
|
height={150}
|
||||||
|
seed={1}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{ borderRadius: "8px", border: "4px solid #3b82f6" }}
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>Rounded + Border</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Placeholder.Image
|
||||||
|
width={150}
|
||||||
|
height={150}
|
||||||
|
seed={2}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{ boxShadow: "0 10px 15px rgba(0, 0, 0, 0.3)" }}
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>With Shadow</p>
|
||||||
|
</VStack>
|
||||||
|
<VStack h="center" gap={2}>
|
||||||
|
<Placeholder.Image
|
||||||
|
width={150}
|
||||||
|
height={150}
|
||||||
|
seed={3}
|
||||||
|
objectFit="cover"
|
||||||
|
style={{
|
||||||
|
borderRadius: "9999px",
|
||||||
|
border: "4px solid #22c55e",
|
||||||
|
boxShadow: "0 10px 15px rgba(0, 0, 0, 0.3)",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<p style={{ fontSize: "14px" }}>Circular + Effects</p>
|
||||||
|
</VStack>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Type definitions
|
||||||
|
type PlaceholderAvatarProps = Omit<AvatarProps, "src"> & {
|
||||||
|
seed?: string
|
||||||
|
type?: DicebearStyleName
|
||||||
|
transparent?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
type PlaceholderImageProps = Omit<ImageProps, "src" | "alt"> & {
|
||||||
|
width?: number
|
||||||
|
height?: number
|
||||||
|
seed?: number
|
||||||
|
alt?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
// All supported DiceBear HTTP styleNames. Source: https://www.dicebear.com/styles
|
||||||
|
const allStyles = [
|
||||||
|
"adventurer",
|
||||||
|
"adventurer-neutral",
|
||||||
|
"avataaars",
|
||||||
|
"avataaars-neutral",
|
||||||
|
"big-ears",
|
||||||
|
"big-ears-neutral",
|
||||||
|
"big-smile",
|
||||||
|
"bottts",
|
||||||
|
"bottts-neutral",
|
||||||
|
"croodles",
|
||||||
|
"croodles-neutral",
|
||||||
|
"dylan",
|
||||||
|
"fun-emoji",
|
||||||
|
"glass",
|
||||||
|
"icons",
|
||||||
|
"identicon",
|
||||||
|
"initials",
|
||||||
|
"lorelei",
|
||||||
|
"lorelei-neutral",
|
||||||
|
"micah",
|
||||||
|
"miniavs",
|
||||||
|
"notionists",
|
||||||
|
"notionists-neutral",
|
||||||
|
"open-peeps",
|
||||||
|
"personas",
|
||||||
|
"pixel-art",
|
||||||
|
"pixel-art-neutral",
|
||||||
|
"rings",
|
||||||
|
"shapes",
|
||||||
|
"thumbs",
|
||||||
|
] as const
|
||||||
|
|
||||||
|
type DicebearStyleName = (typeof allStyles)[number]
|
||||||
|
|
||||||
|
// Default export for convenience
|
||||||
|
export default Placeholder
|
||||||
263
src/select.tsx
Normal file
263
src/select.tsx
Normal file
|
|
@ -0,0 +1,263 @@
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { JSX, FC } from "hono/jsx"
|
||||||
|
import { VStack, HStack } from "./stack"
|
||||||
|
|
||||||
|
export type SelectOption = {
|
||||||
|
value: string
|
||||||
|
label: string
|
||||||
|
disabled?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SelectProps = Omit<JSX.IntrinsicElements["select"], "children"> & {
|
||||||
|
options: SelectOption[]
|
||||||
|
placeholder?: string
|
||||||
|
labelPosition?: "above" | "left" | "right"
|
||||||
|
children?: any
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Select: FC<SelectProps> = (props) => {
|
||||||
|
const { options, placeholder, labelPosition = "above", children, style, ...selectProps } = props
|
||||||
|
|
||||||
|
// If a label is provided but no id, generate a random id so the label can be clicked
|
||||||
|
if (children && !selectProps.id) {
|
||||||
|
selectProps.id = `random-${Math.random().toString(36)}`
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectStyle: JSX.CSSProperties = {
|
||||||
|
height: "40px",
|
||||||
|
padding: "8px 32px 8px 12px",
|
||||||
|
borderRadius: "6px",
|
||||||
|
border: "1px solid #d1d5db",
|
||||||
|
backgroundColor: "white",
|
||||||
|
fontSize: "14px",
|
||||||
|
outline: "none",
|
||||||
|
appearance: "none",
|
||||||
|
backgroundImage: `url("")`,
|
||||||
|
backgroundRepeat: "no-repeat",
|
||||||
|
backgroundPosition: "right 8px center",
|
||||||
|
backgroundSize: "16px 16px",
|
||||||
|
...style,
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectElement = (
|
||||||
|
<select style={selectStyle} {...selectProps}>
|
||||||
|
{placeholder && (
|
||||||
|
<option value="" disabled>
|
||||||
|
{placeholder}
|
||||||
|
</option>
|
||||||
|
)}
|
||||||
|
{options.map((option) => (
|
||||||
|
<option
|
||||||
|
key={option.value}
|
||||||
|
value={option.value}
|
||||||
|
disabled={option.disabled}
|
||||||
|
selected={selectProps.value === option.value}
|
||||||
|
>
|
||||||
|
{option.label}
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
)
|
||||||
|
|
||||||
|
if (!children) {
|
||||||
|
return selectElement
|
||||||
|
}
|
||||||
|
|
||||||
|
const labelStyle: JSX.CSSProperties = {
|
||||||
|
fontSize: "14px",
|
||||||
|
fontWeight: "500",
|
||||||
|
color: "#111827",
|
||||||
|
}
|
||||||
|
|
||||||
|
const labelElement = (
|
||||||
|
<label for={selectProps.id} style={labelStyle}>
|
||||||
|
{children}
|
||||||
|
</label>
|
||||||
|
)
|
||||||
|
|
||||||
|
if (labelPosition === "above") {
|
||||||
|
return (
|
||||||
|
<div style={{ display: "flex", flexDirection: "column", gap: "4px", flex: 1, minWidth: 0 }}>
|
||||||
|
{labelElement}
|
||||||
|
{selectElement}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (labelPosition === "left") {
|
||||||
|
return (
|
||||||
|
<div style={{ display: "flex", alignItems: "center", gap: "4px", flex: 1 }}>
|
||||||
|
{labelElement}
|
||||||
|
<select style={{ ...selectStyle, flex: 1 }} {...selectProps}>
|
||||||
|
{placeholder && (
|
||||||
|
<option value="" disabled>
|
||||||
|
{placeholder}
|
||||||
|
</option>
|
||||||
|
)}
|
||||||
|
{options.map((option) => (
|
||||||
|
<option key={option.value} value={option.value} disabled={option.disabled}>
|
||||||
|
{option.label}
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (labelPosition === "right") {
|
||||||
|
return (
|
||||||
|
<div style={{ display: "flex", alignItems: "center", gap: "4px", flex: 1 }}>
|
||||||
|
<select style={{ ...selectStyle, flex: 1 }} {...selectProps}>
|
||||||
|
{placeholder && (
|
||||||
|
<option value="" disabled>
|
||||||
|
{placeholder}
|
||||||
|
</option>
|
||||||
|
)}
|
||||||
|
{options.map((option) => (
|
||||||
|
<option key={option.value} value={option.value} disabled={option.disabled}>
|
||||||
|
{option.label}
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
{labelElement}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
const months = [
|
||||||
|
{ value: "01", label: "January" },
|
||||||
|
{ value: "02", label: "February" },
|
||||||
|
{ value: "03", label: "March" },
|
||||||
|
{ value: "04", label: "April" },
|
||||||
|
{ value: "05", label: "May" },
|
||||||
|
{ value: "06", label: "June" },
|
||||||
|
{ value: "07", label: "July" },
|
||||||
|
{ value: "08", label: "August" },
|
||||||
|
{ value: "09", label: "September" },
|
||||||
|
{ value: "10", label: "October" },
|
||||||
|
{ value: "11", label: "November" },
|
||||||
|
{ value: "12", label: "December" },
|
||||||
|
]
|
||||||
|
|
||||||
|
const years = Array.from({ length: 10 }, (_, i) => ({
|
||||||
|
value: String(2024 + i),
|
||||||
|
label: String(2024 + i),
|
||||||
|
}))
|
||||||
|
|
||||||
|
const countries = [
|
||||||
|
{ value: "us", label: "United States" },
|
||||||
|
{ value: "ca", label: "Canada" },
|
||||||
|
{ value: "uk", label: "United Kingdom" },
|
||||||
|
{ value: "de", label: "Germany" },
|
||||||
|
{ value: "fr", label: "France" },
|
||||||
|
{ value: "au", label: "Australia", disabled: true },
|
||||||
|
]
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VStack gap={8} style={{ padding: "24px", maxWidth: "448px" }}>
|
||||||
|
{/* Basic selects */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Basic Selects</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Select options={months} placeholder="Select month" />
|
||||||
|
<Select options={years} placeholder="Select year" />
|
||||||
|
<Select options={countries} placeholder="Select country" />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* With values */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>With Values</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Select options={months} value="03" />
|
||||||
|
<Select options={years} value="2025" />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Disabled state */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Disabled State</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Select
|
||||||
|
options={months}
|
||||||
|
disabled
|
||||||
|
placeholder="Disabled select"
|
||||||
|
style={{ opacity: 0.5, cursor: "not-allowed" }}
|
||||||
|
/>
|
||||||
|
<Select options={years} disabled value="2024" style={{ opacity: 0.5, cursor: "not-allowed" }} />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Custom styling */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Custom Styling</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Select options={countries} style={{ borderColor: "#93c5fd" }} placeholder="Custom styled select" />
|
||||||
|
<Select options={months} style={{ height: "32px", fontSize: "12px" }} placeholder="Small select" />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Label above */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Label Above</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Select options={months} placeholder="Select month">
|
||||||
|
Birth Month
|
||||||
|
</Select>
|
||||||
|
<Select options={years} placeholder="Select year">
|
||||||
|
Birth Year
|
||||||
|
</Select>
|
||||||
|
<Select options={countries} placeholder="Select country">
|
||||||
|
Country
|
||||||
|
</Select>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Label to the left */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Label Left</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Select labelPosition="left" options={months} placeholder="Select month">
|
||||||
|
Month
|
||||||
|
</Select>
|
||||||
|
<Select labelPosition="left" options={years} placeholder="Select year">
|
||||||
|
Year
|
||||||
|
</Select>
|
||||||
|
<Select labelPosition="left" options={countries} placeholder="Select country">
|
||||||
|
Country
|
||||||
|
</Select>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Label to the right */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Label Right</h2>
|
||||||
|
<VStack gap={4}>
|
||||||
|
<Select labelPosition="right" options={months} placeholder="Select month">
|
||||||
|
Month
|
||||||
|
</Select>
|
||||||
|
<Select labelPosition="right" options={years} placeholder="Select year">
|
||||||
|
Year
|
||||||
|
</Select>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* Horizontal layout (like card form) */}
|
||||||
|
<VStack gap={4}>
|
||||||
|
<h2 style={{ fontSize: "20px", fontWeight: "bold" }}>Horizontal Layout</h2>
|
||||||
|
<HStack gap={4}>
|
||||||
|
<Select options={months} placeholder="MM">
|
||||||
|
Expires
|
||||||
|
</Select>
|
||||||
|
<Select options={years} placeholder="YYYY">
|
||||||
|
Year
|
||||||
|
</Select>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
193
src/stack.tsx
Normal file
193
src/stack.tsx
Normal file
|
|
@ -0,0 +1,193 @@
|
||||||
|
import type { TailwindSize } from "./types"
|
||||||
|
import "hono/jsx"
|
||||||
|
import type { FC, PropsWithChildren, JSX } from "hono/jsx"
|
||||||
|
import { Grid } from "./grid"
|
||||||
|
|
||||||
|
export const VStack: FC<VStackProps> = (props) => {
|
||||||
|
return (
|
||||||
|
<Stack
|
||||||
|
direction="col"
|
||||||
|
mainAxis={props.v}
|
||||||
|
crossAxis={props.h}
|
||||||
|
wrap={props.wrap}
|
||||||
|
gap={props.gap}
|
||||||
|
style={props.style}
|
||||||
|
>
|
||||||
|
{props.children}
|
||||||
|
</Stack>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const HStack: FC<HStackProps> = (props) => {
|
||||||
|
return (
|
||||||
|
<Stack
|
||||||
|
direction="row"
|
||||||
|
mainAxis={props.h}
|
||||||
|
crossAxis={props.v}
|
||||||
|
wrap={props.wrap}
|
||||||
|
gap={props.gap}
|
||||||
|
style={props.style}
|
||||||
|
>
|
||||||
|
{props.children}
|
||||||
|
</Stack>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const Stack: FC<StackProps> = (props) => {
|
||||||
|
const gapPx = props.gap ? props.gap * 4 : 0
|
||||||
|
|
||||||
|
const baseStyles: JSX.CSSProperties = {
|
||||||
|
display: "flex",
|
||||||
|
flexDirection: props.direction === "row" ? "row" : "column",
|
||||||
|
flexWrap: props.wrap ? "wrap" : "nowrap",
|
||||||
|
gap: `${gapPx}px`,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.mainAxis) {
|
||||||
|
baseStyles.justifyContent = getJustifyContent(props.mainAxis)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.crossAxis) {
|
||||||
|
baseStyles.alignItems = getAlignItems(props.crossAxis)
|
||||||
|
}
|
||||||
|
|
||||||
|
const combinedStyles = {
|
||||||
|
...baseStyles,
|
||||||
|
...props.style,
|
||||||
|
}
|
||||||
|
|
||||||
|
return <div style={combinedStyles}>{props.children}</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Test = () => {
|
||||||
|
const mainAxisOpts: MainAxisOpts[] = ["start", "center", "end", "between", "around", "evenly"]
|
||||||
|
const crossAxisOpts: CrossAxisOpts[] = ["start", "center", "end", "stretch", "baseline"]
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VStack gap={8} style={{ padding: "16px" }}>
|
||||||
|
{/* HStack layout matrix */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h2 style={{ fontSize: "18px", fontWeight: "bold" }}>HStack Layout</h2>
|
||||||
|
<div style={{ overflowX: "auto" }}>
|
||||||
|
<Grid cols={7} gap={1} style={{ gridTemplateColumns: "auto repeat(6, 1fr)" }}>
|
||||||
|
{/* Header row: blank + h labels */}
|
||||||
|
<div></div>
|
||||||
|
{mainAxisOpts.map((h) => (
|
||||||
|
<div key={h} style={{ fontSize: "14px", fontWeight: "500", textAlign: "center" }}>
|
||||||
|
h: {h}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
|
||||||
|
{/* Each row: v label + HStack cells */}
|
||||||
|
{crossAxisOpts.map((v) => [
|
||||||
|
<div key={v} style={{ fontSize: "14px", fontWeight: "500" }}>
|
||||||
|
v: {v}
|
||||||
|
</div>,
|
||||||
|
...mainAxisOpts.map((h) => (
|
||||||
|
<HStack
|
||||||
|
key={`${h}-${v}`}
|
||||||
|
h={h}
|
||||||
|
v={v}
|
||||||
|
style={{ backgroundColor: "#f3f4f6", padding: "8px", height: "96px", border: "1px solid #9ca3af" }}
|
||||||
|
>
|
||||||
|
<div style={{ textAlign: "center", padding: "4px", backgroundColor: "#ef4444" }}>Aa</div>
|
||||||
|
<div style={{ textAlign: "center", padding: "4px", backgroundColor: "#22c55e" }}>Aa</div>
|
||||||
|
<div style={{ textAlign: "center", padding: "4px", backgroundColor: "#3b82f6" }}>Aa</div>
|
||||||
|
</HStack>
|
||||||
|
)),
|
||||||
|
])}
|
||||||
|
</Grid>
|
||||||
|
</div>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
{/* VStack layout matrix */}
|
||||||
|
<VStack gap={2}>
|
||||||
|
<h2 style={{ fontSize: "18px", fontWeight: "bold" }}>VStack Layout</h2>
|
||||||
|
<div style={{ overflowX: "auto" }}>
|
||||||
|
<Grid cols={6} gap={1} style={{ gridTemplateColumns: "auto repeat(5, 1fr)" }}>
|
||||||
|
{/* Header row: blank + h labels */}
|
||||||
|
<div></div>
|
||||||
|
{crossAxisOpts.map((h) => (
|
||||||
|
<div key={h} style={{ fontSize: "14px", fontWeight: "500", textAlign: "center" }}>
|
||||||
|
h: {h}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
|
||||||
|
{/* Each row: v label + VStack cells */}
|
||||||
|
{mainAxisOpts.map((v) => [
|
||||||
|
<div key={v} style={{ fontSize: "14px", fontWeight: "500" }}>
|
||||||
|
v: {v}
|
||||||
|
</div>,
|
||||||
|
...crossAxisOpts.map((h) => (
|
||||||
|
<VStack
|
||||||
|
key={`${h}-${v}`}
|
||||||
|
v={v}
|
||||||
|
h={h}
|
||||||
|
style={{ backgroundColor: "#f3f4f6", padding: "8px", height: "168px", border: "1px solid #9ca3af" }}
|
||||||
|
>
|
||||||
|
<div style={{ textAlign: "center", padding: "4px", backgroundColor: "#ef4444" }}>Aa</div>
|
||||||
|
<div style={{ textAlign: "center", padding: "4px", backgroundColor: "#22c55e" }}>Aa</div>
|
||||||
|
<div style={{ textAlign: "center", padding: "4px", backgroundColor: "#3b82f6" }}>Aa</div>
|
||||||
|
</VStack>
|
||||||
|
)),
|
||||||
|
])}
|
||||||
|
</Grid>
|
||||||
|
</div>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type StackDirection = "row" | "col"
|
||||||
|
|
||||||
|
type StackProps = {
|
||||||
|
direction: StackDirection
|
||||||
|
mainAxis?: string
|
||||||
|
crossAxis?: string
|
||||||
|
wrap?: boolean
|
||||||
|
gap?: TailwindSize
|
||||||
|
style?: JSX.CSSProperties
|
||||||
|
children?: any
|
||||||
|
}
|
||||||
|
|
||||||
|
type MainAxisOpts = "start" | "center" | "end" | "between" | "around" | "evenly"
|
||||||
|
type CrossAxisOpts = "start" | "center" | "end" | "stretch" | "baseline"
|
||||||
|
|
||||||
|
type CommonStackProps = PropsWithChildren & {
|
||||||
|
wrap?: boolean
|
||||||
|
gap?: TailwindSize
|
||||||
|
style?: JSX.CSSProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
type VStackProps = CommonStackProps & {
|
||||||
|
v?: MainAxisOpts // main axis for vertical stack
|
||||||
|
h?: CrossAxisOpts // cross axis for vertical stack
|
||||||
|
}
|
||||||
|
|
||||||
|
type HStackProps = CommonStackProps & {
|
||||||
|
h?: MainAxisOpts // main axis for horizontal stack
|
||||||
|
v?: CrossAxisOpts // cross axis for horizontal stack
|
||||||
|
}
|
||||||
|
|
||||||
|
function getJustifyContent(axis: string): string {
|
||||||
|
const map: Record<string, string> = {
|
||||||
|
start: "flex-start",
|
||||||
|
center: "center",
|
||||||
|
end: "flex-end",
|
||||||
|
between: "space-between",
|
||||||
|
around: "space-around",
|
||||||
|
evenly: "space-evenly",
|
||||||
|
}
|
||||||
|
return map[axis] || "flex-start"
|
||||||
|
}
|
||||||
|
|
||||||
|
function getAlignItems(axis: string): string {
|
||||||
|
const map: Record<string, string> = {
|
||||||
|
start: "flex-start",
|
||||||
|
center: "center",
|
||||||
|
end: "flex-end",
|
||||||
|
stretch: "stretch",
|
||||||
|
baseline: "baseline",
|
||||||
|
}
|
||||||
|
return map[axis] || "stretch"
|
||||||
|
}
|
||||||
1
src/types.ts
Normal file
1
src/types.ts
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
export type TailwindSize = 0 | 0.5 | 1 | 1.5 | 2 | 2.5 | 3 | 3.5 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 14 | 16 | 20 | 24 | 28 | 32 | 36 | 40 | 44 | 48 | 52 | 56 | 60 | 64 | 72 | 80 | 96
|
||||||
37
test/server.tsx
Normal file
37
test/server.tsx
Normal file
|
|
@ -0,0 +1,37 @@
|
||||||
|
import { Hono } from 'hono'
|
||||||
|
import { readdirSync } from 'fs'
|
||||||
|
import { join } from 'path'
|
||||||
|
|
||||||
|
const port = process.env.PORT ?? '3100'
|
||||||
|
const app = new Hono()
|
||||||
|
|
||||||
|
app.get('/:file', async c => {
|
||||||
|
const file = c.req.param('file') ?? ''
|
||||||
|
const fileName = (file).replace('.', '')
|
||||||
|
const path = join(process.env.PWD ?? '.', `/src/${fileName}.tsx`)
|
||||||
|
|
||||||
|
if (!(await Bun.file(path).exists()))
|
||||||
|
return c.text('404 Not Found', 404)
|
||||||
|
|
||||||
|
const page = await import(path + `?t=${Date.now()}`)
|
||||||
|
return c.html(<><h1>{file}</h1><page.Test req={c.req} /></>)
|
||||||
|
})
|
||||||
|
|
||||||
|
app.get('/', c => {
|
||||||
|
return c.html(<>
|
||||||
|
<h1>Test Files</h1>
|
||||||
|
<ul style='font-size:150%'>{testFiles().map(x => <li><a href={`/${x}`}>{x}</a></li>)}</ul>
|
||||||
|
</>)
|
||||||
|
})
|
||||||
|
|
||||||
|
function testFiles(): string[] {
|
||||||
|
return readdirSync('./test')
|
||||||
|
.filter(x => x.endsWith('.tsx') && !x.startsWith('server'))
|
||||||
|
.map(x => x.replace('.tsx', ''))
|
||||||
|
.sort()
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
fetch: app.fetch,
|
||||||
|
port
|
||||||
|
}
|
||||||
36
tsconfig.json
Normal file
36
tsconfig.json
Normal file
|
|
@ -0,0 +1,36 @@
|
||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
// Environment setup & latest features
|
||||||
|
"lib": [
|
||||||
|
"ESNext",
|
||||||
|
"DOM"
|
||||||
|
],
|
||||||
|
"target": "ESNext",
|
||||||
|
"module": "Preserve",
|
||||||
|
"moduleDetection": "force",
|
||||||
|
"jsx": "react-jsx",
|
||||||
|
"jsxImportSource": "hono/jsx",
|
||||||
|
"allowJs": true,
|
||||||
|
// Bundler mode
|
||||||
|
"moduleResolution": "bundler",
|
||||||
|
"allowImportingTsExtensions": true,
|
||||||
|
"verbatimModuleSyntax": true,
|
||||||
|
"noEmit": true,
|
||||||
|
// Best practices
|
||||||
|
"strict": true,
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
"noUncheckedIndexedAccess": true,
|
||||||
|
"noImplicitOverride": true,
|
||||||
|
// Some stricter flags (disabled by default)
|
||||||
|
"noUnusedLocals": false,
|
||||||
|
"noUnusedParameters": false,
|
||||||
|
"noPropertyAccessFromIndexSignature": false,
|
||||||
|
"baseUrl": ".",
|
||||||
|
"paths": {
|
||||||
|
"#*": [
|
||||||
|
"src/*"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue
Block a user