Client Components
hono/jsx
不仅支持服务端,还支持客户端。这意味着可以创建在浏览器中运行的交互式 UI。我们称之为客户端组件或 hono/jsx/dom
。
它速度快且体积非常小。使用 Brotli 压缩后,hono/jsx/dom
中的计数器程序仅为 2.8KB。相比之下,React 为 47.8KB。
本节将介绍客户端组件的特定功能。
计数器示例
这是一个简单的计数器示例,相同的代码也适用于 React。
import { useState } from 'hono/jsx'
import { render } from 'hono/jsx/dom'
function Counter() {
const [count, setCount] = useState(0)
return (
<div>
<p>Count: {count}</p>
<button onClick={() => setCount(count + 1)}>Increment</button>
</div>
)
}
function App() {
return (
<html>
<body>
<Counter />
</body>
</html>
)
}
const root = document.getElementById('root')
render(<App />, root)
render()
你可以使用 render()
函数将 JSX 组件插入到指定的 HTML 元素中。
render(<Component />, container)
兼容 React 的 Hooks
hono/jsx/dom 拥有与 React 兼容或部分兼容的 Hooks。你可以通过查看 React 文档 了解这些 API。
useState()
useEffect()
useRef()
useCallback()
use()
startTransition()
useTransition()
useDeferredValue()
useMemo()
useLayoutEffect()
useReducer()
useDebugValue()
createElement()
memo()
isValidElement()
useId()
createRef()
forwardRef()
useImperativeHandle()
useSyncExternalStore()
useInsertionEffect()
useFormStatus()
useActionState()
useOptimistic()
startViewTransition()
系列
startViewTransition()
系列包含原创的 hooks 和函数,可以轻松处理 View Transitions API。以下是如何使用它们的示例。
1. 最简单的示例
你可以使用 startViewTransition()
通过简短的 document.startViewTransition
来编写过渡效果。
import { useState, startViewTransition } from 'hono/jsx'
import { css, Style } from 'hono/css'
export default function App() {
const [showLargeImage, setShowLargeImage] = useState(false)
return (
<>
<Style />
<button
onClick={() =>
startViewTransition(() =>
setShowLargeImage((state) => !state)
)
}
>
Click!
</button>
<div>
{!showLargeImage ? (
<img src='https://hono-dev.myedgetech.com/images/logo.png' />
) : (
<div
class={css`
background: url('https://hono-dev.myedgetech.com/images/logo-large.png');
background-size: contain;
background-repeat: no-repeat;
background-position: center;
width: 600px;
height: 600px;
`}
></div>
)}
</div>
</>
)
}
2. 将 viewTransition()
与 keyframes()
结合使用
viewTransition()
函数允许你获取唯一的 view-transition-name
。
你可以将其与 keyframes()
一起使用。 ::view-transition-old()
会被转换为 ::view-transition-old(${uniqueName})
。
import { useState, startViewTransition } from 'hono/jsx'
import { viewTransition } from 'hono/jsx/dom/css'
import { css, keyframes, Style } from 'hono/css'
const rotate = keyframes`
from {
rotate: 0deg;
}
to {
rotate: 360deg;
}
`
export default function App() {
const [showLargeImage, setShowLargeImage] = useState(false)
const [transitionNameClass] = useState(() =>
viewTransition(css`
::view-transition-old() {
animation-name: ${rotate};
}
::view-transition-new() {
animation-name: ${rotate};
}
`)
)
return (
<>
<Style />
<button
onClick={() =>
startViewTransition(() =>
setShowLargeImage((state) => !state)
)
}
>
Click!
</button>
<div>
{!showLargeImage ? (
<img src='https://hono-dev.myedgetech.com/images/logo.png' />
) : (
<div
class={css`
${transitionNameClass}
background: url('https://hono-dev.myedgetech.com/images/logo-large.png');
background-size: contain;
background-repeat: no-repeat;
background-position: center;
width: 600px;
height: 600px;
`}
></div>
)}
</div>
</>
)
}
3. 使用 useViewTransition
如果你只想在动画期间更改样式,可以使用 useViewTransition()
。这个 Hook 返回 [boolean, (callback: () => void) => void]
,它们分别是 isUpdating
标志和 startViewTransition()
函数。
当使用此 Hook 时,Component 会在以下两个时间点进行评估:
- 在调用
startViewTransition()
的回调函数内部。 - 当 the
finish
promise becomes fulfilled (finish
promise 变为 fulfilled)时。
import { useState, useViewTransition } from 'hono/jsx'
import { viewTransition } from 'hono/jsx/dom/css'
import { css, keyframes, Style } from 'hono/css'
const rotate = keyframes`
from {
rotate: 0deg;
}
to {
rotate: 360deg;
}
`
export default function App() {
const [isUpdating, startViewTransition] = useViewTransition()
const [showLargeImage, setShowLargeImage] = useState(false)
const [transitionNameClass] = useState(() =>
viewTransition(css`
::view-transition-old() {
animation-name: ${rotate};
}
::view-transition-new() {
animation-name: ${rotate};
}
`)
)
return (
<>
<Style />
<button
onClick={() =>
startViewTransition(() =>
setShowLargeImage((state) => !state)
)
}
>
Click!
</button>
<div>
{!showLargeImage ? (
<img src='https://hono-dev.myedgetech.com/images/logo.png' />
) : (
<div
class={css`
${transitionNameClass}
background: url('https://hono-dev.myedgetech.com/images/logo-large.png');
background-size: contain;
background-repeat: no-repeat;
background-position: center;
width: 600px;
height: 600px;
position: relative;
${isUpdating &&
css`
&:before {
content: 'Loading...';
position: absolute;
top: 50%;
left: 50%;
}
`}
`}
></div>
)}
</div>
</>
)
}
hono/jsx/dom
运行时
客户端组件有一个小的 JSX 运行时。使用它可以获得比使用 hono/jsx
更小的打包结果。在 tsconfig.json
中指定 hono/jsx/dom
。对于 Deno,修改 deno.json。
{
"compilerOptions": {
"jsx": "react-jsx",
"jsxImportSource": "hono/jsx/dom"
}
}