@hanzo/ui 5.0.2 → 5.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (371) hide show
  1. package/dist/accordion.js +1 -0
  2. package/dist/accordion.mjs +1 -0
  3. package/dist/alert-dialog.js +1 -0
  4. package/dist/alert-dialog.mjs +1 -0
  5. package/dist/alert.js +1 -0
  6. package/dist/alert.mjs +1 -0
  7. package/dist/avatar.js +1 -0
  8. package/dist/avatar.mjs +1 -0
  9. package/dist/badge.js +1 -0
  10. package/dist/badge.mjs +1 -0
  11. package/dist/breadcrumb.js +1 -0
  12. package/dist/breadcrumb.mjs +1 -0
  13. package/dist/calendar.js +1 -0
  14. package/dist/calendar.mjs +1 -0
  15. package/dist/carousel.js +1 -0
  16. package/dist/carousel.mjs +1 -0
  17. package/dist/checkbox.js +1 -0
  18. package/dist/checkbox.mjs +1 -0
  19. package/dist/chunk-3H5S2OQ3.mjs +1 -0
  20. package/dist/chunk-5GRJ7UQX.js +1 -0
  21. package/dist/chunk-63HNMH7C.js +1 -0
  22. package/dist/chunk-72TOQ4DM.mjs +1 -0
  23. package/dist/chunk-7AEFTV5R.mjs +1 -0
  24. package/dist/chunk-7M4AVV2R.js +1 -0
  25. package/dist/chunk-DKPVJSBC.js +1 -0
  26. package/dist/chunk-EI7MMDWY.js +1 -0
  27. package/dist/chunk-GANGDIZG.mjs +1 -0
  28. package/dist/chunk-GRGT2Z4K.js +1 -0
  29. package/dist/chunk-JCUUC6NY.mjs +1 -0
  30. package/dist/chunk-JUQMWLIN.js +1 -0
  31. package/dist/chunk-PRVEIITE.js +1 -0
  32. package/dist/chunk-SH52AKNZ.js +1 -0
  33. package/dist/chunk-TU67EJEW.mjs +1 -0
  34. package/dist/chunk-WN5KN73U.mjs +1 -0
  35. package/dist/chunk-YSXGDEY5.mjs +1 -0
  36. package/dist/chunk-Z76OOVUE.mjs +1 -0
  37. package/dist/collapsible.js +1 -0
  38. package/dist/collapsible.mjs +1 -0
  39. package/dist/command.js +1 -0
  40. package/dist/command.mjs +1 -0
  41. package/dist/context-menu.js +1 -0
  42. package/dist/context-menu.mjs +1 -0
  43. package/dist/dialog.js +1 -0
  44. package/dist/dialog.mjs +1 -0
  45. package/dist/drawer.js +1 -0
  46. package/dist/drawer.mjs +1 -0
  47. package/dist/dropdown-menu.js +1 -0
  48. package/dist/dropdown-menu.mjs +1 -0
  49. package/dist/form.js +1 -0
  50. package/dist/form.mjs +1 -0
  51. package/dist/hover-card.js +1 -0
  52. package/dist/hover-card.mjs +1 -0
  53. package/dist/index.js +1 -9079
  54. package/dist/index.mjs +1 -8700
  55. package/dist/input-otp.js +1 -0
  56. package/dist/input-otp.mjs +1 -0
  57. package/dist/lib/utils.js +1 -0
  58. package/dist/lib/utils.mjs +1 -0
  59. package/dist/navigation-menu.js +1 -0
  60. package/dist/navigation-menu.mjs +1 -0
  61. package/dist/popover.js +1 -0
  62. package/dist/popover.mjs +1 -0
  63. package/dist/progress.js +1 -0
  64. package/dist/progress.mjs +1 -0
  65. package/dist/radio-group.js +1 -0
  66. package/dist/radio-group.mjs +1 -0
  67. package/dist/resizable.js +1 -0
  68. package/dist/resizable.mjs +1 -0
  69. package/dist/scroll-area.js +1 -0
  70. package/dist/scroll-area.mjs +1 -0
  71. package/dist/select.js +1 -0
  72. package/dist/select.mjs +1 -0
  73. package/dist/separator.js +1 -0
  74. package/dist/separator.mjs +1 -0
  75. package/dist/sheet.js +1 -0
  76. package/dist/sheet.mjs +1 -0
  77. package/dist/skeleton.js +1 -0
  78. package/dist/skeleton.mjs +1 -0
  79. package/dist/slider.js +1 -0
  80. package/dist/slider.mjs +1 -0
  81. package/dist/sonner.js +1 -0
  82. package/dist/sonner.mjs +1 -0
  83. package/dist/src/utils.js +1 -0
  84. package/dist/src/utils.mjs +1 -0
  85. package/dist/switch.js +1 -0
  86. package/dist/switch.mjs +1 -0
  87. package/dist/table.js +1 -0
  88. package/dist/table.mjs +1 -0
  89. package/dist/tabs.js +1 -0
  90. package/dist/tabs.mjs +1 -0
  91. package/dist/tailwind/index.js +1 -0
  92. package/dist/tailwind/index.mjs +1 -0
  93. package/dist/textarea.js +1 -0
  94. package/dist/textarea.mjs +1 -0
  95. package/dist/toggle-group.js +1 -0
  96. package/dist/toggle-group.mjs +1 -0
  97. package/dist/toggle.js +1 -0
  98. package/dist/toggle.mjs +1 -0
  99. package/dist/tooltip.js +1 -0
  100. package/dist/tooltip.mjs +1 -0
  101. package/dist/types/index.js +1 -0
  102. package/dist/types/index.mjs +1 -0
  103. package/package.json +110 -81
  104. package/assets/ai-icons.tsx +0 -207
  105. package/assets/crypto.tsx +0 -33
  106. package/assets/file-type-icon.tsx +0 -66
  107. package/assets/file.tsx +0 -45
  108. package/assets/general.tsx +0 -2318
  109. package/assets/hanzo-logo.svg +0 -9
  110. package/assets/hanzo-logo.tsx +0 -17
  111. package/assets/index.ts +0 -122
  112. package/assets/index.tsx +0 -4
  113. package/assets/llm-provider.tsx +0 -1094
  114. package/blocks/auth/index.ts +0 -6
  115. package/blocks/auth/login-2fa.tsx +0 -165
  116. package/blocks/auth/login-basic.tsx +0 -94
  117. package/blocks/auth/login-social.tsx +0 -148
  118. package/blocks/auth/magic-link.tsx +0 -129
  119. package/blocks/auth/password-reset.tsx +0 -97
  120. package/blocks/auth/signup.tsx +0 -157
  121. package/blocks/components/accordian-block.tsx +0 -48
  122. package/blocks/components/block-component-props.ts +0 -11
  123. package/blocks/components/bullet-cards-block.tsx +0 -46
  124. package/blocks/components/card-block/index.tsx +0 -171
  125. package/blocks/components/card-block/link-out-button.tsx +0 -20
  126. package/blocks/components/card-block/util.ts +0 -28
  127. package/blocks/components/carte-blanche-block/index.tsx +0 -127
  128. package/blocks/components/carte-blanche-block/variant-content-left.tsx +0 -49
  129. package/blocks/components/content.tsx +0 -70
  130. package/blocks/components/cta-block.tsx +0 -115
  131. package/blocks/components/enh-heading-block.tsx +0 -204
  132. package/blocks/components/grid-block/grid-block-mutator.ts +0 -12
  133. package/blocks/components/grid-block/index.tsx +0 -83
  134. package/blocks/components/grid-block/mutator-registry.ts +0 -10
  135. package/blocks/components/grid-block/table-borders.mutator.ts +0 -47
  136. package/blocks/components/group-block.tsx +0 -83
  137. package/blocks/components/heading-block.tsx +0 -88
  138. package/blocks/components/image-block.tsx +0 -111
  139. package/blocks/components/index.ts +0 -30
  140. package/blocks/components/screenful-block/content.tsx +0 -123
  141. package/blocks/components/screenful-block/index.tsx +0 -107
  142. package/blocks/components/screenful-block/poster-background.tsx +0 -34
  143. package/blocks/components/screenful-block/video-background.tsx +0 -45
  144. package/blocks/components/space-block.tsx +0 -66
  145. package/blocks/components/video-block.tsx +0 -138
  146. package/blocks/data-display/activity-feed.tsx +0 -242
  147. package/blocks/data-display/data-table.tsx +0 -235
  148. package/blocks/data-display/stats-grid.tsx +0 -194
  149. package/blocks/def/accordian-block.ts +0 -14
  150. package/blocks/def/block.ts +0 -7
  151. package/blocks/def/bullet-cards-block.ts +0 -22
  152. package/blocks/def/card-block.ts +0 -22
  153. package/blocks/def/carte-blanche-block.ts +0 -21
  154. package/blocks/def/cta-block.ts +0 -19
  155. package/blocks/def/element-block.ts +0 -11
  156. package/blocks/def/enh-heading-block.ts +0 -44
  157. package/blocks/def/grid-block.ts +0 -16
  158. package/blocks/def/group-block.ts +0 -11
  159. package/blocks/def/heading-block.ts +0 -15
  160. package/blocks/def/image-block.ts +0 -31
  161. package/blocks/def/index.ts +0 -35
  162. package/blocks/def/screenful-block.ts +0 -54
  163. package/blocks/def/space-block.ts +0 -64
  164. package/blocks/def/video-block.ts +0 -9
  165. package/blocks/ecommerce/checkout.tsx +0 -242
  166. package/blocks/ecommerce/index.ts +0 -7
  167. package/blocks/ecommerce/product-detail.tsx +0 -257
  168. package/blocks/ecommerce/product-grid.tsx +0 -148
  169. package/blocks/ecommerce/shopping-cart.tsx +0 -181
  170. package/blocks/index.ts +0 -2
  171. package/blocks/marketing/cta-section.tsx +0 -207
  172. package/blocks/marketing/faq.tsx +0 -159
  173. package/blocks/marketing/features-grid.tsx +0 -156
  174. package/blocks/marketing/hero-section.tsx +0 -192
  175. package/blocks/marketing/index.ts +0 -6
  176. package/blocks/marketing/pricing-table.tsx +0 -121
  177. package/blocks/marketing/testimonials.tsx +0 -196
  178. package/components/index.ts +0 -9
  179. package/dist/index.js.map +0 -1
  180. package/dist/index.mjs.map +0 -1
  181. package/dist/tailwind.js +0 -2025
  182. package/dist/tailwind.js.map +0 -1
  183. package/dist/tailwind.mjs +0 -2013
  184. package/dist/tailwind.mjs.map +0 -1
  185. package/dist/types.js +0 -59
  186. package/dist/types.js.map +0 -1
  187. package/dist/types.mjs +0 -53
  188. package/dist/types.mjs.map +0 -1
  189. package/dist/utils.js +0 -30
  190. package/dist/utils.js.map +0 -1
  191. package/dist/utils.mjs +0 -26
  192. package/dist/utils.mjs.map +0 -1
  193. package/frameworks/core/index.ts +0 -6
  194. package/frameworks/core/utils/index.ts +0 -64
  195. package/frameworks/react/components/button.tsx +0 -26
  196. package/frameworks/react/components/index.ts +0 -5
  197. package/frameworks/react/hooks/index.ts +0 -5
  198. package/frameworks/react/index.ts +0 -9
  199. package/frameworks/react/package.json +0 -8
  200. package/frameworks/react/utils/index.ts +0 -2
  201. package/frameworks/react-native/index.ts +0 -9
  202. package/frameworks/react-native/package.json +0 -8
  203. package/frameworks/registry.json +0 -371
  204. package/frameworks/setup.sh +0 -69
  205. package/frameworks/svelte/index.ts +0 -9
  206. package/frameworks/svelte/package.json +0 -8
  207. package/frameworks/tracker.json +0 -1854
  208. package/frameworks/vue/index.ts +0 -9
  209. package/frameworks/vue/package.json +0 -8
  210. package/helpers/file.ts +0 -33
  211. package/helpers/memoization.ts +0 -40
  212. package/primitives/accordion.tsx +0 -74
  213. package/primitives/action-button.tsx +0 -42
  214. package/primitives/alert-dialog.tsx +0 -185
  215. package/primitives/alert.tsx +0 -74
  216. package/primitives/apply-typography.tsx +0 -55
  217. package/primitives/aspect-ratio.tsx +0 -5
  218. package/primitives/avatar.tsx +0 -57
  219. package/primitives/background-beams.tsx +0 -142
  220. package/primitives/badge.tsx +0 -45
  221. package/primitives/breadcrumb.tsx +0 -130
  222. package/primitives/breakpoint-indicator.tsx +0 -19
  223. package/primitives/button.tsx +0 -72
  224. package/primitives/calendar.tsx +0 -72
  225. package/primitives/card.tsx +0 -97
  226. package/primitives/carousel.tsx +0 -238
  227. package/primitives/chat/chat-input-area.tsx +0 -88
  228. package/primitives/chat/chat-input.tsx +0 -71
  229. package/primitives/chat/files-preview.tsx +0 -331
  230. package/primitives/chat/index.ts +0 -6
  231. package/primitives/chat/json-form.tsx +0 -8
  232. package/primitives/chat/message-list.tsx +0 -308
  233. package/primitives/chat/message.tsx +0 -569
  234. package/primitives/chat/sqlite-preview.tsx +0 -215
  235. package/primitives/checkbox.tsx +0 -32
  236. package/primitives/collapsible.tsx +0 -9
  237. package/primitives/combobox.tsx +0 -239
  238. package/primitives/command.tsx +0 -151
  239. package/primitives/context-menu.tsx +0 -206
  240. package/primitives/copy-to-clipboard-icon.tsx +0 -60
  241. package/primitives/dialog-video-controller.tsx +0 -38
  242. package/primitives/dialog.tsx +0 -128
  243. package/primitives/dot-pattern.tsx +0 -57
  244. package/primitives/dots-loader.tsx +0 -13
  245. package/primitives/drawer.tsx +0 -113
  246. package/primitives/dropdown-menu.tsx +0 -199
  247. package/primitives/error-message.tsx +0 -19
  248. package/primitives/file-uploader.tsx +0 -203
  249. package/primitives/form.tsx +0 -185
  250. package/primitives/hover-card.tsx +0 -28
  251. package/primitives/icons/github.tsx +0 -14
  252. package/primitives/icons/index.ts +0 -18
  253. package/primitives/icons/youtube-logo.tsx +0 -59
  254. package/primitives/index-client.ts +0 -4
  255. package/primitives/index-common.ts +0 -304
  256. package/primitives/index-next.ts +0 -4
  257. package/primitives/input-otp.tsx +0 -65
  258. package/primitives/input.tsx +0 -128
  259. package/primitives/label.tsx +0 -21
  260. package/primitives/list-adaptor.ts +0 -12
  261. package/primitives/list-box.tsx +0 -74
  262. package/primitives/loading-spinner.tsx +0 -33
  263. package/primitives/markdown-preview.tsx +0 -612
  264. package/primitives/mermaid.tsx +0 -196
  265. package/primitives/navigation-menu.tsx +0 -147
  266. package/primitives/next/image.tsx +0 -91
  267. package/primitives/next/index.ts +0 -7
  268. package/primitives/next/inline-icon.tsx +0 -36
  269. package/primitives/next/link-element.tsx +0 -109
  270. package/primitives/next/mdx-link.tsx +0 -22
  271. package/primitives/next/media-stack.tsx +0 -52
  272. package/primitives/next/nav-items.tsx +0 -45
  273. package/primitives/next/youtube-embed.tsx +0 -83
  274. package/primitives/pagination.tsx +0 -117
  275. package/primitives/popover.tsx +0 -34
  276. package/primitives/pretty-json-print.tsx +0 -28
  277. package/primitives/progress.tsx +0 -27
  278. package/primitives/prompt-textarea.tsx +0 -72
  279. package/primitives/qr-code.tsx +0 -112
  280. package/primitives/radio-group.tsx +0 -42
  281. package/primitives/resizable.tsx +0 -47
  282. package/primitives/scroll-area.tsx +0 -57
  283. package/primitives/search-input.tsx +0 -66
  284. package/primitives/select.tsx +0 -122
  285. package/primitives/separator.tsx +0 -26
  286. package/primitives/sheet.tsx +0 -139
  287. package/primitives/skeleton.tsx +0 -18
  288. package/primitives/slider.tsx +0 -63
  289. package/primitives/sonner.tsx +0 -35
  290. package/primitives/step-indicator.tsx +0 -69
  291. package/primitives/stepper.tsx +0 -272
  292. package/primitives/switch.tsx +0 -27
  293. package/primitives/table.tsx +0 -105
  294. package/primitives/tabs.tsx +0 -50
  295. package/primitives/text-area.tsx +0 -26
  296. package/primitives/text-link.tsx +0 -27
  297. package/primitives/textarea.tsx +0 -64
  298. package/primitives/textfield.tsx +0 -78
  299. package/primitives/toast.tsx +0 -30
  300. package/primitives/toggle-group.tsx +0 -63
  301. package/primitives/toggle.tsx +0 -44
  302. package/primitives/tooltip.tsx +0 -47
  303. package/primitives/video-player.tsx +0 -23
  304. package/src/button.ts +0 -1
  305. package/src/hooks/index.ts +0 -7
  306. package/src/hooks/use-click-away.ts +0 -31
  307. package/src/hooks/use-combined-refs.ts +0 -22
  308. package/src/hooks/use-copy-clipboard.ts +0 -30
  309. package/src/hooks/use-debounce.ts +0 -17
  310. package/src/hooks/use-fill-ids.ts +0 -25
  311. package/src/hooks/use-map.ts +0 -26
  312. package/src/hooks/use-measure.ts +0 -42
  313. package/src/hooks/use-reverse-video-playback.ts +0 -43
  314. package/src/hooks/use-scroll-restoration.ts +0 -50
  315. package/src/index-lean.ts +0 -87
  316. package/src/index.ts +0 -54
  317. package/src/mcp/README.md +0 -141
  318. package/src/mcp/enhanced-server.ts +0 -1208
  319. package/src/mcp/index.ts +0 -518
  320. package/src/mcp/package.json +0 -10
  321. package/src/registry/api.ts +0 -164
  322. package/src/registry/index.ts +0 -60
  323. package/src/registry/package.json +0 -10
  324. package/src/utils.ts +0 -19
  325. package/tailwind/colors.tailwind.js +0 -53
  326. package/tailwind/fontFamily.tailwind.ts +0 -7
  327. package/tailwind/fontSize.tailwind.ts +0 -13
  328. package/tailwind/index.ts +0 -7
  329. package/tailwind/safelist.tailwind.js +0 -26
  330. package/tailwind/screens.tailwind.js +0 -8
  331. package/tailwind/spacing.tailwind.js +0 -65
  332. package/tailwind/tailwind.config.hanzo-preset.d.ts +0 -5
  333. package/tailwind/tailwind.config.hanzo-preset.js +0 -915
  334. package/tailwind/tw-font-desc.ts +0 -15
  335. package/tailwind/typo-plugin/get-plugin-styles.js +0 -679
  336. package/tailwind/typo-plugin/index.d.ts +0 -9
  337. package/tailwind/typo-plugin/index.js +0 -141
  338. package/tailwind/typo-plugin/utils.js +0 -60
  339. package/tailwind/typography-test.mdx +0 -35
  340. package/tailwind/z-index.tailwind.js +0 -71
  341. package/types/animation-def.ts +0 -3
  342. package/types/breakpoints.ts +0 -11
  343. package/types/bullet-item.ts +0 -10
  344. package/types/button-def.ts +0 -39
  345. package/types/dimensions.ts +0 -8
  346. package/types/grid-def.ts +0 -56
  347. package/types/image-def.ts +0 -32
  348. package/types/index.ts +0 -30
  349. package/types/link-def.ts +0 -56
  350. package/types/media-stack-def.ts +0 -31
  351. package/types/t-shirt-size.ts +0 -5
  352. package/types/tshirt-dimensions.ts +0 -20
  353. package/types/video-def.ts +0 -25
  354. package/util/blob.ts +0 -33
  355. package/util/copy-to-clipboard.ts +0 -17
  356. package/util/create-shadow-root.ts +0 -22
  357. package/util/date.ts +0 -84
  358. package/util/debounce.ts +0 -11
  359. package/util/file.ts +0 -15
  360. package/util/format-and-abbreviate-as-currency.ts +0 -125
  361. package/util/format-text.ts +0 -34
  362. package/util/format-to-max-char.ts +0 -68
  363. package/util/index-client.ts +0 -3
  364. package/util/index.ts +0 -112
  365. package/util/number-abbreviate.ts +0 -49
  366. package/util/specifier.ts +0 -43
  367. package/util/spread-to-transform.ts +0 -25
  368. package/util/step-animation.ts +0 -90
  369. package/util/timing.ts +0 -3
  370. package/util/toasts.tsx +0 -17
  371. package/util/two-way-map.ts +0 -19
