@nous-research/ui 0.16.0 → 0.17.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.
- package/CHANGELOG.md +39 -0
- package/dist/hooks/use-below-breakpoint.d.ts +2 -0
- package/dist/hooks/use-below-breakpoint.js +17 -0
- package/dist/hooks/use-confirm-delete.d.ts +10 -0
- package/dist/hooks/use-confirm-delete.js +35 -0
- package/dist/hooks/use-toast.d.ts +7 -0
- package/dist/hooks/use-toast.js +21 -0
- package/dist/index.d.ts +11 -1
- package/dist/index.js +23 -2
- package/dist/ui/components/bottom-sheet.d.ts +15 -0
- package/dist/ui/components/bottom-sheet.js +192 -0
- package/dist/ui/components/card.d.ts +5 -0
- package/dist/ui/components/card.js +74 -0
- package/dist/ui/components/checkbox.d.ts +1 -1
- package/dist/ui/components/checkbox.js +2 -2
- package/dist/ui/components/confirm-dialog.d.ts +13 -0
- package/dist/ui/components/confirm-dialog.js +113 -0
- package/dist/ui/components/dialog.d.ts +15 -0
- package/dist/ui/components/dialog.js +171 -0
- package/dist/ui/components/input.d.ts +1 -0
- package/dist/ui/components/input.js +21 -0
- package/dist/ui/components/label.d.ts +1 -0
- package/dist/ui/components/label.js +18 -0
- package/dist/ui/components/separator.d.ts +5 -0
- package/dist/ui/components/separator.js +22 -0
- package/dist/ui/components/toast.d.ts +8 -0
- package/dist/ui/components/toast.js +39 -0
- package/dist/ui/globals.css +14 -2
- package/package.json +2 -2
- package/src/hooks/use-below-breakpoint.ts +21 -0
- package/src/hooks/use-confirm-delete.ts +43 -0
- package/src/hooks/use-toast.ts +29 -0
- package/src/index.ts +22 -1
- package/src/ui/components/animated-count.stories.tsx +1 -1
- package/src/ui/components/ascii.stories.tsx +1 -1
- package/src/ui/components/badge.stories.tsx +1 -1
- package/src/ui/components/blend-mode.stories.tsx +1 -1
- package/src/ui/components/blink.stories.tsx +1 -1
- package/src/ui/components/bottom-sheet.stories.tsx +43 -0
- package/src/ui/components/bottom-sheet.tsx +227 -0
- package/src/ui/components/button.stories.tsx +1 -1
- package/src/ui/components/card.stories.tsx +63 -0
- package/src/ui/components/card.tsx +85 -0
- package/src/ui/components/checkbox.stories.tsx +1 -1
- package/src/ui/components/checkbox.tsx +1 -1
- package/src/ui/components/command-block.stories.tsx +1 -1
- package/src/ui/components/confirm-dialog.stories.tsx +91 -0
- package/src/ui/components/confirm-dialog.tsx +130 -0
- package/src/ui/components/dialog.stories.tsx +169 -0
- package/src/ui/components/dialog.tsx +177 -0
- package/src/ui/components/dropdown-menu.stories.tsx +1 -1
- package/src/ui/components/fit-text/index.stories.tsx +1 -1
- package/src/ui/components/forms.stories.tsx +173 -0
- package/src/ui/components/graphs/index.stories.tsx +1 -1
- package/src/ui/components/hover-bg.stories.tsx +1 -1
- package/src/ui/components/image-distortion.stories.tsx +1 -1
- package/src/ui/components/input.stories.tsx +39 -0
- package/src/ui/components/input.tsx +20 -0
- package/src/ui/components/label.stories.tsx +26 -0
- package/src/ui/components/label.tsx +16 -0
- package/src/ui/components/list-item.stories.tsx +1 -1
- package/src/ui/components/poster.stories.tsx +1 -1
- package/src/ui/components/progress.stories.tsx +1 -1
- package/src/ui/components/scramble.stories.tsx +1 -1
- package/src/ui/components/segmented.stories.tsx +1 -1
- package/src/ui/components/select.stories.tsx +1 -1
- package/src/ui/components/separator.stories.tsx +33 -0
- package/src/ui/components/separator.tsx +24 -0
- package/src/ui/components/spinner.stories.tsx +1 -1
- package/src/ui/components/stats.stories.tsx +1 -1
- package/src/ui/components/switch.stories.tsx +1 -1
- package/src/ui/components/tabs.stories.tsx +1 -1
- package/src/ui/components/terminal-demo.stories.tsx +1 -1
- package/src/ui/components/theme-toggle.stories.tsx +1 -1
- package/src/ui/components/tier-card.stories.tsx +1 -1
- package/src/ui/components/toast.stories.tsx +55 -0
- package/src/ui/components/toast.tsx +49 -0
- package/src/ui/components/tv.stories.tsx +1 -1
- package/src/ui/components/watchlist.stories.tsx +1 -1
- package/src/ui/globals.css +14 -2
- package/dist/ui/components/modal/index.d.ts +0 -8
- package/dist/ui/components/modal/index.js +0 -35
- package/dist/ui/components/modal/modal.css +0 -36
- package/src/ui/components/modal/index.stories.tsx +0 -46
- package/src/ui/components/modal/index.tsx +0 -48
- package/src/ui/components/modal/modal.css +0 -36
package/CHANGELOG.md
CHANGED
|
@@ -2,6 +2,45 @@
|
|
|
2
2
|
|
|
3
3
|
All notable changes to `@nous-research/ui` are documented here. Format based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/).
|
|
4
4
|
|
|
5
|
+
## 0.17.0
|
|
6
|
+
|
|
7
|
+
### Component promotion and Radix Dialog consolidation
|
|
8
|
+
|
|
9
|
+
Promotes reusable components and hooks from `hermes-agent/web` into the shared design system, and consolidates all dialog/modal implementations on Radix UI primitives.
|
|
10
|
+
|
|
11
|
+
#### Added
|
|
12
|
+
|
|
13
|
+
- **`Dialog` compound component** — general-purpose modal built on Radix Dialog primitive. Compound API: `Dialog`, `DialogTrigger`, `DialogContent`, `DialogHeader`, `DialogFooter`, `DialogTitle`, `DialogDescription`, `DialogClose`. Dismissible via backdrop click or ESC.
|
|
14
|
+
- **`Input`** — styled text input with focus ring, disabled state, courier font.
|
|
15
|
+
- **`Label`** — form label with mondwest font, uppercase tracking, peer-disabled opacity.
|
|
16
|
+
- **`Separator`** — horizontal/vertical divider with `role="separator"`.
|
|
17
|
+
- **`Card`** compound component — `Card`, `CardHeader`, `CardTitle`, `CardDescription`, `CardContent` with themeable `--component-card-*` CSS custom properties.
|
|
18
|
+
- **`Toast`** — portal-based success/error notification with enter/exit animations.
|
|
19
|
+
- **`BottomSheet`** — mobile-first bottom sheet with drag-to-dismiss, backdrop, reduced-motion support.
|
|
20
|
+
- **`useToast`** — toast state management with auto-dismiss timer.
|
|
21
|
+
- **`useBelowBreakpoint`** — `matchMedia`-based responsive breakpoint query (SSR-safe).
|
|
22
|
+
- **`useConfirmDelete`** — generic confirm-delete flow state machine.
|
|
23
|
+
- **Forms overview story** — showcases all form controls in one view.
|
|
24
|
+
- Toast keyframes (`toast-in`, `toast-out`) in `globals.css`.
|
|
25
|
+
|
|
26
|
+
#### Changed
|
|
27
|
+
|
|
28
|
+
- **`ConfirmDialog` rebuilt on Radix AlertDialog** — same external API (`open`, `onCancel`, `onConfirm`, `title`, `description`, `destructive`, `loading`), but ESC handling, focus trapping, scroll lock, and portal rendering are now handled by Radix. Backdrop click does not dismiss (correct for destructive confirms).
|
|
29
|
+
- **`Checkbox` migrated to unified `radix-ui` package** — import changed from `@radix-ui/react-checkbox` to `radix-ui`.
|
|
30
|
+
- **Dependency:** `@radix-ui/react-checkbox` replaced with unified `radix-ui` package.
|
|
31
|
+
- Storybook stories organized into categories (Forms, Feedback, Overlays, Data Display, Layout, Effects).
|
|
32
|
+
- Kitchen-sink demo updated to use new `Dialog`.
|
|
33
|
+
|
|
34
|
+
#### Removed
|
|
35
|
+
|
|
36
|
+
- **`Modal` component** (`modal/index.tsx`, `modal.css`) — unused by any consumer app; replaced by `Dialog`.
|
|
37
|
+
- **`useModalBehavior` hook** — functionality now provided natively by Radix Dialog/AlertDialog primitives.
|
|
38
|
+
- Stale `modal.css` import from `globals.css`.
|
|
39
|
+
|
|
40
|
+
#### Fixed
|
|
41
|
+
|
|
42
|
+
- Portal text color: dialogs now use `text-foreground-base` for correct rendering outside the blend-mode stack.
|
|
43
|
+
|
|
5
44
|
## 0.16.0
|
|
6
45
|
|
|
7
46
|
### Usability overhaul — typography, contrast, theming
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { useEffect, useState } from "react";
|
|
3
|
+
export function useBelowBreakpoint(px) {
|
|
4
|
+
const query = `(max-width: ${px - 1}px)`;
|
|
5
|
+
const [matches, setMatches] = useState(
|
|
6
|
+
() => typeof window !== "undefined" ? window.matchMedia(query).matches : false
|
|
7
|
+
);
|
|
8
|
+
useEffect(() => {
|
|
9
|
+
const mql = window.matchMedia(query);
|
|
10
|
+
const sync = () => setMatches(mql.matches);
|
|
11
|
+
sync();
|
|
12
|
+
mql.addEventListener("change", sync);
|
|
13
|
+
return () => mql.removeEventListener("change", sync);
|
|
14
|
+
}, [query]);
|
|
15
|
+
return matches;
|
|
16
|
+
}
|
|
17
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZUVmZmVjdCwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcblxuLyoqIFRydWUgd2hlbiB2aWV3cG9ydCB3aWR0aCBpcyBzdHJpY3RseSBiZWxvdyBgcHhgLiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHVzZUJlbG93QnJlYWtwb2ludChweDogbnVtYmVyKSB7XG4gIGNvbnN0IHF1ZXJ5ID0gYChtYXgtd2lkdGg6ICR7cHggLSAxfXB4KWBcbiAgY29uc3QgW21hdGNoZXMsIHNldE1hdGNoZXNdID0gdXNlU3RhdGUoKCkgPT5cbiAgICB0eXBlb2Ygd2luZG93ICE9PSAndW5kZWZpbmVkJyA/IHdpbmRvdy5tYXRjaE1lZGlhKHF1ZXJ5KS5tYXRjaGVzIDogZmFsc2VcbiAgKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3QgbXFsID0gd2luZG93Lm1hdGNoTWVkaWEocXVlcnkpXG4gICAgY29uc3Qgc3luYyA9ICgpID0+IHNldE1hdGNoZXMobXFsLm1hdGNoZXMpXG4gICAgc3luYygpXG4gICAgbXFsLmFkZEV2ZW50TGlzdGVuZXIoJ2NoYW5nZScsIHN5bmMpXG4gICAgcmV0dXJuICgpID0+IG1xbC5yZW1vdmVFdmVudExpc3RlbmVyKCdjaGFuZ2UnLCBzeW5jKVxuICB9LCBbcXVlcnldKVxuXG4gIHJldHVybiBtYXRjaGVzXG59XG4iXSwKICAibWFwcGluZ3MiOiAiO0FBRUEsU0FBUyxXQUFXLGdCQUFnQjtBQUc3QixnQkFBUyxtQkFBbUIsSUFBWTtBQUM3QyxRQUFNLFFBQVEsZUFBZSxLQUFLLENBQUM7QUFDbkMsUUFBTSxDQUFDLFNBQVMsVUFBVSxJQUFJO0FBQUEsSUFBUyxNQUNyQyxPQUFPLFdBQVcsY0FBYyxPQUFPLFdBQVcsS0FBSyxFQUFFLFVBQVU7QUFBQSxFQUNyRTtBQUVBLFlBQVUsTUFBTTtBQUNkLFVBQU0sTUFBTSxPQUFPLFdBQVcsS0FBSztBQUNuQyxVQUFNLE9BQU8sTUFBTSxXQUFXLElBQUksT0FBTztBQUN6QyxTQUFLO0FBQ0wsUUFBSSxpQkFBaUIsVUFBVSxJQUFJO0FBQ25DLFdBQU8sTUFBTSxJQUFJLG9CQUFvQixVQUFVLElBQUk7QUFBQSxFQUNyRCxHQUFHLENBQUMsS0FBSyxDQUFDO0FBRVYsU0FBTztBQUNUOyIsCiAgIm5hbWVzIjogW10KfQo=
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export declare function useConfirmDelete<TId>({ onDelete }: {
|
|
2
|
+
onDelete: (id: TId) => Promise<void>;
|
|
3
|
+
}): {
|
|
4
|
+
readonly cancel: () => void;
|
|
5
|
+
readonly confirm: () => Promise<void>;
|
|
6
|
+
readonly isDeleting: boolean;
|
|
7
|
+
readonly isOpen: boolean;
|
|
8
|
+
readonly pendingId: TId | null;
|
|
9
|
+
readonly requestDelete: (id: TId) => void;
|
|
10
|
+
};
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { useCallback, useState } from "react";
|
|
3
|
+
export function useConfirmDelete({
|
|
4
|
+
onDelete
|
|
5
|
+
}) {
|
|
6
|
+
const [pendingId, setPendingId] = useState(null);
|
|
7
|
+
const [isDeleting, setIsDeleting] = useState(false);
|
|
8
|
+
const requestDelete = useCallback((id) => {
|
|
9
|
+
setPendingId(id);
|
|
10
|
+
}, []);
|
|
11
|
+
const cancel = useCallback(() => {
|
|
12
|
+
if (!isDeleting) setPendingId(null);
|
|
13
|
+
}, [isDeleting]);
|
|
14
|
+
const confirm = useCallback(async () => {
|
|
15
|
+
if (pendingId === null) return;
|
|
16
|
+
const id = pendingId;
|
|
17
|
+
setIsDeleting(true);
|
|
18
|
+
try {
|
|
19
|
+
await onDelete(id);
|
|
20
|
+
setPendingId(null);
|
|
21
|
+
} catch {
|
|
22
|
+
} finally {
|
|
23
|
+
setIsDeleting(false);
|
|
24
|
+
}
|
|
25
|
+
}, [pendingId, onDelete]);
|
|
26
|
+
return {
|
|
27
|
+
cancel,
|
|
28
|
+
confirm,
|
|
29
|
+
isDeleting,
|
|
30
|
+
isOpen: pendingId !== null,
|
|
31
|
+
pendingId,
|
|
32
|
+
requestDelete
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuXG5leHBvcnQgZnVuY3Rpb24gdXNlQ29uZmlybURlbGV0ZTxUSWQ+KHtcbiAgb25EZWxldGVcbn06IHtcbiAgb25EZWxldGU6IChpZDogVElkKSA9PiBQcm9taXNlPHZvaWQ+XG59KSB7XG4gIGNvbnN0IFtwZW5kaW5nSWQsIHNldFBlbmRpbmdJZF0gPSB1c2VTdGF0ZTxUSWQgfCBudWxsPihudWxsKVxuICBjb25zdCBbaXNEZWxldGluZywgc2V0SXNEZWxldGluZ10gPSB1c2VTdGF0ZShmYWxzZSlcblxuICBjb25zdCByZXF1ZXN0RGVsZXRlID0gdXNlQ2FsbGJhY2soKGlkOiBUSWQpID0+IHtcbiAgICBzZXRQZW5kaW5nSWQoaWQpXG4gIH0sIFtdKVxuXG4gIGNvbnN0IGNhbmNlbCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBpZiAoIWlzRGVsZXRpbmcpIHNldFBlbmRpbmdJZChudWxsKVxuICB9LCBbaXNEZWxldGluZ10pXG5cbiAgY29uc3QgY29uZmlybSA9IHVzZUNhbGxiYWNrKGFzeW5jICgpID0+IHtcbiAgICBpZiAocGVuZGluZ0lkID09PSBudWxsKSByZXR1cm5cbiAgICBjb25zdCBpZCA9IHBlbmRpbmdJZFxuICAgIHNldElzRGVsZXRpbmcodHJ1ZSlcbiAgICB0cnkge1xuICAgICAgYXdhaXQgb25EZWxldGUoaWQpXG4gICAgICBzZXRQZW5kaW5nSWQobnVsbClcbiAgICB9IGNhdGNoIHtcbiAgICAgIC8vIERpYWxvZyBzdGF5cyBvcGVuOyBjYWxsZXIgY2FuIHN1cmZhY2UgZXJyb3JzIGluIG9uRGVsZXRlXG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIHNldElzRGVsZXRpbmcoZmFsc2UpXG4gICAgfVxuICB9LCBbcGVuZGluZ0lkLCBvbkRlbGV0ZV0pXG5cbiAgcmV0dXJuIHtcbiAgICBjYW5jZWwsXG4gICAgY29uZmlybSxcbiAgICBpc0RlbGV0aW5nLFxuICAgIGlzT3BlbjogcGVuZGluZ0lkICE9PSBudWxsLFxuICAgIHBlbmRpbmdJZCxcbiAgICByZXF1ZXN0RGVsZXRlXG4gIH0gYXMgY29uc3Rcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7QUFFQSxTQUFTLGFBQWEsZ0JBQWdCO0FBRS9CLGdCQUFTLGlCQUFzQjtBQUFBLEVBQ3BDO0FBQ0YsR0FFRztBQUNELFFBQU0sQ0FBQyxXQUFXLFlBQVksSUFBSSxTQUFxQixJQUFJO0FBQzNELFFBQU0sQ0FBQyxZQUFZLGFBQWEsSUFBSSxTQUFTLEtBQUs7QUFFbEQsUUFBTSxnQkFBZ0IsWUFBWSxDQUFDLE9BQVk7QUFDN0MsaUJBQWEsRUFBRTtBQUFBLEVBQ2pCLEdBQUcsQ0FBQyxDQUFDO0FBRUwsUUFBTSxTQUFTLFlBQVksTUFBTTtBQUMvQixRQUFJLENBQUMsV0FBWSxjQUFhLElBQUk7QUFBQSxFQUNwQyxHQUFHLENBQUMsVUFBVSxDQUFDO0FBRWYsUUFBTSxVQUFVLFlBQVksWUFBWTtBQUN0QyxRQUFJLGNBQWMsS0FBTTtBQUN4QixVQUFNLEtBQUs7QUFDWCxrQkFBYyxJQUFJO0FBQ2xCLFFBQUk7QUFDRixZQUFNLFNBQVMsRUFBRTtBQUNqQixtQkFBYSxJQUFJO0FBQUEsSUFDbkIsUUFBUTtBQUFBLElBRVIsVUFBRTtBQUNBLG9CQUFjLEtBQUs7QUFBQSxJQUNyQjtBQUFBLEVBQ0YsR0FBRyxDQUFDLFdBQVcsUUFBUSxDQUFDO0FBRXhCLFNBQU87QUFBQSxJQUNMO0FBQUEsSUFDQTtBQUFBLElBQ0E7QUFBQSxJQUNBLFFBQVEsY0FBYztBQUFBLElBQ3RCO0FBQUEsSUFDQTtBQUFBLEVBQ0Y7QUFDRjsiLAogICJuYW1lcyI6IFtdCn0K
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { useCallback, useEffect, useRef, useState } from "react";
|
|
3
|
+
export function useToast(duration = 3e3) {
|
|
4
|
+
const [toast, setToast] = useState(null);
|
|
5
|
+
const timerRef = useRef(null);
|
|
6
|
+
useEffect(() => {
|
|
7
|
+
return () => {
|
|
8
|
+
if (timerRef.current) clearTimeout(timerRef.current);
|
|
9
|
+
};
|
|
10
|
+
}, []);
|
|
11
|
+
const showToast = useCallback(
|
|
12
|
+
(message, type) => {
|
|
13
|
+
if (timerRef.current) clearTimeout(timerRef.current);
|
|
14
|
+
setToast({ message, type });
|
|
15
|
+
timerRef.current = setTimeout(() => setToast(null), duration);
|
|
16
|
+
},
|
|
17
|
+
[duration]
|
|
18
|
+
);
|
|
19
|
+
return { showToast, toast };
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiJ3VzZSBjbGllbnQnXG5cbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VFZmZlY3QsIHVzZVJlZiwgdXNlU3RhdGUgfSBmcm9tICdyZWFjdCdcblxuZXhwb3J0IGZ1bmN0aW9uIHVzZVRvYXN0KGR1cmF0aW9uID0gMzAwMCkge1xuICBjb25zdCBbdG9hc3QsIHNldFRvYXN0XSA9IHVzZVN0YXRlPHtcbiAgICBtZXNzYWdlOiBzdHJpbmdcbiAgICB0eXBlOiAnZXJyb3InIHwgJ3N1Y2Nlc3MnXG4gIH0gfCBudWxsPihudWxsKVxuXG4gIGNvbnN0IHRpbWVyUmVmID0gdXNlUmVmPFJldHVyblR5cGU8dHlwZW9mIHNldFRpbWVvdXQ+IHwgbnVsbD4obnVsbClcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIHJldHVybiAoKSA9PiB7XG4gICAgICBpZiAodGltZXJSZWYuY3VycmVudCkgY2xlYXJUaW1lb3V0KHRpbWVyUmVmLmN1cnJlbnQpXG4gICAgfVxuICB9LCBbXSlcblxuICBjb25zdCBzaG93VG9hc3QgPSB1c2VDYWxsYmFjayhcbiAgICAobWVzc2FnZTogc3RyaW5nLCB0eXBlOiAnZXJyb3InIHwgJ3N1Y2Nlc3MnKSA9PiB7XG4gICAgICBpZiAodGltZXJSZWYuY3VycmVudCkgY2xlYXJUaW1lb3V0KHRpbWVyUmVmLmN1cnJlbnQpXG4gICAgICBzZXRUb2FzdCh7IG1lc3NhZ2UsIHR5cGUgfSlcbiAgICAgIHRpbWVyUmVmLmN1cnJlbnQgPSBzZXRUaW1lb3V0KCgpID0+IHNldFRvYXN0KG51bGwpLCBkdXJhdGlvbilcbiAgICB9LFxuICAgIFtkdXJhdGlvbl1cbiAgKVxuXG4gIHJldHVybiB7IHNob3dUb2FzdCwgdG9hc3QgfVxufVxuIl0sCiAgIm1hcHBpbmdzIjogIjtBQUVBLFNBQVMsYUFBYSxXQUFXLFFBQVEsZ0JBQWdCO0FBRWxELGdCQUFTLFNBQVMsV0FBVyxLQUFNO0FBQ3hDLFFBQU0sQ0FBQyxPQUFPLFFBQVEsSUFBSSxTQUdoQixJQUFJO0FBRWQsUUFBTSxXQUFXLE9BQTZDLElBQUk7QUFFbEUsWUFBVSxNQUFNO0FBQ2QsV0FBTyxNQUFNO0FBQ1gsVUFBSSxTQUFTLFFBQVMsY0FBYSxTQUFTLE9BQU87QUFBQSxJQUNyRDtBQUFBLEVBQ0YsR0FBRyxDQUFDLENBQUM7QUFFTCxRQUFNLFlBQVk7QUFBQSxJQUNoQixDQUFDLFNBQWlCLFNBQThCO0FBQzlDLFVBQUksU0FBUyxRQUFTLGNBQWEsU0FBUyxPQUFPO0FBQ25ELGVBQVMsRUFBRSxTQUFTLEtBQUssQ0FBQztBQUMxQixlQUFTLFVBQVUsV0FBVyxNQUFNLFNBQVMsSUFBSSxHQUFHLFFBQVE7QUFBQSxJQUM5RDtBQUFBLElBQ0EsQ0FBQyxRQUFRO0FBQUEsRUFDWDtBQUVBLFNBQU8sRUFBRSxXQUFXLE1BQU07QUFDNUI7IiwKICAibmFtZXMiOiBbXQp9Cg==
|
package/dist/index.d.ts
CHANGED
|
@@ -1,13 +1,16 @@
|
|
|
1
1
|
export { AnimatedCount, useAnimatedCount } from './ui/components/animated-count';
|
|
2
2
|
export { AsciiSkeleton, Scramble as AsciiScramble } from './ui/components/ascii';
|
|
3
3
|
export { Badge } from './ui/components/badge';
|
|
4
|
+
export { BottomSheet } from './ui/components/bottom-sheet';
|
|
4
5
|
export { NousGirlBadge } from './ui/components/badges/nous-girl';
|
|
5
6
|
export { BlendMode, useBlendMode, withBlendMode } from './ui/components/blend-mode';
|
|
6
7
|
export type { BlendModeProps } from './ui/components/blend-mode';
|
|
7
8
|
export { Blink } from './ui/components/blink';
|
|
8
9
|
export { Button } from './ui/components/button';
|
|
10
|
+
export { Card, CardContent, CardDescription, CardHeader, CardTitle } from './ui/components/card';
|
|
9
11
|
export { Checkbox } from './ui/components/checkbox';
|
|
10
12
|
export { CommandBlock, CopyButton } from './ui/components/command-block';
|
|
13
|
+
export { ConfirmDialog } from './ui/components/confirm-dialog';
|
|
11
14
|
export { Cursor } from './ui/components/cursor';
|
|
12
15
|
export { DropdownMenu } from './ui/components/dropdown-menu';
|
|
13
16
|
export { FitText } from './ui/components/fit-text';
|
|
@@ -19,9 +22,11 @@ export { DiscordIcon } from './ui/components/icons/discord';
|
|
|
19
22
|
export { GitHubIcon } from './ui/components/icons/github';
|
|
20
23
|
export { ImageDistortion } from './ui/components/image-distortion';
|
|
21
24
|
export type { AutoPlayPattern } from './ui/components/image-distortion';
|
|
25
|
+
export { Input } from './ui/components/input';
|
|
26
|
+
export { Label } from './ui/components/label';
|
|
22
27
|
export { LevaClient } from './ui/components/leva-client';
|
|
23
28
|
export { ListItem } from './ui/components/list-item';
|
|
24
|
-
export {
|
|
29
|
+
export { Dialog, DialogClose, DialogContent, DialogDescription, DialogFooter, DialogHeader, DialogOverlay, DialogPortal, DialogTitle, DialogTrigger } from './ui/components/dialog';
|
|
25
30
|
export { FilterGroup, Segmented } from './ui/components/segmented';
|
|
26
31
|
export { Switch } from './ui/components/switch';
|
|
27
32
|
export { Tabs, TabsList, TabsTrigger } from './ui/components/tabs';
|
|
@@ -35,12 +40,14 @@ export { SceneCanvas } from './ui/components/scene-canvas';
|
|
|
35
40
|
export { Scramble } from './ui/components/scramble';
|
|
36
41
|
export { Select, SelectOption } from './ui/components/select';
|
|
37
42
|
export { SelectionSwitcher } from './ui/components/selection-switcher';
|
|
43
|
+
export { Separator } from './ui/components/separator';
|
|
38
44
|
export { Spinner } from './ui/components/spinner';
|
|
39
45
|
export { Stats } from './ui/components/stats';
|
|
40
46
|
export { TerminalDemo } from './ui/components/terminal-demo';
|
|
41
47
|
export type { TerminalDemoStep } from './ui/components/terminal-demo';
|
|
42
48
|
export { ThemeToggle } from './ui/components/theme-toggle';
|
|
43
49
|
export { TierCard } from './ui/components/tier-card';
|
|
50
|
+
export { Toast } from './ui/components/toast';
|
|
44
51
|
export type { TierCardPrice, TierCardProps } from './ui/components/tier-card';
|
|
45
52
|
export { TV } from './ui/components/tv';
|
|
46
53
|
export { Watchlist } from './ui/components/watchlist';
|
|
@@ -63,7 +70,10 @@ export { cn, clamp, smoothstep, hexToVec3, truncate, stripWpStyles } from './uti
|
|
|
63
70
|
export { polyRef } from './utils';
|
|
64
71
|
export type { PolyComponent, PolyProps, PolyRef } from './utils';
|
|
65
72
|
export { hexToRgb, rgbToHex, colorDodge, colorMix } from './utils/color';
|
|
73
|
+
export { useBelowBreakpoint } from './hooks/use-below-breakpoint';
|
|
66
74
|
export { useCappedFrame } from './hooks/use-capped-frame';
|
|
75
|
+
export { useConfirmDelete } from './hooks/use-confirm-delete';
|
|
67
76
|
export { useCssVarDims } from './hooks/use-css-var-dims';
|
|
68
77
|
export { $gpuTier, useGpuTier } from './hooks/use-gpu-tier';
|
|
69
78
|
export { useSmoothControls, getControlAtom, setControlValue } from './hooks/use-smooth-controls';
|
|
79
|
+
export { useToast } from './hooks/use-toast';
|
package/dist/index.js
CHANGED
|
@@ -1,12 +1,15 @@
|
|
|
1
1
|
export { AnimatedCount, useAnimatedCount } from "./ui/components/animated-count.js";
|
|
2
2
|
export { AsciiSkeleton, Scramble as AsciiScramble } from "./ui/components/ascii.js";
|
|
3
3
|
export { Badge } from "./ui/components/badge.js";
|
|
4
|
+
export { BottomSheet } from "./ui/components/bottom-sheet.js";
|
|
4
5
|
export { NousGirlBadge } from "./ui/components/badges/nous-girl.js";
|
|
5
6
|
export { BlendMode, useBlendMode, withBlendMode } from "./ui/components/blend-mode.js";
|
|
6
7
|
export { Blink } from "./ui/components/blink.js";
|
|
7
8
|
export { Button } from "./ui/components/button.js";
|
|
9
|
+
export { Card, CardContent, CardDescription, CardHeader, CardTitle } from "./ui/components/card.js";
|
|
8
10
|
export { Checkbox } from "./ui/components/checkbox.js";
|
|
9
11
|
export { CommandBlock, CopyButton } from "./ui/components/command-block.js";
|
|
12
|
+
export { ConfirmDialog } from "./ui/components/confirm-dialog.js";
|
|
10
13
|
export { Cursor } from "./ui/components/cursor.js";
|
|
11
14
|
export { DropdownMenu } from "./ui/components/dropdown-menu.js";
|
|
12
15
|
export { FitText } from "./ui/components/fit-text/index.js";
|
|
@@ -17,9 +20,22 @@ export * as Icons from "./ui/components/icons/index.js";
|
|
|
17
20
|
export { DiscordIcon } from "./ui/components/icons/discord.js";
|
|
18
21
|
export { GitHubIcon } from "./ui/components/icons/github.js";
|
|
19
22
|
export { ImageDistortion } from "./ui/components/image-distortion.js";
|
|
23
|
+
export { Input } from "./ui/components/input.js";
|
|
24
|
+
export { Label } from "./ui/components/label.js";
|
|
20
25
|
export { LevaClient } from "./ui/components/leva-client.js";
|
|
21
26
|
export { ListItem } from "./ui/components/list-item.js";
|
|
22
|
-
export {
|
|
27
|
+
export {
|
|
28
|
+
Dialog,
|
|
29
|
+
DialogClose,
|
|
30
|
+
DialogContent,
|
|
31
|
+
DialogDescription,
|
|
32
|
+
DialogFooter,
|
|
33
|
+
DialogHeader,
|
|
34
|
+
DialogOverlay,
|
|
35
|
+
DialogPortal,
|
|
36
|
+
DialogTitle,
|
|
37
|
+
DialogTrigger
|
|
38
|
+
} from "./ui/components/dialog.js";
|
|
23
39
|
export { FilterGroup, Segmented } from "./ui/components/segmented.js";
|
|
24
40
|
export { Switch } from "./ui/components/switch.js";
|
|
25
41
|
export { Tabs, TabsList, TabsTrigger } from "./ui/components/tabs.js";
|
|
@@ -48,11 +64,13 @@ export { SceneCanvas } from "./ui/components/scene-canvas.js";
|
|
|
48
64
|
export { Scramble } from "./ui/components/scramble.js";
|
|
49
65
|
export { Select, SelectOption } from "./ui/components/select.js";
|
|
50
66
|
export { SelectionSwitcher } from "./ui/components/selection-switcher.js";
|
|
67
|
+
export { Separator } from "./ui/components/separator.js";
|
|
51
68
|
export { Spinner } from "./ui/components/spinner.js";
|
|
52
69
|
export { Stats } from "./ui/components/stats.js";
|
|
53
70
|
export { TerminalDemo } from "./ui/components/terminal-demo.js";
|
|
54
71
|
export { ThemeToggle } from "./ui/components/theme-toggle.js";
|
|
55
72
|
export { TierCard } from "./ui/components/tier-card.js";
|
|
73
|
+
export { Toast } from "./ui/components/toast.js";
|
|
56
74
|
export { TV } from "./ui/components/tv.js";
|
|
57
75
|
export { Watchlist } from "./ui/components/watchlist.js";
|
|
58
76
|
export { Typography } from "./ui/components/typography/index.js";
|
|
@@ -75,7 +93,9 @@ export {
|
|
|
75
93
|
export { cn, clamp, smoothstep, hexToVec3, truncate, stripWpStyles } from "./utils/index.js";
|
|
76
94
|
export { polyRef } from "./utils/index.js";
|
|
77
95
|
export { hexToRgb, rgbToHex, colorDodge, colorMix } from "./utils/color.js";
|
|
96
|
+
export { useBelowBreakpoint } from "./hooks/use-below-breakpoint.js";
|
|
78
97
|
export { useCappedFrame } from "./hooks/use-capped-frame.js";
|
|
98
|
+
export { useConfirmDelete } from "./hooks/use-confirm-delete.js";
|
|
79
99
|
export { useCssVarDims } from "./hooks/use-css-var-dims.js";
|
|
80
100
|
export { $gpuTier, useGpuTier } from "./hooks/use-gpu-tier.js";
|
|
81
101
|
export {
|
|
@@ -83,4 +103,5 @@ export {
|
|
|
83
103
|
getControlAtom,
|
|
84
104
|
setControlValue
|
|
85
105
|
} from "./hooks/use-smooth-controls.js";
|
|
86
|
-
|
|
106
|
+
export { useToast } from "./hooks/use-toast.js";
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["<stdin>"],
  "sourcesContent": ["export { AnimatedCount, useAnimatedCount } from './ui/components/animated-count'\nexport { AsciiSkeleton, Scramble as AsciiScramble } from './ui/components/ascii'\nexport { Badge } from './ui/components/badge'\nexport { BottomSheet } from './ui/components/bottom-sheet'\nexport { NousGirlBadge } from './ui/components/badges/nous-girl'\nexport { BlendMode, useBlendMode, withBlendMode } from './ui/components/blend-mode'\nexport type { BlendModeProps } from './ui/components/blend-mode'\nexport { Blink } from './ui/components/blink'\nexport { Button } from './ui/components/button'\nexport { Card, CardContent, CardDescription, CardHeader, CardTitle } from './ui/components/card'\nexport { Checkbox } from './ui/components/checkbox'\nexport { CommandBlock, CopyButton } from './ui/components/command-block'\nexport { ConfirmDialog } from './ui/components/confirm-dialog'\nexport { Cursor } from './ui/components/cursor'\nexport { DropdownMenu } from './ui/components/dropdown-menu'\nexport { FitText } from './ui/components/fit-text'\nexport { BarChart, LineChart } from './ui/components/graphs'\nexport { Cell, Grid } from './ui/components/grid'\nexport { HoverBg } from './ui/components/hover-bg'\nexport * as Icons from './ui/components/icons'\nexport { DiscordIcon } from './ui/components/icons/discord'\nexport { GitHubIcon } from './ui/components/icons/github'\nexport { ImageDistortion } from './ui/components/image-distortion'\nexport type { AutoPlayPattern } from './ui/components/image-distortion'\nexport { Input } from './ui/components/input'\nexport { Label } from './ui/components/label'\nexport { LevaClient } from './ui/components/leva-client'\nexport { ListItem } from './ui/components/list-item'\nexport {\n  Dialog,\n  DialogClose,\n  DialogContent,\n  DialogDescription,\n  DialogFooter,\n  DialogHeader,\n  DialogOverlay,\n  DialogPortal,\n  DialogTitle,\n  DialogTrigger\n} from './ui/components/dialog'\nexport { FilterGroup, Segmented } from './ui/components/segmented'\nexport { Switch } from './ui/components/switch'\nexport { Tabs, TabsList, TabsTrigger } from './ui/components/tabs'\nexport { Poster } from './ui/components/poster'\nexport type {\n  PosterAspect,\n  PosterProps,\n  PosterVariant\n} from './ui/components/poster'\nexport {\n  applyLens,\n  BLEND_MODES,\n  LENSES,\n  LENS_0,\n  LENS_5I,\n  lens0,\n  lens5i,\n  toggleLens,\n  $lightMode\n} from './ui/components/overlays'\nexport {\n  Glitch,\n  Greys,\n  Lens,\n  Noise,\n  Overlays,\n  Vignette\n} from './ui/components/overlays'\nexport type { LensPreset } from './ui/components/overlays'\nexport { Progress } from './ui/components/progress'\nexport { SceneCanvas } from './ui/components/scene-canvas'\nexport { Scramble } from './ui/components/scramble'\nexport { Select, SelectOption } from './ui/components/select'\nexport { SelectionSwitcher } from './ui/components/selection-switcher'\nexport { Separator } from './ui/components/separator'\nexport { Spinner } from './ui/components/spinner'\nexport { Stats } from './ui/components/stats'\nexport { TerminalDemo } from './ui/components/terminal-demo'\nexport type { TerminalDemoStep } from './ui/components/terminal-demo'\nexport { ThemeToggle } from './ui/components/theme-toggle'\nexport { TierCard } from './ui/components/tier-card'\nexport { Toast } from './ui/components/toast'\nexport type { TierCardPrice, TierCardProps } from './ui/components/tier-card'\nexport { TV } from './ui/components/tv'\nexport { Watchlist } from './ui/components/watchlist'\n\nexport { Typography } from './ui/components/typography'\nexport type { TypographyProps } from './ui/components/typography'\nexport { H1 } from './ui/components/typography/h1'\nexport { H2 } from './ui/components/typography/h2'\nexport { Legend } from './ui/components/typography/legend'\nexport { Small } from './ui/components/typography/small'\n\nexport { BasicPage } from './ui/basic-page'\nexport { Header } from './ui/header'\nexport type { HeaderLink, HeaderProps, HeaderSocial } from './ui/header'\nexport { Footer } from './ui/footer'\nexport type {\n  FooterGroup,\n  FooterLink,\n  FooterProps\n} from './ui/footer'\nexport { Socials } from './ui/components/socials'\nexport type { SocialLink } from './ui/components/socials'\nexport { LayoutWrapper } from './ui/layout-wrapper'\n\nexport {\n  FONT_SANS,\n  FONT_MONO,\n  FONT_RULES_COMPRESSED,\n  FONT_RULES_EXPANDED,\n  FONT_MONDWEST\n} from './fonts'\n\nexport { cn, clamp, smoothstep, hexToVec3, truncate, stripWpStyles } from './utils'\nexport { polyRef } from './utils'\nexport type { PolyComponent, PolyProps, PolyRef } from './utils'\nexport { hexToRgb, rgbToHex, colorDodge, colorMix } from './utils/color'\n\nexport { useBelowBreakpoint } from './hooks/use-below-breakpoint'\nexport { useCappedFrame } from './hooks/use-capped-frame'\nexport { useConfirmDelete } from './hooks/use-confirm-delete'\nexport { useCssVarDims } from './hooks/use-css-var-dims'\nexport { $gpuTier, useGpuTier } from './hooks/use-gpu-tier'\nexport {\n  useSmoothControls,\n  getControlAtom,\n  setControlValue\n} from './hooks/use-smooth-controls'\nexport { useToast } from './hooks/use-toast'\n"],
  "mappings": "AAAA,SAAS,eAAe,wBAAwB;AAChD,SAAS,eAAe,YAAY,qBAAqB;AACzD,SAAS,aAAa;AACtB,SAAS,mBAAmB;AAC5B,SAAS,qBAAqB;AAC9B,SAAS,WAAW,cAAc,qBAAqB;AAEvD,SAAS,aAAa;AACtB,SAAS,cAAc;AACvB,SAAS,MAAM,aAAa,iBAAiB,YAAY,iBAAiB;AAC1E,SAAS,gBAAgB;AACzB,SAAS,cAAc,kBAAkB;AACzC,SAAS,qBAAqB;AAC9B,SAAS,cAAc;AACvB,SAAS,oBAAoB;AAC7B,SAAS,eAAe;AACxB,SAAS,UAAU,iBAAiB;AACpC,SAAS,MAAM,YAAY;AAC3B,SAAS,eAAe;AACxB,YAAY,WAAW;AACvB,SAAS,mBAAmB;AAC5B,SAAS,kBAAkB;AAC3B,SAAS,uBAAuB;AAEhC,SAAS,aAAa;AACtB,SAAS,aAAa;AACtB,SAAS,kBAAkB;AAC3B,SAAS,gBAAgB;AACzB;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,aAAa,iBAAiB;AACvC,SAAS,cAAc;AACvB,SAAS,MAAM,UAAU,mBAAmB;AAC5C,SAAS,cAAc;AAMvB;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AAEP,SAAS,gBAAgB;AACzB,SAAS,mBAAmB;AAC5B,SAAS,gBAAgB;AACzB,SAAS,QAAQ,oBAAoB;AACrC,SAAS,yBAAyB;AAClC,SAAS,iBAAiB;AAC1B,SAAS,eAAe;AACxB,SAAS,aAAa;AACtB,SAAS,oBAAoB;AAE7B,SAAS,mBAAmB;AAC5B,SAAS,gBAAgB;AACzB,SAAS,aAAa;AAEtB,SAAS,UAAU;AACnB,SAAS,iBAAiB;AAE1B,SAAS,kBAAkB;AAE3B,SAAS,UAAU;AACnB,SAAS,UAAU;AACnB,SAAS,cAAc;AACvB,SAAS,aAAa;AAEtB,SAAS,iBAAiB;AAC1B,SAAS,cAAc;AAEvB,SAAS,cAAc;AAMvB,SAAS,eAAe;AAExB,SAAS,qBAAqB;AAE9B;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AAEP,SAAS,IAAI,OAAO,YAAY,WAAW,UAAU,qBAAqB;AAC1E,SAAS,eAAe;AAExB,SAAS,UAAU,UAAU,YAAY,gBAAgB;AAEzD,SAAS,0BAA0B;AACnC,SAAS,sBAAsB;AAC/B,SAAS,wBAAwB;AACjC,SAAS,qBAAqB;AAC9B,SAAS,UAAU,kBAAkB;AACrC;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,gBAAgB;",
  "names": []
}

|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { type ReactNode } from 'react';
|
|
2
|
+
/**
|
|
3
|
+
* Mobile-first bottom sheet with slide + fade enter/exit, drag-to-dismiss
|
|
4
|
+
* handle, body scroll lock, and reduced-motion support. Portaled to
|
|
5
|
+
* `document.body` to escape ancestor stacking contexts.
|
|
6
|
+
*/
|
|
7
|
+
export declare function BottomSheet({ backdropDismissLabel, children, onClose, open, title }: BottomSheetProps): import("react").ReactPortal | null;
|
|
8
|
+
interface BottomSheetProps {
|
|
9
|
+
backdropDismissLabel?: string;
|
|
10
|
+
children: ReactNode;
|
|
11
|
+
onClose: () => void;
|
|
12
|
+
open: boolean;
|
|
13
|
+
title: string;
|
|
14
|
+
}
|
|
15
|
+
export {};
|
|
@@ -0,0 +1,192 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
3
|
+
import {
|
|
4
|
+
useEffect,
|
|
5
|
+
useRef,
|
|
6
|
+
useState
|
|
7
|
+
} from "react";
|
|
8
|
+
import { createPortal } from "react-dom";
|
|
9
|
+
import { cn } from "../../utils/index.js";
|
|
10
|
+
import { Typography } from "./typography/index.js";
|
|
11
|
+
const CLOSE_DRAG_MIN_PX = 72;
|
|
12
|
+
const CLOSE_DRAG_RATIO = 0.18;
|
|
13
|
+
const SHEET_TRANSITION_MS = 280;
|
|
14
|
+
export function BottomSheet({
|
|
15
|
+
backdropDismissLabel = "Dismiss",
|
|
16
|
+
children,
|
|
17
|
+
onClose,
|
|
18
|
+
open,
|
|
19
|
+
title
|
|
20
|
+
}) {
|
|
21
|
+
const [renderPortal, setRenderPortal] = useState(open);
|
|
22
|
+
const [entered, setEntered] = useState(false);
|
|
23
|
+
const [dragOffsetPx, setDragOffsetPx] = useState(0);
|
|
24
|
+
const [dragActive, setDragActive] = useState(false);
|
|
25
|
+
const closeTimerRef = useRef(null);
|
|
26
|
+
const sheetRef = useRef(null);
|
|
27
|
+
const dragTrackingRef = useRef(false);
|
|
28
|
+
const dragStartYRef = useRef(0);
|
|
29
|
+
const dragOffsetRef = useRef(0);
|
|
30
|
+
const reducedMotion = typeof window !== "undefined" && window.matchMedia("(prefers-reduced-motion: reduce)").matches;
|
|
31
|
+
const syncDragPx = (next) => {
|
|
32
|
+
dragOffsetRef.current = next;
|
|
33
|
+
setDragOffsetPx(next);
|
|
34
|
+
};
|
|
35
|
+
useEffect(() => {
|
|
36
|
+
if (closeTimerRef.current) {
|
|
37
|
+
clearTimeout(closeTimerRef.current);
|
|
38
|
+
closeTimerRef.current = null;
|
|
39
|
+
}
|
|
40
|
+
const ms = reducedMotion ? 0 : SHEET_TRANSITION_MS;
|
|
41
|
+
let openRafId = 0;
|
|
42
|
+
let exitRafId = 0;
|
|
43
|
+
if (open) {
|
|
44
|
+
openRafId = requestAnimationFrame(() => {
|
|
45
|
+
dragTrackingRef.current = false;
|
|
46
|
+
dragOffsetRef.current = 0;
|
|
47
|
+
setDragActive(false);
|
|
48
|
+
setDragOffsetPx(0);
|
|
49
|
+
setRenderPortal(true);
|
|
50
|
+
requestAnimationFrame(() => {
|
|
51
|
+
requestAnimationFrame(() => setEntered(true));
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
} else {
|
|
55
|
+
exitRafId = requestAnimationFrame(() => {
|
|
56
|
+
dragTrackingRef.current = false;
|
|
57
|
+
setDragActive(false);
|
|
58
|
+
setEntered(false);
|
|
59
|
+
closeTimerRef.current = window.setTimeout(() => {
|
|
60
|
+
dragOffsetRef.current = 0;
|
|
61
|
+
setDragOffsetPx(0);
|
|
62
|
+
setRenderPortal(false);
|
|
63
|
+
closeTimerRef.current = null;
|
|
64
|
+
}, ms);
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
return () => {
|
|
68
|
+
cancelAnimationFrame(openRafId);
|
|
69
|
+
cancelAnimationFrame(exitRafId);
|
|
70
|
+
if (closeTimerRef.current) {
|
|
71
|
+
clearTimeout(closeTimerRef.current);
|
|
72
|
+
closeTimerRef.current = null;
|
|
73
|
+
}
|
|
74
|
+
};
|
|
75
|
+
}, [open, reducedMotion]);
|
|
76
|
+
useEffect(() => {
|
|
77
|
+
if (!renderPortal) return;
|
|
78
|
+
const prev = document.body.style.overflow;
|
|
79
|
+
document.body.style.overflow = "hidden";
|
|
80
|
+
return () => {
|
|
81
|
+
document.body.style.overflow = prev;
|
|
82
|
+
};
|
|
83
|
+
}, [renderPortal]);
|
|
84
|
+
if (!renderPortal || typeof document === "undefined") return null;
|
|
85
|
+
const durationClass = reducedMotion ? "duration-0" : "duration-[280ms]";
|
|
86
|
+
const draggingVisual = dragActive || dragOffsetPx > 0;
|
|
87
|
+
const onDragPointerDown = (e) => {
|
|
88
|
+
if (reducedMotion || !entered) return;
|
|
89
|
+
if (e.pointerType === "mouse" && e.button !== 0) return;
|
|
90
|
+
dragTrackingRef.current = true;
|
|
91
|
+
setDragActive(true);
|
|
92
|
+
dragStartYRef.current = e.clientY;
|
|
93
|
+
syncDragPx(0);
|
|
94
|
+
e.currentTarget.setPointerCapture(e.pointerId);
|
|
95
|
+
};
|
|
96
|
+
const onDragPointerMove = (e) => {
|
|
97
|
+
if (!dragTrackingRef.current) return;
|
|
98
|
+
const dy = e.clientY - dragStartYRef.current;
|
|
99
|
+
const next = Math.max(0, dy);
|
|
100
|
+
const sheetH = sheetRef.current?.offsetHeight ?? 560;
|
|
101
|
+
syncDragPx(Math.min(next, sheetH));
|
|
102
|
+
};
|
|
103
|
+
const endDrag = (e) => {
|
|
104
|
+
if (!dragTrackingRef.current) return;
|
|
105
|
+
dragTrackingRef.current = false;
|
|
106
|
+
setDragActive(false);
|
|
107
|
+
try {
|
|
108
|
+
e.currentTarget.releasePointerCapture(e.pointerId);
|
|
109
|
+
} catch {
|
|
110
|
+
}
|
|
111
|
+
const sheetH = sheetRef.current?.offsetHeight ?? 560;
|
|
112
|
+
const threshold = Math.max(CLOSE_DRAG_MIN_PX, sheetH * CLOSE_DRAG_RATIO);
|
|
113
|
+
const d = dragOffsetRef.current;
|
|
114
|
+
if (d >= threshold) {
|
|
115
|
+
onClose();
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
syncDragPx(0);
|
|
119
|
+
};
|
|
120
|
+
return createPortal(
|
|
121
|
+
/* @__PURE__ */ jsxs("div", { className: "fixed inset-0 z-[200] flex flex-col justify-end", children: [
|
|
122
|
+
/* @__PURE__ */ jsx(
|
|
123
|
+
"button",
|
|
124
|
+
{
|
|
125
|
+
"aria-label": backdropDismissLabel,
|
|
126
|
+
className: cn(
|
|
127
|
+
"absolute inset-0 bg-black/55 backdrop-blur-[2px]",
|
|
128
|
+
"transition-opacity ease-out motion-reduce:transition-none",
|
|
129
|
+
durationClass,
|
|
130
|
+
entered ? "opacity-100" : "opacity-0"
|
|
131
|
+
),
|
|
132
|
+
onClick: onClose,
|
|
133
|
+
type: "button"
|
|
134
|
+
}
|
|
135
|
+
),
|
|
136
|
+
/* @__PURE__ */ jsxs(
|
|
137
|
+
"div",
|
|
138
|
+
{
|
|
139
|
+
"aria-label": title,
|
|
140
|
+
"aria-modal": "true",
|
|
141
|
+
className: cn(
|
|
142
|
+
"relative flex max-h-[85dvh] min-h-0 flex-col rounded-t-xl border border-current/20",
|
|
143
|
+
"bg-background-base/98 pb-[max(1rem,env(safe-area-inset-bottom))]",
|
|
144
|
+
"shadow-[0_-12px_40px_-8px_rgba(0,0,0,0.55)] backdrop-blur-md",
|
|
145
|
+
"ease-out motion-reduce:transition-none transform-gpu",
|
|
146
|
+
draggingVisual ? "transition-none" : cn("transition-transform", durationClass),
|
|
147
|
+
entered ? "translate-y-0" : "translate-y-full"
|
|
148
|
+
),
|
|
149
|
+
ref: sheetRef,
|
|
150
|
+
role: "dialog",
|
|
151
|
+
style: entered && dragOffsetPx > 0 ? { transform: `translateY(${dragOffsetPx}px)` } : void 0,
|
|
152
|
+
children: [
|
|
153
|
+
/* @__PURE__ */ jsxs(
|
|
154
|
+
"div",
|
|
155
|
+
{
|
|
156
|
+
className: cn(
|
|
157
|
+
"flex shrink-0 flex-col gap-2 border-b border-current/15 px-4 pb-3 pt-2",
|
|
158
|
+
"touch-none select-none",
|
|
159
|
+
reducedMotion ? "cursor-default" : "cursor-grab active:cursor-grabbing"
|
|
160
|
+
),
|
|
161
|
+
onPointerCancel: endDrag,
|
|
162
|
+
onPointerDown: onDragPointerDown,
|
|
163
|
+
onPointerMove: onDragPointerMove,
|
|
164
|
+
onPointerUp: endDrag,
|
|
165
|
+
children: [
|
|
166
|
+
/* @__PURE__ */ jsx(
|
|
167
|
+
"div",
|
|
168
|
+
{
|
|
169
|
+
"aria-hidden": true,
|
|
170
|
+
className: "mx-auto h-1 w-10 shrink-0 rounded-full bg-current/20"
|
|
171
|
+
}
|
|
172
|
+
),
|
|
173
|
+
/* @__PURE__ */ jsx(
|
|
174
|
+
Typography,
|
|
175
|
+
{
|
|
176
|
+
className: "text-[0.65rem] tracking-[0.15em] uppercase text-midground/70",
|
|
177
|
+
mondwest: true,
|
|
178
|
+
children: title
|
|
179
|
+
}
|
|
180
|
+
)
|
|
181
|
+
]
|
|
182
|
+
}
|
|
183
|
+
),
|
|
184
|
+
/* @__PURE__ */ jsx("div", { className: "min-h-0 flex-1 overflow-y-auto overscroll-contain", children })
|
|
185
|
+
]
|
|
186
|
+
}
|
|
187
|
+
)
|
|
188
|
+
] }),
|
|
189
|
+
document.body
|
|
190
|
+
);
|
|
191
|
+
}
|
|
192
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["<stdin>"],
  "sourcesContent": ["'use client'\n\nimport {\n  type PointerEvent as ReactPointerEvent,\n  type ReactNode,\n  useEffect,\n  useRef,\n  useState\n} from 'react'\nimport { createPortal } from 'react-dom'\n\nimport { cn } from '../../utils'\nimport { Typography } from './typography'\n\nconst CLOSE_DRAG_MIN_PX = 72\nconst CLOSE_DRAG_RATIO = 0.18\nconst SHEET_TRANSITION_MS = 280\n\n/**\n * Mobile-first bottom sheet with slide + fade enter/exit, drag-to-dismiss\n * handle, body scroll lock, and reduced-motion support. Portaled to\n * `document.body` to escape ancestor stacking contexts.\n */\nexport function BottomSheet({\n  backdropDismissLabel = 'Dismiss',\n  children,\n  onClose,\n  open,\n  title\n}: BottomSheetProps) {\n  const [renderPortal, setRenderPortal] = useState(open)\n  const [entered, setEntered] = useState(false)\n  const [dragOffsetPx, setDragOffsetPx] = useState(0)\n  const [dragActive, setDragActive] = useState(false)\n\n  const closeTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null)\n  const sheetRef = useRef<HTMLDivElement>(null)\n  const dragTrackingRef = useRef(false)\n  const dragStartYRef = useRef(0)\n  const dragOffsetRef = useRef(0)\n\n  const reducedMotion =\n    typeof window !== 'undefined' &&\n    window.matchMedia('(prefers-reduced-motion: reduce)').matches\n\n  const syncDragPx = (next: number) => {\n    dragOffsetRef.current = next\n    setDragOffsetPx(next)\n  }\n\n  useEffect(() => {\n    if (closeTimerRef.current) {\n      clearTimeout(closeTimerRef.current)\n      closeTimerRef.current = null\n    }\n\n    const ms = reducedMotion ? 0 : SHEET_TRANSITION_MS\n\n    let openRafId = 0\n    let exitRafId = 0\n\n    if (open) {\n      openRafId = requestAnimationFrame(() => {\n        dragTrackingRef.current = false\n        dragOffsetRef.current = 0\n        setDragActive(false)\n        setDragOffsetPx(0)\n        setRenderPortal(true)\n        requestAnimationFrame(() => {\n          requestAnimationFrame(() => setEntered(true))\n        })\n      })\n    } else {\n      exitRafId = requestAnimationFrame(() => {\n        dragTrackingRef.current = false\n        setDragActive(false)\n        setEntered(false)\n        closeTimerRef.current = window.setTimeout(() => {\n          dragOffsetRef.current = 0\n          setDragOffsetPx(0)\n          setRenderPortal(false)\n          closeTimerRef.current = null\n        }, ms)\n      })\n    }\n\n    return () => {\n      cancelAnimationFrame(openRafId)\n      cancelAnimationFrame(exitRafId)\n      if (closeTimerRef.current) {\n        clearTimeout(closeTimerRef.current)\n        closeTimerRef.current = null\n      }\n    }\n  }, [open, reducedMotion])\n\n  useEffect(() => {\n    if (!renderPortal) return\n    const prev = document.body.style.overflow\n    document.body.style.overflow = 'hidden'\n    return () => {\n      document.body.style.overflow = prev\n    }\n  }, [renderPortal])\n\n  if (!renderPortal || typeof document === 'undefined') return null\n\n  const durationClass = reducedMotion ? 'duration-0' : 'duration-[280ms]'\n  const draggingVisual = dragActive || dragOffsetPx > 0\n\n  const onDragPointerDown = (e: ReactPointerEvent<HTMLDivElement>) => {\n    if (reducedMotion || !entered) return\n    if (e.pointerType === 'mouse' && e.button !== 0) return\n\n    dragTrackingRef.current = true\n    setDragActive(true)\n    dragStartYRef.current = e.clientY\n    syncDragPx(0)\n    e.currentTarget.setPointerCapture(e.pointerId)\n  }\n\n  const onDragPointerMove = (e: ReactPointerEvent<HTMLDivElement>) => {\n    if (!dragTrackingRef.current) return\n    const dy = e.clientY - dragStartYRef.current\n    const next = Math.max(0, dy)\n    const sheetH = sheetRef.current?.offsetHeight ?? 560\n    syncDragPx(Math.min(next, sheetH))\n  }\n\n  const endDrag = (e: ReactPointerEvent<HTMLDivElement>) => {\n    if (!dragTrackingRef.current) return\n    dragTrackingRef.current = false\n    setDragActive(false)\n    try {\n      e.currentTarget.releasePointerCapture(e.pointerId)\n    } catch {\n      /* already released */\n    }\n\n    const sheetH = sheetRef.current?.offsetHeight ?? 560\n    const threshold = Math.max(CLOSE_DRAG_MIN_PX, sheetH * CLOSE_DRAG_RATIO)\n    const d = dragOffsetRef.current\n\n    if (d >= threshold) {\n      onClose()\n      return\n    }\n    syncDragPx(0)\n  }\n\n  return createPortal(\n    <div className=\"fixed inset-0 z-[200] flex flex-col justify-end\">\n      <button\n        aria-label={backdropDismissLabel}\n        className={cn(\n          'absolute inset-0 bg-black/55 backdrop-blur-[2px]',\n          'transition-opacity ease-out motion-reduce:transition-none',\n          durationClass,\n          entered ? 'opacity-100' : 'opacity-0'\n        )}\n        onClick={onClose}\n        type=\"button\"\n      />\n\n      <div\n        aria-label={title}\n        aria-modal=\"true\"\n        className={cn(\n          'relative flex max-h-[85dvh] min-h-0 flex-col rounded-t-xl border border-current/20',\n          'bg-background-base/98 pb-[max(1rem,env(safe-area-inset-bottom))]',\n          'shadow-[0_-12px_40px_-8px_rgba(0,0,0,0.55)] backdrop-blur-md',\n          'ease-out motion-reduce:transition-none transform-gpu',\n          draggingVisual\n            ? 'transition-none'\n            : cn('transition-transform', durationClass),\n          entered ? 'translate-y-0' : 'translate-y-full'\n        )}\n        ref={sheetRef}\n        role=\"dialog\"\n        style={\n          entered && dragOffsetPx > 0\n            ? { transform: `translateY(${dragOffsetPx}px)` }\n            : undefined\n        }\n      >\n        <div\n          className={cn(\n            'flex shrink-0 flex-col gap-2 border-b border-current/15 px-4 pb-3 pt-2',\n            'touch-none select-none',\n            reducedMotion\n              ? 'cursor-default'\n              : 'cursor-grab active:cursor-grabbing'\n          )}\n          onPointerCancel={endDrag}\n          onPointerDown={onDragPointerDown}\n          onPointerMove={onDragPointerMove}\n          onPointerUp={endDrag}\n        >\n          <div\n            aria-hidden\n            className=\"mx-auto h-1 w-10 shrink-0 rounded-full bg-current/20\"\n          />\n\n          <Typography\n            className=\"text-[0.65rem] tracking-[0.15em] uppercase text-midground/70\"\n            mondwest\n          >\n            {title}\n          </Typography>\n        </div>\n\n        <div className=\"min-h-0 flex-1 overflow-y-auto overscroll-contain\">\n          {children}\n        </div>\n      </div>\n    </div>,\n    document.body\n  )\n}\n\ninterface BottomSheetProps {\n  backdropDismissLabel?: string\n  children: ReactNode\n  onClose: () => void\n  open: boolean\n  title: string\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
export declare function Card({ className, style, ...props }: React.HTMLAttributes<HTMLDivElement>): import("react").JSX.Element;
|
|
2
|
+
export declare function CardHeader({ className, ...props }: React.HTMLAttributes<HTMLDivElement>): import("react").JSX.Element;
|
|
3
|
+
export declare function CardTitle({ className, ...props }: React.HTMLAttributes<HTMLHeadingElement>): import("react").JSX.Element;
|
|
4
|
+
export declare function CardDescription({ className, ...props }: React.HTMLAttributes<HTMLParagraphElement>): import("react").JSX.Element;
|
|
5
|
+
export declare function CardContent({ className, ...props }: React.HTMLAttributes<HTMLDivElement>): import("react").JSX.Element;
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { jsx } from "react/jsx-runtime";
|
|
2
|
+
import { cn } from "../../utils/index.js";
|
|
3
|
+
const CARD_STYLE = {
|
|
4
|
+
background: "var(--component-card-background)",
|
|
5
|
+
borderImage: "var(--component-card-border-image)",
|
|
6
|
+
boxShadow: "var(--component-card-box-shadow)",
|
|
7
|
+
clipPath: "var(--component-card-clip-path)"
|
|
8
|
+
};
|
|
9
|
+
export function Card({
|
|
10
|
+
className,
|
|
11
|
+
style,
|
|
12
|
+
...props
|
|
13
|
+
}) {
|
|
14
|
+
return /* @__PURE__ */ jsx(
|
|
15
|
+
"div",
|
|
16
|
+
{
|
|
17
|
+
className: cn(
|
|
18
|
+
"border border-midground/15 bg-background-base/80 text-midground w-full",
|
|
19
|
+
className
|
|
20
|
+
),
|
|
21
|
+
style: { ...CARD_STYLE, ...style },
|
|
22
|
+
...props
|
|
23
|
+
}
|
|
24
|
+
);
|
|
25
|
+
}
|
|
26
|
+
export function CardHeader({
|
|
27
|
+
className,
|
|
28
|
+
...props
|
|
29
|
+
}) {
|
|
30
|
+
return /* @__PURE__ */ jsx(
|
|
31
|
+
"div",
|
|
32
|
+
{
|
|
33
|
+
className: cn(
|
|
34
|
+
"flex flex-col gap-1.5 p-4 border-b border-midground/15",
|
|
35
|
+
className
|
|
36
|
+
),
|
|
37
|
+
...props
|
|
38
|
+
}
|
|
39
|
+
);
|
|
40
|
+
}
|
|
41
|
+
export function CardTitle({
|
|
42
|
+
className,
|
|
43
|
+
...props
|
|
44
|
+
}) {
|
|
45
|
+
return /* @__PURE__ */ jsx(
|
|
46
|
+
"h3",
|
|
47
|
+
{
|
|
48
|
+
className: cn(
|
|
49
|
+
"font-expanded text-sm font-bold tracking-[0.08em] uppercase",
|
|
50
|
+
className
|
|
51
|
+
),
|
|
52
|
+
...props
|
|
53
|
+
}
|
|
54
|
+
);
|
|
55
|
+
}
|
|
56
|
+
export function CardDescription({
|
|
57
|
+
className,
|
|
58
|
+
...props
|
|
59
|
+
}) {
|
|
60
|
+
return /* @__PURE__ */ jsx(
|
|
61
|
+
"p",
|
|
62
|
+
{
|
|
63
|
+
className: cn("font-mondwest text-xs text-midground/60", className),
|
|
64
|
+
...props
|
|
65
|
+
}
|
|
66
|
+
);
|
|
67
|
+
}
|
|
68
|
+
export function CardContent({
|
|
69
|
+
className,
|
|
70
|
+
...props
|
|
71
|
+
}) {
|
|
72
|
+
return /* @__PURE__ */ jsx("div", { className: cn("p-4", className), ...props });
|
|
73
|
+
}
|
|
74
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiPHN0ZGluPiJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiaW1wb3J0IHsgY24gfSBmcm9tICcuLi8uLi91dGlscydcblxuLyoqXG4gKiBUaGVtZWFibGUgY2FyZCBwcmltaXRpdmUuIFRoZW1lcyBjYW4gcmVzdHlsZSBldmVyeSBjYXJkIGJ5IHNldHRpbmcgQ1NTXG4gKiBjdXN0b20gcHJvcGVydGllczpcbiAqXG4gKiAgIC0tY29tcG9uZW50LWNhcmQtY2xpcC1wYXRoXG4gKiAgIC0tY29tcG9uZW50LWNhcmQtYm9yZGVyLWltYWdlXG4gKiAgIC0tY29tcG9uZW50LWNhcmQtYmFja2dyb3VuZFxuICogICAtLWNvbXBvbmVudC1jYXJkLWJveC1zaGFkb3dcbiAqXG4gKiBBbGwgYXJlIG9wdGlvbmFsIFx1MjAxNCB1bnNldCB2YXJzIGNvbXB1dGUgdG8gdGhlaXIgQ1NTIGluaXRpYWwgdmFsdWUuXG4gKi9cbmNvbnN0IENBUkRfU1RZTEU6IFJlYWN0LkNTU1Byb3BlcnRpZXMgPSB7XG4gIGJhY2tncm91bmQ6ICd2YXIoLS1jb21wb25lbnQtY2FyZC1iYWNrZ3JvdW5kKScsXG4gIGJvcmRlckltYWdlOiAndmFyKC0tY29tcG9uZW50LWNhcmQtYm9yZGVyLWltYWdlKScsXG4gIGJveFNoYWRvdzogJ3ZhcigtLWNvbXBvbmVudC1jYXJkLWJveC1zaGFkb3cpJyxcbiAgY2xpcFBhdGg6ICd2YXIoLS1jb21wb25lbnQtY2FyZC1jbGlwLXBhdGgpJ1xufVxuXG5leHBvcnQgZnVuY3Rpb24gQ2FyZCh7XG4gIGNsYXNzTmFtZSxcbiAgc3R5bGUsXG4gIC4uLnByb3BzXG59OiBSZWFjdC5IVE1MQXR0cmlidXRlczxIVE1MRGl2RWxlbWVudD4pIHtcbiAgcmV0dXJuIChcbiAgICA8ZGl2XG4gICAgICBjbGFzc05hbWU9e2NuKFxuICAgICAgICAnYm9yZGVyIGJvcmRlci1taWRncm91bmQvMTUgYmctYmFja2dyb3VuZC1iYXNlLzgwIHRleHQtbWlkZ3JvdW5kIHctZnVsbCcsXG4gICAgICAgIGNsYXNzTmFtZVxuICAgICAgKX1cbiAgICAgIHN0eWxlPXt7IC4uLkNBUkRfU1RZTEUsIC4uLnN0eWxlIH19XG4gICAgICB7Li4ucHJvcHN9XG4gICAgLz5cbiAgKVxufVxuXG5leHBvcnQgZnVuY3Rpb24gQ2FyZEhlYWRlcih7XG4gIGNsYXNzTmFtZSxcbiAgLi4ucHJvcHNcbn06IFJlYWN0LkhUTUxBdHRyaWJ1dGVzPEhUTUxEaXZFbGVtZW50Pikge1xuICByZXR1cm4gKFxuICAgIDxkaXZcbiAgICAgIGNsYXNzTmFtZT17Y24oXG4gICAgICAgICdmbGV4IGZsZXgtY29sIGdhcC0xLjUgcC00IGJvcmRlci1iIGJvcmRlci1taWRncm91bmQvMTUnLFxuICAgICAgICBjbGFzc05hbWVcbiAgICAgICl9XG4gICAgICB7Li4ucHJvcHN9XG4gICAgLz5cbiAgKVxufVxuXG5leHBvcnQgZnVuY3Rpb24gQ2FyZFRpdGxlKHtcbiAgY2xhc3NOYW1lLFxuICAuLi5wcm9wc1xufTogUmVhY3QuSFRNTEF0dHJpYnV0ZXM8SFRNTEhlYWRpbmdFbGVtZW50Pikge1xuICByZXR1cm4gKFxuICAgIDxoM1xuICAgICAgY2xhc3NOYW1lPXtjbihcbiAgICAgICAgJ2ZvbnQtZXhwYW5kZWQgdGV4dC1zbSBmb250LWJvbGQgdHJhY2tpbmctWzAuMDhlbV0gdXBwZXJjYXNlJyxcbiAgICAgICAgY2xhc3NOYW1lXG4gICAgICApfVxuICAgICAgey4uLnByb3BzfVxuICAgIC8+XG4gIClcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIENhcmREZXNjcmlwdGlvbih7XG4gIGNsYXNzTmFtZSxcbiAgLi4ucHJvcHNcbn06IFJlYWN0LkhUTUxBdHRyaWJ1dGVzPEhUTUxQYXJhZ3JhcGhFbGVtZW50Pikge1xuICByZXR1cm4gKFxuICAgIDxwXG4gICAgICBjbGFzc05hbWU9e2NuKCdmb250LW1vbmR3ZXN0IHRleHQteHMgdGV4dC1taWRncm91bmQvNjAnLCBjbGFzc05hbWUpfVxuICAgICAgey4uLnByb3BzfVxuICAgIC8+XG4gIClcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIENhcmRDb250ZW50KHtcbiAgY2xhc3NOYW1lLFxuICAuLi5wcm9wc1xufTogUmVhY3QuSFRNTEF0dHJpYnV0ZXM8SFRNTERpdkVsZW1lbnQ+KSB7XG4gIHJldHVybiA8ZGl2IGNsYXNzTmFtZT17Y24oJ3AtNCcsIGNsYXNzTmFtZSl9IHsuLi5wcm9wc30gLz5cbn1cbiJdLAogICJtYXBwaW5ncyI6ICJBQTBCSTtBQTFCSixTQUFTLFVBQVU7QUFhbkIsTUFBTSxhQUFrQztBQUFBLEVBQ3RDLFlBQVk7QUFBQSxFQUNaLGFBQWE7QUFBQSxFQUNiLFdBQVc7QUFBQSxFQUNYLFVBQVU7QUFDWjtBQUVPLGdCQUFTLEtBQUs7QUFBQSxFQUNuQjtBQUFBLEVBQ0E7QUFBQSxFQUNBLEdBQUc7QUFDTCxHQUF5QztBQUN2QyxTQUNFO0FBQUEsSUFBQztBQUFBO0FBQUEsTUFDQyxXQUFXO0FBQUEsUUFDVDtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBQUEsTUFDQSxPQUFPLEVBQUUsR0FBRyxZQUFZLEdBQUcsTUFBTTtBQUFBLE1BQ2hDLEdBQUc7QUFBQTtBQUFBLEVBQ047QUFFSjtBQUVPLGdCQUFTLFdBQVc7QUFBQSxFQUN6QjtBQUFBLEVBQ0EsR0FBRztBQUNMLEdBQXlDO0FBQ3ZDLFNBQ0U7QUFBQSxJQUFDO0FBQUE7QUFBQSxNQUNDLFdBQVc7QUFBQSxRQUNUO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFBQSxNQUNDLEdBQUc7QUFBQTtBQUFBLEVBQ047QUFFSjtBQUVPLGdCQUFTLFVBQVU7QUFBQSxFQUN4QjtBQUFBLEVBQ0EsR0FBRztBQUNMLEdBQTZDO0FBQzNDLFNBQ0U7QUFBQSxJQUFDO0FBQUE7QUFBQSxNQUNDLFdBQVc7QUFBQSxRQUNUO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFBQSxNQUNDLEdBQUc7QUFBQTtBQUFBLEVBQ047QUFFSjtBQUVPLGdCQUFTLGdCQUFnQjtBQUFBLEVBQzlCO0FBQUEsRUFDQSxHQUFHO0FBQ0wsR0FBK0M7QUFDN0MsU0FDRTtBQUFBLElBQUM7QUFBQTtBQUFBLE1BQ0MsV0FBVyxHQUFHLDJDQUEyQyxTQUFTO0FBQUEsTUFDakUsR0FBRztBQUFBO0FBQUEsRUFDTjtBQUVKO0FBRU8sZ0JBQVMsWUFBWTtBQUFBLEVBQzFCO0FBQUEsRUFDQSxHQUFHO0FBQ0wsR0FBeUM7QUFDdkMsU0FBTyxvQkFBQyxTQUFJLFdBQVcsR0FBRyxPQUFPLFNBQVMsR0FBSSxHQUFHLE9BQU87QUFDMUQ7IiwKICAibmFtZXMiOiBbXQp9Cg==
|