@proveanything/smartlinks-utils-ui 0.1.0 → 0.1.1

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.
package/README.md CHANGED
@@ -16,6 +16,16 @@ This package requires the following peer dependencies in your app:
16
16
  npm install react react-dom @proveanything/smartlinks
17
17
  ```
18
18
 
19
+ ## Setup
20
+
21
+ Import the pre-compiled styles in your app's entry point:
22
+
23
+ ```tsx
24
+ import '@proveanything/smartlinks-ui/styles.css';
25
+ ```
26
+
27
+ This provides all the Tailwind utility classes used by the components. Your app still needs to define the CSS variables (e.g., `--primary`, `--border`) — these come from your own design system (shadcn, custom theme, etc.).
28
+
19
29
  ## Components
20
30
 
21
31
  ### Asset Picker
@@ -0,0 +1,2 @@
1
+
2
+ export { }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@proveanything/smartlinks-utils-ui",
3
- "version": "0.1.0",
3
+ "version": "0.1.1",
4
4
  "description": "Reusable React components for SmartLinks microapps — Asset Picker, Conditions Editor, Icon Picker, and more.",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
@@ -11,6 +11,7 @@
11
11
  "types": "./dist/index.d.ts",
12
12
  "import": "./dist/index.js"
13
13
  },
14
+ "./styles.css": "./dist/styles.css",
14
15
  "./asset-picker": {
15
16
  "types": "./dist/components/AssetPicker/index.d.ts",
16
17
  "import": "./dist/components/AssetPicker/index.js"
@@ -37,25 +38,28 @@
37
38
  },
38
39
  "publishConfig": {
39
40
  "access": "public"
40
- },
41
+ },
41
42
  "peerDependencies": {
43
+ "@proveanything/smartlinks": "^1.3.0",
42
44
  "react": "^18.0.0",
43
- "react-dom": "^18.0.0",
44
- "@proveanything/smartlinks": "^1.3.0"
45
+ "react-dom": "^18.0.0"
45
46
  },
46
47
  "devDependencies": {
47
- "react": "^18.3.1",
48
- "react-dom": "^18.3.1",
49
48
  "@proveanything/smartlinks": "^1.3.17",
50
49
  "@types/react": "^18.3.0",
51
50
  "@types/react-dom": "^18.3.0",
51
+ "react": "^18.3.1",
52
+ "react-dom": "^18.3.1",
52
53
  "tsup": "^8.0.0",
53
54
  "typescript": "^5.5.0"
54
55
  },
55
56
  "dependencies": {
56
- "lucide-react": "^0.462.0",
57
+ "@tailwindcss/postcss": "^4.2.1",
58
+ "autoprefixer": "^10.4.24",
57
59
  "clsx": "^2.1.1",
58
- "tailwind-merge": "^2.5.2"
60
+ "lucide-react": "^0.462.0",
61
+ "tailwind-merge": "^2.5.2",
62
+ "tailwindcss": "^4.2.1"
59
63
  },
60
64
  "keywords": [
61
65
  "smartlinks",
@@ -1,518 +0,0 @@
1
- import { cn } from './chunk-L7FQ52F5.js';
2
- import { useState, useCallback, useRef, useEffect, useMemo } from 'react';
3
- import { ChevronRight, Loader2, AlertCircle, Search, Smile, ChevronLeft, X } from 'lucide-react';
4
- import { jsx, jsxs, Fragment } from 'react/jsx-runtime';
5
-
6
- // src/components/IconPicker/icon-index.ts
7
- var FA_API = "https://api.fontawesome.com";
8
- var FA_VERSION = "7.x";
9
- var SEARCH_QUERY = `
10
- query Search($version: String!, $query: String!, $first: Int) {
11
- search(version: $version, query: $query, first: $first) {
12
- id
13
- label
14
- unicode
15
- familyStylesByLicense {
16
- pro {
17
- family
18
- style
19
- }
20
- }
21
- }
22
- }
23
- `;
24
- function mapRawIcon(raw) {
25
- const proStyles = raw.familyStylesByLicense?.pro || [];
26
- const families = /* @__PURE__ */ new Set();
27
- const styles = /* @__PURE__ */ new Set();
28
- let isBrand = false;
29
- for (const fs of proStyles) {
30
- if (fs.style === "brands" || fs.family === "brands") {
31
- families.add("brands");
32
- isBrand = true;
33
- } else if (fs.family === "classic" || fs.family === "sharp") {
34
- families.add("classic");
35
- if (["solid", "regular", "light"].includes(fs.style)) styles.add(fs.style);
36
- } else if (fs.family === "duotone") {
37
- families.add("duotone");
38
- if (["solid", "regular", "light"].includes(fs.style)) styles.add(fs.style);
39
- }
40
- }
41
- return {
42
- id: raw.id,
43
- label: raw.label || raw.id,
44
- unicode: raw.unicode || "",
45
- families: [...families],
46
- styles: [...styles],
47
- isBrand,
48
- terms: [raw.id, raw.label?.toLowerCase() || ""]
49
- };
50
- }
51
- async function gqlRequest(query, variables) {
52
- const res = await fetch(FA_API, {
53
- method: "POST",
54
- headers: { "Content-Type": "application/json" },
55
- body: JSON.stringify({ query, variables })
56
- });
57
- if (!res.ok) throw new Error(`FA API error: ${res.status}`);
58
- const json = await res.json();
59
- if (json.errors?.length) throw new Error(json.errors[0].message);
60
- return json.data;
61
- }
62
- async function searchIcons(query, first = 200) {
63
- const data = await gqlRequest(SEARCH_QUERY, {
64
- version: FA_VERSION,
65
- query,
66
- first
67
- });
68
- return (data.search || []).map(mapRawIcon);
69
- }
70
- var _catalog = /* @__PURE__ */ new Map();
71
- var _catalogListeners = /* @__PURE__ */ new Set();
72
- var _catalogLoading = false;
73
- var _catalogLoaded = false;
74
- function getCatalog() {
75
- return [..._catalog.values()].sort((a, b) => a.id.localeCompare(b.id));
76
- }
77
- function isCatalogLoaded() {
78
- return _catalogLoaded;
79
- }
80
- function subscribeCatalog(fn) {
81
- _catalogListeners.add(fn);
82
- return () => {
83
- _catalogListeners.delete(fn);
84
- };
85
- }
86
- function notifyListeners() {
87
- for (const fn of _catalogListeners) fn();
88
- }
89
- var BROWSE_QUERIES = [
90
- "a",
91
- "b",
92
- "c",
93
- "d",
94
- "e",
95
- "f",
96
- "g",
97
- "h",
98
- "i",
99
- "j",
100
- "k",
101
- "l",
102
- "m",
103
- "n",
104
- "o",
105
- "p",
106
- "q",
107
- "r",
108
- "s",
109
- "t",
110
- "u",
111
- "v",
112
- "w",
113
- "x",
114
- "y",
115
- "z",
116
- "0",
117
- "1",
118
- "2",
119
- "3",
120
- "4",
121
- "5"
122
- ];
123
- async function loadCatalogInBackground() {
124
- if (_catalogLoaded || _catalogLoading) return;
125
- _catalogLoading = true;
126
- notifyListeners();
127
- for (let i = 0; i < BROWSE_QUERIES.length; i += 3) {
128
- const batch = BROWSE_QUERIES.slice(i, i + 3);
129
- const results = await Promise.all(
130
- batch.map((q) => searchIcons(q, 200).catch(() => []))
131
- );
132
- for (const icons of results) {
133
- for (const icon of icons) {
134
- if (!_catalog.has(icon.id)) _catalog.set(icon.id, icon);
135
- }
136
- }
137
- notifyListeners();
138
- }
139
- _catalogLoaded = true;
140
- _catalogLoading = false;
141
- notifyListeners();
142
- }
143
- function toFaClass(id, family, style) {
144
- if (family === "brands") return `fa-brands fa-${id}`;
145
- const stylePrefix = style ? `fa-${style}` : "fa-solid";
146
- if (family === "duotone") return `fa-duotone ${stylePrefix} fa-${id}`;
147
- return `${stylePrefix} fa-${id}`;
148
- }
149
- function parseFaClass(cls) {
150
- if (!cls) return null;
151
- const parts = cls.trim().split(/\s+/);
152
- if (parts.length < 2) return null;
153
- const hasDuotone = parts.includes("fa-duotone");
154
- const hasBrands = parts.includes("fa-brands");
155
- const prefixes = /* @__PURE__ */ new Set(["fa-solid", "fa-regular", "fa-light", "fa-thin", "fa-brands", "fa-duotone"]);
156
- const namePart = parts.find((p) => p.startsWith("fa-") && !prefixes.has(p));
157
- if (!namePart) return null;
158
- const id = namePart.replace(/^fa-/, "");
159
- if (hasBrands) return { id, family: "brands", style: null };
160
- let style = "solid";
161
- if (parts.includes("fa-regular")) style = "regular";
162
- else if (parts.includes("fa-light")) style = "light";
163
- return { id, family: hasDuotone ? "duotone" : "classic", style };
164
- }
165
-
166
- // src/components/IconPicker/useIconSearch.ts
167
- function useIconSearch(options = {}) {
168
- const { families, styles, pageSize = 100 } = options;
169
- const [catalog, setCatalog] = useState(() => getCatalog());
170
- const [catalogReady, setCatalogReady] = useState(() => isCatalogLoaded());
171
- const [searchResults, setSearchResults] = useState(null);
172
- const [loading, setLoading] = useState(true);
173
- const [searching, setSearching] = useState(false);
174
- const [error, setError] = useState(null);
175
- const [query, setQuery] = useState("");
176
- const [activeFamily, setActiveFamily] = useState("classic");
177
- const [activeStyle, setActiveStyle] = useState(null);
178
- const [page, setPage] = useState(0);
179
- const debounceRef = useRef();
180
- useEffect(() => {
181
- const unsub = subscribeCatalog(() => {
182
- setCatalog(getCatalog());
183
- setCatalogReady(isCatalogLoaded());
184
- });
185
- return unsub;
186
- }, []);
187
- useEffect(() => {
188
- loadCatalogInBackground().then(() => setLoading(false)).catch((err) => {
189
- setError(err.message);
190
- setLoading(false);
191
- });
192
- }, []);
193
- useEffect(() => {
194
- if (catalog.length > 0) setLoading(false);
195
- }, [catalog.length]);
196
- const setSearch = useCallback((q) => {
197
- if (debounceRef.current) clearTimeout(debounceRef.current);
198
- debounceRef.current = setTimeout(async () => {
199
- setPage(0);
200
- const trimmed = q.trim();
201
- setQuery(trimmed);
202
- if (!trimmed) {
203
- setSearchResults(null);
204
- setSearching(false);
205
- return;
206
- }
207
- setSearching(true);
208
- try {
209
- const results = await searchIcons(trimmed, 300);
210
- setSearchResults(results);
211
- } catch {
212
- setSearchResults(null);
213
- } finally {
214
- setSearching(false);
215
- }
216
- }, 250);
217
- }, []);
218
- const source = searchResults ?? catalog;
219
- const availableFamilies = useMemo(() => {
220
- const set = /* @__PURE__ */ new Set();
221
- for (const icon of source) {
222
- for (const f of icon.families) {
223
- if (!families || families.includes(f)) set.add(f);
224
- }
225
- }
226
- const order = ["classic", "duotone", "brands"];
227
- return order.filter((f) => set.has(f));
228
- }, [source, families]);
229
- useEffect(() => {
230
- if (availableFamilies.length > 0 && !availableFamilies.includes(activeFamily)) {
231
- setActiveFamily(availableFamilies[0]);
232
- }
233
- }, [availableFamilies, activeFamily]);
234
- const availableStyles = useMemo(() => {
235
- if (activeFamily === "brands") return [];
236
- const set = /* @__PURE__ */ new Set();
237
- for (const icon of source) {
238
- if (!icon.families.includes(activeFamily)) continue;
239
- for (const s of icon.styles) {
240
- if (!styles || styles.includes(s)) set.add(s);
241
- }
242
- }
243
- const order = ["solid", "regular", "light"];
244
- return order.filter((s) => set.has(s));
245
- }, [source, activeFamily, styles]);
246
- const handleSetFamily = useCallback((f) => {
247
- setActiveFamily(f);
248
- setActiveStyle(null);
249
- setPage(0);
250
- }, []);
251
- const filtered = useMemo(() => {
252
- let result = source;
253
- result = result.filter((icon) => icon.families.includes(activeFamily));
254
- if (activeStyle && activeFamily !== "brands") {
255
- result = result.filter((icon) => icon.styles.includes(activeStyle));
256
- }
257
- if (query && !searchResults) {
258
- const lower = query.toLowerCase();
259
- const terms = lower.split(/\s+/);
260
- result = result.filter(
261
- (icon) => terms.every((term) => icon.terms.some((t) => t.includes(term)))
262
- );
263
- }
264
- return result;
265
- }, [source, searchResults, activeFamily, activeStyle, query]);
266
- const totalPages = Math.ceil(filtered.length / pageSize);
267
- const pageIcons = useMemo(
268
- () => filtered.slice(page * pageSize, (page + 1) * pageSize),
269
- [filtered, page, pageSize]
270
- );
271
- return {
272
- loading: loading && catalog.length === 0,
273
- searching,
274
- error,
275
- query,
276
- setSearch,
277
- activeFamily,
278
- setActiveFamily: handleSetFamily,
279
- availableFamilies,
280
- activeStyle,
281
- setActiveStyle: useCallback((s) => {
282
- setActiveStyle(s);
283
- setPage(0);
284
- }, []),
285
- availableStyles,
286
- filtered,
287
- pageIcons,
288
- page,
289
- setPage,
290
- totalPages,
291
- totalCount: filtered.length,
292
- catalogReady
293
- };
294
- }
295
- var FAMILY_LABELS = {
296
- classic: "Classic",
297
- duotone: "Duotone",
298
- brands: "Brands"
299
- };
300
- var IconPickerContent = ({
301
- value,
302
- onSelect,
303
- onConfirm,
304
- families: allowedFamilies,
305
- styles: allowedStyles,
306
- pageSize = 100,
307
- className
308
- }) => {
309
- const {
310
- loading,
311
- searching,
312
- error,
313
- setSearch,
314
- activeFamily,
315
- setActiveFamily,
316
- availableFamilies,
317
- activeStyle,
318
- setActiveStyle,
319
- availableStyles,
320
- pageIcons,
321
- page,
322
- setPage,
323
- totalPages,
324
- totalCount
325
- } = useIconSearch({ families: allowedFamilies, styles: allowedStyles, pageSize });
326
- const inputRef = useRef(null);
327
- const parsed = value ? parseFaClass(value) : null;
328
- const [hoveredId, setHoveredId] = useState(null);
329
- useEffect(() => {
330
- inputRef.current?.focus();
331
- }, [loading]);
332
- const handleSelect = useCallback((id) => {
333
- const family = activeFamily;
334
- const style = family === "brands" ? null : activeStyle || "solid";
335
- const icon = {
336
- name: toFaClass(id, family, style),
337
- family,
338
- style,
339
- label: id
340
- };
341
- onSelect?.(icon);
342
- onConfirm?.(icon);
343
- }, [onSelect, onConfirm, activeFamily, activeStyle]);
344
- if (loading) {
345
- return /* @__PURE__ */ jsxs("div", { className: cn("flex flex-col items-center justify-center py-12 gap-3", className), children: [
346
- /* @__PURE__ */ jsx(Loader2, { className: "w-6 h-6 animate-spin text-muted-foreground" }),
347
- /* @__PURE__ */ jsx("p", { className: "text-sm text-muted-foreground", children: "Loading icons\u2026" })
348
- ] });
349
- }
350
- if (error) {
351
- return /* @__PURE__ */ jsxs("div", { className: cn("flex flex-col items-center justify-center py-12 gap-3", className), children: [
352
- /* @__PURE__ */ jsx(AlertCircle, { className: "w-6 h-6 text-destructive" }),
353
- /* @__PURE__ */ jsx("p", { className: "text-sm text-destructive", children: error })
354
- ] });
355
- }
356
- return /* @__PURE__ */ jsxs("div", { className: cn("space-y-3", className), children: [
357
- /* @__PURE__ */ jsxs("div", { className: "relative", children: [
358
- /* @__PURE__ */ jsx(Search, { className: "absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-muted-foreground" }),
359
- searching && /* @__PURE__ */ jsx(Loader2, { className: "absolute right-3 top-1/2 -translate-y-1/2 w-4 h-4 animate-spin text-muted-foreground" }),
360
- /* @__PURE__ */ jsx(
361
- "input",
362
- {
363
- ref: inputRef,
364
- type: "text",
365
- placeholder: "Search icons\u2026",
366
- onChange: (e) => setSearch(e.target.value),
367
- className: "w-full pl-9 pr-9 py-2 text-sm rounded-md border border-border bg-transparent focus:outline-none focus:ring-1 focus:ring-ring"
368
- }
369
- )
370
- ] }),
371
- availableFamilies.length > 1 && /* @__PURE__ */ jsx("div", { className: "flex gap-1 border-b border-border pb-2", children: availableFamilies.map((f) => /* @__PURE__ */ jsx(
372
- "button",
373
- {
374
- onClick: () => setActiveFamily(f),
375
- className: cn(
376
- "px-3 py-1.5 text-xs font-semibold rounded-md transition-colors",
377
- f === activeFamily ? "bg-primary text-primary-foreground" : "text-muted-foreground hover:bg-accent hover:text-accent-foreground"
378
- ),
379
- children: FAMILY_LABELS[f]
380
- },
381
- f
382
- )) }),
383
- availableStyles.length > 1 && /* @__PURE__ */ jsxs("div", { className: "flex gap-1 flex-wrap items-center", children: [
384
- /* @__PURE__ */ jsx(
385
- "button",
386
- {
387
- onClick: () => setActiveStyle(null),
388
- className: cn(
389
- "px-2.5 py-1 text-xs font-medium rounded-full transition-colors",
390
- !activeStyle ? "bg-foreground text-background" : "bg-muted text-muted-foreground hover:bg-accent"
391
- ),
392
- children: "All"
393
- }
394
- ),
395
- availableStyles.map((s) => /* @__PURE__ */ jsx(
396
- "button",
397
- {
398
- onClick: () => setActiveStyle(s === activeStyle ? null : s),
399
- className: cn(
400
- "px-2.5 py-1 text-xs font-medium rounded-full capitalize transition-colors",
401
- s === activeStyle ? "bg-foreground text-background" : "bg-muted text-muted-foreground hover:bg-accent"
402
- ),
403
- children: s
404
- },
405
- s
406
- )),
407
- /* @__PURE__ */ jsxs("span", { className: "ml-auto text-[11px] text-muted-foreground", children: [
408
- totalCount,
409
- " icons"
410
- ] })
411
- ] }),
412
- pageIcons.length === 0 ? /* @__PURE__ */ jsxs("div", { className: "flex flex-col items-center justify-center py-10 gap-2", children: [
413
- /* @__PURE__ */ jsx(Smile, { className: "w-6 h-6 text-muted-foreground/40" }),
414
- /* @__PURE__ */ jsx("p", { className: "text-sm text-muted-foreground", children: "No icons found" })
415
- ] }) : /* @__PURE__ */ jsx("div", { className: "grid grid-cols-8 sm:grid-cols-10 md:grid-cols-12 gap-1", children: pageIcons.map((icon) => {
416
- const isSelected = parsed?.id === icon.id && parsed?.family === activeFamily;
417
- const isHovered = hoveredId === icon.id;
418
- const displayStyle = activeFamily === "brands" ? null : activeStyle || "solid";
419
- const cssClass = toFaClass(icon.id, activeFamily, displayStyle);
420
- return /* @__PURE__ */ jsx(
421
- "button",
422
- {
423
- onClick: () => handleSelect(icon.id),
424
- onMouseEnter: () => setHoveredId(icon.id),
425
- onMouseLeave: () => setHoveredId(null),
426
- title: icon.label,
427
- className: cn(
428
- "aspect-square flex items-center justify-center rounded-md text-base transition-all",
429
- isSelected ? "bg-primary/10 text-primary ring-2 ring-primary" : isHovered ? "bg-accent text-accent-foreground scale-110" : "text-muted-foreground hover:bg-accent/50"
430
- ),
431
- children: /* @__PURE__ */ jsx("i", { className: cssClass })
432
- },
433
- icon.id
434
- );
435
- }) }),
436
- /* @__PURE__ */ jsx("div", { className: "h-6 text-center", children: hoveredId && /* @__PURE__ */ jsxs("code", { className: "text-[11px] text-muted-foreground bg-muted px-2 py-0.5 rounded", children: [
437
- "fa-",
438
- hoveredId
439
- ] }) }),
440
- totalPages > 1 && /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-center gap-2", children: [
441
- /* @__PURE__ */ jsx(
442
- "button",
443
- {
444
- onClick: () => setPage(Math.max(0, page - 1)),
445
- disabled: page === 0,
446
- className: "p-1 rounded hover:bg-accent disabled:opacity-30 transition-colors",
447
- children: /* @__PURE__ */ jsx(ChevronLeft, { className: "w-4 h-4" })
448
- }
449
- ),
450
- /* @__PURE__ */ jsxs("span", { className: "text-xs text-muted-foreground", children: [
451
- page + 1,
452
- " / ",
453
- totalPages
454
- ] }),
455
- /* @__PURE__ */ jsx(
456
- "button",
457
- {
458
- onClick: () => setPage(Math.min(totalPages - 1, page + 1)),
459
- disabled: page >= totalPages - 1,
460
- className: "p-1 rounded hover:bg-accent disabled:opacity-30 transition-colors",
461
- children: /* @__PURE__ */ jsx(ChevronRight, { className: "w-4 h-4" })
462
- }
463
- )
464
- ] })
465
- ] });
466
- };
467
- var PickerDialog = ({ open, onClose, children }) => {
468
- if (!open) return null;
469
- return /* @__PURE__ */ jsxs("div", { className: "fixed inset-0 z-50 flex items-center justify-center", children: [
470
- /* @__PURE__ */ jsx("div", { className: "absolute inset-0 bg-black/50 backdrop-blur-sm", onClick: onClose }),
471
- /* @__PURE__ */ jsxs("div", { className: "relative z-10 w-full max-w-xl max-h-[80vh] bg-background rounded-xl shadow-2xl border border-border flex flex-col overflow-hidden mx-4", children: [
472
- /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between px-4 py-3 border-b border-border", children: [
473
- /* @__PURE__ */ jsx("h3", { className: "text-sm font-semibold text-foreground", children: "Select Icon" }),
474
- /* @__PURE__ */ jsx("button", { onClick: onClose, className: "p-1 rounded hover:bg-accent transition-colors", children: /* @__PURE__ */ jsx(X, { className: "w-4 h-4 text-muted-foreground" }) })
475
- ] }),
476
- /* @__PURE__ */ jsx("div", { className: "flex-1 overflow-y-auto p-4", children })
477
- ] })
478
- ] });
479
- };
480
- var IconPicker = (props) => {
481
- const { mode = "inline", trigger, open: controlledOpen, onClose, value, onSelect, className, ...rest } = props;
482
- const [internalOpen, setInternalOpen] = useState(false);
483
- const isOpen = controlledOpen ?? internalOpen;
484
- const handleClose = useCallback(() => {
485
- setInternalOpen(false);
486
- onClose?.();
487
- }, [onClose]);
488
- const handleSelectAndClose = useCallback((icon) => {
489
- onSelect?.(icon);
490
- handleClose();
491
- }, [onSelect, handleClose]);
492
- if (mode === "inline") {
493
- return /* @__PURE__ */ jsx(IconPickerContent, { value, onSelect, className, ...rest });
494
- }
495
- const parsed = value ? parseFaClass(value) : null;
496
- const triggerElement = trigger || /* @__PURE__ */ jsxs("div", { className: cn(
497
- "inline-flex items-center gap-2 px-3 py-2 rounded-md border border-border cursor-pointer",
498
- "hover:border-ring transition-colors",
499
- className
500
- ), children: [
501
- parsed ? /* @__PURE__ */ jsxs(Fragment, { children: [
502
- /* @__PURE__ */ jsx("i", { className: value }),
503
- /* @__PURE__ */ jsxs("span", { className: "text-sm text-muted-foreground", children: [
504
- "fa-",
505
- parsed.id
506
- ] })
507
- ] }) : /* @__PURE__ */ jsx("span", { className: "text-sm text-muted-foreground", children: "Choose icon\u2026" }),
508
- /* @__PURE__ */ jsx(ChevronRight, { className: "w-3.5 h-3.5 text-muted-foreground ml-auto" })
509
- ] });
510
- return /* @__PURE__ */ jsxs(Fragment, { children: [
511
- /* @__PURE__ */ jsx("span", { onClick: () => setInternalOpen(true), className: "cursor-pointer", children: triggerElement }),
512
- /* @__PURE__ */ jsx(PickerDialog, { open: isOpen, onClose: handleClose, children: /* @__PURE__ */ jsx(IconPickerContent, { value, onSelect: handleSelectAndClose, ...rest }) })
513
- ] });
514
- };
515
-
516
- export { IconPicker, parseFaClass, toFaClass };
517
- //# sourceMappingURL=chunk-HLFNSOPD.js.map
518
- //# sourceMappingURL=chunk-HLFNSOPD.js.map