package/src/mcp/index.ts DELETED
@@ -1,518 +0,0 @@
1
- import { registrySchema, Registry, RegistryItem } from "../registry"
2
- import { fetchRegistry, getRegistryItem, getRegistryItemUrl } from "../registry/api"
3
- import { Server } from "@modelcontextprotocol/sdk/server/index.js"
4
- import {
5
- CallToolRequestSchema,
6
- ListToolsRequestSchema,
7
- } from "@modelcontextprotocol/sdk/types.js"
8
- import { z } from "zod"
9
- import { zodToJsonSchema } from "zod-to-json-schema"
10
-
11
- // Get package version
12
- let packageVersion = "4.5.0"
13
- try {
14
- // eslint-disable-next-line @typescript-eslint/no-var-requires
15
- const packageJson = require("../../package.json")
16
- packageVersion = packageJson.version || packageVersion
17
- } catch (error) {
18
- console.error("Could not load package.json for version", error)
19
- }
20
-
21
- /**
22
- * MCP Server implementation for @hanzo/ui registry
23
- * Provides tools for interacting with the registry
24
- */
25
- export const server = new Server(
26
- {
27
- name: "hanzo-ui",
28
- version: packageVersion,
29
- },
30
- {
31
- capabilities: {
32
- resources: {},
33
- tools: {},
34
- },
35
- }
36
- )
37
-
38
- // Register the available tools
39
- server.setRequestHandler(ListToolsRequestSchema, async () => {
40
- return {
41
- tools: [
42
- {
43
- name: "init",
44
- description:
45
- "Initialize a new project using @hanzo/ui components and styles.",
46
- inputSchema: zodToJsonSchema(z.object({
47
- style: z.string().optional().describe("The style to use for the project (e.g., 'default' or 'new-york')"),
48
- })),
49
- },
50
- {
51
- name: "list_components",
52
- description: "List all available components in the registry",
53
- inputSchema: zodToJsonSchema(z.object({
54
- type: z.string().optional().describe("Filter components by type (e.g., 'ui', 'block')"),
55
- category: z.string().optional().describe("Filter components by category"),
56
- })),
57
- },
58
- {
59
- name: "get_component",
60
- description: "Get detailed information about a specific component",
61
- inputSchema: zodToJsonSchema(
62
- z.object({
63
- name: z
64
- .string()
65
- .describe("The name of the component to get from the registry"),
66
- })
67
- ),
68
- },
69
- {
70
- name: "add_component",
71
- description: "Get instructions for adding a component to a project",
72
- inputSchema: zodToJsonSchema(
73
- z.object({
74
- name: z
75
- .string()
76
- .describe("The name of the component to add"),
77
- style: z
78
- .string()
79
- .optional()
80
- .describe("The style to use (default, new-york, etc.)"),
81
- })
82
- ),
83
- },
84
- {
85
- name: "list_styles",
86
- description: "List all available styles in the registry",
87
- inputSchema: zodToJsonSchema(z.object({})),
88
- },
89
- {
90
- name: "search_registry",
91
- description: "Search the registry for components matching criteria",
92
- inputSchema: zodToJsonSchema(
93
- z.object({
94
- query: z
95
- .string()
96
- .describe("Search term to look for in component names and descriptions"),
97
- })
98
- ),
99
- },
100
- ],
101
- }
102
- })
103
-
104
- // Handle tool calls
105
- server.setRequestHandler(CallToolRequestSchema, async (request) => {
106
- try {
107
- // Ensure we have arguments
108
- const args = request.params.arguments || {}
109
-
110
- // Get registry URL from environment variable or use default
111
- const REGISTRY_URL = process.env.REGISTRY_URL || "https://ui.hanzo.ai/registry/registry.json"
112
- const registry = await getRegistry(REGISTRY_URL)
113
-
114
- switch (request.params.name) {
115
- case "init": {
116
- const style = args.style || "default"
117
-
118
- let text = `# Initialize a new project with @hanzo/ui
119
-
120
- To create a new project with @hanzo/ui components, follow these steps:
121
-
122
- 1. Create a new project first:
123
- \`\`\`bash
124
- # Using npm
125
- npm create next-app@latest my-app
126
-
127
- # Or using yarn
128
- yarn create next-app my-app
129
-
130
- # Or using pnpm
131
- pnpm create next-app my-app
132
- \`\`\`
133
-
134
- 2. Navigate to your project directory:
135
- \`\`\`bash
136
- cd my-app
137
- \`\`\`
138
-
139
- 3. Install @hanzo/ui and initialize it:
140
- \`\`\`bash
141
- npx @hanzo/ui@latest init${style ? ` --style=${style}` : ""}
142
- \`\`\`
143
-
144
- This will:
145
- - Install all necessary dependencies
146
- - Set up the project structure for @hanzo/ui
147
- - Configure tailwind.css with the appropriate theme
148
- - Add component configuration
149
- `
150
-
151
- // Check if the registry has any additional setup instructions
152
- const setupConfig = registry.items.find(
153
- (item) => item.type === "registry:config" && item.name === "setup"
154
- )
155
-
156
- if (setupConfig) {
157
- text += `
158
- ## Additional Setup
159
-
160
- ${setupConfig.description || "Follow any additional setup instructions provided during initialization."}
161
- `
162
- }
163
-
164
- return {
165
- content: [
166
- { type: "text", text }
167
- ],
168
- }
169
- }
170
-
171
- case "list_components": {
172
- // Filter components based on type and category if provided
173
- const typeFilter = args.type ? `ui:${args.type}` : null
174
- const categoryFilter = args.category || null
175
-
176
- let components = registry.items.filter(item => {
177
- // Filter by type if specified
178
- if (typeFilter && !item.type.includes(typeFilter)) {
179
- return false
180
- }
181
-
182
- // Filter by category if specified
183
- if (categoryFilter && item.category !== categoryFilter) {
184
- return false
185
- }
186
-
187
- return true
188
- })
189
-
190
- // Group components by category if they have categories
191
- const hasCategories = components.some(item => item.category)
192
- let responseText = "# Available Components\n\n"
193
-
194
- if (hasCategories) {
195
- // Group by category
196
- const categorized = components.reduce((acc, item) => {
197
- const category = item.category || "Uncategorized"
198
- if (!acc[category]) {
199
- acc[category] = []
200
- }
201
- acc[category].push(item)
202
- return acc
203
- }, {} as Record<string, RegistryItem[]>)
204
-
205
- // Build response
206
- for (const [category, items] of Object.entries(categorized)) {
207
- responseText += `## ${category}\n\n`
208
-
209
- items.forEach(item => {
210
- responseText += `- **${item.name}** - ${item.description || 'No description available'}\n`
211
- })
212
-
213
- responseText += "\n"
214
- }
215
- } else {
216
- // Simple list
217
- components.forEach(item => {
218
- responseText += `- **${item.name}** (${item.type}) - ${item.description || 'No description available'}\n`
219
- })
220
- }
221
-
222
- responseText += "\nTo get detailed information about a specific component, use the `get_component` tool.\n"
223
- responseText += "To add a component to your project, use the `add_component` tool.\n"
224
-
225
- return {
226
- content: [{ type: "text", text: responseText }],
227
- }
228
- }
229
-
230
- case "get_component": {
231
- const name = z.string().parse(args.name)
232
-
233
- if (!name) {
234
- throw new Error("Component name is required")
235
- }
236
-
237
- const itemUrl = getRegistryItemUrl(name, REGISTRY_URL)
238
- const component = await getRegistryItem(itemUrl, "")
239
-
240
- if (!component) {
241
- return {
242
- content: [
243
- {
244
- type: "text",
245
- text: `Component '${name}' not found in the registry.`,
246
- },
247
- ],
248
- }
249
- }
250
-
251
- // Build a markdown description of the component
252
- let componentDetails = `# ${component.name}\n\n`
253
-
254
- if (component.description) {
255
- componentDetails += `${component.description}\n\n`
256
- }
257
-
258
- componentDetails += `**Type:** ${component.type}\n`
259
-
260
- if (component.category) {
261
- componentDetails += `**Category:** ${component.category}\n`
262
- }
263
-
264
- // Dependencies
265
- if (component.dependencies && component.dependencies.length > 0) {
266
- componentDetails += `\n## Dependencies\n\n`
267
- component.dependencies.forEach(dep => {
268
- componentDetails += `- ${dep}\n`
269
- })
270
- }
271
-
272
- // Registry Dependencies
273
- if (component.registryDependencies && component.registryDependencies.length > 0) {
274
- componentDetails += `\n## Required Components\n\n`
275
- componentDetails += `This component requires the following components that will be installed automatically:\n\n`
276
- component.registryDependencies.forEach(dep => {
277
- componentDetails += `- ${dep}\n`
278
- })
279
- }
280
-
281
- // Files
282
- if (component.files && component.files.length > 0) {
283
- componentDetails += `\n## Files\n\n`
284
- component.files.forEach(file => {
285
- componentDetails += `- ${file.path} (${file.type})\n`
286
- })
287
- }
288
-
289
- // Show example usage if available
290
- const exampleFile = component.files?.find(file => file.type === "registry:example")
291
- if (exampleFile && exampleFile.content) {
292
- componentDetails += `\n## Example Usage\n\n\`\`\`tsx\n${exampleFile.content}\n\`\`\`\n`
293
- }
294
-
295
- // Add installation instructions
296
- componentDetails += `\n## Installation\n\n`
297
- componentDetails += `Run the following command to add this component to your project:\n\n`
298
- componentDetails += `\`\`\`bash\n`
299
- componentDetails += `npx @hanzo/ui@latest add ${name}\n`
300
- componentDetails += `\`\`\`\n`
301
-
302
- return {
303
- content: [{ type: "text", text: componentDetails }],
304
- }
305
- }
306
-
307
- case "add_component": {
308
- const name = z.string().parse(args.name)
309
- const style = args.style || "default"
310
-
311
- if (!name) {
312
- throw new Error("Component name is required")
313
- }
314
-
315
- const itemUrl = getRegistryItemUrl(name, REGISTRY_URL)
316
- const component = await getRegistryItem(itemUrl, "")
317
-
318
- if (!component) {
319
- return {
320
- content: [
321
- {
322
- type: "text",
323
- text: `Component '${name}' not found in the registry.`,
324
- },
325
- ],
326
- }
327
- }
328
-
329
- // Check if the component has dependencies
330
- const hasDeps = component.registryDependencies && component.registryDependencies.length > 0
331
-
332
- let instructions = `# Adding the ${name} Component\n\n`
333
-
334
- // Main installation command
335
- instructions += `Run the following command to add the \`${name}\` component to your project:\n\n`
336
- instructions += `\`\`\`bash\n`
337
- instructions += `npx @hanzo/ui@latest add ${name} --style=${style}\n`
338
- instructions += `\`\`\`\n\n`
339
-
340
- // Explain what this will do
341
- instructions += `This will:\n`
342
- instructions += `- Add the component file(s) to your project\n`
343
- instructions += `- Install any required npm dependencies\n`
344
-
345
- if (hasDeps) {
346
- instructions += `- Install required component dependencies: ${component.registryDependencies!.join(", ")}\n`
347
- }
348
-
349
- // Usage example
350
- instructions += `\n## Usage\n\nAfter adding the component, you can import it in your project:\n\n`
351
- instructions += `\`\`\`tsx\n`
352
- instructions += `import { ${name.charAt(0).toUpperCase() + name.slice(1)} } from "@/components/ui/${name}"\n\n`
353
-
354
- // Add a simple example if possible
355
- if (component.type.includes("ui:")) {
356
- instructions += `export default function Example() {\n`
357
- instructions += ` return <${name.charAt(0).toUpperCase() + name.slice(1)} />\n`
358
- instructions += `}\n`
359
- }
360
-
361
- instructions += `\`\`\`\n`
362
-
363
- return {
364
- content: [
365
- {
366
- type: "text",
367
- text: instructions,
368
- },
369
- ],
370
- }
371
- }
372
-
373
- case "list_styles": {
374
- // Find all style-related items in the registry
375
- const styles = registry.items.filter(item =>
376
- item.type === "registry:style" || item.type.includes("style")
377
- )
378
-
379
- let styleInfo = "# Available Styles\n\n"
380
-
381
- if (styles.length === 0) {
382
- styleInfo += "The registry does not define any specific styles.\n\n"
383
- styleInfo += "The default style will be used when installing components.\n"
384
- } else {
385
- styles.forEach(style => {
386
- styleInfo += `## ${style.name}\n\n`
387
- styleInfo += `${style.description || 'No description available'}\n\n`
388
-
389
- // Add usage info
390
- styleInfo += `**Usage:**\n\n`
391
- styleInfo += `\`\`\`bash\n`
392
- styleInfo += `# Initialize with this style\n`
393
- styleInfo += `npx @hanzo/ui@latest init --style=${style.name}\n\n`
394
- styleInfo += `# Install components with this style\n`
395
- styleInfo += `npx @hanzo/ui@latest add [component] --style=${style.name}\n`
396
- styleInfo += `\`\`\`\n\n`
397
- })
398
- }
399
-
400
- return {
401
- content: [{ type: "text", text: styleInfo }],
402
- }
403
- }
404
-
405
- case "search_registry": {
406
- const query = z.string().parse(args.query).toLowerCase()
407
-
408
- if (!query || query.length < 2) {
409
- return {
410
- content: [
411
- {
412
- type: "text",
413
- text: "Please provide a search term with at least 2 characters.",
414
- },
415
- ],
416
- }
417
- }
418
-
419
- // Search through all items in the registry
420
- const results = registry.items.filter(item => {
421
- // Search in name
422
- if (item.name.toLowerCase().includes(query)) {
423
- return true
424
- }
425
-
426
- // Search in description
427
- if (item.description?.toLowerCase().includes(query)) {
428
- return true
429
- }
430
-
431
- // Search in category
432
- if (item.category?.toLowerCase().includes(query)) {
433
- return true
434
- }
435
-
436
- return false
437
- })
438
-
439
- if (results.length === 0) {
440
- return {
441
- content: [
442
- {
443
- type: "text",
444
- text: `No components found matching '${query}'. Try a different search term.`,
445
- },
446
- ],
447
- }
448
- }
449
-
450
- // Format the results
451
- let searchResults = `# Search Results for '${query}'\n\n`
452
- searchResults += `Found ${results.length} matching components\n\n`
453
-
454
- results.forEach(item => {
455
- searchResults += `## ${item.name}\n\n`
456
- searchResults += `**Type:** ${item.type}\n`
457
-
458
- if (item.category) {
459
- searchResults += `**Category:** ${item.category}\n`
460
- }
461
-
462
- if (item.description) {
463
- searchResults += `\n${item.description}\n`
464
- }
465
-
466
- searchResults += `\n**Installation:**\n\n`
467
- searchResults += `\`\`\`bash\nnpx @hanzo/ui@latest add ${item.name}\n\`\`\`\n\n`
468
- })
469
-
470
- return {
471
- content: [{ type: "text", text: searchResults }],
472
- }
473
- }
474
-
475
- default:
476
- throw new Error(`Tool ${request.params.name} not found`)
477
- }
478
- } catch (error) {
479
- if (error instanceof z.ZodError) {
480
- return {
481
- content: [
482
- {
483
- type: "text",
484
- text: `Invalid input: ${JSON.stringify(error.errors)}`,
485
- },
486
- ],
487
- }
488
- }
489
-
490
- throw error
491
- }
492
- })
493
-
494
- /**
495
- * Fetches and parses the registry from the given URL
496
- * @param registryUrl - URL to the registry.json file
497
- * @returns The parsed registry object
498
- */
499
- async function getRegistry(registryUrl: string): Promise<Registry> {
500
- try {
501
- const [registryJson] = await fetchRegistry([registryUrl], {
502
- useCache: false,
503
- })
504
- return registrySchema.parse(registryJson)
505
- } catch (error) {
506
- console.error(`Failed to fetch registry from ${registryUrl}:`, error)
507
- // Return a minimal valid registry if we can't fetch the real one
508
- return {
509
- name: "hanzo-ui",
510
- items: [],
511
- }
512
- }
513
- }
514
-
515
- // Support CommonJS
516
- if (typeof module !== 'undefined') {
517
- module.exports = { server }
518
- }
@@ -1,10 +0,0 @@
1
- {
2
- "name": "@hanzo/ui-mcp",
3
- "private": true,
4
- "type": "module",
5
- "main": "../dist/mcp/index.js",
6
- "types": "../dist/mcp/index.d.ts",
7
- "scripts": {
8
- "build": "tsc"
9
- }
10
- }
@@ -1,164 +0,0 @@
1
- import { registryItemSchema, RegistryItem, Registry } from "./index"
2
- import { z } from "zod"
3
-
4
- const registryCache = new Map<string, Promise<any>>()
5
-
6
- /**
7
- * Gets the full registry URL based on the registry name or URL provided
8
- * @param registry - Registry name, URL, or file path
9
- * @returns The complete registry URL
10
- */
11
- export function getRegistryUrl(registry: string): string {
12
- // Check if registry is a full URL
13
- if (registry.startsWith("http")) {
14
- return registry
15
- }
16
-
17
- // Default to Hanzo registry if REGISTRY_BASE_URL is defined, otherwise use hanzo
18
- const baseUrl = process.env.REGISTRY_BASE_URL || "https://ui.hanzo.com/registry"
19
-
20
- // Check if registry is a path to a JSON file
21
- if (registry.endsWith(".json")) {
22
- return `${baseUrl}/${registry}`
23
- }
24
-
25
- // Assume registry is a registry style
26
- return `${baseUrl}/${registry}/registry.json`
27
- }
28
-
29
- /**
30
- * Fetches a specific registry item by URL
31
- * @param itemUrl - The URL of the item to fetch
32
- * @param basePath - Base path for resolving imports
33
- * @returns The registry item or null if not found
34
- */
35
- export async function getRegistryItem(
36
- itemUrl: string,
37
- basePath: string
38
- ): Promise<RegistryItem | null> {
39
- try {
40
- const response = await fetch(itemUrl)
41
-
42
- if (!response.ok) {
43
- console.error(`Failed to fetch registry item from ${itemUrl}: ${response.status} ${response.statusText}`)
44
- return null
45
- }
46
-
47
- const item = await response.json()
48
- const validatedItem = registryItemSchema.safeParse(item)
49
-
50
- if (!validatedItem.success) {
51
- console.error(`Invalid registry item format: ${JSON.stringify(validatedItem.error)}`)
52
- return null
53
- }
54
-
55
- // Process and resolve file paths
56
- if (validatedItem.data.files) {
57
- for (const file of validatedItem.data.files) {
58
- if (!file.content) {
59
- continue
60
- }
61
-
62
- // If the file has a target, we resolve it from the target
63
- if (file.target) {
64
- file.target = resolveImport(file.target, basePath)
65
- }
66
- }
67
- }
68
-
69
- return validatedItem.data
70
- } catch (error) {
71
- console.error(`Error fetching registry item: ${error}`)
72
- return null
73
- }
74
- }
75
-
76
- /**
77
- * Resolves an import path relative to a base path
78
- * @param importPath - The import path to resolve
79
- * @param basePath - The base path to resolve against
80
- * @returns The resolved import path
81
- */
82
- export function resolveImport(importPath: string, basePath: string): string {
83
- const importParts = importPath?.split("/")
84
-
85
- // Check if it's a relative import
86
- if (importPath.startsWith(".")) {
87
- return [basePath.replace(/^@/, ""), ...importParts].filter(Boolean).join("/")
88
- }
89
-
90
- // Check if it's a scope package
91
- if (importParts?.[0]?.startsWith("@")) {
92
- return importPath
93
- }
94
-
95
- return importPath
96
- }
97
-
98
- /**
99
- * Fetches registry data from one or more paths
100
- * @param paths - Array of registry paths to fetch
101
- * @param options - Options for fetching the registry
102
- * @returns Array of registry data
103
- */
104
- export async function fetchRegistry(
105
- paths: string[],
106
- options: { useCache?: boolean } = {}
107
- ): Promise<Registry[]> {
108
- options = {
109
- useCache: true,
110
- ...options,
111
- }
112
-
113
- try {
114
- const results = await Promise.all(
115
- paths.map(async (path) => {
116
- const url = getRegistryUrl(path)
117
-
118
- // Check cache first if caching is enabled
119
- if (options.useCache && registryCache.has(url)) {
120
- return registryCache.get(url)
121
- }
122
-
123
- // Store the promise in the cache before awaiting if caching is enabled
124
- const fetchPromise = (async () => {
125
- const response = await fetch(url)
126
-
127
- if (!response.ok) {
128
- throw new Error(
129
- `Failed to fetch registry from ${url}. ${response.status} ${response.statusText}`
130
- )
131
- }
132
-
133
- return response.json()
134
- })()
135
-
136
- if (options.useCache) {
137
- registryCache.set(url, fetchPromise)
138
- }
139
- return fetchPromise
140
- })
141
- )
142
-
143
- return results
144
- } catch (error) {
145
- console.error(`Error fetching registry: ${error}`)
146
- throw error
147
- }
148
- }
149
-
150
- /**
151
- * Gets the URL for a specific registry item
152
- * @param itemName - The name of the registry item
153
- * @param registryUrl - The base registry URL
154
- * @returns The URL for the specific registry item
155
- */
156
- export function getRegistryItemUrl(itemName: string, registryUrl: string): string {
157
- const registryBaseUrl = registryUrl.replace(/\/registry\.json$/, "")
158
- return `${registryBaseUrl}/r/${itemName}.json`
159
- }
160
-
161
- // Support CommonJS
162
- if (typeof module !== 'undefined') {
163
- module.exports = { getRegistryUrl, getRegistryItem, resolveImport, fetchRegistry, getRegistryItemUrl }
164
- }