mirror of
https://github.com/johndoe6345789/low-code-react-app-b.git
synced 2026-04-25 14:14:57 +00:00
Compare commits
2 Commits
tier-1-boo
...
codex/chan
| Author | SHA1 | Date | |
|---|---|---|---|
| 418e3aa657 | |||
| cac24c0716 |
@@ -3,57 +3,7 @@
|
|||||||
"allow": [
|
"allow": [
|
||||||
"Bash(ls:*)",
|
"Bash(ls:*)",
|
||||||
"Bash(find:*)",
|
"Bash(find:*)",
|
||||||
"Bash(grep:*)",
|
"Bash(grep:*)"
|
||||||
"Bash(wc:*)",
|
|
||||||
"Bash(for file in accordion alert aspect-ratio avatar badge button card checkbox collapsible dialog hover-card input label popover progress radio-group resizable scroll-area separator skeleton sheet switch tabs textarea toggle tooltip)",
|
|
||||||
"Bash(do)",
|
|
||||||
"Bash([ -f \"src/config/pages/ui/$file.json\" ])",
|
|
||||||
"Bash(echo:*)",
|
|
||||||
"Bash(done)",
|
|
||||||
"Bash(for file in data-source-card editor-toolbar empty-editor-state monaco-editor-panel search-bar)",
|
|
||||||
"Bash([ -f \"src/config/pages/molecules/$file.json\" ])",
|
|
||||||
"Bash(for file in empty-canvas-state page-header schema-editor-canvas schema-editor-properties-panel schema-editor-sidebar schema-editor-status-bar schema-editor-toolbar toolbar-actions)",
|
|
||||||
"Bash([ -f \"src/config/pages/organisms/$file.json\" ])",
|
|
||||||
"Bash([ -f \"src/config/pages/atoms/input.json\" ])",
|
|
||||||
"Bash(npm run tsx:*)",
|
|
||||||
"Bash(npx tsx:*)",
|
|
||||||
"Bash(npm run test:e2e:*)",
|
|
||||||
"Bash(npx playwright:*)",
|
|
||||||
"Bash(timeout 15 npm run dev:*)",
|
|
||||||
"Bash(netstat:*)",
|
|
||||||
"Bash(findstr:*)",
|
|
||||||
"Bash(taskkill:*)",
|
|
||||||
"Bash(xargs:*)",
|
|
||||||
"Bash(npm run build:*)",
|
|
||||||
"Bash(npm install)",
|
|
||||||
"Bash(for file in ComponentBindingDialog.tsx DataSourceEditorDialog.tsx GitHubBuildStatus.tsx SaveIndicator.tsx ComponentTree.tsx SeedDataManager.tsx LazyD3BarChart.tsx StorageSettings.tsx)",
|
|
||||||
"Bash(do if [ -f \"$file\" ])",
|
|
||||||
"Bash(then echo \"EXISTS: $file\")",
|
|
||||||
"Bash(fi)",
|
|
||||||
"Bash(npm install:*)",
|
|
||||||
"Bash(for file in AppBranding BindingEditor Breadcrumb CanvasRenderer CodeExplanationDialog ComponentPalette)",
|
|
||||||
"Bash(do echo -n \"$file: \")",
|
|
||||||
"Bash(tsx scripts/audit-json-components.ts:*)",
|
|
||||||
"Bash(npm run audit:json:*)",
|
|
||||||
"Bash(bash:*)",
|
|
||||||
"Bash(git restore:*)",
|
|
||||||
"Bash(tree:*)",
|
|
||||||
"Bash(powershell:*)",
|
|
||||||
"Bash(npx tsc:*)",
|
|
||||||
"Bash(node -c:*)",
|
|
||||||
"Bash(tee:*)",
|
|
||||||
"Bash(git log:*)",
|
|
||||||
"Bash(__NEW_LINE_d25a97dbcf730748__ git show 5a70926:src/components/molecules/FileTabs.tsx)",
|
|
||||||
"Bash(__NEW_LINE_133e2efdc2fa9e3f__ cp /tmp/EditorToolbar.tsx /Users/rmac/Documents/GitHub/low-code-react-app-b/src/components/molecules/)",
|
|
||||||
"Bash(__NEW_LINE_60a67b52c0555a4f__ ls -la /Users/rmac/Documents/GitHub/low-code-react-app-b/src/components/molecules/File*)",
|
|
||||||
"Bash(__NEW_LINE_db723daab8184235__ git show 5a70926:src/components/atoms/ComponentTreeNode.tsx)",
|
|
||||||
"Bash(__NEW_LINE_225433a826af561c__ git show 5a70926:src/components/molecules/index.ts)",
|
|
||||||
"Bash(__NEW_LINE_3ace36b310bc6599__ head -5 /Users/rmac/Documents/GitHub/low-code-react-app-b/src/components/atoms/DataTable.tsx)",
|
|
||||||
"Bash(__NEW_LINE_5257b4833161515f__ echo \"=== BUILD SUCCESS ===\")",
|
|
||||||
"Bash(__NEW_LINE_609b943e39e1ac8d__ cp /tmp/COMPREHENSIVE_BUILD_REPORT.md /Users/rmac/Documents/GitHub/low-code-react-app-b/BUILD_REPORT.md)",
|
|
||||||
"Bash(for component in \"Accordion\" \"CopyButton\" \"FileUpload\" \"FilterInput\" \"Image\" \"Input\" \"PasswordInput\" \"Popover\")",
|
|
||||||
"Bash(git add:*)",
|
|
||||||
"Bash(git commit -m \"$\\(cat <<''EOF''\nfeat: Complete JSON component migration for 9 components \\(atoms + BindingEditor\\)\n\nMigration complete for:\n- 5 atoms: Accordion, CopyButton, FileUpload, FilterInput, Image, Input, PasswordInput, Popover \\(8 total\\)\n- 1 molecule: BindingEditor\n\nChanges:\n- Deleted 9 legacy TSX files that have complete JSON equivalents\n- Exported BindingEditor from json-components.ts with useBindingEditor hook\n- Registered useBindingEditor in hooks-registry.ts\n- Updated all imports across codebase to use JSON-based components\n- Fixed build errors: schema-loader dynamic import, DataSourceGroupSection\n- Cleaned up component index exports\n\nBuild status: ✅ PASSING\n- 0 TypeScript errors\n- All 9,408 modules transformed successfully\n- No blocking build warnings\n\nNext steps:\n- 3 organisms still need conversion: DataSourceManager, NavigationMenu, TreeListPanel\n- 120+ additional components have TSX versions \\(need individual migration\\)\n- 22 JSON components now available for use throughout the app\n\nCo-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>\nEOF\n\\)\")"
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
260
BUILD_REPORT.md
260
BUILD_REPORT.md
@@ -1,260 +0,0 @@
|
|||||||
# Comprehensive Build Analysis Report
|
|
||||||
|
|
||||||
## Executive Summary
|
|
||||||
After the EditorToolbar fix and restoration of 56+ deleted component files, **the build now completes successfully with ZERO TypeScript errors**. All blocking build issues have been resolved.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Build Status: ✅ PASSING
|
|
||||||
|
|
||||||
```
|
|
||||||
✓ 9,417 modules transformed
|
|
||||||
✓ Built in 8.81s
|
|
||||||
✓ All 330+ component types generated
|
|
||||||
✓ No TypeScript errors
|
|
||||||
✓ Zero blocking issues
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Issues Identified & Fixed
|
|
||||||
|
|
||||||
### Phase 1: Initial Build Failure
|
|
||||||
|
|
||||||
**Error:** Missing EditorToolbar component import in CodeEditor.tsx
|
|
||||||
```
|
|
||||||
error during build:
|
|
||||||
[vite:load-fallback] Could not load /src/components/molecules/EditorToolbar
|
|
||||||
```
|
|
||||||
|
|
||||||
**Root Cause:** EditorToolbar.tsx was deleted in a recent commit as part of the JSON migration strategy, but CodeEditor.tsx still imports it.
|
|
||||||
|
|
||||||
**Fix Applied:** Restored EditorToolbar.tsx from commit 5a70926
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
### Phase 2: Cascading Missing Dependencies
|
|
||||||
|
|
||||||
As imports were resolved, more missing files were discovered:
|
|
||||||
- SearchBar.tsx
|
|
||||||
- ComponentTreeNode.tsx
|
|
||||||
- PropertyEditorField.tsx
|
|
||||||
- And 50+ others
|
|
||||||
|
|
||||||
**Root Cause:** A bulk deletion of component files occurred during the JSON migration cleanup without corresponding updates to dependent components.
|
|
||||||
|
|
||||||
**Fix Applied:** Systematically restored all 56+ deleted component files from commit 5a70926 using:
|
|
||||||
```bash
|
|
||||||
git ls-tree -r --name-only 5a70926 -- src/components | grep "\.tsx$" | while read file; do
|
|
||||||
if [ ! -f "$file" ]; then
|
|
||||||
git show 5a70926:"$file" > "$file"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
### Phase 3: Export Index Errors
|
|
||||||
|
|
||||||
**Error:** Missing exports in component index files
|
|
||||||
```
|
|
||||||
"AppLogo" is not exported by "src/components/atoms/index.ts"
|
|
||||||
Could not resolve "./TreeListHeader" from "src/components/molecules/index.ts"
|
|
||||||
Could not resolve "./TreeCard" from "src/components/molecules/index.ts"
|
|
||||||
```
|
|
||||||
|
|
||||||
**Root Cause:**
|
|
||||||
- Component index files were not updated when files were restored
|
|
||||||
- Some exports referenced non-existent files (TreeCard, TreeListHeader)
|
|
||||||
|
|
||||||
**Fixes Applied:**
|
|
||||||
1. Restored `src/components/atoms/index.ts` from commit 5a70926 (126 exports)
|
|
||||||
2. Restored `src/components/molecules/index.ts`
|
|
||||||
3. Restored `src/components/organisms/index.ts`
|
|
||||||
4. Restored `src/components/index.ts`
|
|
||||||
5. Removed orphaned export references:
|
|
||||||
- Removed: `export { TreeListHeader } from './TreeListHeader'`
|
|
||||||
- Removed: `export { TreeCard } from './TreeCard'`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Complete List of Restored Files
|
|
||||||
|
|
||||||
### Atoms (87 total)
|
|
||||||
```
|
|
||||||
Accordion, ActionButton, ActionCard, ActionIcon, Alert, AppLogo, Avatar,
|
|
||||||
AvatarGroup, Badge, BindingIndicator, Breadcrumb, Button, ButtonGroup,
|
|
||||||
Calendar, Card, Checkbox, Chip, CircularProgress, Code, ColorSwatch,
|
|
||||||
CommandPalette, CompletionCard, ComponentPaletteItem, ComponentTreeNode,
|
|
||||||
ConfirmButton, Container, ContextMenu, CopyButton, CountBadge, DataList,
|
|
||||||
DataSourceBadge, DataTable, DatePicker, DetailRow, Divider, Dot, Drawer,
|
|
||||||
EmptyMessage, EmptyState, EmptyStateIcon, ErrorBadge, FileIcon, FileUpload,
|
|
||||||
FilterInput, Flex, Form, GlowCard, Grid, Heading, HelperText, HoverCard,
|
|
||||||
IconButton, IconText, IconWrapper, Image, InfoBox, InfoPanel, Input, Kbd,
|
|
||||||
KeyValue, Label, Link, List, ListItem, LiveIndicator, LoadingSpinner,
|
|
||||||
LoadingState, Menu, MetricCard, MetricDisplay, Modal, Notification,
|
|
||||||
NumberInput, PageHeader, PanelHeader, PasswordInput, Popover, ProgressBar,
|
|
||||||
PropertyEditorField, Pulse, QuickActionButton, Radio, RangeSlider, Rating,
|
|
||||||
ResponsiveGrid, ScrollArea, SearchInput, Section, SeedDataStatus, Select,
|
|
||||||
Separator, Skeleton, Slider, Spacer, Sparkle, Spinner, Stack, StatCard,
|
|
||||||
StatusBadge, StatusIcon, StepIndicator, Stepper, Switch, TabIcon, Table,
|
|
||||||
Tabs, Tag, Text, TextArea, TextGradient, TextHighlight, Timeline, Timestamp,
|
|
||||||
TipsCard, Toggle, Tooltip, TreeIcon
|
|
||||||
```
|
|
||||||
|
|
||||||
### Molecules (32 total)
|
|
||||||
```
|
|
||||||
AppBranding, Breadcrumb, CanvasRenderer, CodeExplanationDialog,
|
|
||||||
ComponentBindingDialog, ComponentPalette, ComponentTree, DataSourceCard,
|
|
||||||
DataSourceEditorDialog, EditorActions, EditorToolbar, EmptyEditorState,
|
|
||||||
FileTabs, GitHubBuildStatus, LazyBarChart, LazyD3BarChart,
|
|
||||||
LazyInlineMonacoEditor, LazyLineChart, LazyMonacoEditor, MonacoEditorPanel,
|
|
||||||
NavigationGroupHeader, PropertyEditor, SaveIndicator, SearchBar,
|
|
||||||
SearchInput, SeedDataManager, StorageSettings, ToolbarButton,
|
|
||||||
TreeFormDialog
|
|
||||||
```
|
|
||||||
|
|
||||||
### Organisms (11 total)
|
|
||||||
```
|
|
||||||
AppHeader, EmptyCanvasState, PageHeader, SchemaCodeViewer,
|
|
||||||
SchemaEditorCanvas, SchemaEditorLayout, SchemaEditorPropertiesPanel,
|
|
||||||
SchemaEditorSidebar, SchemaEditorStatusBar, SchemaEditorToolbar,
|
|
||||||
ToolbarActions
|
|
||||||
```
|
|
||||||
|
|
||||||
**Total Files Restored: 130**
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Build Metrics
|
|
||||||
|
|
||||||
| Metric | Value |
|
|
||||||
|--------|-------|
|
|
||||||
| Build Time | 8.81s |
|
|
||||||
| Total Bundle Size | 8.9 MB |
|
|
||||||
| Main JS Bundle | 1,737 KB (437 KB gzip) |
|
|
||||||
| CSS Bundle | 481 KB (81 KB gzip) |
|
|
||||||
| Modules Transformed | 9,417 |
|
|
||||||
| Component Types Generated | 330+ |
|
|
||||||
| TypeScript Errors | 0 |
|
|
||||||
| Build-blocking Errors | 0 |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Non-Blocking Warnings
|
|
||||||
|
|
||||||
### 1. Dynamic/Static Import Conflicts (8 instances)
|
|
||||||
```
|
|
||||||
(!) /src/config/pages/component-tree.json is dynamically imported
|
|
||||||
by /src/hooks/use-schema-loader.ts but also statically imported
|
|
||||||
by /src/components/JSONComponentTreeManager.tsx
|
|
||||||
```
|
|
||||||
|
|
||||||
**Impact:** None - warnings only, build succeeds
|
|
||||||
**Action Needed:** Can be fixed by standardizing import style (dynamic vs static)
|
|
||||||
|
|
||||||
### 2. Chunk Size Warnings
|
|
||||||
```
|
|
||||||
(!) Some chunks are larger than 1000 kB after minification.
|
|
||||||
```
|
|
||||||
|
|
||||||
**Impact:** Performance advisory only
|
|
||||||
**Action Needed:** Optional code-splitting optimization
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## File Changes Summary
|
|
||||||
|
|
||||||
### Modified Files (10)
|
|
||||||
- `.claude/settings.local.json`
|
|
||||||
- `src/components/atoms/index.ts` (updated exports)
|
|
||||||
- `src/components/molecules/index.ts` (updated exports, removed 2 orphaned)
|
|
||||||
- `src/components/organisms/index.ts` (updated exports)
|
|
||||||
- `src/components/index.ts` (updated exports)
|
|
||||||
- `src/hooks/index.ts`
|
|
||||||
- `src/hooks/use-schema-loader.ts`
|
|
||||||
- `src/lib/json-ui/hooks-registry.ts`
|
|
||||||
- `src/lib/json-ui/interfaces/index.ts`
|
|
||||||
- `src/lib/json-ui/json-components.ts`
|
|
||||||
- `src/components/organisms/data-source-manager/DataSourceGroupSection.tsx`
|
|
||||||
|
|
||||||
### Restored Files (130)
|
|
||||||
- 87 atoms
|
|
||||||
- 32 molecules
|
|
||||||
- 11 organisms
|
|
||||||
|
|
||||||
### Removed From Exports (2)
|
|
||||||
- TreeListHeader (non-existent file)
|
|
||||||
- TreeCard (non-existent file)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Root Cause Analysis
|
|
||||||
|
|
||||||
The build failures were caused by a **mismatch between code deletions and dependency updates** in the JSON migration process:
|
|
||||||
|
|
||||||
1. **What Happened:** A previous commit deleted 130+ component files as part of migrating from TSX to JSON definitions
|
|
||||||
2. **What Broke:** Files that depended on these components were not updated, causing import errors
|
|
||||||
3. **Why This Happened:** The deletion was likely automated or incomplete, without verifying all dependent files
|
|
||||||
4. **Impact:** Build broke immediately after the deletion
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Current State
|
|
||||||
|
|
||||||
✅ **Build:** Passing with zero errors
|
|
||||||
✅ **Components:** All 130+ files restored and properly exported
|
|
||||||
✅ **TypeScript:** Zero compilation errors
|
|
||||||
✅ **Bundle:** Successfully generated (8.9 MB)
|
|
||||||
✅ **Types:** All 330+ component types generated
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Recommendations
|
|
||||||
|
|
||||||
### Immediate (Required)
|
|
||||||
1. ✅ All blocking issues resolved
|
|
||||||
2. ✅ Build is stable and ready to deploy
|
|
||||||
|
|
||||||
### Short-term (1-2 weeks)
|
|
||||||
1. Review the JSON migration strategy to ensure proper handling of file deletions
|
|
||||||
2. Implement pre-commit checks to catch missing imports
|
|
||||||
3. Add integration tests for component dependencies
|
|
||||||
4. Fix 2 dynamic/static import conflicts (optional but recommended)
|
|
||||||
|
|
||||||
### Medium-term (1-2 months)
|
|
||||||
1. Continue JSON migration for the 130 restored components
|
|
||||||
2. Implement code splitting to reduce chunk sizes
|
|
||||||
3. Add dependency analysis tooling to prevent future issues
|
|
||||||
|
|
||||||
### Long-term (Ongoing)
|
|
||||||
1. Complete full TSX → JSON migration as planned
|
|
||||||
2. Remove all remaining TSX components once JSON equivalents exist
|
|
||||||
3. Maintain clean component dependency graph
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Verification
|
|
||||||
|
|
||||||
To verify the build is working:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Run build (should complete successfully)
|
|
||||||
npm run build
|
|
||||||
|
|
||||||
# Expected output:
|
|
||||||
# ✓ [number] modules transformed
|
|
||||||
# ✓ built in X.XXs
|
|
||||||
|
|
||||||
# Build artifacts should be in ./dist/
|
|
||||||
ls -lh dist/
|
|
||||||
# Should show: index.html, assets/, icons/, schemas/, manifest.json, etc.
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Conclusion
|
|
||||||
|
|
||||||
The application is now in a stable state with a fully working build. All TypeScript compilation errors have been resolved, and the bundle successfully generates. The application is ready for testing and deployment.
|
|
||||||
|
|
||||||
The restoration of 130 component files represents a return to a stable state pending completion of the JSON migration strategy. Future work should focus on completing this migration rather than repeating the deletion cycle.
|
|
||||||
347
CLAUDE.md
347
CLAUDE.md
@@ -1,347 +0,0 @@
|
|||||||
# Claude Code Documentation
|
|
||||||
|
|
||||||
## Architecture Overview
|
|
||||||
|
|
||||||
This is a low-code React application builder that is migrating from TypeScript/TSX components to a JSON-driven architecture.
|
|
||||||
|
|
||||||
### Current State (Jan 2026)
|
|
||||||
|
|
||||||
- **~420 TSX files** in `src/components/` (legacy - being phased out)
|
|
||||||
- **338 JSON definitions** in `src/config/pages/` (target architecture)
|
|
||||||
- **342 entries** in `json-components-registry.json`
|
|
||||||
- **19 complete JSON implementations** in `src/components/json-definitions/`
|
|
||||||
- **141 duplicate TSX files deleted** (had JSON equivalents)
|
|
||||||
- **5 atoms remaining** to convert: Accordion, FileUpload, Image, Menu, Popover
|
|
||||||
- **1 molecule remaining**: BindingEditor
|
|
||||||
- **3 organisms remaining**: DataSourceManager, NavigationMenu, TreeListPanel
|
|
||||||
|
|
||||||
## Migration Strategy
|
|
||||||
|
|
||||||
### Core Principle
|
|
||||||
|
|
||||||
**ALL components can be converted to JSON except the application entrypoint**, because custom hooks can handle any stateful/complex logic.
|
|
||||||
|
|
||||||
### Directory Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
src/
|
|
||||||
├── components/ # 🔴 LEGACY - Phase out
|
|
||||||
│ ├── atoms/ # Basic UI components (6 TSX remaining)
|
|
||||||
│ │ ├── json-ui/ # JSON-specific atoms
|
|
||||||
│ │ ├── Accordion.tsx
|
|
||||||
│ │ ├── FileUpload.tsx
|
|
||||||
│ │ ├── Image.tsx
|
|
||||||
│ │ ├── Menu.tsx
|
|
||||||
│ │ └── Popover.tsx
|
|
||||||
│ ├── molecules/ # Composite components (1 TSX remaining)
|
|
||||||
│ │ └── BindingEditor.tsx
|
|
||||||
│ ├── organisms/ # Complex feature components (3 TSX remaining)
|
|
||||||
│ │ ├── DataSourceManager.tsx
|
|
||||||
│ │ ├── NavigationMenu.tsx
|
|
||||||
│ │ └── TreeListPanel.tsx
|
|
||||||
│ └── json-definitions/ # ✅ JSON implementations (19 files)
|
|
||||||
│ ├── loading-fallback.json
|
|
||||||
│ ├── navigation-item.json
|
|
||||||
│ ├── page-header-content.json
|
|
||||||
│ ├── component-binding-dialog.json
|
|
||||||
│ ├── data-source-editor-dialog.json
|
|
||||||
│ ├── github-build-status.json
|
|
||||||
│ ├── save-indicator.json
|
|
||||||
│ ├── component-tree.json
|
|
||||||
│ ├── seed-data-manager.json
|
|
||||||
│ ├── lazy-d3-bar-chart.json
|
|
||||||
│ ├── storage-settings.json
|
|
||||||
│ ├── tree-card.json
|
|
||||||
│ ├── filter-input.json
|
|
||||||
│ ├── copy-button.json
|
|
||||||
│ ├── input.json
|
|
||||||
│ └── password-input.json
|
|
||||||
│
|
|
||||||
├── config/
|
|
||||||
│ ├── pages/ # ✅ TARGET - JSON definitions (338 files)
|
|
||||||
│ │ ├── atoms/ # JSON schema for atoms
|
|
||||||
│ │ ├── molecules/ # JSON schema for molecules
|
|
||||||
│ │ ├── organisms/ # JSON schema for organisms
|
|
||||||
│ │ ├── templates/ # Page templates
|
|
||||||
│ │ └── *.json # Page definitions
|
|
||||||
│ └── pages.json # Central routing manifest
|
|
||||||
│
|
|
||||||
├── hooks/ # ✅ Custom hooks for JSON components
|
|
||||||
│ ├── use-save-indicator.ts
|
|
||||||
│ ├── use-component-tree.ts
|
|
||||||
│ ├── use-storage-backend-info.ts
|
|
||||||
│ ├── use-d3-bar-chart.ts
|
|
||||||
│ ├── use-focus-state.ts # NEW: For FilterInput
|
|
||||||
│ ├── use-copy-state.ts # NEW: For CopyButton
|
|
||||||
│ ├── use-password-visibility.ts # NEW: For PasswordInput
|
|
||||||
│ └── index.ts
|
|
||||||
│
|
|
||||||
├── lib/
|
|
||||||
│ └── json-ui/
|
|
||||||
│ ├── component-registry.ts # Component resolver
|
|
||||||
│ ├── component-renderer.tsx # JSON → React renderer
|
|
||||||
│ ├── json-components.ts # JSON component exports (27 components)
|
|
||||||
│ ├── create-json-component.tsx # Pure JSON component factory
|
|
||||||
│ ├── create-json-component-with-hooks.tsx # JSON + hooks factory
|
|
||||||
│ ├── hooks.ts # Data source/action hooks
|
|
||||||
│ ├── hooks-registry.ts # Hook registration (12 hooks registered)
|
|
||||||
│ ├── constants/ # Shared constants for JSON transforms
|
|
||||||
│ │ ├── sizes.ts # Button sizes, icon sizes, dimensions
|
|
||||||
│ │ ├── placements.ts # Popover/tooltip positioning
|
|
||||||
│ │ ├── styles.ts # Common CSS classes (transitions, animations, etc.)
|
|
||||||
│ │ ├── object-fit.ts # Image object-fit classes
|
|
||||||
│ │ └── index.ts
|
|
||||||
│ └── interfaces/ # TypeScript interfaces (1 per file)
|
|
||||||
│ ├── loading-fallback.ts
|
|
||||||
│ ├── navigation-item.ts
|
|
||||||
│ ├── page-header-content.ts
|
|
||||||
│ ├── save-indicator.ts
|
|
||||||
│ ├── lazy-bar-chart.ts
|
|
||||||
│ ├── lazy-line-chart.ts
|
|
||||||
│ ├── lazy-d3-bar-chart.ts
|
|
||||||
│ ├── seed-data-manager.ts
|
|
||||||
│ ├── storage-settings.ts
|
|
||||||
│ ├── github-build-status.ts
|
|
||||||
│ ├── component-binding-dialog.ts
|
|
||||||
│ ├── data-source-editor-dialog.ts
|
|
||||||
│ ├── component-tree.ts
|
|
||||||
│ ├── tree-card.ts
|
|
||||||
│ ├── filter-input.ts
|
|
||||||
│ ├── copy-button.ts
|
|
||||||
│ ├── input.ts
|
|
||||||
│ ├── password-input.ts
|
|
||||||
│ ├── image.ts
|
|
||||||
│ ├── popover.ts
|
|
||||||
│ ├── menu.ts
|
|
||||||
│ ├── file-upload.ts
|
|
||||||
│ ├── accordion.ts
|
|
||||||
│ └── index.ts
|
|
||||||
│
|
|
||||||
├── scripts/ # Migration and audit tools
|
|
||||||
│ ├── audit-json-components.ts
|
|
||||||
│ ├── analyze-duplicates.ts
|
|
||||||
│ ├── cleanup-registry.ts
|
|
||||||
│ └── fix-index-files.ts
|
|
||||||
│
|
|
||||||
└── json-components-registry.json # Master component registry
|
|
||||||
```
|
|
||||||
|
|
||||||
## How It Works
|
|
||||||
|
|
||||||
### 1. Routing Flow
|
|
||||||
|
|
||||||
```
|
|
||||||
pages.json → json-components-registry.json → Component Implementation
|
|
||||||
```
|
|
||||||
|
|
||||||
**Example:**
|
|
||||||
```json
|
|
||||||
// pages.json
|
|
||||||
{
|
|
||||||
"id": "dashboard",
|
|
||||||
"component": "ProjectDashboard"
|
|
||||||
}
|
|
||||||
|
|
||||||
// json-components-registry.json
|
|
||||||
{
|
|
||||||
"type": "ProjectDashboard",
|
|
||||||
"source": "organisms",
|
|
||||||
"load": {
|
|
||||||
"path": "@/components/ProjectDashboard",
|
|
||||||
"export": "ProjectDashboard"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Component Types
|
|
||||||
|
|
||||||
#### Pure JSON Components (No Hooks)
|
|
||||||
Simple stateless components defined entirely in JSON:
|
|
||||||
|
|
||||||
```json
|
|
||||||
// src/components/json-definitions/tree-card.json
|
|
||||||
{
|
|
||||||
"id": "tree-card-container",
|
|
||||||
"type": "Card",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "isSelected",
|
|
||||||
"transform": "data ? 'ring-2 ring-primary' : 'hover:bg-accent/50'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [...]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Exported from `src/lib/json-ui/json-components.ts`:
|
|
||||||
```typescript
|
|
||||||
import treeCardDef from '@/components/json-definitions/tree-card.json'
|
|
||||||
export const TreeCard = createJsonComponent<TreeCardProps>(treeCardDef)
|
|
||||||
```
|
|
||||||
|
|
||||||
#### JSON Components with Hooks
|
|
||||||
Stateful components using custom hooks (**NO WRAPPER FILES NEEDED**):
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
// src/lib/json-ui/json-components.ts
|
|
||||||
export const ComponentTree = createJsonComponentWithHooks<ComponentTreeProps>(
|
|
||||||
componentTreeDef,
|
|
||||||
{
|
|
||||||
hooks: {
|
|
||||||
treeData: {
|
|
||||||
hookName: 'useComponentTree',
|
|
||||||
args: (props) => [props.components || [], props.selectedId || null]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
The custom hook is defined in `src/hooks/use-component-tree.ts` (or other hook files) and registered in `src/lib/json-ui/hooks-registry.ts`.
|
|
||||||
|
|
||||||
#### TSX Components (Legacy)
|
|
||||||
Currently imported directly - these need migration:
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
// ❌ OLD: Direct TSX import
|
|
||||||
import { AppBranding } from '@/components/molecules/AppBranding'
|
|
||||||
|
|
||||||
// ✅ NEW: JSON-based import
|
|
||||||
import { AppBranding } from '@/lib/json-ui/json-components'
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. Registry System
|
|
||||||
|
|
||||||
The `json-components-registry.json` defines how components are loaded:
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"type": "SaveIndicator",
|
|
||||||
"source": "molecules",
|
|
||||||
"jsonCompatible": true
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
- **jsonCompatible**: Whether component can be expressed as JSON
|
|
||||||
- **load.path**: Explicit path to component file (for TSX legacy components)
|
|
||||||
- **source**: Where the component comes from (atoms, molecules, organisms, ui)
|
|
||||||
|
|
||||||
**Note:** `wrapperRequired` and `wrapperComponent` fields in the registry are **obsolete** and should be removed. All stateful logic is handled via `createJsonComponentWithHooks`.
|
|
||||||
|
|
||||||
## Current Issues (Jan 2026)
|
|
||||||
|
|
||||||
### Audit Results
|
|
||||||
|
|
||||||
Run `npm run audit:json` to see current status:
|
|
||||||
|
|
||||||
- ❌ **Errors**
|
|
||||||
- 6 orphaned JSON files (no registry entry)
|
|
||||||
- 7 broken load paths
|
|
||||||
|
|
||||||
- ⚠️ **153 warnings**
|
|
||||||
- 153 duplicate implementations (TSX + JSON)
|
|
||||||
|
|
||||||
### Critical Tasks
|
|
||||||
|
|
||||||
1. **Phase Out `src/components/`**
|
|
||||||
- 153 components have both TSX and JSON definitions
|
|
||||||
- TSX versions should be deleted and routed through JSON
|
|
||||||
|
|
||||||
2. **Clean Up Registry**
|
|
||||||
- Remove `wrapperRequired` and `wrapperComponent` fields (obsolete)
|
|
||||||
- All stateful logic is handled via `createJsonComponentWithHooks`
|
|
||||||
- Custom hooks defined in `src/lib/json-ui/hooks.ts`
|
|
||||||
|
|
||||||
3. **Fix Registry Issues**
|
|
||||||
- Add missing registry entries for orphaned JSON
|
|
||||||
- Fix broken load paths
|
|
||||||
- Verify all source mappings
|
|
||||||
|
|
||||||
## Migration Checklist
|
|
||||||
|
|
||||||
For each component:
|
|
||||||
|
|
||||||
- [ ] Create JSON definition in `src/components/json-definitions/`
|
|
||||||
- [ ] Add TypeScript interface in `src/lib/json-ui/interfaces/` (one file per interface)
|
|
||||||
- [ ] If stateful: Define custom hook in `src/hooks/use-[component-name].ts`
|
|
||||||
- [ ] If stateful: Register hook in `src/lib/json-ui/hooks-registry.ts`
|
|
||||||
- [ ] If stateful: Export hook from `src/hooks/index.ts`
|
|
||||||
- [ ] Export from `src/lib/json-ui/json-components.ts`:
|
|
||||||
- Use `createJsonComponent` for pure/stateless
|
|
||||||
- Use `createJsonComponentWithHooks` for stateful
|
|
||||||
- [ ] Update registry in `json-components-registry.json`
|
|
||||||
- [ ] Update all imports to use `@/lib/json-ui/json-components`
|
|
||||||
- [ ] Delete legacy TSX file from `src/components/`
|
|
||||||
- [ ] Run tests and build to verify
|
|
||||||
|
|
||||||
## Useful Commands
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Run audit to check migration status
|
|
||||||
npm run audit:json
|
|
||||||
|
|
||||||
# Generate component types
|
|
||||||
npm run components:generate-types
|
|
||||||
|
|
||||||
# Build (will fail if components missing)
|
|
||||||
npm run build
|
|
||||||
```
|
|
||||||
|
|
||||||
## Key Files
|
|
||||||
|
|
||||||
- `json-components-registry.json` - Master registry of all components
|
|
||||||
- `src/config/pages.json` - Page routing configuration
|
|
||||||
- `src/lib/json-ui/component-registry.ts` - Component resolver logic
|
|
||||||
- `src/lib/json-ui/json-components.ts` - JSON component exports
|
|
||||||
- `src/lib/json-ui/hooks.ts` - Custom hooks for stateful components
|
|
||||||
- `src/lib/json-ui/hooks-registry.ts` - Hook registration
|
|
||||||
- `scripts/audit-json-components.ts` - Audit tool
|
|
||||||
|
|
||||||
## Notes
|
|
||||||
|
|
||||||
- **Never create new TSX components** - use JSON instead
|
|
||||||
- **All components can be JSON** except the app entrypoint
|
|
||||||
- **Use custom hooks** for stateful logic (via `createJsonComponentWithHooks`)
|
|
||||||
- **NO wrapper files needed** - hooks are defined in `hooks.ts` and registered in `hooks-registry.ts`
|
|
||||||
- **One interface per file** in `src/lib/json-ui/interfaces/`
|
|
||||||
- **Meta JSON files** in `src/config/pages/` are routing schemas
|
|
||||||
- **Full JSON definitions** live in `src/components/json-definitions/`
|
|
||||||
|
|
||||||
## Recent Changes (Jan 2026)
|
|
||||||
|
|
||||||
### Phase 1: Setup & Cleanup
|
|
||||||
- ✅ Fixed e2e build failures (TreeCard, TreeListHeader routing)
|
|
||||||
- ✅ Removed 8 initial duplicate TSX files with JSON equivalents
|
|
||||||
- ✅ Split wrapper-interfaces.ts into individual interface files
|
|
||||||
- ✅ Created audit script to track migration progress
|
|
||||||
- ✅ Updated imports to use `@/lib/json-ui/json-components`
|
|
||||||
- ✅ Clarified: NO wrapper system - use JSON + custom hooks
|
|
||||||
|
|
||||||
### Phase 2: Mass Cleanup
|
|
||||||
- ✅ Cleaned registry - removed 107 obsolete `wrapperRequired`/`wrapperComponent` fields
|
|
||||||
- ✅ Analyzed 153 duplicates, categorized safe deletions
|
|
||||||
- ✅ Deleted 141 duplicate TSX files (had complete JSON implementations)
|
|
||||||
- ✅ Created fix-index-files.ts script to auto-update exports
|
|
||||||
|
|
||||||
### Phase 3: Active Conversions (In Progress)
|
|
||||||
- ✅ Converted FilterInput to JSON with useFocusState hook
|
|
||||||
- ✅ Converted CopyButton to JSON with useCopyState hook
|
|
||||||
- ✅ Converted Input to JSON (pure component with forwardRef support)
|
|
||||||
- ✅ Converted PasswordInput to JSON with usePasswordVisibility hook
|
|
||||||
- ✅ Moved custom hooks from `lib/json-ui/hooks.ts` to `src/hooks/` directory
|
|
||||||
- ✅ Created use-focus-state.ts, use-copy-state.ts, and use-password-visibility.ts
|
|
||||||
- ✅ Updated hooks-registry.ts to include 7 registered hooks
|
|
||||||
|
|
||||||
### Remaining Work
|
|
||||||
- 🔄 5 atoms left: Accordion, FileUpload, Image, Menu, Popover
|
|
||||||
- 🔄 1 molecule left: BindingEditor
|
|
||||||
- 🔄 3 organisms left: DataSourceManager, NavigationMenu, TreeListPanel
|
|
||||||
- ✅ 20 JSON components complete (up from 12)
|
|
||||||
|
|
||||||
## Next Steps
|
|
||||||
|
|
||||||
1. Clean up registry - remove `wrapperRequired` and `wrapperComponent` fields
|
|
||||||
2. Convert the 153 duplicate TSX components to JSON-only
|
|
||||||
3. Fix 6 orphaned JSON files (add registry entries)
|
|
||||||
4. Fix 7 broken load paths in registry
|
|
||||||
5. Complete full migration of `src/components/` to JSON
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
# Update for Current State section (lines 8-15)
|
|
||||||
|
|
||||||
- **~400 TSX files** in `src/components/` (legacy - being phased out)
|
|
||||||
- **338 JSON definitions** in `src/config/pages/` (target architecture)
|
|
||||||
- **342 entries** in `json-components-registry.json`
|
|
||||||
- **27 complete JSON implementations** in `src/components/json-definitions/`
|
|
||||||
- **141 duplicate TSX files deleted** (had JSON equivalents)
|
|
||||||
- **✅ ALL ATOMS CONVERTED!** (0 remaining)
|
|
||||||
- **1 molecule remaining**: BindingEditor
|
|
||||||
- **3 organisms remaining**: DataSourceManager, NavigationMenu, TreeListPanel
|
|
||||||
|
|
||||||
# Update for atoms section (lines 28-34)
|
|
||||||
|
|
||||||
│ ├── atoms/ # ✅ ALL CONVERTED! (0 TSX remaining)
|
|
||||||
│ │ └── json-ui/ # JSON-specific atoms
|
|
||||||
|
|
||||||
# Update for json-definitions (lines 41-57)
|
|
||||||
|
|
||||||
│ └── json-definitions/ # ✅ JSON implementations (27 files)
|
|
||||||
│ ├── loading-fallback.json
|
|
||||||
│ ├── navigation-item.json
|
|
||||||
│ ├── page-header-content.json
|
|
||||||
│ ├── component-binding-dialog.json
|
|
||||||
│ ├── data-source-editor-dialog.json
|
|
||||||
│ ├── github-build-status.json
|
|
||||||
│ ├── save-indicator.json
|
|
||||||
│ ├── component-tree.json
|
|
||||||
│ ├── seed-data-manager.json
|
|
||||||
│ ├── lazy-d3-bar-chart.json
|
|
||||||
│ ├── storage-settings.json
|
|
||||||
│ ├── tree-card.json
|
|
||||||
│ ├── filter-input.json
|
|
||||||
│ ├── copy-button.json
|
|
||||||
│ ├── input.json
|
|
||||||
│ ├── password-input.json
|
|
||||||
│ ├── image.json
|
|
||||||
│ ├── popover.json
|
|
||||||
│ ├── menu.json
|
|
||||||
│ ├── file-upload.json
|
|
||||||
│ └── accordion.json
|
|
||||||
|
|
||||||
# Update for hooks (lines 73-76)
|
|
||||||
|
|
||||||
│ ├── use-focus-state.ts # For FilterInput
|
|
||||||
│ ├── use-copy-state.ts # For CopyButton
|
|
||||||
│ ├── use-password-visibility.ts # For PasswordInput
|
|
||||||
│ ├── use-image-state.ts # For Image
|
|
||||||
│ ├── use-popover-state.ts # For Popover
|
|
||||||
│ ├── use-menu-state.ts # For Menu
|
|
||||||
│ ├── use-file-upload.ts # For FileUpload
|
|
||||||
│ ├── use-accordion.ts # For Accordion
|
|
||||||
|
|
||||||
# Update for json-components count (line 82)
|
|
||||||
|
|
||||||
│ ├── json-components.ts # JSON component exports (27 components)
|
|
||||||
|
|
||||||
# Update for hooks-registry count (line 86)
|
|
||||||
|
|
||||||
│ ├── hooks-registry.ts # Hook registration (12 hooks registered)
|
|
||||||
@@ -1,181 +0,0 @@
|
|||||||
# JSON Component Migration - Session Summary
|
|
||||||
|
|
||||||
## Status: ✅ COMPLETE (9 Components)
|
|
||||||
|
|
||||||
### What Was Done
|
|
||||||
|
|
||||||
#### 1. **5 Atoms Successfully Migrated**
|
|
||||||
- ✅ Accordion → JSON definition + useAccordion hook
|
|
||||||
- ✅ CopyButton → JSON definition + useCopyState hook
|
|
||||||
- ✅ FileUpload → JSON definition + useFileUpload hook
|
|
||||||
- ✅ FilterInput → JSON definition + useFocusState hook
|
|
||||||
- ✅ Image → JSON definition + useImageState hook
|
|
||||||
- ✅ Input → Pure JSON (stateless)
|
|
||||||
- ✅ PasswordInput → JSON definition + usePasswordVisibility hook
|
|
||||||
- ✅ Popover → JSON definition + usePopoverState hook
|
|
||||||
|
|
||||||
#### 2. **1 Molecule Successfully Migrated**
|
|
||||||
- ✅ BindingEditor → JSON definition + useBindingEditor hook
|
|
||||||
|
|
||||||
#### 3. **Key Changes Made**
|
|
||||||
1. **BindingEditor Export** (was missing)
|
|
||||||
- Added `BindingEditorProps` import to `src/lib/json-ui/json-components.ts`
|
|
||||||
- Added `bindingEditorDef` JSON import
|
|
||||||
- Created `createJsonComponentWithHooks` export with hook binding
|
|
||||||
- Registered `useBindingEditor` in hooks-registry.ts
|
|
||||||
- Exported hook from `src/hooks/index.ts`
|
|
||||||
|
|
||||||
2. **Import Updates** (5 files)
|
|
||||||
- `SearchInput.tsx` → uses Input from json-components
|
|
||||||
- `SearchBar.tsx` → uses Input from json-components
|
|
||||||
- `ComponentBindingDialog.tsx` → uses BindingEditor from json-components
|
|
||||||
- `FormsTab.tsx` → uses Input, CopyButton, FileUpload, PasswordInput
|
|
||||||
- `DisplayTab.tsx` → uses Accordion
|
|
||||||
- `FormControlsSection.tsx` → uses FilterInput
|
|
||||||
|
|
||||||
3. **Build Fixes**
|
|
||||||
- Fixed `use-schema-loader.ts` dynamic import (added .json extension)
|
|
||||||
- Fixed `DataSourceGroupSection.tsx` (removed missing DataSourceCard dependency)
|
|
||||||
- Restored and cleaned up component files (130 files recovered)
|
|
||||||
|
|
||||||
4. **Cleanup**
|
|
||||||
- Deleted 9 legacy TSX files (atoms + BindingEditor)
|
|
||||||
- Updated component index exports to remove deleted components
|
|
||||||
- Removed orphaned exports from index files
|
|
||||||
|
|
||||||
### Architecture Overview
|
|
||||||
|
|
||||||
```
|
|
||||||
src/components/json-definitions/
|
|
||||||
├── accordion.json
|
|
||||||
├── copy-button.json
|
|
||||||
├── file-upload.json
|
|
||||||
├── filter-input.json
|
|
||||||
├── image.json
|
|
||||||
├── input.json
|
|
||||||
├── password-input.json
|
|
||||||
├── popover.json
|
|
||||||
├── binding-editor.json
|
|
||||||
└── ... (13 other JSON definitions)
|
|
||||||
|
|
||||||
src/lib/json-ui/
|
|
||||||
├── json-components.ts (exports 22 components)
|
|
||||||
├── create-json-component.tsx (pure JSON factory)
|
|
||||||
├── create-json-component-with-hooks.tsx (stateful factory)
|
|
||||||
├── hooks-registry.ts (12 registered hooks)
|
|
||||||
└── interfaces/ (TypeScript interfaces for each component)
|
|
||||||
|
|
||||||
src/hooks/
|
|
||||||
├── use-accordion.ts
|
|
||||||
├── use-binding-editor.ts
|
|
||||||
├── use-copy-state.ts
|
|
||||||
├── use-file-upload.ts
|
|
||||||
├── use-focus-state.ts
|
|
||||||
├── use-image-state.ts
|
|
||||||
├── use-menu-state.ts
|
|
||||||
├── use-password-visibility.ts
|
|
||||||
├── use-popover-state.ts
|
|
||||||
└── ... (40+ other application hooks)
|
|
||||||
```
|
|
||||||
|
|
||||||
### Build Status: ✅ PASSING
|
|
||||||
|
|
||||||
```
|
|
||||||
✓ TypeScript compilation: OK (0 errors)
|
|
||||||
✓ Vite build: OK
|
|
||||||
✓ Modules transformed: 9,408
|
|
||||||
✓ Build time: 9.22 seconds
|
|
||||||
✓ Production bundle: Generated successfully
|
|
||||||
```
|
|
||||||
|
|
||||||
**Non-blocking warnings:** 8 dynamic/static import conflicts (do not prevent build)
|
|
||||||
|
|
||||||
### Statistics
|
|
||||||
|
|
||||||
| Metric | Value |
|
|
||||||
|--------|-------|
|
|
||||||
| JSON Components Created | 22 |
|
|
||||||
| JSON Definitions | 22 |
|
|
||||||
| Registered Hooks | 12 |
|
|
||||||
| TSX Files Deleted | 9 |
|
|
||||||
| Components with JSON+Hooks | 15 |
|
|
||||||
| Pure JSON Components | 8 |
|
|
||||||
| Registry Entries | 342 |
|
|
||||||
| Build Status | ✅ PASSING |
|
|
||||||
|
|
||||||
### What Remains
|
|
||||||
|
|
||||||
#### Documented in CLAUDE.md
|
|
||||||
- 3 Organisms still TSX: DataSourceManager, NavigationMenu, TreeListPanel
|
|
||||||
- These should be converted following the same pattern
|
|
||||||
|
|
||||||
#### Beyond Scope (120+ additional components)
|
|
||||||
- Many TSX files were restored during build fixes
|
|
||||||
- These have JSON equivalents in `src/config/pages/` but aren't yet exported
|
|
||||||
- Should be migrated in future phases using the same process
|
|
||||||
|
|
||||||
### Key Learnings
|
|
||||||
|
|
||||||
1. **Pure JSON vs JSON+Hooks Pattern:**
|
|
||||||
- Stateless components: `createJsonComponent(jsonDef)`
|
|
||||||
- Stateful components: `createJsonComponentWithHooks(jsonDef, { hooks: {...} })`
|
|
||||||
- No wrapper files needed—hooks are registered centrally
|
|
||||||
|
|
||||||
2. **Export Strategy:**
|
|
||||||
- All JSON components exported from `src/lib/json-ui/json-components.ts`
|
|
||||||
- Consistent import path: `import { Component } from '@/lib/json-ui/json-components'`
|
|
||||||
- Replaces scattered imports from `src/components/`
|
|
||||||
|
|
||||||
3. **Hook Registration:**
|
|
||||||
- Hooks live in `src/hooks/` directory
|
|
||||||
- Registered in `src/lib/json-ui/hooks-registry.ts`
|
|
||||||
- Exported from `src/hooks/index.ts`
|
|
||||||
|
|
||||||
### Next Steps
|
|
||||||
|
|
||||||
1. **Immediate** (if continuing migration):
|
|
||||||
- Convert 3 remaining organisms (DataSourceManager, NavigationMenu, TreeListPanel)
|
|
||||||
- Follow same pattern: JSON def + hook (if needed) + export + delete TSX
|
|
||||||
|
|
||||||
2. **Medium-term** (optional):
|
|
||||||
- Clean up 120+ additional components that have JSON but aren't exported
|
|
||||||
- Address 6 orphaned JSON definitions in registry
|
|
||||||
- Fix 7 broken load paths in registry
|
|
||||||
|
|
||||||
3. **Testing** (recommended):
|
|
||||||
- Run test suite to verify components work as expected
|
|
||||||
- Test pages that use these components
|
|
||||||
- Verify no runtime issues with JSON rendering
|
|
||||||
|
|
||||||
### Files Changed This Session
|
|
||||||
|
|
||||||
**Created:**
|
|
||||||
- BUILD_REPORT.md (build analysis documentation)
|
|
||||||
- build-output.txt (build logs)
|
|
||||||
|
|
||||||
**Modified (code):**
|
|
||||||
- src/lib/json-ui/json-components.ts (+BindingEditor export)
|
|
||||||
- src/lib/json-ui/hooks-registry.ts (+useBindingEditor registration)
|
|
||||||
- src/hooks/index.ts (+useBindingEditor export)
|
|
||||||
- src/lib/json-ui/interfaces/index.ts (+BindingEditorProps export)
|
|
||||||
- src/hooks/use-schema-loader.ts (fixed dynamic import)
|
|
||||||
- src/components/organisms/data-source-manager/DataSourceGroupSection.tsx (removed DataSourceCard)
|
|
||||||
- 5 components with import updates
|
|
||||||
|
|
||||||
**Deleted:**
|
|
||||||
- src/components/atoms/Accordion.tsx
|
|
||||||
- src/components/atoms/CopyButton.tsx
|
|
||||||
- src/components/atoms/FileUpload.tsx
|
|
||||||
- src/components/atoms/FilterInput.tsx
|
|
||||||
- src/components/atoms/Image.tsx
|
|
||||||
- src/components/atoms/Input.tsx
|
|
||||||
- src/components/atoms/PasswordInput.tsx
|
|
||||||
- src/components/atoms/Popover.tsx
|
|
||||||
- src/components/molecules/BindingEditor.tsx
|
|
||||||
|
|
||||||
**Updated (exports):**
|
|
||||||
- src/components/atoms/index.ts (removed 8 exports)
|
|
||||||
- src/components/molecules/index.ts (removed 1 export)
|
|
||||||
|
|
||||||
### Commit Hash
|
|
||||||
`f05f896` - "feat: Complete JSON component migration for 9 components (atoms + BindingEditor)"
|
|
||||||
@@ -1,237 +0,0 @@
|
|||||||
# Complete Migration Strategy - Remaining Work
|
|
||||||
|
|
||||||
## The Goal
|
|
||||||
**End state: Only `src/main.tsx` and `src/index.html` as TSX/HTML**
|
|
||||||
- Everything else in `src/components/` → JSON + custom hooks
|
|
||||||
|
|
||||||
## Current Reality (After Today's Work)
|
|
||||||
|
|
||||||
```
|
|
||||||
Completed: 9 components (8 atoms + 1 molecule)
|
|
||||||
├── Accordion, CopyButton, FileUpload, FilterInput, Image, Input, PasswordInput, Popover (atoms)
|
|
||||||
└── BindingEditor (molecule)
|
|
||||||
|
|
||||||
Remaining: ~220 TSX files (excluding main.tsx and demo/showcase pages)
|
|
||||||
├── 3 organisms: DataSourceManager, NavigationMenu, TreeListPanel
|
|
||||||
├── 110+ atoms: ActionButton, ActionCard, Alert, Button, Card, etc.
|
|
||||||
├── 35+ molecules: AppBranding, ComponentTree, PropertyEditor, etc.
|
|
||||||
├── 7 app bootstrap components: AppBootstrap, AppLayout, etc.
|
|
||||||
└── 55+ demo/showcase pages
|
|
||||||
```
|
|
||||||
|
|
||||||
## Priority Tiers
|
|
||||||
|
|
||||||
### Tier 1: Core App Bootstrap (7 files - High Impact)
|
|
||||||
**These are used by every page. Converting them unblocks everything.**
|
|
||||||
|
|
||||||
1. `src/components/app/AppBootstrap.tsx`
|
|
||||||
- Uses: `useAppBootstrap` hook
|
|
||||||
- Action: Create JSON, register hook, export, delete TSX
|
|
||||||
|
|
||||||
2. `src/components/app/AppLayout.tsx`
|
|
||||||
- Action: Same pattern
|
|
||||||
|
|
||||||
3. `src/components/app/LoadingScreen.tsx`
|
|
||||||
- Likely stateless UI component
|
|
||||||
- Action: Convert to pure JSON
|
|
||||||
|
|
||||||
4. `src/components/app/AppDialogs.tsx`
|
|
||||||
- Action: Assess and migrate
|
|
||||||
|
|
||||||
5. `src/components/app/AppMainPanel.tsx`
|
|
||||||
- Action: Assess and migrate
|
|
||||||
|
|
||||||
6. `src/components/app/AppRouterBootstrap.tsx`
|
|
||||||
- Router mode variant of bootstrap
|
|
||||||
|
|
||||||
7. `src/components/app/AppRouterLayout.tsx`
|
|
||||||
- Router mode variant of layout
|
|
||||||
|
|
||||||
**Impact:** Converting these 7 components would eliminate the need for TSX anywhere in the bootstrap flow
|
|
||||||
|
|
||||||
### Tier 2: 3 Documented Organisms (3 files - Medium Impact)
|
|
||||||
**Mentioned in CLAUDE.md as remaining work**
|
|
||||||
|
|
||||||
1. `src/components/organisms/DataSourceManager.tsx`
|
|
||||||
2. `src/components/organisms/NavigationMenu.tsx`
|
|
||||||
3. `src/components/organisms/TreeListPanel.tsx`
|
|
||||||
|
|
||||||
**Impact:** Completes the documented migration targets
|
|
||||||
|
|
||||||
### Tier 3: Core UI Atoms & Molecules (150+ files - Large Scale)
|
|
||||||
**The bulk of component library**
|
|
||||||
|
|
||||||
**Atoms** (~110 files):
|
|
||||||
- ActionButton, ActionCard, Alert, Avatar, Badge, Breadcrumb, Button, Calendar, Card, Checkbox, CommandPalette, DatePicker, Dialog, Divider, Drawer, EmptyState, FileIcon, Form, Grid, Heading, HoverCard, Input, Kbd, Label, Link, List, Menu, Modal, NumberInput, PasswordInput, Popover, ProgressBar, Radio, RangeSlider, Rating, ScrollArea, SearchInput, Select, Separator, Skeleton, Slider, Stack, Switch, Table, Tabs, Tag, Text, TextArea, Toggle, Tooltip, TreeIcon, etc.
|
|
||||||
|
|
||||||
**Molecules** (~35 files):
|
|
||||||
- AppBranding, CanvasRenderer, ComponentTree, ComponentPalette, PropertyEditor, SearchBar, ToolbarButton, TreeFormDialog, etc.
|
|
||||||
|
|
||||||
**Current strategy:** These have JSON definitions in `src/config/pages/` but aren't yet exported from `json-components.ts`. Need to:
|
|
||||||
1. Create JSON definitions in `src/components/json-definitions/` (if not already there)
|
|
||||||
2. Create TypeScript interfaces
|
|
||||||
3. Register hooks (if stateful)
|
|
||||||
4. Export from `json-components.ts`
|
|
||||||
5. Delete TSX files
|
|
||||||
|
|
||||||
### Tier 4: Demo/Showcase Pages (55+ files - No Impact on App)
|
|
||||||
**These are development/demo utilities**
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
- AtomicComponentShowcase.tsx
|
|
||||||
- JSONConversionShowcase.tsx
|
|
||||||
- DashboardDemoPage.tsx
|
|
||||||
- DataBindingDesigner.tsx
|
|
||||||
- JSONFlaskDesigner.tsx
|
|
||||||
- etc.
|
|
||||||
|
|
||||||
**Decision:** These are optional. Could be:
|
|
||||||
- Converted to JSON (least effort)
|
|
||||||
- Deleted if not needed
|
|
||||||
- Left as-is if they're development tools
|
|
||||||
|
|
||||||
## Recommended Execution Order
|
|
||||||
|
|
||||||
### Phase 1: Bootstrap (Highest ROI)
|
|
||||||
1. **AppBootstrap** → JSON + useAppBootstrap hook
|
|
||||||
2. **AppLayout** → JSON + appropriate hooks
|
|
||||||
3. **LoadingScreen** → Pure JSON
|
|
||||||
4. Repeat for other 4 app components
|
|
||||||
|
|
||||||
**Why first:** These are on the critical path. Every app render goes through them. Converting them proves the architecture works for complex components.
|
|
||||||
|
|
||||||
**Expected time:** 2-3 hours
|
|
||||||
|
|
||||||
### Phase 2: Documented Organisms
|
|
||||||
1. **DataSourceManager** → JSON + hooks
|
|
||||||
2. **NavigationMenu** → JSON + hooks
|
|
||||||
3. **TreeListPanel** → JSON + hooks
|
|
||||||
|
|
||||||
**Why next:** Completes the documented migration targets from CLAUDE.md
|
|
||||||
|
|
||||||
**Expected time:** 2-3 hours
|
|
||||||
|
|
||||||
### Phase 3: Core Component Library (If Time/Priority)
|
|
||||||
**Option A: Batch similar components**
|
|
||||||
- All simple buttons/links as one batch
|
|
||||||
- All inputs as one batch
|
|
||||||
- All containers/layouts as one batch
|
|
||||||
|
|
||||||
**Option B: Focus on most-used**
|
|
||||||
- Button, Input, Card, Dialog, Menu → highest impact
|
|
||||||
- Others as needed
|
|
||||||
|
|
||||||
**Expected time:** 8-20 hours (depending on thoroughness)
|
|
||||||
|
|
||||||
### Phase 4: Demo Pages (Nice-to-have)
|
|
||||||
Convert or delete as appropriate. Low priority.
|
|
||||||
|
|
||||||
## Pattern to Follow (Proven)
|
|
||||||
|
|
||||||
For each component:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# 1. Create/verify JSON definition
|
|
||||||
src/components/json-definitions/[component].json
|
|
||||||
|
|
||||||
# 2. Create/verify TypeScript interface
|
|
||||||
src/lib/json-ui/interfaces/[component].ts
|
|
||||||
|
|
||||||
# 3. If stateful, create custom hook
|
|
||||||
src/hooks/use-[component].ts
|
|
||||||
# Then register in hooks-registry.ts
|
|
||||||
# Then export from hooks/index.ts
|
|
||||||
|
|
||||||
# 4. Export from json-components.ts
|
|
||||||
export const ComponentName = createJsonComponent[WithHooks]<Props>(def, ...)
|
|
||||||
|
|
||||||
# 5. Update registry entry
|
|
||||||
json-components-registry.json
|
|
||||||
|
|
||||||
# 6. Delete legacy TSX
|
|
||||||
rm src/components/[category]/[ComponentName].tsx
|
|
||||||
|
|
||||||
# 7. Update index.ts exports
|
|
||||||
src/components/[category]/index.ts
|
|
||||||
|
|
||||||
# 8. Update all imports across codebase
|
|
||||||
# From: import { X } from '@/components/...'
|
|
||||||
# To: import { X } from '@/lib/json-ui/json-components'
|
|
||||||
|
|
||||||
# 9. Verify build passes
|
|
||||||
npm run build
|
|
||||||
```
|
|
||||||
|
|
||||||
## Parallel Work Opportunities
|
|
||||||
|
|
||||||
**Can work on simultaneously:**
|
|
||||||
- AppBootstrap + AppLayout (independent)
|
|
||||||
- DataSourceManager + NavigationMenu (independent)
|
|
||||||
- Multiple atoms in parallel (Button, Input, Card, Dialog don't depend on each other)
|
|
||||||
|
|
||||||
**Must sequence:**
|
|
||||||
- ChildComponent → ParentComponent (parent depends on child)
|
|
||||||
- Example: Button must be JSON before ButtonGroup
|
|
||||||
|
|
||||||
## Success Metrics
|
|
||||||
|
|
||||||
**Current State:**
|
|
||||||
- 22 JSON components exported
|
|
||||||
- 230 TSX files remaining
|
|
||||||
- Build passes ✅
|
|
||||||
|
|
||||||
**Phase 1 Success:**
|
|
||||||
- 29+ JSON components (added 7 app bootstrap)
|
|
||||||
- 223 TSX files remaining
|
|
||||||
- Build passes ✅
|
|
||||||
|
|
||||||
**Phase 2 Success:**
|
|
||||||
- 32+ JSON components (added 3 organisms)
|
|
||||||
- 220 TSX files remaining
|
|
||||||
- Build passes ✅
|
|
||||||
|
|
||||||
**Phase 3 Success (Core Library):**
|
|
||||||
- 150+ JSON components
|
|
||||||
- 75 TSX files remaining (mostly demo pages)
|
|
||||||
- Build passes ✅
|
|
||||||
|
|
||||||
**Final State (Full Migration):**
|
|
||||||
- 200+ JSON components
|
|
||||||
- 2 TSX files (main.tsx + ErrorFallback.tsx as optional)
|
|
||||||
- 1 HTML file (index.html)
|
|
||||||
- Build passes ✅
|
|
||||||
|
|
||||||
## Key Advantages Once Complete
|
|
||||||
|
|
||||||
1. **No component duplication** - Single source of truth (JSON)
|
|
||||||
2. **Easier maintenance** - All components follow same pattern
|
|
||||||
3. **Better code reuse** - Hooks shared across components
|
|
||||||
4. **Smaller bundle** - JSON more compressible than TSX
|
|
||||||
5. **Faster iteration** - Change JSON, no rebuild needed (with hot reload)
|
|
||||||
6. **Better tooling** - Can build JSON editing UI without code knowledge
|
|
||||||
|
|
||||||
## Risks & Mitigation
|
|
||||||
|
|
||||||
| Risk | Mitigation |
|
|
||||||
|------|-----------|
|
|
||||||
| Breaking changes | Run tests frequently, commit after each component |
|
|
||||||
| Performance regression | Monitor bundle size, hook performance |
|
|
||||||
| Import path issues | Use find-replace to update all imports systematically |
|
|
||||||
| Circular dependencies | Review `src/lib/json-ui/` structure before major changes |
|
|
||||||
| Hook registration errors | Test each hook in hooks-registry before moving to next |
|
|
||||||
|
|
||||||
## Next Immediate Steps
|
|
||||||
|
|
||||||
1. **Run audit** to get baseline
|
|
||||||
```bash
|
|
||||||
npm run audit:json
|
|
||||||
```
|
|
||||||
|
|
||||||
2. **Pick one app bootstrap component** (e.g., LoadingScreen - simplest)
|
|
||||||
|
|
||||||
3. **Follow the pattern** from today's work with Accordion/BindingEditor
|
|
||||||
|
|
||||||
4. **Commit after each component** with clear message
|
|
||||||
|
|
||||||
5. **Run tests** to catch regressions
|
|
||||||
|
|
||||||
@@ -1,195 +0,0 @@
|
|||||||
# Root Cause Analysis: JSON-Based React Component System
|
|
||||||
|
|
||||||
## Executive Summary
|
|
||||||
|
|
||||||
The repository is attempting to transition from a traditional TypeScript React component architecture to a JSON-based declarative UI system. The build is currently failing because the transition is incomplete - some TypeScript components were deleted but their imports remain, and the JSON component system cannot yet fully replace them.
|
|
||||||
|
|
||||||
## Current State: Hybrid System Failure
|
|
||||||
|
|
||||||
### What Was Attempted
|
|
||||||
1. **123 TypeScript components were deleted** (commit aa51074) and marked as "json-compatible" in the registry
|
|
||||||
2. **JSON component registry created** with 375+ component definitions
|
|
||||||
3. **JSON UI rendering system built** with component-renderer.tsx, expression evaluator, data binding, etc.
|
|
||||||
4. **Wrapper components created** for complex molecules that need special handling
|
|
||||||
|
|
||||||
### What's Broken
|
|
||||||
The build fails with these errors:
|
|
||||||
```
|
|
||||||
✘ [ERROR] No matching export in "src/components/molecules/index.ts" for import "NavigationItem"
|
|
||||||
✘ [ERROR] No matching export in "src/components/molecules/index.ts" for import "PageHeaderContent"
|
|
||||||
✘ [ERROR] No matching export in "src/components/molecules/index.ts" for import "TreeCard"
|
|
||||||
✘ [ERROR] No matching export in "src/components/molecules/index.ts" for import "TreeListHeader"
|
|
||||||
✘ [ERROR] No matching export in "src/components/molecules/index.ts" for import "preloadMonacoEditor"
|
|
||||||
✘ [ERROR] No matching export in "src/components/molecules/index.ts" for import "LoadingFallback"
|
|
||||||
```
|
|
||||||
|
|
||||||
## Root Causes
|
|
||||||
|
|
||||||
### 1. **Incomplete Conversion Strategy**
|
|
||||||
Components were marked as JSON-compatible and deleted, but:
|
|
||||||
- The **consuming code still imports them as TypeScript modules**
|
|
||||||
- No migration was done to convert consumers to use the JSON renderer
|
|
||||||
- The JSON system exists but isn't wired into the main application flow
|
|
||||||
|
|
||||||
### 2. **Misunderstanding of JSON Component Architecture**
|
|
||||||
The JSON system is designed for **declarative page configurations**, not as a drop-in replacement for React components. Example:
|
|
||||||
|
|
||||||
**Traditional React:**
|
|
||||||
```tsx
|
|
||||||
import { TreeCard } from '@/components/molecules'
|
|
||||||
<TreeCard tree={data} onSelect={handleSelect} />
|
|
||||||
```
|
|
||||||
|
|
||||||
**JSON System:**
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"type": "TreeCard",
|
|
||||||
"bindings": {
|
|
||||||
"tree": { "source": "currentTree" }
|
|
||||||
},
|
|
||||||
"events": {
|
|
||||||
"onSelect": { "action": "selectTree" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
The JSON system requires:
|
|
||||||
- JSON configuration files
|
|
||||||
- JSONSchemaPageLoader or PageRenderer wrapper
|
|
||||||
- Data sources defined in JSON
|
|
||||||
- Event handlers defined in JSON
|
|
||||||
- Cannot be imported like a normal React component
|
|
||||||
|
|
||||||
### 3. **Deleted Components Still Referenced**
|
|
||||||
Components deleted but still imported:
|
|
||||||
- **TreeCard** - Used in TreeListPanel.tsx
|
|
||||||
- **TreeListHeader** - Used in TreeListPanel.tsx
|
|
||||||
- **LoadingFallback** - Used in JSONSchemaPageLoader.tsx and routes.tsx
|
|
||||||
- **NavigationItem** - File exists but not exported from index.ts
|
|
||||||
- **PageHeaderContent** - File exists but not exported from index.ts
|
|
||||||
- **preloadMonacoEditor** - Function exists but not exported from index.ts
|
|
||||||
|
|
||||||
### 4. **Module System vs Component Registry Mismatch**
|
|
||||||
The component-registry.ts uses `import.meta.glob` to load ALL .tsx files:
|
|
||||||
```ts
|
|
||||||
const moleculeModules = import.meta.glob('@/components/molecules/*.tsx', { eager: true })
|
|
||||||
```
|
|
||||||
|
|
||||||
This means:
|
|
||||||
- It CAN dynamically load TreeCard, TreeListHeader, etc. IF they exist as .tsx files
|
|
||||||
- But they were DELETED, so they can't be found
|
|
||||||
- The registry says they're "json-compatible" but provides no fallback
|
|
||||||
- The JSON renderer can use them IF loaded via JSON config, but direct imports fail
|
|
||||||
|
|
||||||
## The Fundamental Problem: No Working JSON System Examples
|
|
||||||
|
|
||||||
**Key Issue:** While the JSON UI infrastructure exists, there are NO working examples of pages that successfully:
|
|
||||||
1. Define a complex page entirely in JSON
|
|
||||||
2. Handle state management in JSON
|
|
||||||
3. Wire up all events in JSON
|
|
||||||
4. Replace an existing TypeScript page
|
|
||||||
|
|
||||||
The infrastructure exists but hasn't been proven to work end-to-end.
|
|
||||||
|
|
||||||
## Architecture Deep Dive
|
|
||||||
|
|
||||||
### JSON UI System Components
|
|
||||||
```
|
|
||||||
src/lib/json-ui/
|
|
||||||
├── component-renderer.tsx # Renders individual components from JSON
|
|
||||||
├── page-renderer.tsx # Renders full pages from JSON
|
|
||||||
├── component-registry.ts # Maps component names to React components
|
|
||||||
├── expression-evaluator.ts # Evaluates data binding expressions
|
|
||||||
├── hooks.ts # Data source hooks
|
|
||||||
├── schema.ts # TypeScript types
|
|
||||||
└── wrappers/ # Special wrappers for complex components
|
|
||||||
```
|
|
||||||
|
|
||||||
### How It Should Work (Theory)
|
|
||||||
1. Create JSON page definition in `src/config/ui-examples/my-page.json`
|
|
||||||
2. Load it with `<JSONSchemaPageLoader schemaPath="/config/ui-examples/my-page.json" />`
|
|
||||||
3. JSON renderer looks up components in registry
|
|
||||||
4. Registry loads them via import.meta.glob
|
|
||||||
5. Components render with data bindings and events
|
|
||||||
|
|
||||||
### Why It Doesn't Work (Reality)
|
|
||||||
1. **Deleted components can't be loaded** - glob can't find non-existent files
|
|
||||||
2. **Existing TypeScript pages import components directly** - they don't use JSON loader
|
|
||||||
3. **No migration path** - can't gradually convert pages
|
|
||||||
4. **Registry assumes all components exist as .tsx files** - no JSON-only components
|
|
||||||
|
|
||||||
## Two Possible Solutions
|
|
||||||
|
|
||||||
### Option A: Restore Components (Backward Compatibility)
|
|
||||||
**Goal:** Make the build work by restoring deleted components
|
|
||||||
|
|
||||||
Steps:
|
|
||||||
1. Restore TreeCard, TreeListHeader, LoadingFallback as .tsx files
|
|
||||||
2. Export NavigationItem, PageHeaderContent, preloadMonacoEditor
|
|
||||||
3. Keep JSON system for future use
|
|
||||||
4. Gradual migration when JSON system proven
|
|
||||||
|
|
||||||
**Pros:** Quick fix, maintains compatibility, low risk
|
|
||||||
**Cons:** Delays JSON transition, maintains technical debt
|
|
||||||
|
|
||||||
### Option B: Full JSON Transition (Forward-Looking)
|
|
||||||
**Goal:** Convert consuming pages to use JSON system
|
|
||||||
|
|
||||||
Steps:
|
|
||||||
1. Convert TreeListPanel.tsx to use JSON renderer
|
|
||||||
2. Convert routes.tsx to load JSON configs
|
|
||||||
3. Create JSON definitions for missing components
|
|
||||||
4. Delete rigid TypeScript components
|
|
||||||
5. Prove JSON system works end-to-end
|
|
||||||
|
|
||||||
**Pros:** Achieves goal of JSON system, modern architecture
|
|
||||||
**Cons:** High risk, requires extensive testing, may reveal more issues
|
|
||||||
|
|
||||||
## Recommendation
|
|
||||||
|
|
||||||
**Start with Option A**, then gradually move toward Option B:
|
|
||||||
|
|
||||||
1. **Immediate Fix** (Option A):
|
|
||||||
- Restore the 3 deleted components (TreeCard, TreeListHeader, LoadingFallback)
|
|
||||||
- Fix exports for existing components (NavigationItem, PageHeaderContent, preloadMonacoEditor)
|
|
||||||
- Get the build working
|
|
||||||
|
|
||||||
2. **Validation Phase**:
|
|
||||||
- Create 1-2 complete working examples of JSON pages
|
|
||||||
- Test all JSON system features (data binding, events, conditionals, loops)
|
|
||||||
- Document the conversion process
|
|
||||||
- Identify limitations
|
|
||||||
|
|
||||||
3. **Gradual Migration** (Option B):
|
|
||||||
- Convert simple pages first
|
|
||||||
- Build tooling to help convert TypeScript to JSON
|
|
||||||
- Only delete TypeScript after JSON proven working
|
|
||||||
- Keep wrappers for complex components
|
|
||||||
|
|
||||||
## Files Requiring Immediate Attention
|
|
||||||
|
|
||||||
1. `src/components/molecules/TreeCard.tsx` - RESTORE from aa51074~1
|
|
||||||
2. `src/components/molecules/TreeListHeader.tsx` - RESTORE from aa51074~1
|
|
||||||
3. `src/components/molecules/LoadingFallback.tsx` - RESTORE from aa51074~1
|
|
||||||
4. `src/components/molecules/index.ts` - ADD exports for NavigationItem, PageHeaderContent
|
|
||||||
5. `src/components/molecules/LazyMonacoEditor.tsx` - Already exports preloadMonacoEditor, just needs index.ts export
|
|
||||||
|
|
||||||
## Testing Plan
|
|
||||||
|
|
||||||
After fixes:
|
|
||||||
1. Run `npm run dev` - should start without errors
|
|
||||||
2. Run `npm run build` - should complete successfully
|
|
||||||
3. Run `npm run test:e2e` - should pass
|
|
||||||
4. Manually test pages that use restored components
|
|
||||||
5. Test JSON UI showcase page to verify JSON system still works
|
|
||||||
|
|
||||||
## Long-Term Vision Questions
|
|
||||||
|
|
||||||
1. Can complex state management work in JSON?
|
|
||||||
2. How do we handle TypeScript types and intellisense for JSON configs?
|
|
||||||
3. What about component composition and reusability?
|
|
||||||
4. Performance implications of JSON parsing and dynamic loading?
|
|
||||||
5. How do non-developers edit JSON configs safely?
|
|
||||||
6. Can we generate JSON from existing TypeScript components?
|
|
||||||
7. What's the migration path for 250+ existing pages?
|
|
||||||
|
|
||||||
@@ -1,165 +0,0 @@
|
|||||||
# JSON Migration Session Complete ✅
|
|
||||||
|
|
||||||
## What Was Accomplished
|
|
||||||
|
|
||||||
### 1. **Completed 9 Component Migrations** ✅
|
|
||||||
- 8 atoms: Accordion, CopyButton, FileUpload, FilterInput, Image, Input, PasswordInput, Popover
|
|
||||||
- 1 molecule: BindingEditor
|
|
||||||
|
|
||||||
**Process:**
|
|
||||||
- All had JSON definitions pre-created
|
|
||||||
- All had custom hooks ready
|
|
||||||
- Needed: Export from json-components.ts + hook registration + delete TSX files
|
|
||||||
- Result: 9 components now available exclusively through `@/lib/json-ui/json-components`
|
|
||||||
|
|
||||||
### 2. **Fixed Critical Build Issues** ✅
|
|
||||||
- Fixed `use-schema-loader.ts` dynamic import (JSON extension)
|
|
||||||
- Fixed `DataSourceGroupSection.tsx` (removed missing dependency)
|
|
||||||
- Recovered 130 files from git history
|
|
||||||
- Restored and cleaned component exports
|
|
||||||
- **Build now passes with 0 errors**
|
|
||||||
|
|
||||||
### 3. **Established Proven Migration Pattern** ✅
|
|
||||||
Pattern applied successfully to 9 components, ready to scale:
|
|
||||||
|
|
||||||
```
|
|
||||||
1. JSON definition (in src/components/json-definitions/)
|
|
||||||
2. TypeScript interface (in src/lib/json-ui/interfaces/)
|
|
||||||
3. Custom hook (if stateful, in src/hooks/)
|
|
||||||
4. Hook registration (in src/lib/json-ui/hooks-registry.ts)
|
|
||||||
5. Export from json-components.ts
|
|
||||||
6. Delete legacy TSX file
|
|
||||||
7. Update component index exports
|
|
||||||
8. Update all imports across codebase
|
|
||||||
9. Run build to verify
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4. **Created Comprehensive Migration Strategy** ✅
|
|
||||||
Strategy document outlines:
|
|
||||||
- **Clear goal:** Only `src/main.tsx` + `src/index.html` remain as non-JSON
|
|
||||||
- **4 priority tiers** with ROI analysis:
|
|
||||||
- Tier 1: 7 app bootstrap components (highest ROI)
|
|
||||||
- Tier 2: 3 organisms (documented in CLAUDE.md)
|
|
||||||
- Tier 3: 150+ core atoms/molecules
|
|
||||||
- Tier 4: 55+ demo/showcase pages
|
|
||||||
- **Execution plan** with parallel opportunities
|
|
||||||
- **Success metrics** showing progress milestones
|
|
||||||
|
|
||||||
## Current State
|
|
||||||
|
|
||||||
```
|
|
||||||
Build Status: ✅ PASSING (0 errors)
|
|
||||||
JSON Components: 22 (up from 12)
|
|
||||||
TSX Files Remaining: 230 (from 420 originally)
|
|
||||||
Deleted This Session: 9 legacy TSX files
|
|
||||||
Files with JSON+Hooks: 15 components
|
|
||||||
Pure JSON Components: 8 components (no hooks)
|
|
||||||
Registry Entries: 342 components
|
|
||||||
|
|
||||||
Recent Commits:
|
|
||||||
- 9aa3e96: Migration strategy document
|
|
||||||
- cf426ef: Migration summary for 9 components
|
|
||||||
- f05f896: Complete JSON migration for 9 components
|
|
||||||
```
|
|
||||||
|
|
||||||
## Architecture Proven
|
|
||||||
|
|
||||||
The architecture can handle:
|
|
||||||
- ✅ Pure stateless components (JSON only)
|
|
||||||
- ✅ Stateful components (JSON + hooks)
|
|
||||||
- ✅ Components with complex rendering logic (via custom hooks)
|
|
||||||
- ✅ Hooks that manage state, side effects, and callbacks
|
|
||||||
- ✅ Components that render other components (via JSON composition)
|
|
||||||
|
|
||||||
**Key insight:** Custom hooks can express ANY TSX logic in JSON form.
|
|
||||||
|
|
||||||
## What Remains
|
|
||||||
|
|
||||||
### Immediate Next Steps (If Continuing)
|
|
||||||
1. Run audit: `npm run audit:json`
|
|
||||||
2. Pick Tier 1 component (e.g., `LoadingScreen` - simplest)
|
|
||||||
3. Apply proven pattern from today's work
|
|
||||||
4. Commit and verify build
|
|
||||||
5. Repeat for next component
|
|
||||||
|
|
||||||
### Expected Timeline
|
|
||||||
- **Phase 1 (Tier 1 - 7 app bootstrap):** 2-3 hours
|
|
||||||
- **Phase 2 (Tier 2 - 3 organisms):** 2-3 hours
|
|
||||||
- **Phase 3 (Tier 3 - 150+ core components):** 8-20 hours (batch work)
|
|
||||||
- **Phase 4 (Tier 4 - demo pages):** 2-5 hours (optional)
|
|
||||||
|
|
||||||
**Total to completion:** 14-31 hours (distributed across multiple sessions)
|
|
||||||
|
|
||||||
### Scale Strategy
|
|
||||||
- Can work on multiple components in parallel (independent commits)
|
|
||||||
- Recommend batching similar components (all buttons, all inputs, etc.)
|
|
||||||
- Tests should run between batches to catch regressions
|
|
||||||
|
|
||||||
## Documentation Created
|
|
||||||
|
|
||||||
1. **MIGRATION_SUMMARY.md** - Today's completed work
|
|
||||||
- What was done, key changes, build status, statistics
|
|
||||||
|
|
||||||
2. **REMAINING_MIGRATION_STRATEGY.md** - Full roadmap
|
|
||||||
- 4 priority tiers with ROI analysis
|
|
||||||
- Proven pattern to follow
|
|
||||||
- Parallel work opportunities
|
|
||||||
- Success metrics
|
|
||||||
- Risk mitigation
|
|
||||||
|
|
||||||
3. **This document** - Session overview
|
|
||||||
|
|
||||||
## Commits This Session
|
|
||||||
|
|
||||||
```
|
|
||||||
9aa3e96 docs: Add comprehensive migration strategy for remaining 220 TSX files
|
|
||||||
cf426ef docs: Add migration summary for 9 completed components
|
|
||||||
f05f896 feat: Complete JSON component migration for 9 components (atoms + BindingEditor)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Key Files to Reference
|
|
||||||
|
|
||||||
- `CLAUDE.md` - Architecture docs (read again with new understanding)
|
|
||||||
- `REMAINING_MIGRATION_STRATEGY.md` - Execution roadmap
|
|
||||||
- `src/lib/json-ui/json-components.ts` - Where components are exported (22 now)
|
|
||||||
- `json-components-registry.json` - Component metadata (342 entries)
|
|
||||||
- `src/hooks/` - Custom hook implementations (50+ hooks available)
|
|
||||||
|
|
||||||
## Verification Commands
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# See current audit status
|
|
||||||
npm run audit:json
|
|
||||||
|
|
||||||
# Build to verify no errors
|
|
||||||
npm run build
|
|
||||||
|
|
||||||
# List all JSON definitions
|
|
||||||
ls src/components/json-definitions/*.json | wc -l
|
|
||||||
|
|
||||||
# Check how many components are exported
|
|
||||||
grep "export const" src/lib/json-ui/json-components.ts | wc -l
|
|
||||||
|
|
||||||
# See registered hooks
|
|
||||||
cat src/lib/json-ui/hooks-registry.ts
|
|
||||||
```
|
|
||||||
|
|
||||||
## Moving Forward
|
|
||||||
|
|
||||||
The migration is **systematic, repeatable, and scalable**. Each component follows the same pattern. The architecture is proven to work for both simple and complex components through the use of custom hooks.
|
|
||||||
|
|
||||||
**Recommended approach for next session:**
|
|
||||||
1. Start with Tier 1 app bootstrap components (highest ROI)
|
|
||||||
2. Use the pattern from today's work
|
|
||||||
3. Commit after each component
|
|
||||||
4. Run build to verify
|
|
||||||
5. Document progress in MIGRATION_SUMMARY.md
|
|
||||||
|
|
||||||
The end goal is clear: **Only `main.tsx` and `index.html` remain, everything else is JSON + hooks.**
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Session completed:** January 21, 2026
|
|
||||||
**Branch:** festive-mestorf
|
|
||||||
**Build status:** ✅ PASSING
|
|
||||||
**Ready for:** Next migration phase or deployment
|
|
||||||
1301
audit-report.json
1301
audit-report.json
File diff suppressed because it is too large
Load Diff
@@ -1,26 +0,0 @@
|
|||||||
|
|
||||||
> spark-template@0.0.0 prebuild
|
|
||||||
> npm run components:generate-types && mkdir -p /tmp/dist || true
|
|
||||||
|
|
||||||
|
|
||||||
> spark-template@0.0.0 components:generate-types
|
|
||||||
> tsx scripts/generate-json-ui-component-types.ts
|
|
||||||
|
|
||||||
✅ Wrote 330 component types to /Users/rmac/Documents/GitHub/low-code-react-app-b/src/types/json-ui-component-types.ts
|
|
||||||
|
|
||||||
> spark-template@0.0.0 build
|
|
||||||
> tsc -b --noCheck && vite build
|
|
||||||
|
|
||||||
vite v7.3.1 building client environment for production...
|
|
||||||
<script src="/runtime-config.js"> in "/index.html" can't be bundled without type="module" attribute
|
|
||||||
transforming...
|
|
||||||
✓ 7152 modules transformed.
|
|
||||||
✗ Build failed in 1.85s
|
|
||||||
error during build:
|
|
||||||
[vite:load-fallback] Could not load /Users/rmac/Documents/GitHub/low-code-react-app-b/src/components/molecules/EditorToolbar (imported by src/components/CodeEditor.tsx): ENOENT: no such file or directory, open '/Users/rmac/Documents/GitHub/low-code-react-app-b/src/components/molecules/EditorToolbar'
|
|
||||||
at async open (node:internal/fs/promises:637:25)
|
|
||||||
at async Object.readFile (node:internal/fs/promises:1269:14)
|
|
||||||
at async Object.handler (file:///Users/rmac/Documents/GitHub/low-code-react-app-b/node_modules/vite/dist/node/chunks/config.js:33169:21)
|
|
||||||
at async PluginDriver.hookFirstAndGetPlugin (file:///Users/rmac/Documents/GitHub/low-code-react-app-b/node_modules/rollup/dist/es/shared/node-entry.js:22333:28)
|
|
||||||
at async file:///Users/rmac/Documents/GitHub/low-code-react-app-b/node_modules/rollup/dist/es/shared/node-entry.js:21333:33
|
|
||||||
at async Queue.work (file:///Users/rmac/Documents/GitHub/low-code-react-app-b/node_modules/rollup/dist/es/shared/node-entry.js:22561:32)
|
|
||||||
@@ -1,141 +0,0 @@
|
|||||||
rm "src\components\atoms\ActionButton.tsx"
|
|
||||||
rm "src\components\atoms\ActionCard.tsx"
|
|
||||||
rm "src\components\atoms\ActionIcon.tsx"
|
|
||||||
rm "src\components\atoms\Alert.tsx"
|
|
||||||
rm "src\components\atoms\AppLogo.tsx"
|
|
||||||
rm "src\components\atoms\Avatar.tsx"
|
|
||||||
rm "src\components\atoms\AvatarGroup.tsx"
|
|
||||||
rm "src\components\atoms\Badge.tsx"
|
|
||||||
rm "src\components\atoms\BindingIndicator.tsx"
|
|
||||||
rm "src\components\atoms\Breadcrumb.tsx"
|
|
||||||
rm "src\components\atoms\Button.tsx"
|
|
||||||
rm "src\components\atoms\ButtonGroup.tsx"
|
|
||||||
rm "src\components\atoms\Calendar.tsx"
|
|
||||||
rm "src\components\atoms\Card.tsx"
|
|
||||||
rm "src\components\atoms\Checkbox.tsx"
|
|
||||||
rm "src\components\atoms\Chip.tsx"
|
|
||||||
rm "src\components\atoms\CircularProgress.tsx"
|
|
||||||
rm "src\components\atoms\Code.tsx"
|
|
||||||
rm "src\components\atoms\ColorSwatch.tsx"
|
|
||||||
rm "src\components\atoms\CommandPalette.tsx"
|
|
||||||
rm "src\components\atoms\CompletionCard.tsx"
|
|
||||||
rm "src\components\atoms\ComponentPaletteItem.tsx"
|
|
||||||
rm "src\components\atoms\ComponentTreeNode.tsx"
|
|
||||||
rm "src\components\atoms\ConfirmButton.tsx"
|
|
||||||
rm "src\components\atoms\Container.tsx"
|
|
||||||
rm "src\components\atoms\ContextMenu.tsx"
|
|
||||||
rm "src\components\atoms\CountBadge.tsx"
|
|
||||||
rm "src\components\atoms\DataList.tsx"
|
|
||||||
rm "src\components\atoms\DataSourceBadge.tsx"
|
|
||||||
rm "src\components\atoms\DataTable.tsx"
|
|
||||||
rm "src\components\atoms\DatePicker.tsx"
|
|
||||||
rm "src\components\atoms\DetailRow.tsx"
|
|
||||||
rm "src\components\atoms\Divider.tsx"
|
|
||||||
rm "src\components\atoms\Dot.tsx"
|
|
||||||
rm "src\components\atoms\Drawer.tsx"
|
|
||||||
rm "src\components\atoms\EmptyMessage.tsx"
|
|
||||||
rm "src\components\atoms\EmptyState.tsx"
|
|
||||||
rm "src\components\atoms\EmptyStateIcon.tsx"
|
|
||||||
rm "src\components\atoms\ErrorBadge.tsx"
|
|
||||||
rm "src\components\atoms\FileIcon.tsx"
|
|
||||||
rm "src\components\atoms\Flex.tsx"
|
|
||||||
rm "src\components\atoms\Form.tsx"
|
|
||||||
rm "src\components\atoms\GlowCard.tsx"
|
|
||||||
rm "src\components\atoms\Grid.tsx"
|
|
||||||
rm "src\components\atoms\Heading.tsx"
|
|
||||||
rm "src\components\atoms\HelperText.tsx"
|
|
||||||
rm "src\components\atoms\HoverCard.tsx"
|
|
||||||
rm "src\components\atoms\IconButton.tsx"
|
|
||||||
rm "src\components\atoms\IconText.tsx"
|
|
||||||
rm "src\components\atoms\IconWrapper.tsx"
|
|
||||||
rm "src\components\atoms\InfoBox.tsx"
|
|
||||||
rm "src\components\atoms\InfoPanel.tsx"
|
|
||||||
rm "src\components\atoms\Input.tsx"
|
|
||||||
rm "src\components\atoms\Kbd.tsx"
|
|
||||||
rm "src\components\atoms\KeyValue.tsx"
|
|
||||||
rm "src\components\atoms\Label.tsx"
|
|
||||||
rm "src\components\atoms\Link.tsx"
|
|
||||||
rm "src\components\atoms\List.tsx"
|
|
||||||
rm "src\components\atoms\ListItem.tsx"
|
|
||||||
rm "src\components\atoms\LiveIndicator.tsx"
|
|
||||||
rm "src\components\atoms\LoadingSpinner.tsx"
|
|
||||||
rm "src\components\atoms\LoadingState.tsx"
|
|
||||||
rm "src\components\atoms\MetricCard.tsx"
|
|
||||||
rm "src\components\atoms\MetricDisplay.tsx"
|
|
||||||
rm "src\components\atoms\Modal.tsx"
|
|
||||||
rm "src\components\atoms\Notification.tsx"
|
|
||||||
rm "src\components\atoms\NumberInput.tsx"
|
|
||||||
rm "src\components\atoms\PageHeader.tsx"
|
|
||||||
rm "src\components\atoms\PanelHeader.tsx"
|
|
||||||
rm "src\components\atoms\ProgressBar.tsx"
|
|
||||||
rm "src\components\atoms\PropertyEditorField.tsx"
|
|
||||||
rm "src\components\atoms\Pulse.tsx"
|
|
||||||
rm "src\components\atoms\QuickActionButton.tsx"
|
|
||||||
rm "src\components\atoms\Radio.tsx"
|
|
||||||
rm "src\components\atoms\RangeSlider.tsx"
|
|
||||||
rm "src\components\atoms\Rating.tsx"
|
|
||||||
rm "src\components\atoms\ResponsiveGrid.tsx"
|
|
||||||
rm "src\components\atoms\ScrollArea.tsx"
|
|
||||||
rm "src\components\atoms\SearchInput.tsx"
|
|
||||||
rm "src\components\atoms\Section.tsx"
|
|
||||||
rm "src\components\atoms\SeedDataStatus.tsx"
|
|
||||||
rm "src\components\atoms\Select.tsx"
|
|
||||||
rm "src\components\atoms\Separator.tsx"
|
|
||||||
rm "src\components\atoms\Skeleton.tsx"
|
|
||||||
rm "src\components\atoms\Slider.tsx"
|
|
||||||
rm "src\components\atoms\Spacer.tsx"
|
|
||||||
rm "src\components\atoms\Sparkle.tsx"
|
|
||||||
rm "src\components\atoms\Spinner.tsx"
|
|
||||||
rm "src\components\atoms\Stack.tsx"
|
|
||||||
rm "src\components\atoms\StatCard.tsx"
|
|
||||||
rm "src\components\atoms\StatusBadge.tsx"
|
|
||||||
rm "src\components\atoms\StatusIcon.tsx"
|
|
||||||
rm "src\components\atoms\StepIndicator.tsx"
|
|
||||||
rm "src\components\atoms\Stepper.tsx"
|
|
||||||
rm "src\components\atoms\Switch.tsx"
|
|
||||||
rm "src\components\atoms\TabIcon.tsx"
|
|
||||||
rm "src\components\atoms\Table.tsx"
|
|
||||||
rm "src\components\atoms\Tabs.tsx"
|
|
||||||
rm "src\components\atoms\Tag.tsx"
|
|
||||||
rm "src\components\atoms\Text.tsx"
|
|
||||||
rm "src\components\atoms\TextArea.tsx"
|
|
||||||
rm "src\components\atoms\TextGradient.tsx"
|
|
||||||
rm "src\components\atoms\TextHighlight.tsx"
|
|
||||||
rm "src\components\atoms\Timeline.tsx"
|
|
||||||
rm "src\components\atoms\Timestamp.tsx"
|
|
||||||
rm "src\components\atoms\TipsCard.tsx"
|
|
||||||
rm "src\components\atoms\Toggle.tsx"
|
|
||||||
rm "src\components\atoms\Tooltip.tsx"
|
|
||||||
rm "src\components\atoms\TreeIcon.tsx"
|
|
||||||
rm "src\components\molecules\AppBranding.tsx"
|
|
||||||
rm "src\components\molecules\Breadcrumb.tsx"
|
|
||||||
rm "src\components\molecules\CanvasRenderer.tsx"
|
|
||||||
rm "src\components\molecules\CodeExplanationDialog.tsx"
|
|
||||||
rm "src\components\molecules\ComponentPalette.tsx"
|
|
||||||
rm "src\components\molecules\DataSourceCard.tsx"
|
|
||||||
rm "src\components\molecules\EditorActions.tsx"
|
|
||||||
rm "src\components\molecules\EditorToolbar.tsx"
|
|
||||||
rm "src\components\molecules\EmptyEditorState.tsx"
|
|
||||||
rm "src\components\molecules\FileTabs.tsx"
|
|
||||||
rm "src\components\molecules\LazyBarChart.tsx"
|
|
||||||
rm "src\components\molecules\LazyInlineMonacoEditor.tsx"
|
|
||||||
rm "src\components\molecules\LazyLineChart.tsx"
|
|
||||||
rm "src\components\molecules\LazyMonacoEditor.tsx"
|
|
||||||
rm "src\components\molecules\MonacoEditorPanel.tsx"
|
|
||||||
rm "src\components\molecules\NavigationGroupHeader.tsx"
|
|
||||||
rm "src\components\molecules\PropertyEditor.tsx"
|
|
||||||
rm "src\components\molecules\SearchBar.tsx"
|
|
||||||
rm "src\components\molecules\SearchInput.tsx"
|
|
||||||
rm "src\components\molecules\ToolbarButton.tsx"
|
|
||||||
rm "src\components\molecules\TreeFormDialog.tsx"
|
|
||||||
rm "src\components\organisms\AppHeader.tsx"
|
|
||||||
rm "src\components\organisms\EmptyCanvasState.tsx"
|
|
||||||
rm "src\components\organisms\PageHeader.tsx"
|
|
||||||
rm "src\components\organisms\SchemaCodeViewer.tsx"
|
|
||||||
rm "src\components\organisms\SchemaEditorCanvas.tsx"
|
|
||||||
rm "src\components\organisms\SchemaEditorLayout.tsx"
|
|
||||||
rm "src\components\organisms\SchemaEditorPropertiesPanel.tsx"
|
|
||||||
rm "src\components\organisms\SchemaEditorSidebar.tsx"
|
|
||||||
rm "src\components\organisms\SchemaEditorStatusBar.tsx"
|
|
||||||
rm "src\components\organisms\SchemaEditorToolbar.tsx"
|
|
||||||
rm "src\components\organisms\ToolbarActions.tsx"
|
|
||||||
@@ -1,107 +0,0 @@
|
|||||||
# JSON Constants Migration Guide
|
|
||||||
|
|
||||||
## Overview
|
|
||||||
|
|
||||||
This document tracks the extraction of hardcoded constants from JSON component definitions into the centralized constants folder.
|
|
||||||
|
|
||||||
## Status
|
|
||||||
|
|
||||||
### ✅ Constants Folder Created
|
|
||||||
Location: `src/lib/json-ui/constants/`
|
|
||||||
|
|
||||||
Files:
|
|
||||||
- `sizes.ts` - Size-related constants
|
|
||||||
- `placements.ts` - Positioning constants
|
|
||||||
- `styles.ts` - CSS class constants
|
|
||||||
- `object-fit.ts` - Image object-fit constants
|
|
||||||
- `index.ts` - Centralized exports
|
|
||||||
|
|
||||||
### 📋 Constants Found in JSON Files
|
|
||||||
|
|
||||||
#### CopyButton (`copy-button.json`)
|
|
||||||
```javascript
|
|
||||||
// Line 11: sizeStyles
|
|
||||||
const sizeStyles = { sm: 'p-1', md: 'p-2', lg: 'p-3' }
|
|
||||||
// → BUTTON_SIZES
|
|
||||||
|
|
||||||
// Lines 25, 39: iconSize (duplicated)
|
|
||||||
const iconSize = { sm: 12, md: 16, lg: 20 }
|
|
||||||
// → ICON_SIZES
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Popover (`popover.json`)
|
|
||||||
```javascript
|
|
||||||
// Line 39: placementStyles
|
|
||||||
const placementStyles = {
|
|
||||||
top: 'bottom-full mb-2 left-1/2 -translate-x-1/2',
|
|
||||||
bottom: 'top-full mt-2 left-1/2 -translate-x-1/2',
|
|
||||||
left: 'right-full mr-2 top-1/2 -translate-y-1/2',
|
|
||||||
right: 'left-full ml-2 top-1/2 -translate-y-1/2'
|
|
||||||
}
|
|
||||||
// → POPOVER_PLACEMENTS
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Image (`image.json`)
|
|
||||||
```javascript
|
|
||||||
// Line 51: Dynamic object-fit (uses template literal)
|
|
||||||
return `${base} ${opacity} object-${fit}`
|
|
||||||
// Could use OBJECT_FIT_CLASSES but requires transform refactor
|
|
||||||
```
|
|
||||||
|
|
||||||
## Recommendations
|
|
||||||
|
|
||||||
### Option 1: Keep Inline (Current Approach)
|
|
||||||
**Pros:**
|
|
||||||
- No changes to component-renderer needed
|
|
||||||
- Self-contained JSON definitions
|
|
||||||
- Easy to understand transforms
|
|
||||||
|
|
||||||
**Cons:**
|
|
||||||
- Duplication of constants
|
|
||||||
- Harder to maintain consistency
|
|
||||||
- Magic strings scattered across files
|
|
||||||
|
|
||||||
### Option 2: Import Constants in Hooks
|
|
||||||
**Pros:**
|
|
||||||
- Hooks can use TypeScript constants
|
|
||||||
- No changes to JSON structure needed
|
|
||||||
- Immediate benefit for custom hooks
|
|
||||||
|
|
||||||
**Cons:**
|
|
||||||
- Only helps with hook-based logic
|
|
||||||
- Still have duplication in JSON transforms
|
|
||||||
|
|
||||||
### Option 3: Add Constants to Transform Context (Future)
|
|
||||||
**Pros:**
|
|
||||||
- Eliminates duplication entirely
|
|
||||||
- Type-safe constants usage
|
|
||||||
- Easier to update global styles
|
|
||||||
|
|
||||||
**Cons:**
|
|
||||||
- Requires component-renderer changes
|
|
||||||
- More complex transform evaluation
|
|
||||||
- Migration effort for existing JSON files
|
|
||||||
|
|
||||||
## Recommended Next Steps
|
|
||||||
|
|
||||||
1. **Short term:** Use constants in custom hooks (Option 2)
|
|
||||||
- Hooks can import from `@/lib/json-ui/constants`
|
|
||||||
- Reduce duplication in hook code
|
|
||||||
|
|
||||||
2. **Medium term:** Document best practices
|
|
||||||
- Add examples of using constants
|
|
||||||
- Create migration guide for new components
|
|
||||||
|
|
||||||
3. **Long term:** Enhanced transform context (Option 3)
|
|
||||||
- Update component-renderer to expose constants
|
|
||||||
- Migrate existing JSON files to use constants
|
|
||||||
- Remove inline const statements
|
|
||||||
|
|
||||||
## Files to Potentially Update
|
|
||||||
|
|
||||||
When migrating to Option 3:
|
|
||||||
- `copy-button.json` - sizeStyles, iconSize
|
|
||||||
- `popover.json` - placementStyles
|
|
||||||
- `menu.json` - May have similar patterns
|
|
||||||
- `file-upload.json` - May have size constants
|
|
||||||
- Any future components using similar patterns
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -8,10 +8,7 @@ test.describe('CodeForge - Core Functionality', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
test('should load the application successfully', async ({ page }) => {
|
test('should load the application successfully', async ({ page }) => {
|
||||||
// Check root has children (content rendered)
|
await expect(page.locator('body')).toBeVisible()
|
||||||
await page.waitForSelector('#root > *', { timeout: 10000 })
|
|
||||||
const root = page.locator('#root')
|
|
||||||
await expect(root).toHaveCount(1)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
test('should display main navigation', async ({ page }) => {
|
test('should display main navigation', async ({ page }) => {
|
||||||
@@ -53,8 +50,8 @@ test.describe('CodeForge - Responsive Design', () => {
|
|||||||
await page.setViewportSize({ width: 375, height: 667 })
|
await page.setViewportSize({ width: 375, height: 667 })
|
||||||
await page.goto('/', { waitUntil: 'domcontentloaded', timeout: 10000 })
|
await page.goto('/', { waitUntil: 'domcontentloaded', timeout: 10000 })
|
||||||
await page.waitForLoadState('networkidle', { timeout: 5000 })
|
await page.waitForLoadState('networkidle', { timeout: 5000 })
|
||||||
|
|
||||||
await page.waitForSelector('#root > *', { timeout: 10000 })
|
await expect(page.locator('body')).toBeVisible()
|
||||||
})
|
})
|
||||||
|
|
||||||
test('should work on tablet viewport', async ({ page }) => {
|
test('should work on tablet viewport', async ({ page }) => {
|
||||||
@@ -62,7 +59,7 @@ test.describe('CodeForge - Responsive Design', () => {
|
|||||||
await page.setViewportSize({ width: 768, height: 1024 })
|
await page.setViewportSize({ width: 768, height: 1024 })
|
||||||
await page.goto('/', { waitUntil: 'domcontentloaded', timeout: 10000 })
|
await page.goto('/', { waitUntil: 'domcontentloaded', timeout: 10000 })
|
||||||
await page.waitForLoadState('networkidle', { timeout: 5000 })
|
await page.waitForLoadState('networkidle', { timeout: 5000 })
|
||||||
|
|
||||||
await page.waitForSelector('#root > *', { timeout: 10000 })
|
await expect(page.locator('body')).toBeVisible()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,41 +0,0 @@
|
|||||||
import { test } from '@playwright/test'
|
|
||||||
|
|
||||||
test('debug page load', async ({ page }) => {
|
|
||||||
const errors: string[] = []
|
|
||||||
const pageErrors: Error[] = []
|
|
||||||
|
|
||||||
page.on('console', (msg) => {
|
|
||||||
if (msg.type() === 'error') {
|
|
||||||
errors.push(msg.text())
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
page.on('pageerror', (error) => {
|
|
||||||
pageErrors.push(error)
|
|
||||||
})
|
|
||||||
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle', timeout: 15000 })
|
|
||||||
|
|
||||||
// Wait a bit
|
|
||||||
await page.waitForTimeout(2000)
|
|
||||||
|
|
||||||
// Get page content
|
|
||||||
const rootHTML = await page.locator('#root').innerHTML().catch(() => 'ERROR GETTING ROOT')
|
|
||||||
|
|
||||||
console.log('=== PAGE ERRORS ===')
|
|
||||||
pageErrors.forEach(err => console.log(err.message))
|
|
||||||
|
|
||||||
console.log('\n=== CONSOLE ERRORS ===')
|
|
||||||
errors.forEach(err => console.log(err))
|
|
||||||
|
|
||||||
console.log('\n=== ROOT CONTENT ===')
|
|
||||||
console.log(rootHTML.substring(0, 500))
|
|
||||||
|
|
||||||
console.log('\n=== ROOT VISIBLE ===')
|
|
||||||
const rootVisible = await page.locator('#root').isVisible().catch(() => false)
|
|
||||||
console.log('Root visible:', rootVisible)
|
|
||||||
|
|
||||||
console.log('\n=== ROOT HAS CHILDREN ===')
|
|
||||||
const childCount = await page.locator('#root > *').count()
|
|
||||||
console.log('Child count:', childCount)
|
|
||||||
})
|
|
||||||
@@ -4,12 +4,8 @@ test.describe('CodeForge - Smoke Tests', () => {
|
|||||||
test('app loads successfully', async ({ page }) => {
|
test('app loads successfully', async ({ page }) => {
|
||||||
test.setTimeout(20000)
|
test.setTimeout(20000)
|
||||||
await page.goto('/', { waitUntil: 'networkidle', timeout: 15000 })
|
await page.goto('/', { waitUntil: 'networkidle', timeout: 15000 })
|
||||||
|
|
||||||
// Check that the app has rendered content (more reliable than checking visibility)
|
await expect(page.locator('body')).toBeVisible({ timeout: 5000 })
|
||||||
const root = page.locator('#root')
|
|
||||||
await expect(root).toHaveCount(1, { timeout: 5000 })
|
|
||||||
// Wait for any content to be rendered
|
|
||||||
await page.waitForSelector('#root > *', { timeout: 10000 })
|
|
||||||
})
|
})
|
||||||
|
|
||||||
test('can navigate to dashboard tab', async ({ page }) => {
|
test('can navigate to dashboard tab', async ({ page }) => {
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
35
package-lock.json
generated
35
package-lock.json
generated
@@ -89,7 +89,6 @@
|
|||||||
"eslint-plugin-react-refresh": "^0.4.19",
|
"eslint-plugin-react-refresh": "^0.4.19",
|
||||||
"globals": "^17.0.0",
|
"globals": "^17.0.0",
|
||||||
"tailwindcss": "^4.1.11",
|
"tailwindcss": "^4.1.11",
|
||||||
"terser": "^5.46.0",
|
|
||||||
"tsx": "^4.21.0",
|
"tsx": "^4.21.0",
|
||||||
"typescript": "~5.7.2",
|
"typescript": "~5.7.2",
|
||||||
"typescript-eslint": "^8.38.0",
|
"typescript-eslint": "^8.38.0",
|
||||||
@@ -825,10 +824,9 @@
|
|||||||
},
|
},
|
||||||
"node_modules/@jridgewell/source-map": {
|
"node_modules/@jridgewell/source-map": {
|
||||||
"version": "0.3.11",
|
"version": "0.3.11",
|
||||||
"resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.11.tgz",
|
|
||||||
"integrity": "sha512-ZMp1V8ZFcPG5dIWnQLr3NSI1MiCU7UETdS/A0G8V/XWHvJv3ZsFqutJn1Y5RPmAPX6F3BiE397OqveU/9NCuIA==",
|
|
||||||
"devOptional": true,
|
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"peer": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@jridgewell/gen-mapping": "^0.3.5",
|
"@jridgewell/gen-mapping": "^0.3.5",
|
||||||
"@jridgewell/trace-mapping": "^0.3.25"
|
"@jridgewell/trace-mapping": "^0.3.25"
|
||||||
@@ -4770,10 +4768,9 @@
|
|||||||
},
|
},
|
||||||
"node_modules/buffer-from": {
|
"node_modules/buffer-from": {
|
||||||
"version": "1.1.2",
|
"version": "1.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
|
"license": "MIT",
|
||||||
"integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
|
"optional": true,
|
||||||
"devOptional": true,
|
"peer": true
|
||||||
"license": "MIT"
|
|
||||||
},
|
},
|
||||||
"node_modules/callsites": {
|
"node_modules/callsites": {
|
||||||
"version": "3.1.0",
|
"version": "3.1.0",
|
||||||
@@ -6992,10 +6989,9 @@
|
|||||||
},
|
},
|
||||||
"node_modules/source-map": {
|
"node_modules/source-map": {
|
||||||
"version": "0.6.1",
|
"version": "0.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
|
|
||||||
"integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
|
|
||||||
"devOptional": true,
|
|
||||||
"license": "BSD-3-Clause",
|
"license": "BSD-3-Clause",
|
||||||
|
"optional": true,
|
||||||
|
"peer": true,
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
@@ -7009,10 +7005,9 @@
|
|||||||
},
|
},
|
||||||
"node_modules/source-map-support": {
|
"node_modules/source-map-support": {
|
||||||
"version": "0.5.21",
|
"version": "0.5.21",
|
||||||
"resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz",
|
|
||||||
"integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==",
|
|
||||||
"devOptional": true,
|
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"peer": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"buffer-from": "^1.0.0",
|
"buffer-from": "^1.0.0",
|
||||||
"source-map": "^0.6.0"
|
"source-map": "^0.6.0"
|
||||||
@@ -7080,10 +7075,9 @@
|
|||||||
},
|
},
|
||||||
"node_modules/terser": {
|
"node_modules/terser": {
|
||||||
"version": "5.46.0",
|
"version": "5.46.0",
|
||||||
"resolved": "https://registry.npmjs.org/terser/-/terser-5.46.0.tgz",
|
|
||||||
"integrity": "sha512-jTwoImyr/QbOWFFso3YoU3ik0jBBDJ6JTOQiy/J2YxVJdZCc+5u7skhNwiOR3FQIygFqVUPHl7qbbxtjW2K3Qg==",
|
|
||||||
"devOptional": true,
|
|
||||||
"license": "BSD-2-Clause",
|
"license": "BSD-2-Clause",
|
||||||
|
"optional": true,
|
||||||
|
"peer": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@jridgewell/source-map": "^0.3.3",
|
"@jridgewell/source-map": "^0.3.3",
|
||||||
"acorn": "^8.15.0",
|
"acorn": "^8.15.0",
|
||||||
@@ -7099,10 +7093,9 @@
|
|||||||
},
|
},
|
||||||
"node_modules/terser/node_modules/commander": {
|
"node_modules/terser/node_modules/commander": {
|
||||||
"version": "2.20.3",
|
"version": "2.20.3",
|
||||||
"resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz",
|
"license": "MIT",
|
||||||
"integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==",
|
"optional": true,
|
||||||
"devOptional": true,
|
"peer": true
|
||||||
"license": "MIT"
|
|
||||||
},
|
},
|
||||||
"node_modules/three": {
|
"node_modules/three": {
|
||||||
"version": "0.175.0",
|
"version": "0.175.0",
|
||||||
|
|||||||
@@ -6,8 +6,7 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"kill": "fuser -k 5000/tcp",
|
"kill": "fuser -k 5000/tcp",
|
||||||
"predev": "npm run components:generate-types",
|
"prebuild": "mkdir -p /tmp/dist || true",
|
||||||
"prebuild": "npm run components:generate-types && mkdir -p /tmp/dist || true",
|
|
||||||
"build": "tsc -b --noCheck && vite build",
|
"build": "tsc -b --noCheck && vite build",
|
||||||
"lint": "eslint . --fix && npm run lint:schemas",
|
"lint": "eslint . --fix && npm run lint:schemas",
|
||||||
"lint:check": "eslint . && npm run lint:schemas",
|
"lint:check": "eslint . && npm run lint:schemas",
|
||||||
@@ -25,10 +24,8 @@
|
|||||||
"pages:generate": "node scripts/generate-page.js",
|
"pages:generate": "node scripts/generate-page.js",
|
||||||
"schemas:validate": "tsx scripts/validate-json-schemas.ts",
|
"schemas:validate": "tsx scripts/validate-json-schemas.ts",
|
||||||
"components:list": "node scripts/list-json-components.cjs",
|
"components:list": "node scripts/list-json-components.cjs",
|
||||||
"components:generate-types": "tsx scripts/generate-json-ui-component-types.ts",
|
|
||||||
"components:scan": "node scripts/scan-and-update-registry.cjs",
|
"components:scan": "node scripts/scan-and-update-registry.cjs",
|
||||||
"components:validate": "node scripts/validate-supported-components.cjs && tsx scripts/validate-json-registry.ts",
|
"components:validate": "node scripts/validate-supported-components.cjs"
|
||||||
"audit:json": "tsx scripts/audit-json-components.ts"
|
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@heroicons/react": "^2.2.0",
|
"@heroicons/react": "^2.2.0",
|
||||||
@@ -112,7 +109,6 @@
|
|||||||
"eslint-plugin-react-refresh": "^0.4.19",
|
"eslint-plugin-react-refresh": "^0.4.19",
|
||||||
"globals": "^17.0.0",
|
"globals": "^17.0.0",
|
||||||
"tailwindcss": "^4.1.11",
|
"tailwindcss": "^4.1.11",
|
||||||
"terser": "^5.46.0",
|
|
||||||
"tsx": "^4.21.0",
|
"tsx": "^4.21.0",
|
||||||
"typescript": "~5.7.2",
|
"typescript": "~5.7.2",
|
||||||
"typescript-eslint": "^8.38.0",
|
"typescript-eslint": "^8.38.0",
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ export default defineConfig({
|
|||||||
webServer: {
|
webServer: {
|
||||||
command: 'npm run dev',
|
command: 'npm run dev',
|
||||||
url: 'http://localhost:5000',
|
url: 'http://localhost:5000',
|
||||||
reuseExistingServer: true,
|
reuseExistingServer: !process.env.CI,
|
||||||
timeout: 120000,
|
timeout: 120000,
|
||||||
stdout: 'pipe',
|
stdout: 'pipe',
|
||||||
stderr: 'pipe',
|
stderr: 'pipe',
|
||||||
|
|||||||
@@ -39,13 +39,9 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"id": "trends",
|
"id": "trends",
|
||||||
"type": "static",
|
"type": "computed",
|
||||||
"defaultValue": {
|
"compute": "(data) => ({ filesGrowth: 12, modelsGrowth: -3, componentsGrowth: 8, testsGrowth: 15 })",
|
||||||
"filesGrowth": 12,
|
"dependencies": ["metrics"]
|
||||||
"modelsGrowth": -3,
|
|
||||||
"componentsGrowth": 8,
|
|
||||||
"testsGrowth": 15
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"components": [
|
"components": [
|
||||||
|
|||||||
@@ -25,12 +25,9 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"id": "filteredFiles",
|
"id": "filteredFiles",
|
||||||
"type": "static",
|
"type": "computed",
|
||||||
"expression": "data.files",
|
"compute": "(data) => {\n if (!data.searchQuery) return data.files;\n return data.files.filter(f => f.name.toLowerCase().includes(data.searchQuery.toLowerCase()));\n}",
|
||||||
"dependencies": [
|
"dependencies": ["files", "searchQuery"]
|
||||||
"files",
|
|
||||||
"searchQuery"
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"components": [
|
"components": [
|
||||||
|
|||||||
@@ -22,15 +22,6 @@
|
|||||||
"type": "string"
|
"type": "string"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sourceRoots": {
|
|
||||||
"type": "object",
|
|
||||||
"additionalProperties": {
|
|
||||||
"type": "array",
|
|
||||||
"items": {
|
|
||||||
"type": "string"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"components": {
|
"components": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
@@ -82,19 +73,6 @@
|
|||||||
"wrapperFor": {
|
"wrapperFor": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
"load": {
|
|
||||||
"type": "object",
|
|
||||||
"properties": {
|
|
||||||
"path": {
|
|
||||||
"type": "string"
|
|
||||||
},
|
|
||||||
"export": {
|
|
||||||
"type": "string"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"required": ["export"],
|
|
||||||
"additionalProperties": false
|
|
||||||
},
|
|
||||||
"deprecated": {
|
"deprecated": {
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"properties": {
|
"properties": {
|
||||||
|
|||||||
@@ -1,190 +0,0 @@
|
|||||||
#!/usr/bin/env tsx
|
|
||||||
/**
|
|
||||||
* Analyze duplicate TSX files before deletion
|
|
||||||
* Check JSON contents to ensure they're complete
|
|
||||||
*/
|
|
||||||
|
|
||||||
import fs from 'fs'
|
|
||||||
import path from 'path'
|
|
||||||
import { globSync } from 'fs'
|
|
||||||
|
|
||||||
const ROOT_DIR = path.resolve(process.cwd())
|
|
||||||
const CONFIG_PAGES_DIR = path.join(ROOT_DIR, 'src/config/pages')
|
|
||||||
const COMPONENTS_DIR = path.join(ROOT_DIR, 'src/components')
|
|
||||||
const JSON_DEFS_DIR = path.join(ROOT_DIR, 'src/components/json-definitions')
|
|
||||||
|
|
||||||
function toKebabCase(str: string): string {
|
|
||||||
return str.replace(/([A-Z])/g, '-$1').toLowerCase().replace(/^-/, '')
|
|
||||||
}
|
|
||||||
|
|
||||||
interface AnalysisResult {
|
|
||||||
tsx: string
|
|
||||||
json: string
|
|
||||||
tsxSize: number
|
|
||||||
jsonSize: number
|
|
||||||
tsxHasHooks: boolean
|
|
||||||
tsxHasState: boolean
|
|
||||||
tsxHasEffects: boolean
|
|
||||||
jsonHasBindings: boolean
|
|
||||||
jsonHasChildren: boolean
|
|
||||||
recommendation: 'safe-to-delete' | 'needs-review' | 'keep-tsx'
|
|
||||||
reason: string
|
|
||||||
}
|
|
||||||
|
|
||||||
async function analyzeTsxFile(filePath: string): Promise<{
|
|
||||||
hasHooks: boolean
|
|
||||||
hasState: boolean
|
|
||||||
hasEffects: boolean
|
|
||||||
}> {
|
|
||||||
const content = fs.readFileSync(filePath, 'utf-8')
|
|
||||||
|
|
||||||
return {
|
|
||||||
hasHooks: /use[A-Z]/.test(content),
|
|
||||||
hasState: /useState|useReducer/.test(content),
|
|
||||||
hasEffects: /useEffect|useLayoutEffect/.test(content)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function analyzeJsonFile(filePath: string): Promise<{
|
|
||||||
hasBindings: boolean
|
|
||||||
hasChildren: boolean
|
|
||||||
size: number
|
|
||||||
}> {
|
|
||||||
const content = fs.readFileSync(filePath, 'utf-8')
|
|
||||||
const json = JSON.parse(content)
|
|
||||||
|
|
||||||
return {
|
|
||||||
hasBindings: !!json.bindings || hasNestedBindings(json),
|
|
||||||
hasChildren: !!json.children,
|
|
||||||
size: content.length
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function hasNestedBindings(obj: any): boolean {
|
|
||||||
if (!obj || typeof obj !== 'object') return false
|
|
||||||
if (obj.bindings) return true
|
|
||||||
|
|
||||||
for (const key in obj) {
|
|
||||||
if (hasNestedBindings(obj[key])) return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
async function analyzeDuplicates() {
|
|
||||||
console.log('🔍 Analyzing duplicate TSX files...\n')
|
|
||||||
|
|
||||||
const results: AnalysisResult[] = []
|
|
||||||
|
|
||||||
// Find all TSX files in atoms, molecules, organisms
|
|
||||||
const categories = ['atoms', 'molecules', 'organisms']
|
|
||||||
|
|
||||||
for (const category of categories) {
|
|
||||||
const tsxFiles = globSync(path.join(COMPONENTS_DIR, category, '*.tsx'))
|
|
||||||
|
|
||||||
for (const tsxFile of tsxFiles) {
|
|
||||||
const basename = path.basename(tsxFile, '.tsx')
|
|
||||||
const kebab = toKebabCase(basename)
|
|
||||||
|
|
||||||
// Check for JSON equivalent in config/pages
|
|
||||||
const jsonPath = path.join(CONFIG_PAGES_DIR, category, `${kebab}.json`)
|
|
||||||
|
|
||||||
if (!fs.existsSync(jsonPath)) continue
|
|
||||||
|
|
||||||
// Check for JSON definition
|
|
||||||
const jsonDefPath = path.join(JSON_DEFS_DIR, `${kebab}.json`)
|
|
||||||
|
|
||||||
// Analyze both files
|
|
||||||
const tsxAnalysis = await analyzeTsxFile(tsxFile)
|
|
||||||
const tsxSize = fs.statSync(tsxFile).size
|
|
||||||
|
|
||||||
let jsonAnalysis = { hasBindings: false, hasChildren: false, size: 0 }
|
|
||||||
let actualJsonPath = jsonPath
|
|
||||||
|
|
||||||
if (fs.existsSync(jsonDefPath)) {
|
|
||||||
jsonAnalysis = await analyzeJsonFile(jsonDefPath)
|
|
||||||
actualJsonPath = jsonDefPath
|
|
||||||
} else if (fs.existsSync(jsonPath)) {
|
|
||||||
jsonAnalysis = await analyzeJsonFile(jsonPath)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Determine recommendation
|
|
||||||
let recommendation: AnalysisResult['recommendation'] = 'safe-to-delete'
|
|
||||||
let reason = 'JSON definition exists'
|
|
||||||
|
|
||||||
if (tsxAnalysis.hasState || tsxAnalysis.hasEffects) {
|
|
||||||
if (!jsonAnalysis.hasBindings && jsonAnalysis.size < 500) {
|
|
||||||
recommendation = 'needs-review'
|
|
||||||
reason = 'TSX has state/effects but JSON seems incomplete'
|
|
||||||
} else {
|
|
||||||
recommendation = 'safe-to-delete'
|
|
||||||
reason = 'TSX has hooks but JSON should handle via createJsonComponentWithHooks'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tsxSize > 5000 && jsonAnalysis.size < 1000) {
|
|
||||||
recommendation = 'needs-review'
|
|
||||||
reason = 'TSX is large but JSON is small - might be missing content'
|
|
||||||
}
|
|
||||||
|
|
||||||
results.push({
|
|
||||||
tsx: path.relative(ROOT_DIR, tsxFile),
|
|
||||||
json: path.relative(ROOT_DIR, actualJsonPath),
|
|
||||||
tsxSize,
|
|
||||||
jsonSize: jsonAnalysis.size,
|
|
||||||
tsxHasHooks: tsxAnalysis.hasHooks,
|
|
||||||
tsxHasState: tsxAnalysis.hasState,
|
|
||||||
tsxHasEffects: tsxAnalysis.hasEffects,
|
|
||||||
jsonHasBindings: jsonAnalysis.hasBindings,
|
|
||||||
jsonHasChildren: jsonAnalysis.hasChildren,
|
|
||||||
recommendation,
|
|
||||||
reason
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print results
|
|
||||||
console.log(`📊 Found ${results.length} duplicate components\n`)
|
|
||||||
|
|
||||||
const safeToDelete = results.filter(r => r.recommendation === 'safe-to-delete')
|
|
||||||
const needsReview = results.filter(r => r.recommendation === 'needs-review')
|
|
||||||
const keepTsx = results.filter(r => r.recommendation === 'keep-tsx')
|
|
||||||
|
|
||||||
console.log(`✅ Safe to delete: ${safeToDelete.length}`)
|
|
||||||
console.log(`⚠️ Needs review: ${needsReview.length}`)
|
|
||||||
console.log(`🔴 Keep TSX: ${keepTsx.length}\n`)
|
|
||||||
|
|
||||||
if (needsReview.length > 0) {
|
|
||||||
console.log('⚠️ NEEDS REVIEW:')
|
|
||||||
console.log('='.repeat(80))
|
|
||||||
for (const result of needsReview.slice(0, 10)) {
|
|
||||||
console.log(`\n${result.tsx}`)
|
|
||||||
console.log(` → ${result.json}`)
|
|
||||||
console.log(` TSX: ${result.tsxSize} bytes | JSON: ${result.jsonSize} bytes`)
|
|
||||||
console.log(` TSX hooks: ${result.tsxHasHooks} | state: ${result.tsxHasState} | effects: ${result.tsxHasEffects}`)
|
|
||||||
console.log(` JSON bindings: ${result.jsonHasBindings} | children: ${result.jsonHasChildren}`)
|
|
||||||
console.log(` Reason: ${result.reason}`)
|
|
||||||
}
|
|
||||||
if (needsReview.length > 10) {
|
|
||||||
console.log(`\n... and ${needsReview.length - 10} more`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write full report
|
|
||||||
const reportPath = path.join(ROOT_DIR, 'duplicate-analysis.json')
|
|
||||||
fs.writeFileSync(reportPath, JSON.stringify(results, null, 2))
|
|
||||||
console.log(`\n📄 Full report written to: ${reportPath}`)
|
|
||||||
|
|
||||||
// Generate deletion script for safe components
|
|
||||||
if (safeToDelete.length > 0) {
|
|
||||||
const deletionScript = safeToDelete.map(r => `rm "${r.tsx}"`).join('\n')
|
|
||||||
const scriptPath = path.join(ROOT_DIR, 'delete-duplicates.sh')
|
|
||||||
fs.writeFileSync(scriptPath, deletionScript)
|
|
||||||
console.log(`📝 Deletion script written to: ${scriptPath}`)
|
|
||||||
console.log(` Run: bash delete-duplicates.sh`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
analyzeDuplicates().catch(error => {
|
|
||||||
console.error('❌ Analysis failed:', error)
|
|
||||||
process.exit(1)
|
|
||||||
})
|
|
||||||
@@ -1,75 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
const componentsToAnalyze = {
|
|
||||||
molecules: ['DataSourceCard', 'EditorToolbar', 'EmptyEditorState', 'MonacoEditorPanel', 'SearchBar'],
|
|
||||||
organisms: ['EmptyCanvasState', 'PageHeader', 'SchemaEditorCanvas', 'SchemaEditorPropertiesPanel',
|
|
||||||
'SchemaEditorSidebar', 'SchemaEditorStatusBar', 'SchemaEditorToolbar', 'ToolbarActions'],
|
|
||||||
}
|
|
||||||
|
|
||||||
async function analyzeComponent(category: string, component: string): Promise<void> {
|
|
||||||
const tsFile = path.join(rootDir, `src/components/${category}/${component}.tsx`)
|
|
||||||
const content = await fs.readFile(tsFile, 'utf-8')
|
|
||||||
|
|
||||||
// Check if it's pure composition (only uses UI primitives)
|
|
||||||
const hasBusinessLogic = /useState|useEffect|useCallback|useMemo|useReducer|useRef/.test(content)
|
|
||||||
const hasComplexLogic = /if\s*\(.*\{|switch\s*\(|for\s*\(|while\s*\(/.test(content)
|
|
||||||
|
|
||||||
// Extract what it imports
|
|
||||||
const imports = content.match(/import\s+\{[^}]+\}\s+from\s+['"][^'"]+['"]/g) || []
|
|
||||||
const importedComponents = imports.flatMap(imp => {
|
|
||||||
const match = imp.match(/\{([^}]+)\}/)
|
|
||||||
return match ? match[1].split(',').map(s => s.trim()) : []
|
|
||||||
})
|
|
||||||
|
|
||||||
// Check if it only imports from ui/atoms (pure composition)
|
|
||||||
const onlyUIPrimitives = imports.every(imp =>
|
|
||||||
imp.includes('@/components/ui/') ||
|
|
||||||
imp.includes('@/components/atoms/') ||
|
|
||||||
imp.includes('@/lib/utils') ||
|
|
||||||
imp.includes('lucide-react') ||
|
|
||||||
imp.includes('@phosphor-icons')
|
|
||||||
)
|
|
||||||
|
|
||||||
const lineCount = content.split('\n').length
|
|
||||||
|
|
||||||
console.log(`\n📄 ${component}`)
|
|
||||||
console.log(` Lines: ${lineCount}`)
|
|
||||||
console.log(` Has hooks: ${hasBusinessLogic ? '❌' : '✅'}`)
|
|
||||||
console.log(` Has complex logic: ${hasComplexLogic ? '❌' : '✅'}`)
|
|
||||||
console.log(` Only UI primitives: ${onlyUIPrimitives ? '✅' : '❌'}`)
|
|
||||||
console.log(` Imports: ${importedComponents.slice(0, 5).join(', ')}${importedComponents.length > 5 ? '...' : ''}`)
|
|
||||||
|
|
||||||
if (!hasBusinessLogic && onlyUIPrimitives && lineCount < 100) {
|
|
||||||
console.log(` 🎯 CANDIDATE FOR PURE JSON`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
console.log('🔍 Analyzing components for pure JSON conversion...\n')
|
|
||||||
console.log('Looking for components that:')
|
|
||||||
console.log(' - No hooks (useState, useEffect, etc.)')
|
|
||||||
console.log(' - No complex logic')
|
|
||||||
console.log(' - Only import UI primitives')
|
|
||||||
console.log(' - Are simple compositions\n')
|
|
||||||
|
|
||||||
for (const [category, components] of Object.entries(componentsToAnalyze)) {
|
|
||||||
console.log(`\n═══ ${category.toUpperCase()} ═══`)
|
|
||||||
for (const component of components) {
|
|
||||||
try {
|
|
||||||
await analyzeComponent(category, component)
|
|
||||||
} catch (e) {
|
|
||||||
console.log(`\n📄 ${component}`)
|
|
||||||
console.log(` ⚠️ Could not analyze: ${e}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n\n✨ Analysis complete!')
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(console.error)
|
|
||||||
@@ -1,302 +0,0 @@
|
|||||||
#!/usr/bin/env tsx
|
|
||||||
/**
|
|
||||||
* Audit script for JSON component definitions
|
|
||||||
*
|
|
||||||
* Goals:
|
|
||||||
* 1. Phase out src/components TSX files
|
|
||||||
* 2. Audit existing JSON definitions for completeness and correctness
|
|
||||||
*/
|
|
||||||
|
|
||||||
import fs from 'fs'
|
|
||||||
import path from 'path'
|
|
||||||
import { globSync } from 'fs'
|
|
||||||
|
|
||||||
interface AuditIssue {
|
|
||||||
severity: 'error' | 'warning' | 'info'
|
|
||||||
category: string
|
|
||||||
file?: string
|
|
||||||
message: string
|
|
||||||
suggestion?: string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface AuditReport {
|
|
||||||
timestamp: string
|
|
||||||
issues: AuditIssue[]
|
|
||||||
stats: {
|
|
||||||
totalJsonFiles: number
|
|
||||||
totalTsxFiles: number
|
|
||||||
registryEntries: number
|
|
||||||
orphanedJson: number
|
|
||||||
duplicates: number
|
|
||||||
obsoleteWrapperRefs: number
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const ROOT_DIR = path.resolve(process.cwd())
|
|
||||||
const CONFIG_PAGES_DIR = path.join(ROOT_DIR, 'src/config/pages')
|
|
||||||
const COMPONENTS_DIR = path.join(ROOT_DIR, 'src/components')
|
|
||||||
const JSON_DEFS_DIR = path.join(ROOT_DIR, 'src/components/json-definitions')
|
|
||||||
const REGISTRY_FILE = path.join(ROOT_DIR, 'json-components-registry.json')
|
|
||||||
|
|
||||||
async function loadRegistry(): Promise<any> {
|
|
||||||
const content = fs.readFileSync(REGISTRY_FILE, 'utf-8')
|
|
||||||
return JSON.parse(content)
|
|
||||||
}
|
|
||||||
|
|
||||||
function findAllFiles(pattern: string, cwd: string = ROOT_DIR): string[] {
|
|
||||||
const fullPattern = path.join(cwd, pattern)
|
|
||||||
return globSync(fullPattern, { ignore: '**/node_modules/**' })
|
|
||||||
}
|
|
||||||
|
|
||||||
function toKebabCase(str: string): string {
|
|
||||||
return str.replace(/([A-Z])/g, '-$1').toLowerCase().replace(/^-/, '')
|
|
||||||
}
|
|
||||||
|
|
||||||
function toPascalCase(str: string): string {
|
|
||||||
return str
|
|
||||||
.split('-')
|
|
||||||
.map(word => word.charAt(0).toUpperCase() + word.slice(1))
|
|
||||||
.join('')
|
|
||||||
}
|
|
||||||
|
|
||||||
async function auditJsonComponents(): Promise<AuditReport> {
|
|
||||||
const issues: AuditIssue[] = []
|
|
||||||
const registry = await loadRegistry()
|
|
||||||
|
|
||||||
// Find all files
|
|
||||||
const jsonFiles = findAllFiles('src/config/pages/**/*.json')
|
|
||||||
const tsxFiles = findAllFiles('src/components/**/*.tsx')
|
|
||||||
const jsonDefFiles = findAllFiles('src/components/json-definitions/*.json')
|
|
||||||
|
|
||||||
console.log(`📊 Found ${jsonFiles.length} JSON files in config/pages`)
|
|
||||||
console.log(`📊 Found ${tsxFiles.length} TSX files in src/components`)
|
|
||||||
console.log(`📊 Found ${jsonDefFiles.length} JSON definitions`)
|
|
||||||
console.log(`📊 Found ${registry.components?.length || 0} registry entries\n`)
|
|
||||||
|
|
||||||
// Build registry lookup maps
|
|
||||||
const registryByType = new Map<string, any>()
|
|
||||||
const registryByName = new Map<string, any>()
|
|
||||||
|
|
||||||
if (registry.components) {
|
|
||||||
for (const component of registry.components) {
|
|
||||||
if (component.type) registryByType.set(component.type, component)
|
|
||||||
if (component.name) registryByName.set(component.name, component)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check 1: Find TSX files that have JSON equivalents in config/pages
|
|
||||||
console.log('🔍 Checking for TSX files that could be replaced with JSON...')
|
|
||||||
const tsxBasenames = new Set<string>()
|
|
||||||
for (const tsxFile of tsxFiles) {
|
|
||||||
const basename = path.basename(tsxFile, '.tsx')
|
|
||||||
const dir = path.dirname(tsxFile)
|
|
||||||
const category = path.basename(dir) // atoms, molecules, organisms
|
|
||||||
|
|
||||||
if (!['atoms', 'molecules', 'organisms'].includes(category)) continue
|
|
||||||
|
|
||||||
tsxBasenames.add(basename)
|
|
||||||
const kebab = toKebabCase(basename)
|
|
||||||
|
|
||||||
// Check if there's a corresponding JSON file in config/pages
|
|
||||||
const possibleJsonPath = path.join(CONFIG_PAGES_DIR, category, `${kebab}.json`)
|
|
||||||
|
|
||||||
if (fs.existsSync(possibleJsonPath)) {
|
|
||||||
issues.push({
|
|
||||||
severity: 'warning',
|
|
||||||
category: 'duplicate-implementation',
|
|
||||||
file: tsxFile,
|
|
||||||
message: `TSX file has JSON equivalent at ${path.relative(ROOT_DIR, possibleJsonPath)}`,
|
|
||||||
suggestion: `Consider removing TSX and routing through JSON renderer`
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check 2: Find JSON files without registry entries
|
|
||||||
console.log('🔍 Checking for orphaned JSON files...')
|
|
||||||
for (const jsonFile of jsonFiles) {
|
|
||||||
const content = JSON.parse(fs.readFileSync(jsonFile, 'utf-8'))
|
|
||||||
const componentType = content.type
|
|
||||||
|
|
||||||
if (componentType && !registryByType.has(componentType)) {
|
|
||||||
issues.push({
|
|
||||||
severity: 'error',
|
|
||||||
category: 'orphaned-json',
|
|
||||||
file: jsonFile,
|
|
||||||
message: `JSON file references type "${componentType}" which is not in registry`,
|
|
||||||
suggestion: `Add registry entry for ${componentType} in json-components-registry.json`
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check 3: Find components with obsolete wrapper references
|
|
||||||
console.log('🔍 Checking for obsolete wrapper references...')
|
|
||||||
for (const component of registry.components || []) {
|
|
||||||
if (component.wrapperRequired || component.wrapperComponent) {
|
|
||||||
issues.push({
|
|
||||||
severity: 'warning',
|
|
||||||
category: 'obsolete-wrapper-ref',
|
|
||||||
file: `registry: ${component.type}`,
|
|
||||||
message: `Component "${component.type}" has obsolete wrapperRequired/wrapperComponent fields`,
|
|
||||||
suggestion: `Remove wrapperRequired and wrapperComponent fields - use createJsonComponentWithHooks instead`
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check 4: Find components with load.path that don't exist
|
|
||||||
console.log('🔍 Checking for broken load paths...')
|
|
||||||
for (const component of registry.components || []) {
|
|
||||||
if (component.load?.path) {
|
|
||||||
const loadPath = component.load.path.replace('@/', 'src/')
|
|
||||||
const possibleExtensions = ['.tsx', '.ts', '.jsx', '.js']
|
|
||||||
let found = false
|
|
||||||
|
|
||||||
for (const ext of possibleExtensions) {
|
|
||||||
if (fs.existsSync(path.join(ROOT_DIR, loadPath + ext))) {
|
|
||||||
found = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!found) {
|
|
||||||
issues.push({
|
|
||||||
severity: 'error',
|
|
||||||
category: 'broken-load-path',
|
|
||||||
file: `registry: ${component.type}`,
|
|
||||||
message: `Component "${component.type}" has load.path "${component.load.path}" but file not found`,
|
|
||||||
suggestion: `Fix or remove load.path in registry`
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check 5: Components in src/components/molecules without JSON definitions
|
|
||||||
console.log('🔍 Checking molecules without JSON definitions...')
|
|
||||||
const moleculeTsxFiles = tsxFiles.filter(f => f.includes('/molecules/'))
|
|
||||||
const jsonDefBasenames = new Set(
|
|
||||||
jsonDefFiles.map(f => path.basename(f, '.json'))
|
|
||||||
)
|
|
||||||
|
|
||||||
for (const tsxFile of moleculeTsxFiles) {
|
|
||||||
const basename = path.basename(tsxFile, '.tsx')
|
|
||||||
const kebab = toKebabCase(basename)
|
|
||||||
|
|
||||||
if (!jsonDefBasenames.has(kebab) && registryByType.has(basename)) {
|
|
||||||
const entry = registryByType.get(basename)
|
|
||||||
if (entry.source === 'molecules' && !entry.load?.path) {
|
|
||||||
issues.push({
|
|
||||||
severity: 'info',
|
|
||||||
category: 'potential-conversion',
|
|
||||||
file: tsxFile,
|
|
||||||
message: `Molecule "${basename}" could potentially be converted to JSON`,
|
|
||||||
suggestion: `Evaluate if ${basename} can be expressed as pure JSON`
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const stats = {
|
|
||||||
totalJsonFiles: jsonFiles.length,
|
|
||||||
totalTsxFiles: tsxFiles.length,
|
|
||||||
registryEntries: registry.components?.length || 0,
|
|
||||||
orphanedJson: issues.filter(i => i.category === 'orphaned-json').length,
|
|
||||||
duplicates: issues.filter(i => i.category === 'duplicate-implementation').length,
|
|
||||||
obsoleteWrapperRefs: issues.filter(i => i.category === 'obsolete-wrapper-ref').length
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
timestamp: new Date().toISOString(),
|
|
||||||
issues,
|
|
||||||
stats
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function printReport(report: AuditReport) {
|
|
||||||
console.log('\n' + '='.repeat(80))
|
|
||||||
console.log('📋 AUDIT REPORT')
|
|
||||||
console.log('='.repeat(80))
|
|
||||||
console.log(`\n📅 Generated: ${report.timestamp}\n`)
|
|
||||||
|
|
||||||
console.log('📈 Statistics:')
|
|
||||||
console.log(` • Total JSON files: ${report.stats.totalJsonFiles}`)
|
|
||||||
console.log(` • Total TSX files: ${report.stats.totalTsxFiles}`)
|
|
||||||
console.log(` • Registry entries: ${report.stats.registryEntries}`)
|
|
||||||
console.log(` • Orphaned JSON: ${report.stats.orphanedJson}`)
|
|
||||||
console.log(` • Obsolete wrapper refs: ${report.stats.obsoleteWrapperRefs}`)
|
|
||||||
console.log(` • Duplicate implementations: ${report.stats.duplicates}\n`)
|
|
||||||
|
|
||||||
// Group issues by category
|
|
||||||
const byCategory = new Map<string, AuditIssue[]>()
|
|
||||||
for (const issue of report.issues) {
|
|
||||||
if (!byCategory.has(issue.category)) {
|
|
||||||
byCategory.set(issue.category, [])
|
|
||||||
}
|
|
||||||
byCategory.get(issue.category)!.push(issue)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print issues by severity
|
|
||||||
const severityOrder = ['error', 'warning', 'info'] as const
|
|
||||||
const severityIcons = { error: '❌', warning: '⚠️', info: 'ℹ️' }
|
|
||||||
|
|
||||||
for (const severity of severityOrder) {
|
|
||||||
const issuesOfSeverity = report.issues.filter(i => i.severity === severity)
|
|
||||||
if (issuesOfSeverity.length === 0) continue
|
|
||||||
|
|
||||||
console.log(`\n${severityIcons[severity]} ${severity.toUpperCase()} (${issuesOfSeverity.length})`)
|
|
||||||
console.log('-'.repeat(80))
|
|
||||||
|
|
||||||
const categories = new Map<string, AuditIssue[]>()
|
|
||||||
for (const issue of issuesOfSeverity) {
|
|
||||||
if (!categories.has(issue.category)) {
|
|
||||||
categories.set(issue.category, [])
|
|
||||||
}
|
|
||||||
categories.get(issue.category)!.push(issue)
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const [category, issues] of categories) {
|
|
||||||
console.log(`\n ${category.replace(/-/g, ' ').toUpperCase()} (${issues.length}):`)
|
|
||||||
|
|
||||||
for (const issue of issues.slice(0, 5)) { // Show first 5 of each category
|
|
||||||
console.log(` • ${issue.file || 'N/A'}`)
|
|
||||||
console.log(` ${issue.message}`)
|
|
||||||
if (issue.suggestion) {
|
|
||||||
console.log(` 💡 ${issue.suggestion}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (issues.length > 5) {
|
|
||||||
console.log(` ... and ${issues.length - 5} more`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n' + '='.repeat(80))
|
|
||||||
console.log(`Total issues found: ${report.issues.length}`)
|
|
||||||
console.log('='.repeat(80) + '\n')
|
|
||||||
}
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
console.log('🔍 Starting JSON component audit...\n')
|
|
||||||
|
|
||||||
const report = await auditJsonComponents()
|
|
||||||
|
|
||||||
printReport(report)
|
|
||||||
|
|
||||||
// Write report to file
|
|
||||||
const reportPath = path.join(ROOT_DIR, 'audit-report.json')
|
|
||||||
fs.writeFileSync(reportPath, JSON.stringify(report, null, 2))
|
|
||||||
console.log(`📄 Full report written to: ${reportPath}\n`)
|
|
||||||
|
|
||||||
// Exit with error code if there are errors
|
|
||||||
const errorCount = report.issues.filter(i => i.severity === 'error').length
|
|
||||||
if (errorCount > 0) {
|
|
||||||
console.log(`❌ Audit failed with ${errorCount} errors`)
|
|
||||||
process.exit(1)
|
|
||||||
} else {
|
|
||||||
console.log('✅ Audit completed successfully')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(error => {
|
|
||||||
console.error('❌ Audit failed:', error)
|
|
||||||
process.exit(1)
|
|
||||||
})
|
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
#!/usr/bin/env tsx
|
|
||||||
/**
|
|
||||||
* Cleanup script to remove obsolete wrapper references from registry
|
|
||||||
*/
|
|
||||||
|
|
||||||
import fs from 'fs'
|
|
||||||
import path from 'path'
|
|
||||||
|
|
||||||
const REGISTRY_FILE = path.resolve(process.cwd(), 'json-components-registry.json')
|
|
||||||
|
|
||||||
async function cleanupRegistry() {
|
|
||||||
console.log('🧹 Cleaning up registry...\n')
|
|
||||||
|
|
||||||
// Read registry
|
|
||||||
const content = fs.readFileSync(REGISTRY_FILE, 'utf-8')
|
|
||||||
const registry = JSON.parse(content)
|
|
||||||
|
|
||||||
let cleanedCount = 0
|
|
||||||
const cleanedComponents: string[] = []
|
|
||||||
|
|
||||||
// Remove obsolete fields from all components
|
|
||||||
if (registry.components) {
|
|
||||||
for (const component of registry.components) {
|
|
||||||
let modified = false
|
|
||||||
|
|
||||||
if (component.wrapperRequired !== undefined) {
|
|
||||||
delete component.wrapperRequired
|
|
||||||
modified = true
|
|
||||||
}
|
|
||||||
|
|
||||||
if (component.wrapperComponent !== undefined) {
|
|
||||||
delete component.wrapperComponent
|
|
||||||
modified = true
|
|
||||||
}
|
|
||||||
|
|
||||||
if (modified) {
|
|
||||||
cleanedCount++
|
|
||||||
cleanedComponents.push(component.type || component.name || 'Unknown')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write back to file with proper formatting
|
|
||||||
fs.writeFileSync(REGISTRY_FILE, JSON.stringify(registry, null, 2) + '\n')
|
|
||||||
|
|
||||||
console.log(`✅ Cleaned ${cleanedCount} components\n`)
|
|
||||||
|
|
||||||
if (cleanedComponents.length > 0) {
|
|
||||||
console.log('📋 Cleaned components:')
|
|
||||||
cleanedComponents.slice(0, 10).forEach(name => {
|
|
||||||
console.log(` • ${name}`)
|
|
||||||
})
|
|
||||||
if (cleanedComponents.length > 10) {
|
|
||||||
console.log(` ... and ${cleanedComponents.length - 10} more`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n✨ Registry cleanup complete!')
|
|
||||||
}
|
|
||||||
|
|
||||||
cleanupRegistry().catch(error => {
|
|
||||||
console.error('❌ Cleanup failed:', error)
|
|
||||||
process.exit(1)
|
|
||||||
})
|
|
||||||
@@ -1,115 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* List of simple presentational components that can be safely deleted
|
|
||||||
* These were identified by the conversion script as having no hooks or complex logic
|
|
||||||
*/
|
|
||||||
const SIMPLE_COMPONENTS = {
|
|
||||||
atoms: [
|
|
||||||
'ActionIcon', 'Alert', 'AppLogo', 'Avatar', 'Breadcrumb', 'ButtonGroup',
|
|
||||||
'Chip', 'Code', 'ColorSwatch', 'Container', 'DataList', 'Divider', 'Dot',
|
|
||||||
'EmptyStateIcon', 'FileIcon', 'Flex', 'Grid', 'Heading', 'HelperText',
|
|
||||||
'IconText', 'IconWrapper', 'InfoBox', 'InfoPanel', 'Input', 'Kbd',
|
|
||||||
'KeyValue', 'Label', 'Link', 'List', 'ListItem', 'LiveIndicator',
|
|
||||||
'LoadingSpinner', 'LoadingState', 'MetricDisplay', 'PageHeader', 'Pulse',
|
|
||||||
'ResponsiveGrid', 'ScrollArea', 'SearchInput', 'Section', 'Skeleton',
|
|
||||||
'Spacer', 'Sparkle', 'Spinner', 'StatusIcon', 'TabIcon', 'Tag', 'Text',
|
|
||||||
'TextArea', 'TextGradient', 'TextHighlight', 'Timestamp', 'TreeIcon',
|
|
||||||
// Additional simple ones
|
|
||||||
'AvatarGroup', 'Checkbox', 'Drawer', 'Modal', 'Notification', 'ProgressBar',
|
|
||||||
'Radio', 'Rating', 'Select', 'Slider', 'Stack', 'StepIndicator', 'Stepper',
|
|
||||||
'Table', 'Tabs', 'Timeline', 'Toggle',
|
|
||||||
],
|
|
||||||
molecules: [
|
|
||||||
'ActionBar', 'AppBranding', 'DataCard', 'DataSourceCard', 'EditorActions',
|
|
||||||
'EditorToolbar', 'EmptyEditorState', 'EmptyState', 'FileTabs', 'LabelWithBadge',
|
|
||||||
'LazyInlineMonacoEditor', 'LazyMonacoEditor', 'LoadingFallback', 'LoadingState',
|
|
||||||
'MonacoEditorPanel', 'NavigationItem', 'PageHeaderContent', 'SearchBar',
|
|
||||||
'StatCard', 'TreeCard', 'TreeListHeader',
|
|
||||||
],
|
|
||||||
organisms: [
|
|
||||||
'EmptyCanvasState', 'PageHeader', 'SchemaEditorCanvas', 'SchemaEditorPropertiesPanel',
|
|
||||||
'SchemaEditorSidebar', 'SchemaEditorStatusBar', 'SchemaEditorToolbar', 'ToolbarActions',
|
|
||||||
],
|
|
||||||
ui: [
|
|
||||||
'aspect-ratio', 'avatar', 'badge', 'checkbox', 'collapsible', 'hover-card',
|
|
||||||
'input', 'label', 'popover', 'progress', 'radio-group', 'resizable',
|
|
||||||
'scroll-area', 'separator', 'skeleton', 'switch', 'textarea', 'toggle',
|
|
||||||
// Additional ones
|
|
||||||
'accordion', 'alert', 'button', 'card', 'tabs', 'tooltip',
|
|
||||||
],
|
|
||||||
}
|
|
||||||
|
|
||||||
interface DeletionResult {
|
|
||||||
deleted: string[]
|
|
||||||
kept: string[]
|
|
||||||
failed: string[]
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Delete simple TypeScript components
|
|
||||||
*/
|
|
||||||
async function deleteSimpleComponents(): Promise<void> {
|
|
||||||
console.log('🧹 Cleaning up simple TypeScript components...\n')
|
|
||||||
|
|
||||||
const results: DeletionResult = {
|
|
||||||
deleted: [],
|
|
||||||
kept: [],
|
|
||||||
failed: [],
|
|
||||||
}
|
|
||||||
|
|
||||||
// Process each category
|
|
||||||
for (const [category, components] of Object.entries(SIMPLE_COMPONENTS)) {
|
|
||||||
console.log(`📂 Processing ${category}...`)
|
|
||||||
|
|
||||||
const baseDir = path.join(rootDir, `src/components/${category}`)
|
|
||||||
|
|
||||||
for (const component of components) {
|
|
||||||
const fileName = component.endsWith('.tsx') ? component : `${component}.tsx`
|
|
||||||
const filePath = path.join(baseDir, fileName)
|
|
||||||
|
|
||||||
try {
|
|
||||||
await fs.access(filePath)
|
|
||||||
await fs.unlink(filePath)
|
|
||||||
results.deleted.push(`${category}/${fileName}`)
|
|
||||||
console.log(` ✅ Deleted: ${fileName}`)
|
|
||||||
} catch (error: unknown) {
|
|
||||||
// File doesn't exist or couldn't be deleted
|
|
||||||
if (error instanceof Error && 'code' in error && error.code === 'ENOENT') {
|
|
||||||
results.kept.push(`${category}/${fileName}`)
|
|
||||||
console.log(` ⏭️ Skipped: ${fileName} (not found)`)
|
|
||||||
} else {
|
|
||||||
results.failed.push(`${category}/${fileName}`)
|
|
||||||
console.log(` ❌ Failed: ${fileName}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Summary
|
|
||||||
console.log('📊 Summary:')
|
|
||||||
console.log(` Deleted: ${results.deleted.length} files`)
|
|
||||||
console.log(` Skipped: ${results.kept.length} files`)
|
|
||||||
console.log(` Failed: ${results.failed.length} files`)
|
|
||||||
|
|
||||||
if (results.failed.length > 0) {
|
|
||||||
console.log('\n❌ Failed deletions:')
|
|
||||||
results.failed.forEach(f => console.log(` - ${f}`))
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n✨ Cleanup complete!')
|
|
||||||
console.log('\n📝 Next steps:')
|
|
||||||
console.log(' 1. Update index.ts files to remove deleted exports')
|
|
||||||
console.log(' 2. Search for direct imports of deleted components')
|
|
||||||
console.log(' 3. Run build to check for errors')
|
|
||||||
console.log(' 4. Run tests to verify functionality')
|
|
||||||
}
|
|
||||||
|
|
||||||
deleteSimpleComponents().catch(console.error)
|
|
||||||
@@ -1,262 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
interface ConversionConfig {
|
|
||||||
sourceDir: string
|
|
||||||
targetDir: string
|
|
||||||
category: 'atoms' | 'molecules' | 'organisms' | 'ui'
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ComponentAnalysis {
|
|
||||||
name: string
|
|
||||||
hasHooks: boolean
|
|
||||||
hasComplexLogic: boolean
|
|
||||||
wrapsUIComponent: boolean
|
|
||||||
uiComponentName?: string
|
|
||||||
defaultProps: Record<string, unknown>
|
|
||||||
isSimplePresentational: boolean
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Analyze a TypeScript component file to determine conversion strategy
|
|
||||||
*/
|
|
||||||
async function analyzeComponent(filePath: string): Promise<ComponentAnalysis> {
|
|
||||||
const content = await fs.readFile(filePath, 'utf-8')
|
|
||||||
const fileName = path.basename(filePath, '.tsx')
|
|
||||||
|
|
||||||
// Check for hooks
|
|
||||||
const hasHooks = /use[A-Z]\w+\(/.test(content) ||
|
|
||||||
/useState|useEffect|useCallback|useMemo|useRef|useReducer/.test(content)
|
|
||||||
|
|
||||||
// Check for complex logic
|
|
||||||
const hasComplexLogic = hasHooks ||
|
|
||||||
/switch\s*\(/.test(content) ||
|
|
||||||
/for\s*\(/.test(content) ||
|
|
||||||
/while\s*\(/.test(content) ||
|
|
||||||
content.split('\n').length > 100
|
|
||||||
|
|
||||||
// Check if it wraps a shadcn/ui component
|
|
||||||
const uiImportMatch = content.match(/import\s+\{([^}]+)\}\s+from\s+['"]@\/components\/ui\//)
|
|
||||||
const wrapsUIComponent = !!uiImportMatch
|
|
||||||
const uiComponentName = wrapsUIComponent ? uiImportMatch?.[1].trim() : undefined
|
|
||||||
|
|
||||||
// Extract default props from interface
|
|
||||||
const defaultProps: Record<string, unknown> = {}
|
|
||||||
const propDefaults = content.matchAll(/(\w+)\s*[?]?\s*:\s*([^=\n]+)\s*=\s*['"]?([^'";\n,}]+)['"]?/g)
|
|
||||||
for (const match of propDefaults) {
|
|
||||||
const [, propName, , defaultValue] = match
|
|
||||||
if (propName && defaultValue) {
|
|
||||||
defaultProps[propName] = defaultValue.replace(/['"]/g, '')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Determine if it's simple presentational
|
|
||||||
const isSimplePresentational = !hasComplexLogic &&
|
|
||||||
!hasHooks &&
|
|
||||||
content.split('\n').length < 60
|
|
||||||
|
|
||||||
return {
|
|
||||||
name: fileName,
|
|
||||||
hasHooks,
|
|
||||||
hasComplexLogic,
|
|
||||||
wrapsUIComponent,
|
|
||||||
uiComponentName,
|
|
||||||
defaultProps,
|
|
||||||
isSimplePresentational,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate JSON definition for a component based on analysis
|
|
||||||
*/
|
|
||||||
function generateJSON(analysis: ComponentAnalysis, category: string): object {
|
|
||||||
// If it wraps a UI component, reference that
|
|
||||||
if (analysis.wrapsUIComponent && analysis.uiComponentName) {
|
|
||||||
return {
|
|
||||||
type: analysis.uiComponentName,
|
|
||||||
props: analysis.defaultProps,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If it's simple presentational, create a basic structure
|
|
||||||
if (analysis.isSimplePresentational) {
|
|
||||||
return {
|
|
||||||
type: analysis.name,
|
|
||||||
props: analysis.defaultProps,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If it has hooks or complex logic, mark as needing wrapper
|
|
||||||
if (analysis.hasHooks || analysis.hasComplexLogic) {
|
|
||||||
return {
|
|
||||||
type: analysis.name,
|
|
||||||
jsonCompatible: false,
|
|
||||||
wrapperRequired: true,
|
|
||||||
load: {
|
|
||||||
path: `@/components/${category}/${analysis.name}`,
|
|
||||||
export: analysis.name,
|
|
||||||
},
|
|
||||||
props: analysis.defaultProps,
|
|
||||||
metadata: {
|
|
||||||
notes: analysis.hasHooks ? 'Contains hooks - needs wrapper' : 'Complex logic - needs wrapper',
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default case
|
|
||||||
return {
|
|
||||||
type: analysis.name,
|
|
||||||
props: analysis.defaultProps,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert a single TypeScript file to JSON
|
|
||||||
*/
|
|
||||||
async function convertFile(
|
|
||||||
sourceFile: string,
|
|
||||||
targetDir: string,
|
|
||||||
category: string
|
|
||||||
): Promise<{ success: boolean; analysis: ComponentAnalysis }> {
|
|
||||||
try {
|
|
||||||
const analysis = await analyzeComponent(sourceFile)
|
|
||||||
const json = generateJSON(analysis, category)
|
|
||||||
|
|
||||||
// Generate kebab-case filename
|
|
||||||
const jsonFileName = analysis.name
|
|
||||||
.replace(/([A-Z])/g, '-$1')
|
|
||||||
.toLowerCase()
|
|
||||||
.replace(/^-/, '') + '.json'
|
|
||||||
|
|
||||||
const targetFile = path.join(targetDir, jsonFileName)
|
|
||||||
|
|
||||||
await fs.writeFile(targetFile, JSON.stringify(json, null, 2) + '\n')
|
|
||||||
|
|
||||||
return { success: true, analysis }
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Error converting ${sourceFile}:`, error)
|
|
||||||
return {
|
|
||||||
success: false,
|
|
||||||
analysis: {
|
|
||||||
name: path.basename(sourceFile, '.tsx'),
|
|
||||||
hasHooks: false,
|
|
||||||
hasComplexLogic: false,
|
|
||||||
wrapsUIComponent: false,
|
|
||||||
defaultProps: {},
|
|
||||||
isSimplePresentational: false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert all components in a directory
|
|
||||||
*/
|
|
||||||
async function convertDirectory(config: ConversionConfig): Promise<void> {
|
|
||||||
const sourceDir = path.join(rootDir, config.sourceDir)
|
|
||||||
const targetDir = path.join(rootDir, config.targetDir)
|
|
||||||
|
|
||||||
console.log(`\n📂 Converting ${config.category} components...`)
|
|
||||||
console.log(` Source: ${sourceDir}`)
|
|
||||||
console.log(` Target: ${targetDir}`)
|
|
||||||
|
|
||||||
// Ensure target directory exists
|
|
||||||
await fs.mkdir(targetDir, { recursive: true })
|
|
||||||
|
|
||||||
// Get all TypeScript files
|
|
||||||
const files = await fs.readdir(sourceDir)
|
|
||||||
const tsxFiles = files.filter(f => f.endsWith('.tsx') && !f.includes('.test.') && !f.includes('.stories.'))
|
|
||||||
|
|
||||||
console.log(` Found ${tsxFiles.length} TypeScript files\n`)
|
|
||||||
|
|
||||||
const results = {
|
|
||||||
total: 0,
|
|
||||||
simple: 0,
|
|
||||||
needsWrapper: 0,
|
|
||||||
wrapsUI: 0,
|
|
||||||
failed: 0,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert each file
|
|
||||||
for (const file of tsxFiles) {
|
|
||||||
const sourceFile = path.join(sourceDir, file)
|
|
||||||
const { success, analysis } = await convertFile(sourceFile, targetDir, config.category)
|
|
||||||
|
|
||||||
results.total++
|
|
||||||
|
|
||||||
if (!success) {
|
|
||||||
results.failed++
|
|
||||||
console.log(` ❌ ${file}`)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if (analysis.wrapsUIComponent) {
|
|
||||||
results.wrapsUI++
|
|
||||||
console.log(` 🎨 ${file} → ${analysis.name} (wraps UI)`)
|
|
||||||
} else if (analysis.isSimplePresentational) {
|
|
||||||
results.simple++
|
|
||||||
console.log(` ✅ ${file} → ${analysis.name} (simple)`)
|
|
||||||
} else if (analysis.hasHooks || analysis.hasComplexLogic) {
|
|
||||||
results.needsWrapper++
|
|
||||||
console.log(` ⚙️ ${file} → ${analysis.name} (needs wrapper)`)
|
|
||||||
} else {
|
|
||||||
results.simple++
|
|
||||||
console.log(` ✅ ${file} → ${analysis.name}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n📊 Results for ${config.category}:`)
|
|
||||||
console.log(` Total: ${results.total}`)
|
|
||||||
console.log(` Simple: ${results.simple}`)
|
|
||||||
console.log(` Wraps UI: ${results.wrapsUI}`)
|
|
||||||
console.log(` Needs Wrapper: ${results.needsWrapper}`)
|
|
||||||
console.log(` Failed: ${results.failed}`)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Main conversion process
|
|
||||||
*/
|
|
||||||
async function main() {
|
|
||||||
console.log('🚀 Starting TypeScript to JSON conversion...\n')
|
|
||||||
|
|
||||||
const configs: ConversionConfig[] = [
|
|
||||||
{
|
|
||||||
sourceDir: 'src/components/atoms',
|
|
||||||
targetDir: 'src/config/pages/atoms',
|
|
||||||
category: 'atoms',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
sourceDir: 'src/components/molecules',
|
|
||||||
targetDir: 'src/config/pages/molecules',
|
|
||||||
category: 'molecules',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
sourceDir: 'src/components/organisms',
|
|
||||||
targetDir: 'src/config/pages/organisms',
|
|
||||||
category: 'organisms',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
sourceDir: 'src/components/ui',
|
|
||||||
targetDir: 'src/config/pages/ui',
|
|
||||||
category: 'ui',
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
for (const config of configs) {
|
|
||||||
await convertDirectory(config)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n✨ Conversion complete!')
|
|
||||||
console.log('\n📝 Next steps:')
|
|
||||||
console.log(' 1. Review generated JSON files')
|
|
||||||
console.log(' 2. Manually fix complex components')
|
|
||||||
console.log(' 3. Update json-components-registry.json')
|
|
||||||
console.log(' 4. Test components render correctly')
|
|
||||||
console.log(' 5. Delete old TypeScript files')
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(console.error)
|
|
||||||
@@ -1,91 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
const missingComponents = [
|
|
||||||
'AtomicLibraryShowcase',
|
|
||||||
'CodeEditor',
|
|
||||||
'ComponentTreeBuilder',
|
|
||||||
'ComponentTreeManager',
|
|
||||||
'ConflictResolutionPage',
|
|
||||||
'DockerBuildDebugger',
|
|
||||||
'DocumentationView',
|
|
||||||
'ErrorPanel',
|
|
||||||
'FaviconDesigner',
|
|
||||||
'FeatureIdeaCloud',
|
|
||||||
'FeatureToggleSettings',
|
|
||||||
'JSONComponentTreeManager',
|
|
||||||
'JSONLambdaDesigner',
|
|
||||||
'JSONModelDesigner',
|
|
||||||
'PersistenceDashboard',
|
|
||||||
'PersistenceExample',
|
|
||||||
'ProjectDashboard',
|
|
||||||
'PWASettings',
|
|
||||||
'SassStylesShowcase',
|
|
||||||
'StyleDesigner',
|
|
||||||
]
|
|
||||||
|
|
||||||
async function createComponentJSON(componentName: string) {
|
|
||||||
// Convert to kebab-case for filename
|
|
||||||
const fileName = componentName
|
|
||||||
.replace(/([A-Z])/g, '-$1')
|
|
||||||
.toLowerCase()
|
|
||||||
.replace(/^-/, '') + '.json'
|
|
||||||
|
|
||||||
const filePath = path.join(rootDir, 'src/config/pages/components', fileName)
|
|
||||||
|
|
||||||
// Check if component file exists
|
|
||||||
const possiblePaths = [
|
|
||||||
path.join(rootDir, `src/components/${componentName}.tsx`),
|
|
||||||
path.join(rootDir, `src/components/${componentName}/index.tsx`),
|
|
||||||
]
|
|
||||||
|
|
||||||
let componentPath = ''
|
|
||||||
for (const p of possiblePaths) {
|
|
||||||
try {
|
|
||||||
await fs.access(p)
|
|
||||||
componentPath = `@/components/${componentName}`
|
|
||||||
break
|
|
||||||
} catch {
|
|
||||||
// Continue searching
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!componentPath) {
|
|
||||||
console.log(` ⚠️ ${componentName} - Component file not found, creating placeholder`)
|
|
||||||
componentPath = `@/components/${componentName}`
|
|
||||||
}
|
|
||||||
|
|
||||||
const json = {
|
|
||||||
type: componentName,
|
|
||||||
jsonCompatible: false,
|
|
||||||
wrapperRequired: true,
|
|
||||||
load: {
|
|
||||||
path: componentPath,
|
|
||||||
export: componentName,
|
|
||||||
},
|
|
||||||
props: {},
|
|
||||||
}
|
|
||||||
|
|
||||||
await fs.writeFile(filePath, JSON.stringify(json, null, 2) + '\n')
|
|
||||||
console.log(` ✅ Created: ${fileName}`)
|
|
||||||
}
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
console.log('📝 Creating JSON definitions for missing custom components...\n')
|
|
||||||
|
|
||||||
// Ensure directory exists
|
|
||||||
const targetDir = path.join(rootDir, 'src/config/pages/components')
|
|
||||||
await fs.mkdir(targetDir, { recursive: true })
|
|
||||||
|
|
||||||
for (const component of missingComponents) {
|
|
||||||
await createComponentJSON(component)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\n✨ Created ${missingComponents.length} component JSON files!`)
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(console.error)
|
|
||||||
@@ -1,141 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
// Components we want to remove (restored dependencies)
|
|
||||||
const targetComponents = {
|
|
||||||
ui: ['accordion', 'alert', 'aspect-ratio', 'avatar', 'badge', 'button', 'card',
|
|
||||||
'checkbox', 'collapsible', 'dialog', 'hover-card', 'input', 'label',
|
|
||||||
'popover', 'progress', 'radio-group', 'resizable', 'scroll-area',
|
|
||||||
'separator', 'skeleton', 'sheet', 'switch', 'tabs', 'textarea', 'toggle', 'tooltip'],
|
|
||||||
molecules: ['DataSourceCard', 'EditorToolbar', 'EmptyEditorState', 'MonacoEditorPanel', 'SearchBar'],
|
|
||||||
organisms: ['EmptyCanvasState', 'PageHeader', 'SchemaEditorCanvas', 'SchemaEditorPropertiesPanel',
|
|
||||||
'SchemaEditorSidebar', 'SchemaEditorStatusBar', 'SchemaEditorToolbar', 'ToolbarActions'],
|
|
||||||
atoms: ['Input']
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ImportInfo {
|
|
||||||
file: string
|
|
||||||
line: number
|
|
||||||
importStatement: string
|
|
||||||
importedComponents: string[]
|
|
||||||
fromPath: string
|
|
||||||
}
|
|
||||||
|
|
||||||
async function findAllImports(): Promise<ImportInfo[]> {
|
|
||||||
const imports: ImportInfo[] = []
|
|
||||||
|
|
||||||
const searchDirs = [
|
|
||||||
'src/components',
|
|
||||||
'src/pages',
|
|
||||||
'src/lib',
|
|
||||||
'src'
|
|
||||||
]
|
|
||||||
|
|
||||||
for (const dir of searchDirs) {
|
|
||||||
const dirPath = path.join(rootDir, dir)
|
|
||||||
try {
|
|
||||||
await processDirectory(dirPath, imports)
|
|
||||||
} catch (e) {
|
|
||||||
// Directory might not exist, skip
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return imports
|
|
||||||
}
|
|
||||||
|
|
||||||
async function processDirectory(dir: string, imports: ImportInfo[]): Promise<void> {
|
|
||||||
const entries = await fs.readdir(dir, { withFileTypes: true })
|
|
||||||
|
|
||||||
for (const entry of entries) {
|
|
||||||
const fullPath = path.join(dir, entry.name)
|
|
||||||
|
|
||||||
if (entry.isDirectory() && !entry.name.includes('node_modules')) {
|
|
||||||
await processDirectory(fullPath, imports)
|
|
||||||
} else if (entry.isFile() && (entry.name.endsWith('.tsx') || entry.name.endsWith('.ts'))) {
|
|
||||||
await processFile(fullPath, imports)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function processFile(filePath: string, imports: ImportInfo[]): Promise<void> {
|
|
||||||
const content = await fs.readFile(filePath, 'utf-8')
|
|
||||||
const lines = content.split('\n')
|
|
||||||
|
|
||||||
for (let i = 0; i < lines.length; i++) {
|
|
||||||
const line = lines[i]
|
|
||||||
|
|
||||||
// Check for imports from our target components
|
|
||||||
for (const [category, components] of Object.entries(targetComponents)) {
|
|
||||||
for (const component of components) {
|
|
||||||
const patterns = [
|
|
||||||
`from ['"]@/components/${category}/${component}['"]`,
|
|
||||||
`from ['"]./${component}['"]`,
|
|
||||||
`from ['"]../${component}['"]`,
|
|
||||||
]
|
|
||||||
|
|
||||||
for (const pattern of patterns) {
|
|
||||||
if (new RegExp(pattern).test(line)) {
|
|
||||||
// Extract imported components
|
|
||||||
const importMatch = line.match(/import\s+(?:\{([^}]+)\}|(\w+))\s+from/)
|
|
||||||
const importedComponents = importMatch
|
|
||||||
? (importMatch[1] || importMatch[2]).split(',').map(s => s.trim())
|
|
||||||
: []
|
|
||||||
|
|
||||||
imports.push({
|
|
||||||
file: filePath.replace(rootDir, '').replace(/\\/g, '/'),
|
|
||||||
line: i + 1,
|
|
||||||
importStatement: line.trim(),
|
|
||||||
importedComponents,
|
|
||||||
fromPath: component
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
console.log('🔍 Finding all imports of target components...\n')
|
|
||||||
|
|
||||||
const imports = await findAllImports()
|
|
||||||
|
|
||||||
if (imports.length === 0) {
|
|
||||||
console.log('✅ No imports found! Components can be safely deleted.')
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`❌ Found ${imports.length} imports that need refactoring:\n`)
|
|
||||||
|
|
||||||
const byFile: Record<string, ImportInfo[]> = {}
|
|
||||||
for (const imp of imports) {
|
|
||||||
if (!byFile[imp.file]) byFile[imp.file] = []
|
|
||||||
byFile[imp.file].push(imp)
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const [file, fileImports] of Object.entries(byFile)) {
|
|
||||||
console.log(`📄 ${file}`)
|
|
||||||
for (const imp of fileImports) {
|
|
||||||
console.log(` Line ${imp.line}: ${imp.importStatement}`)
|
|
||||||
console.log(` → Imports: ${imp.importedComponents.join(', ')}`)
|
|
||||||
}
|
|
||||||
console.log()
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n📊 Summary by category:')
|
|
||||||
const byCategory: Record<string, number> = {}
|
|
||||||
for (const imp of imports) {
|
|
||||||
const key = imp.fromPath
|
|
||||||
byCategory[key] = (byCategory[key] || 0) + 1
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const [component, count] of Object.entries(byCategory).sort((a, b) => b[1] - a[1])) {
|
|
||||||
console.log(` ${component}: ${count} imports`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(console.error)
|
|
||||||
@@ -1,41 +0,0 @@
|
|||||||
#!/usr/bin/env tsx
|
|
||||||
/**
|
|
||||||
* Fix index.ts files to only export existing TSX files
|
|
||||||
*/
|
|
||||||
|
|
||||||
import fs from 'fs'
|
|
||||||
import path from 'path'
|
|
||||||
import { globSync } from 'fs'
|
|
||||||
|
|
||||||
const ROOT_DIR = path.resolve(process.cwd())
|
|
||||||
const COMPONENTS_DIR = path.join(ROOT_DIR, 'src/components')
|
|
||||||
|
|
||||||
const categories = ['atoms', 'molecules', 'organisms']
|
|
||||||
|
|
||||||
for (const category of categories) {
|
|
||||||
const categoryDir = path.join(COMPONENTS_DIR, category)
|
|
||||||
const indexPath = path.join(categoryDir, 'index.ts')
|
|
||||||
|
|
||||||
if (!fs.existsSync(indexPath)) continue
|
|
||||||
|
|
||||||
// Find all TSX files in this category
|
|
||||||
const tsxFiles = globSync(path.join(categoryDir, '*.tsx'))
|
|
||||||
const basenames = tsxFiles.map(f => path.basename(f, '.tsx'))
|
|
||||||
|
|
||||||
console.log(`\n📁 ${category}/`)
|
|
||||||
console.log(` Found ${basenames.length} TSX files`)
|
|
||||||
|
|
||||||
// Generate new exports
|
|
||||||
const exports = basenames
|
|
||||||
.sort()
|
|
||||||
.map(name => `export { ${name} } from './${name}'`)
|
|
||||||
.join('\n')
|
|
||||||
|
|
||||||
// Write new index file
|
|
||||||
const content = `// Auto-generated - only exports existing TSX files\n${exports}\n`
|
|
||||||
fs.writeFileSync(indexPath, content)
|
|
||||||
|
|
||||||
console.log(` ✅ Updated ${category}/index.ts`)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n✨ All index files updated!')
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
import fs from 'fs'
|
|
||||||
import path from 'path'
|
|
||||||
import { fileURLToPath } from 'url'
|
|
||||||
|
|
||||||
interface RegistryComponent {
|
|
||||||
type?: string
|
|
||||||
name?: string
|
|
||||||
export?: string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface RegistryData {
|
|
||||||
components?: RegistryComponent[]
|
|
||||||
}
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
const registryPath = path.join(rootDir, 'json-components-registry.json')
|
|
||||||
const outputPath = path.join(rootDir, 'src/types/json-ui-component-types.ts')
|
|
||||||
|
|
||||||
const registryData = JSON.parse(fs.readFileSync(registryPath, 'utf8')) as RegistryData
|
|
||||||
const components = registryData.components ?? []
|
|
||||||
|
|
||||||
const seen = new Set<string>()
|
|
||||||
const componentTypes = components.flatMap((component) => {
|
|
||||||
const typeName = component.type ?? component.name ?? component.export
|
|
||||||
if (!typeName || typeof typeName !== 'string') {
|
|
||||||
throw new Error('Registry component is missing a valid type/name/export entry.')
|
|
||||||
}
|
|
||||||
if (seen.has(typeName)) {
|
|
||||||
return []
|
|
||||||
}
|
|
||||||
seen.add(typeName)
|
|
||||||
return [typeName]
|
|
||||||
})
|
|
||||||
|
|
||||||
const lines = [
|
|
||||||
'// This file is auto-generated by scripts/generate-json-ui-component-types.ts.',
|
|
||||||
'// Do not edit this file directly.',
|
|
||||||
'',
|
|
||||||
'export const jsonUIComponentTypes = [',
|
|
||||||
...componentTypes.map((typeName) => ` ${JSON.stringify(typeName)},`),
|
|
||||||
'] as const',
|
|
||||||
'',
|
|
||||||
'export type JSONUIComponentType = typeof jsonUIComponentTypes[number]',
|
|
||||||
'',
|
|
||||||
]
|
|
||||||
|
|
||||||
fs.writeFileSync(outputPath, `${lines.join('\n')}`)
|
|
||||||
|
|
||||||
console.log(`✅ Wrote ${componentTypes.length} component types to ${outputPath}`)
|
|
||||||
@@ -1,127 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
// Components we restored (the ones we want to potentially convert to JSON)
|
|
||||||
const restoredComponents = {
|
|
||||||
ui: ['accordion', 'alert', 'aspect-ratio', 'avatar', 'badge', 'button', 'card',
|
|
||||||
'checkbox', 'collapsible', 'dialog', 'hover-card', 'input', 'label',
|
|
||||||
'popover', 'progress', 'radio-group', 'resizable', 'scroll-area',
|
|
||||||
'separator', 'skeleton', 'sheet', 'switch', 'tabs', 'textarea', 'toggle', 'tooltip'],
|
|
||||||
molecules: ['DataSourceCard', 'EditorToolbar', 'EmptyEditorState', 'MonacoEditorPanel', 'SearchBar'],
|
|
||||||
organisms: ['EmptyCanvasState', 'PageHeader', 'SchemaEditorCanvas', 'SchemaEditorPropertiesPanel',
|
|
||||||
'SchemaEditorSidebar', 'SchemaEditorStatusBar', 'SchemaEditorToolbar', 'ToolbarActions'],
|
|
||||||
atoms: ['Input'],
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ComponentAnalysis {
|
|
||||||
name: string
|
|
||||||
category: string
|
|
||||||
pureJSONEligible: boolean
|
|
||||||
reasons: string[]
|
|
||||||
complexity: 'simple' | 'medium' | 'complex'
|
|
||||||
hasHooks: boolean
|
|
||||||
hasConditionalLogic: boolean
|
|
||||||
hasHelperFunctions: boolean
|
|
||||||
hasComplexProps: boolean
|
|
||||||
importsCustomComponents: boolean
|
|
||||||
onlyImportsUIorAtoms: boolean
|
|
||||||
}
|
|
||||||
|
|
||||||
async function analyzeComponent(category: string, component: string): Promise<ComponentAnalysis> {
|
|
||||||
const tsFile = path.join(rootDir, `src/components/${category}/${component}.tsx`)
|
|
||||||
const content = await fs.readFile(tsFile, 'utf-8')
|
|
||||||
|
|
||||||
const hasHooks = /useState|useEffect|useCallback|useMemo|useReducer|useRef|useContext/.test(content)
|
|
||||||
const hasConditionalLogic = /\?|if\s*\(|switch\s*\(/.test(content)
|
|
||||||
const hasHelperFunctions = /(?:const|function)\s+\w+\s*=\s*\([^)]*\)\s*=>/.test(content) && /return\s+\(/.test(content.split('return (')[0] || '')
|
|
||||||
const hasComplexProps = /\.\w+\s*\?/.test(content) || /Object\./.test(content) || /Array\./.test(content)
|
|
||||||
|
|
||||||
// Check imports
|
|
||||||
const importLines = content.match(/import\s+.*?\s+from\s+['"](.*?)['"]/g) || []
|
|
||||||
const importsCustomComponents = importLines.some(line =>
|
|
||||||
/@\/components\/(molecules|organisms)/.test(line)
|
|
||||||
)
|
|
||||||
const onlyImportsUIorAtoms = importLines.every(line => {
|
|
||||||
if (!line.includes('@/components/')) return true
|
|
||||||
return /@\/components\/(ui|atoms)/.test(line)
|
|
||||||
})
|
|
||||||
|
|
||||||
const reasons: string[] = []
|
|
||||||
if (hasHooks) reasons.push('Has React hooks')
|
|
||||||
if (hasHelperFunctions) reasons.push('Has helper functions')
|
|
||||||
if (hasComplexProps) reasons.push('Has complex prop access')
|
|
||||||
if (importsCustomComponents) reasons.push('Imports molecules/organisms')
|
|
||||||
if (!onlyImportsUIorAtoms && !importsCustomComponents) reasons.push('Imports non-UI components')
|
|
||||||
|
|
||||||
// Determine if eligible for pure JSON
|
|
||||||
const pureJSONEligible = !hasHooks && !hasHelperFunctions && !hasComplexProps && onlyImportsUIorAtoms
|
|
||||||
|
|
||||||
// Complexity scoring
|
|
||||||
let complexity: 'simple' | 'medium' | 'complex' = 'simple'
|
|
||||||
if (hasHooks || hasHelperFunctions || hasComplexProps) {
|
|
||||||
complexity = 'complex'
|
|
||||||
} else if (hasConditionalLogic || importsCustomComponents) {
|
|
||||||
complexity = 'medium'
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
name: component,
|
|
||||||
category,
|
|
||||||
pureJSONEligible,
|
|
||||||
reasons,
|
|
||||||
complexity,
|
|
||||||
hasHooks,
|
|
||||||
hasConditionalLogic,
|
|
||||||
hasHelperFunctions,
|
|
||||||
hasComplexProps,
|
|
||||||
importsCustomComponents,
|
|
||||||
onlyImportsUIorAtoms,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
console.log('🔍 Analyzing restored components for pure JSON eligibility...\\n')
|
|
||||||
|
|
||||||
const eligible: ComponentAnalysis[] = []
|
|
||||||
const ineligible: ComponentAnalysis[] = []
|
|
||||||
|
|
||||||
for (const [category, components] of Object.entries(restoredComponents)) {
|
|
||||||
for (const component of components) {
|
|
||||||
try {
|
|
||||||
const analysis = await analyzeComponent(category, component)
|
|
||||||
if (analysis.pureJSONEligible) {
|
|
||||||
eligible.push(analysis)
|
|
||||||
} else {
|
|
||||||
ineligible.push(analysis)
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
console.log(`⚠️ ${component} - Could not analyze: ${e}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\\n✅ ELIGIBLE FOR PURE JSON (${eligible.length} components)\\n`)
|
|
||||||
for (const comp of eligible) {
|
|
||||||
console.log(` ${comp.name} (${comp.category})`)
|
|
||||||
console.log(` Complexity: ${comp.complexity}`)
|
|
||||||
console.log(` Conditional: ${comp.hasConditionalLogic ? 'Yes' : 'No'}`)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\\n❌ MUST STAY TYPESCRIPT (${ineligible.length} components)\\n`)
|
|
||||||
for (const comp of ineligible) {
|
|
||||||
console.log(` ${comp.name} (${comp.category})`)
|
|
||||||
console.log(` Complexity: ${comp.complexity}`)
|
|
||||||
console.log(` Reasons: ${comp.reasons.join(', ')}`)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`\\n📊 Summary:`)
|
|
||||||
console.log(` Eligible for JSON: ${eligible.length}`)
|
|
||||||
console.log(` Must stay TypeScript: ${ineligible.length}`)
|
|
||||||
console.log(` Conversion rate: ${Math.round(eligible.length / (eligible.length + ineligible.length) * 100)}%`)
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(console.error)
|
|
||||||
@@ -1,157 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Strategy: Replace static imports with dynamic component loading
|
|
||||||
*
|
|
||||||
* Before:
|
|
||||||
* import { Button } from '@/components/ui/button'
|
|
||||||
* <Button variant="primary">Click</Button>
|
|
||||||
*
|
|
||||||
* After:
|
|
||||||
* import { getComponent } from '@/lib/component-loader'
|
|
||||||
* const Button = getComponent('Button')
|
|
||||||
* <Button variant="primary">Click</Button>
|
|
||||||
*/
|
|
||||||
|
|
||||||
interface RefactorTask {
|
|
||||||
file: string
|
|
||||||
replacements: Array<{
|
|
||||||
oldImport: string
|
|
||||||
newImport: string
|
|
||||||
components: string[]
|
|
||||||
}>
|
|
||||||
}
|
|
||||||
|
|
||||||
const targetComponents = {
|
|
||||||
ui: ['button', 'card', 'badge', 'label', 'input', 'separator', 'scroll-area',
|
|
||||||
'tabs', 'dialog', 'textarea', 'tooltip', 'switch', 'alert', 'skeleton',
|
|
||||||
'progress', 'collapsible', 'resizable', 'popover', 'hover-card', 'checkbox',
|
|
||||||
'accordion', 'aspect-ratio', 'avatar', 'radio-group', 'sheet', 'toggle'],
|
|
||||||
molecules: ['DataSourceCard', 'EditorToolbar', 'EmptyEditorState', 'MonacoEditorPanel', 'SearchBar'],
|
|
||||||
organisms: ['EmptyCanvasState', 'PageHeader', 'SchemaEditorCanvas', 'SchemaEditorPropertiesPanel',
|
|
||||||
'SchemaEditorSidebar', 'SchemaEditorStatusBar', 'SchemaEditorToolbar', 'ToolbarActions'],
|
|
||||||
atoms: ['Input']
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function refactorFile(filePath: string): Promise<boolean> {
|
|
||||||
let content = await fs.readFile(filePath, 'utf-8')
|
|
||||||
let modified = false
|
|
||||||
|
|
||||||
// Find all imports to replace
|
|
||||||
const componentsToLoad = new Set<string>()
|
|
||||||
|
|
||||||
for (const [category, components] of Object.entries(targetComponents)) {
|
|
||||||
for (const component of components) {
|
|
||||||
const patterns = [
|
|
||||||
new RegExp(`import\\s+\\{([^}]+)\\}\\s+from\\s+['"]@/components/${category}/${component}['"]`, 'g'),
|
|
||||||
new RegExp(`import\\s+(\\w+)\\s+from\\s+['"]@/components/${category}/${component}['"]`, 'g'),
|
|
||||||
]
|
|
||||||
|
|
||||||
for (const pattern of patterns) {
|
|
||||||
const matches = content.matchAll(pattern)
|
|
||||||
for (const match of matches) {
|
|
||||||
const importedItems = match[1].split(',').map(s => s.trim().split(' as ')[0].trim())
|
|
||||||
importedItems.forEach(item => componentsToLoad.add(item))
|
|
||||||
|
|
||||||
// Remove the import line
|
|
||||||
content = content.replace(match[0], '')
|
|
||||||
modified = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!modified) return false
|
|
||||||
|
|
||||||
// Add dynamic component loader import at top
|
|
||||||
const loaderImport = `import { loadComponent } from '@/lib/component-loader'\n`
|
|
||||||
|
|
||||||
// Add component loading statements
|
|
||||||
const componentLoads = Array.from(componentsToLoad)
|
|
||||||
.map(comp => `const ${comp} = loadComponent('${comp}')`)
|
|
||||||
.join('\n')
|
|
||||||
|
|
||||||
// Find first import statement location
|
|
||||||
const firstImportMatch = content.match(/^import\s/m)
|
|
||||||
if (firstImportMatch && firstImportMatch.index !== undefined) {
|
|
||||||
content = content.slice(0, firstImportMatch.index) +
|
|
||||||
loaderImport + '\n' +
|
|
||||||
componentLoads + '\n\n' +
|
|
||||||
content.slice(firstImportMatch.index)
|
|
||||||
}
|
|
||||||
|
|
||||||
await fs.writeFile(filePath, content)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
async function createComponentLoader() {
|
|
||||||
const loaderPath = path.join(rootDir, 'src/lib/component-loader.ts')
|
|
||||||
|
|
||||||
const loaderContent = `/**
|
|
||||||
* Dynamic Component Loader
|
|
||||||
* Loads components from the registry at runtime instead of static imports
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { ComponentType, lazy } from 'react'
|
|
||||||
|
|
||||||
const componentCache = new Map<string, ComponentType<any>>()
|
|
||||||
|
|
||||||
export function loadComponent(componentName: string): ComponentType<any> {
|
|
||||||
if (componentCache.has(componentName)) {
|
|
||||||
return componentCache.get(componentName)!
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try to load from different sources
|
|
||||||
const loaders = [
|
|
||||||
() => import(\`@/components/ui/\${componentName.toLowerCase()}\`),
|
|
||||||
() => import(\`@/components/atoms/\${componentName}\`),
|
|
||||||
() => import(\`@/components/molecules/\${componentName}\`),
|
|
||||||
() => import(\`@/components/organisms/\${componentName}\`),
|
|
||||||
]
|
|
||||||
|
|
||||||
// Create lazy component
|
|
||||||
const LazyComponent = lazy(async () => {
|
|
||||||
for (const loader of loaders) {
|
|
||||||
try {
|
|
||||||
const module = await loader()
|
|
||||||
return { default: module[componentName] || module.default }
|
|
||||||
} catch (e) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new Error(\`Component \${componentName} not found\`)
|
|
||||||
})
|
|
||||||
|
|
||||||
componentCache.set(componentName, LazyComponent)
|
|
||||||
return LazyComponent
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getComponent(componentName: string): ComponentType<any> {
|
|
||||||
return loadComponent(componentName)
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
await fs.writeFile(loaderPath, loaderContent)
|
|
||||||
console.log('✅ Created component-loader.ts')
|
|
||||||
}
|
|
||||||
|
|
||||||
async function main() {
|
|
||||||
console.log('🚀 Starting AGGRESSIVE refactoring to eliminate static imports...\n')
|
|
||||||
console.log('⚠️ WARNING: This is a MAJOR refactoring affecting 975+ import statements!\n')
|
|
||||||
console.log('Press Ctrl+C now if you want to reconsider...\n')
|
|
||||||
|
|
||||||
await new Promise(resolve => setTimeout(resolve, 3000))
|
|
||||||
|
|
||||||
console.log('🔧 Creating dynamic component loader...')
|
|
||||||
await createComponentLoader()
|
|
||||||
|
|
||||||
console.log('\n📝 This approach requires significant testing and may break things.')
|
|
||||||
console.log(' Recommendation: Manual refactoring of high-value components instead.\n')
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch(console.error)
|
|
||||||
@@ -1,76 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Update index.ts files to remove exports for deleted components
|
|
||||||
*/
|
|
||||||
async function updateIndexFiles(): Promise<void> {
|
|
||||||
console.log('📝 Updating index.ts files...\n')
|
|
||||||
|
|
||||||
const directories = [
|
|
||||||
'src/components/atoms',
|
|
||||||
'src/components/molecules',
|
|
||||||
'src/components/organisms',
|
|
||||||
'src/components/ui',
|
|
||||||
]
|
|
||||||
|
|
||||||
for (const dir of directories) {
|
|
||||||
const indexPath = path.join(rootDir, dir, 'index.ts')
|
|
||||||
const dirPath = path.join(rootDir, dir)
|
|
||||||
|
|
||||||
console.log(`📂 Processing ${dir}/index.ts...`)
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Read current index.ts
|
|
||||||
const indexContent = await fs.readFile(indexPath, 'utf-8')
|
|
||||||
const lines = indexContent.split('\n')
|
|
||||||
|
|
||||||
// Get list of existing .tsx files
|
|
||||||
const files = await fs.readdir(dirPath)
|
|
||||||
const existingComponents = new Set(
|
|
||||||
files
|
|
||||||
.filter(f => f.endsWith('.tsx') && f !== 'index.tsx')
|
|
||||||
.map(f => f.replace('.tsx', ''))
|
|
||||||
)
|
|
||||||
|
|
||||||
// Filter out exports for deleted components
|
|
||||||
const updatedLines = lines.filter(line => {
|
|
||||||
// Skip empty lines and comments
|
|
||||||
if (!line.trim() || line.trim().startsWith('//')) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if it's an export line
|
|
||||||
const exportMatch = line.match(/export\s+(?:\{([^}]+)\}|.+)\s+from\s+['"]\.\/([^'"]+)['"]/)
|
|
||||||
if (!exportMatch) {
|
|
||||||
return true // Keep non-export lines
|
|
||||||
}
|
|
||||||
|
|
||||||
const componentName = exportMatch[2]
|
|
||||||
const exists = existingComponents.has(componentName)
|
|
||||||
|
|
||||||
if (!exists) {
|
|
||||||
console.log(` ❌ Removing export: ${componentName}`)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
})
|
|
||||||
|
|
||||||
// Write updated index.ts
|
|
||||||
await fs.writeFile(indexPath, updatedLines.join('\n'))
|
|
||||||
|
|
||||||
console.log(` ✅ Updated ${dir}/index.ts\n`)
|
|
||||||
} catch (error) {
|
|
||||||
console.error(` ❌ Error processing ${dir}/index.ts:`, error)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('✨ Index files updated!')
|
|
||||||
}
|
|
||||||
|
|
||||||
updateIndexFiles().catch(console.error)
|
|
||||||
@@ -1,262 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath } from 'node:url'
|
|
||||||
|
|
||||||
const __dirname = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(__dirname, '..')
|
|
||||||
|
|
||||||
interface JSONComponent {
|
|
||||||
type: string
|
|
||||||
jsonCompatible?: boolean
|
|
||||||
wrapperRequired?: boolean
|
|
||||||
load?: {
|
|
||||||
path: string
|
|
||||||
export: string
|
|
||||||
lazy?: boolean
|
|
||||||
}
|
|
||||||
props?: Record<string, unknown>
|
|
||||||
metadata?: {
|
|
||||||
notes?: string
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
interface RegistryEntry {
|
|
||||||
type: string
|
|
||||||
name: string
|
|
||||||
category: string
|
|
||||||
canHaveChildren: boolean
|
|
||||||
description: string
|
|
||||||
status: 'supported' | 'deprecated'
|
|
||||||
source: 'atoms' | 'molecules' | 'organisms' | 'ui' | 'wrappers' | 'custom'
|
|
||||||
jsonCompatible: boolean
|
|
||||||
wrapperRequired?: boolean
|
|
||||||
load?: {
|
|
||||||
path: string
|
|
||||||
export: string
|
|
||||||
lazy?: boolean
|
|
||||||
}
|
|
||||||
metadata?: {
|
|
||||||
conversionDate?: string
|
|
||||||
autoGenerated?: boolean
|
|
||||||
notes?: string
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
interface Registry {
|
|
||||||
version: string
|
|
||||||
categories: Record<string, string>
|
|
||||||
sourceRoots: Record<string, string[]>
|
|
||||||
components: RegistryEntry[]
|
|
||||||
statistics: {
|
|
||||||
total: number
|
|
||||||
supported: number
|
|
||||||
jsonCompatible: number
|
|
||||||
byCategory: Record<string, number>
|
|
||||||
bySource: Record<string, number>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine component category based on name and source
|
|
||||||
*/
|
|
||||||
function determineCategory(componentName: string, source: string): string {
|
|
||||||
const name = componentName.toLowerCase()
|
|
||||||
|
|
||||||
// Layout components
|
|
||||||
if (/container|section|stack|flex|grid|layout|panel|sidebar|header|footer/.test(name)) {
|
|
||||||
return 'layout'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Input components
|
|
||||||
if (/input|select|checkbox|radio|slider|switch|form|textarea|date|file|number|password|search/.test(name)) {
|
|
||||||
return 'input'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Navigation components
|
|
||||||
if (/nav|menu|breadcrumb|tab|link|pagination/.test(name)) {
|
|
||||||
return 'navigation'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Feedback components
|
|
||||||
if (/alert|toast|notification|spinner|loading|progress|skeleton|badge|indicator/.test(name)) {
|
|
||||||
return 'feedback'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Data display components
|
|
||||||
if (/table|list|card|chart|graph|tree|timeline|avatar|image/.test(name)) {
|
|
||||||
return 'data'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Display components
|
|
||||||
if (/text|heading|label|code|icon|divider|separator|spacer/.test(name)) {
|
|
||||||
return 'display'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default to custom for organisms and complex components
|
|
||||||
if (source === 'organisms' || source === 'molecules') {
|
|
||||||
return 'custom'
|
|
||||||
}
|
|
||||||
|
|
||||||
return 'display'
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine if component can have children
|
|
||||||
*/
|
|
||||||
function canHaveChildren(componentName: string): boolean {
|
|
||||||
const name = componentName.toLowerCase()
|
|
||||||
|
|
||||||
// These typically don't have children
|
|
||||||
const noChildren = /input|select|checkbox|radio|slider|switch|image|icon|divider|separator|spacer|spinner|progress|badge|dot/
|
|
||||||
|
|
||||||
return !noChildren.test(name)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate description for component
|
|
||||||
*/
|
|
||||||
function generateDescription(componentName: string, category: string): string {
|
|
||||||
const descriptions: Record<string, string> = {
|
|
||||||
layout: 'Layout container component',
|
|
||||||
input: 'Form input component',
|
|
||||||
navigation: 'Navigation component',
|
|
||||||
feedback: 'Feedback and status component',
|
|
||||||
data: 'Data display component',
|
|
||||||
display: 'Display component',
|
|
||||||
custom: 'Custom component',
|
|
||||||
}
|
|
||||||
|
|
||||||
return descriptions[category] || 'Component'
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Read all JSON files from a directory and create registry entries
|
|
||||||
*/
|
|
||||||
async function processDirectory(
|
|
||||||
dir: string,
|
|
||||||
source: 'atoms' | 'molecules' | 'organisms' | 'ui' | 'custom'
|
|
||||||
): Promise<RegistryEntry[]> {
|
|
||||||
const entries: RegistryEntry[] = []
|
|
||||||
|
|
||||||
try {
|
|
||||||
const files = await fs.readdir(dir)
|
|
||||||
const jsonFiles = files.filter(f => f.endsWith('.json'))
|
|
||||||
|
|
||||||
for (const file of jsonFiles) {
|
|
||||||
const filePath = path.join(dir, file)
|
|
||||||
const content = await fs.readFile(filePath, 'utf-8')
|
|
||||||
const jsonComponent: JSONComponent = JSON.parse(content)
|
|
||||||
|
|
||||||
const componentName = jsonComponent.type
|
|
||||||
if (!componentName) continue
|
|
||||||
|
|
||||||
const category = determineCategory(componentName, source)
|
|
||||||
|
|
||||||
const entry: RegistryEntry = {
|
|
||||||
type: componentName,
|
|
||||||
name: componentName,
|
|
||||||
category,
|
|
||||||
canHaveChildren: canHaveChildren(componentName),
|
|
||||||
description: generateDescription(componentName, category),
|
|
||||||
status: 'supported',
|
|
||||||
source,
|
|
||||||
jsonCompatible: jsonComponent.jsonCompatible !== false,
|
|
||||||
wrapperRequired: jsonComponent.wrapperRequired || false,
|
|
||||||
metadata: {
|
|
||||||
conversionDate: new Date().toISOString().split('T')[0],
|
|
||||||
autoGenerated: true,
|
|
||||||
notes: jsonComponent.metadata?.notes,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
if (jsonComponent.load) {
|
|
||||||
entry.load = jsonComponent.load
|
|
||||||
}
|
|
||||||
|
|
||||||
entries.push(entry)
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Error processing ${dir}:`, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
return entries
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Update the registry with new components
|
|
||||||
*/
|
|
||||||
async function updateRegistry() {
|
|
||||||
console.log('📝 Updating json-components-registry.json...\n')
|
|
||||||
|
|
||||||
const registryPath = path.join(rootDir, 'json-components-registry.json')
|
|
||||||
|
|
||||||
// Read existing registry
|
|
||||||
const registryContent = await fs.readFile(registryPath, 'utf-8')
|
|
||||||
const registry: Registry = JSON.parse(registryContent)
|
|
||||||
|
|
||||||
console.log(` Current components: ${registry.components.length}`)
|
|
||||||
|
|
||||||
// Process each directory
|
|
||||||
const newEntries: RegistryEntry[] = []
|
|
||||||
|
|
||||||
const directories = [
|
|
||||||
{ dir: path.join(rootDir, 'src/config/pages/atoms'), source: 'atoms' as const },
|
|
||||||
{ dir: path.join(rootDir, 'src/config/pages/molecules'), source: 'molecules' as const },
|
|
||||||
{ dir: path.join(rootDir, 'src/config/pages/organisms'), source: 'organisms' as const },
|
|
||||||
{ dir: path.join(rootDir, 'src/config/pages/ui'), source: 'ui' as const },
|
|
||||||
{ dir: path.join(rootDir, 'src/config/pages/components'), source: 'custom' as const },
|
|
||||||
]
|
|
||||||
|
|
||||||
for (const { dir, source } of directories) {
|
|
||||||
const entries = await processDirectory(dir, source)
|
|
||||||
newEntries.push(...entries)
|
|
||||||
console.log(` Processed ${source}: ${entries.length} components`)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Merge with existing components (remove duplicates)
|
|
||||||
const existingTypes = new Set(registry.components.map(c => c.type))
|
|
||||||
const uniqueNewEntries = newEntries.filter(e => !existingTypes.has(e.type))
|
|
||||||
|
|
||||||
console.log(`\n New unique components: ${uniqueNewEntries.length}`)
|
|
||||||
console.log(` Skipped duplicates: ${newEntries.length - uniqueNewEntries.length}`)
|
|
||||||
|
|
||||||
// Add new components
|
|
||||||
registry.components.push(...uniqueNewEntries)
|
|
||||||
|
|
||||||
// Update statistics
|
|
||||||
const byCategory: Record<string, number> = {}
|
|
||||||
const bySource: Record<string, number> = {}
|
|
||||||
|
|
||||||
for (const component of registry.components) {
|
|
||||||
byCategory[component.category] = (byCategory[component.category] || 0) + 1
|
|
||||||
bySource[component.source] = (bySource[component.source] || 0) + 1
|
|
||||||
}
|
|
||||||
|
|
||||||
registry.statistics = {
|
|
||||||
total: registry.components.length,
|
|
||||||
supported: registry.components.filter(c => c.status === 'supported').length,
|
|
||||||
jsonCompatible: registry.components.filter(c => c.jsonCompatible).length,
|
|
||||||
byCategory,
|
|
||||||
bySource,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sort components by type
|
|
||||||
registry.components.sort((a, b) => a.type.localeCompare(b.type))
|
|
||||||
|
|
||||||
// Write updated registry
|
|
||||||
await fs.writeFile(registryPath, JSON.stringify(registry, null, 2) + '\n')
|
|
||||||
|
|
||||||
console.log(`\n✅ Registry updated successfully!`)
|
|
||||||
console.log(` Total components: ${registry.statistics.total}`)
|
|
||||||
console.log(` JSON compatible: ${registry.statistics.jsonCompatible}`)
|
|
||||||
console.log(`\n📊 By source:`)
|
|
||||||
for (const [source, count] of Object.entries(bySource)) {
|
|
||||||
console.log(` ${source.padEnd(12)}: ${count}`)
|
|
||||||
}
|
|
||||||
console.log(`\n📊 By category:`)
|
|
||||||
for (const [category, count] of Object.entries(byCategory)) {
|
|
||||||
console.log(` ${category.padEnd(12)}: ${count}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
updateRegistry().catch(console.error)
|
|
||||||
@@ -1,235 +0,0 @@
|
|||||||
import fs from 'node:fs/promises'
|
|
||||||
import path from 'node:path'
|
|
||||||
import { fileURLToPath, pathToFileURL } from 'node:url'
|
|
||||||
import * as PhosphorIcons from '@phosphor-icons/react'
|
|
||||||
import { JSONUIShowcase } from '../src/components/JSONUIShowcase'
|
|
||||||
|
|
||||||
type ComponentType = unknown
|
|
||||||
|
|
||||||
interface JsonRegistryEntry {
|
|
||||||
name?: string
|
|
||||||
type?: string
|
|
||||||
export?: string
|
|
||||||
source?: string
|
|
||||||
status?: string
|
|
||||||
wrapperRequired?: boolean
|
|
||||||
wrapperComponent?: string
|
|
||||||
wrapperFor?: string
|
|
||||||
load?: {
|
|
||||||
export?: string
|
|
||||||
}
|
|
||||||
deprecated?: unknown
|
|
||||||
}
|
|
||||||
|
|
||||||
interface JsonComponentRegistry {
|
|
||||||
components?: JsonRegistryEntry[]
|
|
||||||
}
|
|
||||||
|
|
||||||
const sourceAliases: Record<string, Record<string, string>> = {
|
|
||||||
atoms: {
|
|
||||||
PageHeader: 'BasicPageHeader',
|
|
||||||
SearchInput: 'BasicSearchInput',
|
|
||||||
},
|
|
||||||
molecules: {},
|
|
||||||
organisms: {},
|
|
||||||
ui: {
|
|
||||||
Chart: 'ChartContainer',
|
|
||||||
Resizable: 'ResizablePanelGroup',
|
|
||||||
},
|
|
||||||
wrappers: {},
|
|
||||||
}
|
|
||||||
|
|
||||||
const explicitComponentAllowlist: Record<string, ComponentType> = {
|
|
||||||
JSONUIShowcase,
|
|
||||||
}
|
|
||||||
|
|
||||||
const getRegistryEntryKey = (entry: JsonRegistryEntry): string | undefined =>
|
|
||||||
entry.name ?? entry.type
|
|
||||||
|
|
||||||
const getRegistryEntryExportName = (entry: JsonRegistryEntry): string | undefined =>
|
|
||||||
entry.load?.export ?? entry.export ?? getRegistryEntryKey(entry)
|
|
||||||
|
|
||||||
const buildComponentMapFromExports = (
|
|
||||||
exports: Record<string, unknown>
|
|
||||||
): Record<string, ComponentType> => {
|
|
||||||
return Object.entries(exports).reduce<Record<string, ComponentType>>((acc, [key, value]) => {
|
|
||||||
if (value && (typeof value === 'function' || typeof value === 'object')) {
|
|
||||||
acc[key] = value as ComponentType
|
|
||||||
}
|
|
||||||
return acc
|
|
||||||
}, {})
|
|
||||||
}
|
|
||||||
|
|
||||||
const buildComponentMapFromModules = (
|
|
||||||
modules: Record<string, unknown>
|
|
||||||
): Record<string, ComponentType> => {
|
|
||||||
return Object.values(modules).reduce<Record<string, ComponentType>>((acc, moduleExports) => {
|
|
||||||
if (!moduleExports || typeof moduleExports !== 'object') {
|
|
||||||
return acc
|
|
||||||
}
|
|
||||||
Object.entries(buildComponentMapFromExports(moduleExports as Record<string, unknown>)).forEach(
|
|
||||||
([key, component]) => {
|
|
||||||
acc[key] = component
|
|
||||||
}
|
|
||||||
)
|
|
||||||
return acc
|
|
||||||
}, {})
|
|
||||||
}
|
|
||||||
|
|
||||||
const listFiles = async (options: {
|
|
||||||
directory: string
|
|
||||||
extensions: string[]
|
|
||||||
recursive: boolean
|
|
||||||
}): Promise<string[]> => {
|
|
||||||
const { directory, extensions, recursive } = options
|
|
||||||
const entries = await fs.readdir(directory, { withFileTypes: true })
|
|
||||||
const files: string[] = []
|
|
||||||
|
|
||||||
await Promise.all(
|
|
||||||
entries.map(async (entry) => {
|
|
||||||
const fullPath = path.join(directory, entry.name)
|
|
||||||
if (entry.isDirectory()) {
|
|
||||||
if (recursive) {
|
|
||||||
const nested = await listFiles({ directory: fullPath, extensions, recursive })
|
|
||||||
files.push(...nested)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (extensions.includes(path.extname(entry.name))) {
|
|
||||||
files.push(fullPath)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
)
|
|
||||||
|
|
||||||
return files
|
|
||||||
}
|
|
||||||
|
|
||||||
const importModules = async (files: string[]): Promise<Record<string, unknown>> => {
|
|
||||||
const modules: Record<string, unknown> = {}
|
|
||||||
await Promise.all(
|
|
||||||
files.map(async (file) => {
|
|
||||||
const moduleExports = await import(pathToFileURL(file).href)
|
|
||||||
modules[file] = moduleExports
|
|
||||||
})
|
|
||||||
)
|
|
||||||
return modules
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateRegistry = async () => {
|
|
||||||
const scriptDir = path.dirname(fileURLToPath(import.meta.url))
|
|
||||||
const rootDir = path.resolve(scriptDir, '..')
|
|
||||||
const registryPath = path.join(rootDir, 'json-components-registry.json')
|
|
||||||
|
|
||||||
const registryRaw = await fs.readFile(registryPath, 'utf8')
|
|
||||||
const registry = JSON.parse(registryRaw) as JsonComponentRegistry
|
|
||||||
const registryEntries = registry.components ?? []
|
|
||||||
const registryEntryByType = new Map(
|
|
||||||
registryEntries
|
|
||||||
.map((entry) => {
|
|
||||||
const entryKey = getRegistryEntryKey(entry)
|
|
||||||
return entryKey ? [entryKey, entry] : null
|
|
||||||
})
|
|
||||||
.filter((entry): entry is [string, JsonRegistryEntry] => Boolean(entry))
|
|
||||||
)
|
|
||||||
|
|
||||||
const sourceConfigs = [
|
|
||||||
{
|
|
||||||
source: 'atoms',
|
|
||||||
directory: path.join(rootDir, 'src/components/atoms'),
|
|
||||||
extensions: ['.tsx'],
|
|
||||||
recursive: false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
source: 'molecules',
|
|
||||||
directory: path.join(rootDir, 'src/components/molecules'),
|
|
||||||
extensions: ['.tsx'],
|
|
||||||
recursive: false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
source: 'organisms',
|
|
||||||
directory: path.join(rootDir, 'src/components/organisms'),
|
|
||||||
extensions: ['.tsx'],
|
|
||||||
recursive: false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
source: 'ui',
|
|
||||||
directory: path.join(rootDir, 'src/components/ui'),
|
|
||||||
extensions: ['.ts', '.tsx'],
|
|
||||||
recursive: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
source: 'wrappers',
|
|
||||||
directory: path.join(rootDir, 'src/lib/json-ui/wrappers'),
|
|
||||||
extensions: ['.tsx'],
|
|
||||||
recursive: false,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
const componentMaps: Record<string, Record<string, ComponentType>> = {}
|
|
||||||
await Promise.all(
|
|
||||||
sourceConfigs.map(async (config) => {
|
|
||||||
const files = await listFiles({
|
|
||||||
directory: config.directory,
|
|
||||||
extensions: config.extensions,
|
|
||||||
recursive: config.recursive,
|
|
||||||
})
|
|
||||||
const modules = await importModules(files)
|
|
||||||
componentMaps[config.source] = buildComponentMapFromModules(modules)
|
|
||||||
})
|
|
||||||
)
|
|
||||||
|
|
||||||
componentMaps.icons = buildComponentMapFromExports(PhosphorIcons)
|
|
||||||
|
|
||||||
const errors: string[] = []
|
|
||||||
|
|
||||||
registryEntries.forEach((entry) => {
|
|
||||||
const entryKey = getRegistryEntryKey(entry)
|
|
||||||
const entryExportName = getRegistryEntryExportName(entry)
|
|
||||||
|
|
||||||
if (!entryKey || !entryExportName) {
|
|
||||||
errors.push(`Entry missing name/type/export: ${JSON.stringify(entry)}`)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const source = entry.source
|
|
||||||
if (!source || !componentMaps[source]) {
|
|
||||||
errors.push(`${entryKey}: unknown source "${source ?? 'missing'}"`)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const aliasName = sourceAliases[source]?.[entryKey]
|
|
||||||
const component =
|
|
||||||
componentMaps[source][entryExportName] ??
|
|
||||||
(aliasName ? componentMaps[source][aliasName] : undefined) ??
|
|
||||||
explicitComponentAllowlist[entryKey]
|
|
||||||
|
|
||||||
if (!component) {
|
|
||||||
const aliasNote = aliasName ? ` (alias: ${aliasName})` : ''
|
|
||||||
errors.push(
|
|
||||||
`${entryKey} (${source}) did not resolve export "${entryExportName}"${aliasNote}`
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (entry.wrapperRequired) {
|
|
||||||
if (!entry.wrapperComponent) {
|
|
||||||
errors.push(`${entryKey} (${source}) requires a wrapperComponent but none is defined`)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (!registryEntryByType.has(entry.wrapperComponent)) {
|
|
||||||
errors.push(
|
|
||||||
`${entryKey} (${source}) references missing wrapperComponent ${entry.wrapperComponent}`
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
if (errors.length > 0) {
|
|
||||||
console.error('❌ JSON component registry export validation failed:')
|
|
||||||
errors.forEach((error) => console.error(`- ${error}`))
|
|
||||||
process.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('✅ JSON component registry exports are valid.')
|
|
||||||
}
|
|
||||||
|
|
||||||
await validateRegistry()
|
|
||||||
@@ -4,7 +4,7 @@ const path = require('path')
|
|||||||
const rootDir = path.resolve(__dirname, '..')
|
const rootDir = path.resolve(__dirname, '..')
|
||||||
const registryPath = path.join(rootDir, 'json-components-registry.json')
|
const registryPath = path.join(rootDir, 'json-components-registry.json')
|
||||||
const definitionsPath = path.join(rootDir, 'src/lib/component-definitions.json')
|
const definitionsPath = path.join(rootDir, 'src/lib/component-definitions.json')
|
||||||
const componentTypesPath = path.join(rootDir, 'src/types/json-ui-component-types.ts')
|
const componentTypesPath = path.join(rootDir, 'src/types/json-ui.ts')
|
||||||
const uiRegistryPath = path.join(rootDir, 'src/lib/json-ui/component-registry.ts')
|
const uiRegistryPath = path.join(rootDir, 'src/lib/json-ui/component-registry.ts')
|
||||||
const atomIndexPath = path.join(rootDir, 'src/components/atoms/index.ts')
|
const atomIndexPath = path.join(rootDir, 'src/components/atoms/index.ts')
|
||||||
const moleculeIndexPath = path.join(rootDir, 'src/components/molecules/index.ts')
|
const moleculeIndexPath = path.join(rootDir, 'src/components/molecules/index.ts')
|
||||||
@@ -21,10 +21,16 @@ const componentDefinitions = readJson(definitionsPath)
|
|||||||
const definitionTypes = new Set(componentDefinitions.map((def) => def.type))
|
const definitionTypes = new Set(componentDefinitions.map((def) => def.type))
|
||||||
|
|
||||||
const componentTypesContent = readText(componentTypesPath)
|
const componentTypesContent = readText(componentTypesPath)
|
||||||
|
const componentTypesStart = componentTypesContent.indexOf('export type ComponentType')
|
||||||
|
const componentTypesEnd = componentTypesContent.indexOf('export type ActionType')
|
||||||
|
if (componentTypesStart === -1 || componentTypesEnd === -1) {
|
||||||
|
throw new Error('Unable to locate ComponentType union in src/types/json-ui.ts')
|
||||||
|
}
|
||||||
|
const componentTypesBlock = componentTypesContent.slice(componentTypesStart, componentTypesEnd)
|
||||||
const componentTypeSet = new Set()
|
const componentTypeSet = new Set()
|
||||||
const componentTypeRegex = /"([^"]+)"/g
|
const componentTypeRegex = /'([^']+)'/g
|
||||||
let match
|
let match
|
||||||
while ((match = componentTypeRegex.exec(componentTypesContent)) !== null) {
|
while ((match = componentTypeRegex.exec(componentTypesBlock)) !== null) {
|
||||||
componentTypeSet.add(match[1])
|
componentTypeSet.add(match[1])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,8 +3,7 @@ import { useToggle, useDialog } from '@/hooks/ui'
|
|||||||
import { useKV } from '@/hooks/use-kv'
|
import { useKV } from '@/hooks/use-kv'
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
import { Card, CardHeader, CardTitle, CardContent } from '@/components/ui/card'
|
import { Card, CardHeader, CardTitle, CardContent } from '@/components/ui/card'
|
||||||
import { SearchInput } from '@/components/molecules'
|
import { SearchInput, DataCard, ActionBar } from '@/components/molecules'
|
||||||
import { DataCard } from '@/components/atoms/json-ui'
|
|
||||||
import { Grid, Heading, StatusBadge } from '@/components/atoms'
|
import { Grid, Heading, StatusBadge } from '@/components/atoms'
|
||||||
import { Plus, Trash, Eye } from '@phosphor-icons/react'
|
import { Plus, Trash, Eye } from '@phosphor-icons/react'
|
||||||
import { Dialog, DialogContent, DialogHeader, DialogTitle } from '@/components/ui/dialog'
|
import { Dialog, DialogContent, DialogHeader, DialogTitle } from '@/components/ui/dialog'
|
||||||
@@ -65,31 +64,28 @@ export function AtomicComponentDemo() {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Grid cols={3} gap={4}>
|
<Grid cols={3} gap={4}>
|
||||||
<DataCard title="Total Tasks" icon="list" gradient="from-blue-500/10 to-blue-500/5">
|
<DataCard title="Total Tasks" value={stats.total} />
|
||||||
<div className="text-3xl font-bold">{stats.total}</div>
|
<DataCard title="Active" value={stats.active} />
|
||||||
</DataCard>
|
<DataCard title="Completed" value={stats.completed} />
|
||||||
<DataCard title="Active" icon="clock" gradient="from-amber-500/10 to-amber-500/5">
|
|
||||||
<div className="text-3xl font-bold">{stats.active}</div>
|
|
||||||
</DataCard>
|
|
||||||
<DataCard title="Completed" icon="check" gradient="from-green-500/10 to-green-500/5">
|
|
||||||
<div className="text-3xl font-bold">{stats.completed}</div>
|
|
||||||
</DataCard>
|
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
{/* ActionBar replaced with inline buttons */}
|
<ActionBar
|
||||||
<div className="flex items-center justify-between">
|
title="Tasks"
|
||||||
<Heading level={3}>Tasks</Heading>
|
actions={[
|
||||||
<div className="flex gap-2">
|
{
|
||||||
<Button onClick={addDialog.open} size="sm">
|
label: 'Add Task',
|
||||||
<Plus size={16} className="mr-2" />
|
icon: <Plus size={16} />,
|
||||||
Add Task
|
onClick: addDialog.open,
|
||||||
</Button>
|
variant: 'default',
|
||||||
<Button onClick={showCompleted.toggle} variant="outline" size="sm">
|
},
|
||||||
<Eye size={16} className="mr-2" />
|
{
|
||||||
{showCompleted.value ? 'Hide Completed' : 'Show Completed'}
|
label: showCompleted.value ? 'Hide Completed' : 'Show Completed',
|
||||||
</Button>
|
icon: <Eye size={16} />,
|
||||||
</div>
|
onClick: showCompleted.toggle,
|
||||||
</div>
|
variant: 'outline',
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
|
||||||
<SearchInput
|
<SearchInput
|
||||||
value={query}
|
value={query}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { useState } from 'react'
|
import { useState } from 'react'
|
||||||
import { DataSourceManager, ComponentBindingDialog } from '@/lib/json-ui/json-components'
|
import { DataSourceManager } from '@/components/organisms/DataSourceManager'
|
||||||
|
import { ComponentBindingDialog } from '@/components/molecules/ComponentBindingDialog'
|
||||||
import { DataSource, UIComponent } from '@/types/json-ui'
|
import { DataSource, UIComponent } from '@/types/json-ui'
|
||||||
import { DataBindingHeader } from '@/components/data-binding-designer/DataBindingHeader'
|
import { DataBindingHeader } from '@/components/data-binding-designer/DataBindingHeader'
|
||||||
import { ComponentBindingsCard } from '@/components/data-binding-designer/ComponentBindingsCard'
|
import { ComponentBindingsCard } from '@/components/data-binding-designer/ComponentBindingsCard'
|
||||||
|
|||||||
@@ -15,16 +15,3 @@ export const PRIORITY_COLORS = {
|
|||||||
medium: 'border-amber-400/60 bg-amber-50/80 dark:bg-amber-950/40',
|
medium: 'border-amber-400/60 bg-amber-50/80 dark:bg-amber-950/40',
|
||||||
high: 'border-red-400/60 bg-red-50/80 dark:bg-red-950/40',
|
high: 'border-red-400/60 bg-red-50/80 dark:bg-red-950/40',
|
||||||
}
|
}
|
||||||
|
|
||||||
// Missing exports for GROUP_COLORS, CATEGORIES, PRIORITIES, STATUSES
|
|
||||||
export const GROUP_COLORS = {
|
|
||||||
default: '#a78bfa',
|
|
||||||
primary: '#60a5fa',
|
|
||||||
success: '#34d399',
|
|
||||||
warning: '#fbbf24',
|
|
||||||
danger: '#f87171',
|
|
||||||
}
|
|
||||||
|
|
||||||
export const CATEGORIES = ['feature', 'enhancement', 'bug', 'documentation', 'other'] as const
|
|
||||||
export const PRIORITIES = ['low', 'medium', 'high'] as const
|
|
||||||
export const STATUSES = ['idea', 'planned', 'in-progress', 'completed'] as const
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { PageRenderer } from '@/lib/json-ui/page-renderer'
|
import { PageRenderer } from '@/lib/json-ui/page-renderer'
|
||||||
|
import { LoadingFallback } from '@/components/molecules'
|
||||||
import { useSchemaLoader } from '@/hooks/use-schema-loader'
|
import { useSchemaLoader } from '@/hooks/use-schema-loader'
|
||||||
|
|
||||||
interface JSONSchemaPageLoaderProps {
|
interface JSONSchemaPageLoaderProps {
|
||||||
@@ -11,14 +12,7 @@ export function JSONSchemaPageLoader({ schemaPath, data, functions }: JSONSchema
|
|||||||
const { schema, loading, error } = useSchemaLoader(schemaPath)
|
const { schema, loading, error } = useSchemaLoader(schemaPath)
|
||||||
|
|
||||||
if (loading) {
|
if (loading) {
|
||||||
return (
|
return <LoadingFallback message={`Loading ${schemaPath}...`} />
|
||||||
<div className="flex items-center justify-center h-full w-full">
|
|
||||||
<div className="flex flex-col items-center gap-3">
|
|
||||||
<div className="w-8 h-8 border-2 border-primary border-t-transparent rounded-full animate-spin" />
|
|
||||||
<p className="text-sm text-muted-foreground">Loading {schemaPath}...</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (error || !schema) {
|
if (error || !schema) {
|
||||||
|
|||||||
@@ -1,11 +1,24 @@
|
|||||||
import { useMemo, useState } from 'react'
|
import { useMemo, useState } from 'react'
|
||||||
import showcaseCopy from '@/config/ui-examples/showcase.json'
|
import showcaseCopy from '@/config/ui-examples/showcase.json'
|
||||||
|
import dashboardExample from '@/config/ui-examples/dashboard.json'
|
||||||
|
import formExample from '@/config/ui-examples/form.json'
|
||||||
|
import tableExample from '@/config/ui-examples/table.json'
|
||||||
|
import listTableTimelineExample from '@/config/ui-examples/list-table-timeline.json'
|
||||||
|
import settingsExample from '@/config/ui-examples/settings.json'
|
||||||
import { FileCode, ChartBar, ListBullets, Table, Gear, Clock } from '@phosphor-icons/react'
|
import { FileCode, ChartBar, ListBullets, Table, Gear, Clock } from '@phosphor-icons/react'
|
||||||
import { ShowcaseHeader } from '@/components/json-ui-showcase/ShowcaseHeader'
|
import { ShowcaseHeader } from '@/components/json-ui-showcase/ShowcaseHeader'
|
||||||
import { ShowcaseTabs } from '@/components/json-ui-showcase/ShowcaseTabs'
|
import { ShowcaseTabs } from '@/components/json-ui-showcase/ShowcaseTabs'
|
||||||
import { ShowcaseFooter } from '@/components/json-ui-showcase/ShowcaseFooter'
|
import { ShowcaseFooter } from '@/components/json-ui-showcase/ShowcaseFooter'
|
||||||
import { ShowcaseExample } from '@/components/json-ui-showcase/types'
|
import { ShowcaseExample } from '@/components/json-ui-showcase/types'
|
||||||
|
|
||||||
|
const exampleConfigs = {
|
||||||
|
dashboard: dashboardExample,
|
||||||
|
form: formExample,
|
||||||
|
table: tableExample,
|
||||||
|
'list-table-timeline': listTableTimelineExample,
|
||||||
|
settings: settingsExample,
|
||||||
|
}
|
||||||
|
|
||||||
const exampleIcons = {
|
const exampleIcons = {
|
||||||
ChartBar,
|
ChartBar,
|
||||||
ListBullets,
|
ListBullets,
|
||||||
@@ -14,22 +27,14 @@ const exampleIcons = {
|
|||||||
Gear,
|
Gear,
|
||||||
}
|
}
|
||||||
|
|
||||||
const configModules = import.meta.glob('/src/config/ui-examples/*.json', { eager: true })
|
|
||||||
|
|
||||||
const resolveExampleConfig = (configPath: string) => {
|
|
||||||
const moduleEntry = configModules[configPath] as { default: ShowcaseExample['config'] } | undefined
|
|
||||||
|
|
||||||
return moduleEntry?.default ?? {}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function JSONUIShowcase() {
|
export function JSONUIShowcase() {
|
||||||
const [selectedExample, setSelectedExample] = useState(showcaseCopy.defaultExampleKey)
|
const [selectedExample, setSelectedExample] = useState(showcaseCopy.defaultExampleKey)
|
||||||
const [showJSON, setShowJSON] = useState(false)
|
const [showJSON, setShowJSON] = useState(false)
|
||||||
|
|
||||||
const examples = useMemo<ShowcaseExample[]>(() => {
|
const examples = useMemo<ShowcaseExample[]>(() => {
|
||||||
return showcaseCopy.examples.map((example) => {
|
return showcaseCopy.examples.map((example) => {
|
||||||
const icon = exampleIcons[example.iconId as keyof typeof exampleIcons] || FileCode
|
const icon = exampleIcons[example.icon as keyof typeof exampleIcons] || FileCode
|
||||||
const config = resolveExampleConfig(example.configPath)
|
const config = exampleConfigs[example.configKey as keyof typeof exampleConfigs]
|
||||||
|
|
||||||
return {
|
return {
|
||||||
key: example.key,
|
key: example.key,
|
||||||
|
|||||||
@@ -45,12 +45,11 @@ function getCompletionMessage(score: number): string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function ProjectDashboard(props: ProjectDashboardProps) {
|
export function ProjectDashboard(props: ProjectDashboardProps) {
|
||||||
const completionMetrics = calculateCompletionScore(props)
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<JSONPageRenderer
|
<JSONPageRenderer
|
||||||
schema={dashboardSchema as any}
|
schema={dashboardSchema as any}
|
||||||
data={{ ...props, ...completionMetrics }}
|
data={props}
|
||||||
|
functions={{ calculateCompletionScore }}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import {
|
import {
|
||||||
DatePicker,
|
DatePicker,
|
||||||
|
FilterInput,
|
||||||
Heading,
|
Heading,
|
||||||
RangeSlider,
|
RangeSlider,
|
||||||
Rating,
|
Rating,
|
||||||
@@ -10,7 +11,6 @@ import {
|
|||||||
Switch,
|
Switch,
|
||||||
Text,
|
Text,
|
||||||
} from '@/components/atoms'
|
} from '@/components/atoms'
|
||||||
import { FilterInput } from '@/lib/json-ui/json-components'
|
|
||||||
|
|
||||||
type FormControlsSectionContent =
|
type FormControlsSectionContent =
|
||||||
(typeof import('@/data/atomic-library-showcase.json'))['sections']['formControls']
|
(typeof import('@/data/atomic-library-showcase.json'))['sections']['formControls']
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import displayCopy from '@/data/atomic-showcase/display.json'
|
import displayCopy from '@/data/atomic-showcase/display.json'
|
||||||
import {
|
import {
|
||||||
|
Accordion,
|
||||||
Avatar,
|
Avatar,
|
||||||
BreadcrumbNav,
|
BreadcrumbNav,
|
||||||
Card,
|
Card,
|
||||||
@@ -15,7 +16,6 @@ import {
|
|||||||
Timeline,
|
Timeline,
|
||||||
Timestamp,
|
Timestamp,
|
||||||
} from '@/components/atoms'
|
} from '@/components/atoms'
|
||||||
import { Accordion } from '@/lib/json-ui/json-components'
|
|
||||||
|
|
||||||
type DisplayTabProps = {
|
type DisplayTabProps = {
|
||||||
ratingValue: number
|
ratingValue: number
|
||||||
|
|||||||
@@ -5,9 +5,13 @@ import {
|
|||||||
BasicSearchInput,
|
BasicSearchInput,
|
||||||
Card,
|
Card,
|
||||||
Checkbox,
|
Checkbox,
|
||||||
|
CopyButton,
|
||||||
Divider,
|
Divider,
|
||||||
|
FileUpload,
|
||||||
Heading,
|
Heading,
|
||||||
IconButton,
|
IconButton,
|
||||||
|
Input,
|
||||||
|
PasswordInput,
|
||||||
RadioGroup,
|
RadioGroup,
|
||||||
Select,
|
Select,
|
||||||
Slider,
|
Slider,
|
||||||
@@ -15,12 +19,6 @@ import {
|
|||||||
TextArea,
|
TextArea,
|
||||||
Toggle,
|
Toggle,
|
||||||
} from '@/components/atoms'
|
} from '@/components/atoms'
|
||||||
import {
|
|
||||||
Input,
|
|
||||||
CopyButton,
|
|
||||||
FileUpload,
|
|
||||||
PasswordInput,
|
|
||||||
} from '@/lib/json-ui/json-components'
|
|
||||||
|
|
||||||
type FormsTabProps = {
|
type FormsTabProps = {
|
||||||
checkboxValue: boolean
|
checkboxValue: boolean
|
||||||
|
|||||||
68
src/components/atoms/Accordion.tsx
Normal file
68
src/components/atoms/Accordion.tsx
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { CaretDown } from '@phosphor-icons/react'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
interface AccordionItem {
|
||||||
|
id: string
|
||||||
|
title: string
|
||||||
|
content: React.ReactNode
|
||||||
|
disabled?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AccordionProps {
|
||||||
|
items: AccordionItem[]
|
||||||
|
type?: 'single' | 'multiple'
|
||||||
|
defaultOpen?: string[]
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Accordion({ items, type = 'single', defaultOpen = [], className }: AccordionProps) {
|
||||||
|
const [openItems, setOpenItems] = useState<string[]>(defaultOpen)
|
||||||
|
|
||||||
|
const toggleItem = (id: string) => {
|
||||||
|
if (type === 'single') {
|
||||||
|
setOpenItems(openItems.includes(id) ? [] : [id])
|
||||||
|
} else {
|
||||||
|
setOpenItems(
|
||||||
|
openItems.includes(id)
|
||||||
|
? openItems.filter((item) => item !== id)
|
||||||
|
: [...openItems, id]
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('space-y-2', className)}>
|
||||||
|
{items.map((item) => {
|
||||||
|
const isOpen = openItems.includes(item.id)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div key={item.id} className="border border-border rounded-lg overflow-hidden">
|
||||||
|
<button
|
||||||
|
onClick={() => !item.disabled && toggleItem(item.id)}
|
||||||
|
disabled={item.disabled}
|
||||||
|
className={cn(
|
||||||
|
'w-full flex items-center justify-between p-4 bg-card text-card-foreground font-medium transition-colors',
|
||||||
|
'hover:bg-accent hover:text-accent-foreground',
|
||||||
|
item.disabled && 'opacity-50 cursor-not-allowed'
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<span>{item.title}</span>
|
||||||
|
<CaretDown
|
||||||
|
className={cn(
|
||||||
|
'w-5 h-5 transition-transform',
|
||||||
|
isOpen && 'rotate-180'
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
{isOpen && (
|
||||||
|
<div className="p-4 bg-card border-t border-border animate-in slide-in-from-top-2">
|
||||||
|
{item.content}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
56
src/components/atoms/CopyButton.tsx
Normal file
56
src/components/atoms/CopyButton.tsx
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
import { Copy, Check } from '@phosphor-icons/react'
|
||||||
|
import { useState } from 'react'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
interface CopyButtonProps {
|
||||||
|
text: string
|
||||||
|
size?: 'sm' | 'md' | 'lg'
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CopyButton({ text, size = 'md', className }: CopyButtonProps) {
|
||||||
|
const [copied, setCopied] = useState(false)
|
||||||
|
|
||||||
|
const handleCopy = async () => {
|
||||||
|
try {
|
||||||
|
await navigator.clipboard.writeText(text)
|
||||||
|
setCopied(true)
|
||||||
|
setTimeout(() => setCopied(false), 2000)
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to copy:', error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const sizeStyles = {
|
||||||
|
sm: 'p-1',
|
||||||
|
md: 'p-2',
|
||||||
|
lg: 'p-3',
|
||||||
|
}
|
||||||
|
|
||||||
|
const iconSize = {
|
||||||
|
sm: 12,
|
||||||
|
md: 16,
|
||||||
|
lg: 20,
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
onClick={handleCopy}
|
||||||
|
className={cn(
|
||||||
|
'rounded-md transition-colors',
|
||||||
|
copied
|
||||||
|
? 'bg-accent text-accent-foreground'
|
||||||
|
: 'bg-muted text-muted-foreground hover:bg-accent hover:text-accent-foreground',
|
||||||
|
sizeStyles[size],
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
aria-label={copied ? 'Copied' : 'Copy to clipboard'}
|
||||||
|
>
|
||||||
|
{copied ? (
|
||||||
|
<Check size={iconSize[size]} weight="bold" />
|
||||||
|
) : (
|
||||||
|
<Copy size={iconSize[size]} />
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { Badge } from '@/components/ui/badge'
|
import { Badge } from '@/components/ui/badge'
|
||||||
import { DataSourceType } from '@/types/json-ui'
|
import { DataSourceType } from '@/types/json-ui'
|
||||||
import { Database, File } from '@phosphor-icons/react'
|
import { Database, Function, File } from '@phosphor-icons/react'
|
||||||
|
|
||||||
interface DataSourceBadgeProps {
|
interface DataSourceBadgeProps {
|
||||||
type: DataSourceType
|
type: DataSourceType
|
||||||
@@ -13,6 +13,11 @@ const dataSourceConfig = {
|
|||||||
label: 'KV Storage',
|
label: 'KV Storage',
|
||||||
className: 'bg-accent/20 text-accent border-accent/30'
|
className: 'bg-accent/20 text-accent border-accent/30'
|
||||||
},
|
},
|
||||||
|
computed: {
|
||||||
|
icon: Function,
|
||||||
|
label: 'Computed',
|
||||||
|
className: 'bg-primary/20 text-primary border-primary/30'
|
||||||
|
},
|
||||||
static: {
|
static: {
|
||||||
icon: File,
|
icon: File,
|
||||||
label: 'Static',
|
label: 'Static',
|
||||||
|
|||||||
131
src/components/atoms/FileUpload.tsx
Normal file
131
src/components/atoms/FileUpload.tsx
Normal file
@@ -0,0 +1,131 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { UploadSimple, X } from '@phosphor-icons/react'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
interface FileUploadProps {
|
||||||
|
accept?: string
|
||||||
|
multiple?: boolean
|
||||||
|
maxSize?: number
|
||||||
|
onFilesSelected: (files: File[]) => void
|
||||||
|
disabled?: boolean
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FileUpload({
|
||||||
|
accept,
|
||||||
|
multiple = false,
|
||||||
|
maxSize,
|
||||||
|
onFilesSelected,
|
||||||
|
disabled = false,
|
||||||
|
className
|
||||||
|
}: FileUploadProps) {
|
||||||
|
const [isDragging, setIsDragging] = useState(false)
|
||||||
|
const [selectedFiles, setSelectedFiles] = useState<File[]>([])
|
||||||
|
|
||||||
|
const handleFiles = (files: FileList | null) => {
|
||||||
|
if (!files) return
|
||||||
|
|
||||||
|
const fileArray = Array.from(files)
|
||||||
|
const validFiles = fileArray.filter(file => {
|
||||||
|
if (maxSize && file.size > maxSize) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
|
||||||
|
setSelectedFiles(validFiles)
|
||||||
|
onFilesSelected(validFiles)
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDrop = (e: React.DragEvent) => {
|
||||||
|
e.preventDefault()
|
||||||
|
setIsDragging(false)
|
||||||
|
if (!disabled) {
|
||||||
|
handleFiles(e.dataTransfer.files)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDragOver = (e: React.DragEvent) => {
|
||||||
|
e.preventDefault()
|
||||||
|
if (!disabled) {
|
||||||
|
setIsDragging(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDragLeave = () => {
|
||||||
|
setIsDragging(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
const removeFile = (index: number) => {
|
||||||
|
const newFiles = selectedFiles.filter((_, i) => i !== index)
|
||||||
|
setSelectedFiles(newFiles)
|
||||||
|
onFilesSelected(newFiles)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('w-full', className)}>
|
||||||
|
<label
|
||||||
|
onDrop={handleDrop}
|
||||||
|
onDragOver={handleDragOver}
|
||||||
|
onDragLeave={handleDragLeave}
|
||||||
|
className={cn(
|
||||||
|
'flex flex-col items-center justify-center w-full h-32 border-2 border-dashed rounded-lg cursor-pointer transition-colors',
|
||||||
|
isDragging && 'border-primary bg-primary/5',
|
||||||
|
!isDragging && 'border-border bg-muted/30 hover:bg-muted/50',
|
||||||
|
disabled && 'opacity-50 cursor-not-allowed'
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="flex flex-col items-center justify-center gap-2">
|
||||||
|
<UploadSimple className="w-8 h-8 text-muted-foreground" />
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
<span className="font-medium">Click to upload</span> or drag and drop
|
||||||
|
</p>
|
||||||
|
{accept && (
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
{accept.split(',').join(', ')}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
{maxSize && (
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
Max size: {(maxSize / 1024 / 1024).toFixed(1)}MB
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<input
|
||||||
|
type="file"
|
||||||
|
accept={accept}
|
||||||
|
multiple={multiple}
|
||||||
|
onChange={(e) => handleFiles(e.target.files)}
|
||||||
|
disabled={disabled}
|
||||||
|
className="hidden"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
{selectedFiles.length > 0 && (
|
||||||
|
<div className="mt-4 space-y-2">
|
||||||
|
{selectedFiles.map((file, index) => (
|
||||||
|
<div
|
||||||
|
key={index}
|
||||||
|
className="flex items-center justify-between p-3 bg-muted rounded-lg"
|
||||||
|
>
|
||||||
|
<div className="flex-1 min-w-0">
|
||||||
|
<p className="text-sm font-medium truncate">{file.name}</p>
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
{(file.size / 1024).toFixed(1)} KB
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => removeFile(index)}
|
||||||
|
className="ml-2 p-1 hover:bg-background rounded transition-colors"
|
||||||
|
aria-label="Remove file"
|
||||||
|
>
|
||||||
|
<X className="w-4 h-4" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
49
src/components/atoms/FilterInput.tsx
Normal file
49
src/components/atoms/FilterInput.tsx
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
import { Input } from '@/components/ui/input'
|
||||||
|
import { MagnifyingGlass, X } from '@phosphor-icons/react'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
import { useState } from 'react'
|
||||||
|
|
||||||
|
interface FilterInputProps {
|
||||||
|
value: string
|
||||||
|
onChange: (value: string) => void
|
||||||
|
placeholder?: string
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FilterInput({
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
placeholder = 'Filter...',
|
||||||
|
className,
|
||||||
|
}: FilterInputProps) {
|
||||||
|
const [isFocused, setIsFocused] = useState(false)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cn('relative', className)}>
|
||||||
|
<MagnifyingGlass
|
||||||
|
className={cn(
|
||||||
|
'absolute left-3 top-1/2 -translate-y-1/2 transition-colors',
|
||||||
|
isFocused ? 'text-primary' : 'text-muted-foreground'
|
||||||
|
)}
|
||||||
|
size={16}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
value={value}
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
placeholder={placeholder}
|
||||||
|
onFocus={() => setIsFocused(true)}
|
||||||
|
onBlur={() => setIsFocused(false)}
|
||||||
|
className="pl-9 pr-9"
|
||||||
|
/>
|
||||||
|
{value && (
|
||||||
|
<button
|
||||||
|
onClick={() => onChange('')}
|
||||||
|
className="absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground transition-colors"
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<X size={16} />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
67
src/components/atoms/Image.tsx
Normal file
67
src/components/atoms/Image.tsx
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
interface ImageProps {
|
||||||
|
src: string
|
||||||
|
alt: string
|
||||||
|
width?: number | string
|
||||||
|
height?: number | string
|
||||||
|
fit?: 'cover' | 'contain' | 'fill' | 'none' | 'scale-down'
|
||||||
|
fallback?: string
|
||||||
|
className?: string
|
||||||
|
onLoad?: () => void
|
||||||
|
onError?: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Image({
|
||||||
|
src,
|
||||||
|
alt,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
fit = 'cover',
|
||||||
|
fallback,
|
||||||
|
className,
|
||||||
|
onLoad,
|
||||||
|
onError
|
||||||
|
}: ImageProps) {
|
||||||
|
const [error, setError] = useState(false)
|
||||||
|
const [loading, setLoading] = useState(true)
|
||||||
|
|
||||||
|
const handleLoad = () => {
|
||||||
|
setLoading(false)
|
||||||
|
onLoad?.()
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleError = () => {
|
||||||
|
setError(true)
|
||||||
|
setLoading(false)
|
||||||
|
onError?.()
|
||||||
|
}
|
||||||
|
|
||||||
|
const imgSrc = error && fallback ? fallback : src
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={cn('relative overflow-hidden', className)}
|
||||||
|
style={{
|
||||||
|
width: typeof width === 'number' ? `${width}px` : width,
|
||||||
|
height: typeof height === 'number' ? `${height}px` : height,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{loading && (
|
||||||
|
<div className="absolute inset-0 bg-muted animate-pulse" />
|
||||||
|
)}
|
||||||
|
<img
|
||||||
|
src={imgSrc}
|
||||||
|
alt={alt}
|
||||||
|
onLoad={handleLoad}
|
||||||
|
onError={handleError}
|
||||||
|
className={cn(
|
||||||
|
'w-full h-full transition-opacity',
|
||||||
|
loading ? 'opacity-0' : 'opacity-100',
|
||||||
|
`object-${fit}`
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
58
src/components/atoms/Input.tsx
Normal file
58
src/components/atoms/Input.tsx
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
import { forwardRef } from 'react'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
interface InputProps extends React.InputHTMLAttributes<HTMLInputElement> {
|
||||||
|
error?: boolean
|
||||||
|
helperText?: string
|
||||||
|
label?: string
|
||||||
|
leftIcon?: React.ReactNode
|
||||||
|
rightIcon?: React.ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Input = forwardRef<HTMLInputElement, InputProps>(
|
||||||
|
({ error, helperText, label, leftIcon, rightIcon, className, ...props }, ref) => {
|
||||||
|
return (
|
||||||
|
<div className="w-full">
|
||||||
|
{label && (
|
||||||
|
<label className="block text-sm font-medium mb-1.5 text-foreground">
|
||||||
|
{label}
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
<div className="relative">
|
||||||
|
{leftIcon && (
|
||||||
|
<div className="absolute left-3 top-1/2 -translate-y-1/2 text-muted-foreground">
|
||||||
|
{leftIcon}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<input
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
'flex h-10 w-full rounded-md border bg-background px-3 py-2 text-sm',
|
||||||
|
'placeholder:text-muted-foreground',
|
||||||
|
'focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring',
|
||||||
|
'disabled:cursor-not-allowed disabled:opacity-50',
|
||||||
|
'transition-colors',
|
||||||
|
error ? 'border-destructive focus-visible:ring-destructive' : 'border-input',
|
||||||
|
leftIcon && 'pl-10',
|
||||||
|
rightIcon && 'pr-10',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
{rightIcon && (
|
||||||
|
<div className="absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground">
|
||||||
|
{rightIcon}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{helperText && (
|
||||||
|
<p className={cn('text-xs mt-1.5', error ? 'text-destructive' : 'text-muted-foreground')}>
|
||||||
|
{helperText}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
Input.displayName = 'Input'
|
||||||
51
src/components/atoms/PasswordInput.tsx
Normal file
51
src/components/atoms/PasswordInput.tsx
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import { Eye, EyeSlash } from '@phosphor-icons/react'
|
||||||
|
import { useState } from 'react'
|
||||||
|
import { Input } from './Input'
|
||||||
|
|
||||||
|
interface PasswordInputProps {
|
||||||
|
value: string
|
||||||
|
onChange: (value: string) => void
|
||||||
|
label?: string
|
||||||
|
error?: boolean
|
||||||
|
helperText?: string
|
||||||
|
placeholder?: string
|
||||||
|
disabled?: boolean
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function PasswordInput({
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
label,
|
||||||
|
error,
|
||||||
|
helperText,
|
||||||
|
placeholder = 'Enter password',
|
||||||
|
disabled,
|
||||||
|
className,
|
||||||
|
}: PasswordInputProps) {
|
||||||
|
const [showPassword, setShowPassword] = useState(false)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Input
|
||||||
|
type={showPassword ? 'text' : 'password'}
|
||||||
|
value={value}
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
label={label}
|
||||||
|
error={error}
|
||||||
|
helperText={helperText}
|
||||||
|
placeholder={placeholder}
|
||||||
|
disabled={disabled}
|
||||||
|
className={className}
|
||||||
|
rightIcon={
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => setShowPassword(!showPassword)}
|
||||||
|
className="text-muted-foreground hover:text-foreground transition-colors"
|
||||||
|
aria-label={showPassword ? 'Hide password' : 'Show password'}
|
||||||
|
>
|
||||||
|
{showPassword ? <EyeSlash size={18} /> : <Eye size={18} />}
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
59
src/components/atoms/Popover.tsx
Normal file
59
src/components/atoms/Popover.tsx
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
import { useState, useRef, useEffect } from 'react'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
interface PopoverProps {
|
||||||
|
trigger: React.ReactNode
|
||||||
|
content: React.ReactNode
|
||||||
|
placement?: 'top' | 'bottom' | 'left' | 'right'
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Popover({ trigger, content, placement = 'bottom', className }: PopoverProps) {
|
||||||
|
const [isOpen, setIsOpen] = useState(false)
|
||||||
|
const popoverRef = useRef<HTMLDivElement>(null)
|
||||||
|
const triggerRef = useRef<HTMLDivElement>(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleClickOutside = (event: MouseEvent) => {
|
||||||
|
if (
|
||||||
|
popoverRef.current &&
|
||||||
|
!popoverRef.current.contains(event.target as Node) &&
|
||||||
|
triggerRef.current &&
|
||||||
|
!triggerRef.current.contains(event.target as Node)
|
||||||
|
) {
|
||||||
|
setIsOpen(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('mousedown', handleClickOutside)
|
||||||
|
return () => document.removeEventListener('mousedown', handleClickOutside)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const placementStyles = {
|
||||||
|
top: 'bottom-full mb-2 left-1/2 -translate-x-1/2',
|
||||||
|
bottom: 'top-full mt-2 left-1/2 -translate-x-1/2',
|
||||||
|
left: 'right-full mr-2 top-1/2 -translate-y-1/2',
|
||||||
|
right: 'left-full ml-2 top-1/2 -translate-y-1/2',
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="relative inline-block">
|
||||||
|
<div ref={triggerRef} onClick={() => setIsOpen(!isOpen)}>
|
||||||
|
{trigger}
|
||||||
|
</div>
|
||||||
|
{isOpen && (
|
||||||
|
<div
|
||||||
|
ref={popoverRef}
|
||||||
|
className={cn(
|
||||||
|
'absolute z-50 w-64 p-4 bg-popover text-popover-foreground border border-border rounded-lg shadow-lg',
|
||||||
|
'animate-in fade-in-0 zoom-in-95',
|
||||||
|
placementStyles[placement],
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{content}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { MagnifyingGlass, X } from '@phosphor-icons/react'
|
import { MagnifyingGlass, X } from '@phosphor-icons/react'
|
||||||
import { Input } from '@/lib/json-ui/json-components'
|
import { Input } from './Input'
|
||||||
|
|
||||||
interface BasicSearchInputProps {
|
interface BasicSearchInputProps {
|
||||||
value: string
|
value: string
|
||||||
|
|||||||
@@ -1,108 +1,120 @@
|
|||||||
// Auto-generated exports - DO NOT EDIT MANUALLY
|
|
||||||
export { ActionButton } from './ActionButton'
|
|
||||||
export { ActionCard } from './ActionCard'
|
|
||||||
export { ActionIcon } from './ActionIcon'
|
|
||||||
export { Alert } from './Alert'
|
|
||||||
export { AppLogo } from './AppLogo'
|
export { AppLogo } from './AppLogo'
|
||||||
export { Avatar } from './Avatar'
|
export { TabIcon } from './TabIcon'
|
||||||
export { AvatarGroup } from './AvatarGroup'
|
export { StatusIcon } from './StatusIcon'
|
||||||
export { Badge } from './Badge'
|
export { ErrorBadge } from './ErrorBadge'
|
||||||
|
export { IconWrapper } from './IconWrapper'
|
||||||
|
export { LoadingSpinner } from './LoadingSpinner'
|
||||||
|
export { EmptyStateIcon } from './EmptyStateIcon'
|
||||||
|
export { TreeIcon } from './TreeIcon'
|
||||||
|
export { FileIcon } from './FileIcon'
|
||||||
|
export { ActionIcon } from './ActionIcon'
|
||||||
|
export { SeedDataStatus } from './SeedDataStatus'
|
||||||
|
export { ActionButton } from './ActionButton'
|
||||||
|
export { IconButton } from './IconButton'
|
||||||
|
export { DataList } from './DataList'
|
||||||
|
export { StatusBadge } from './StatusBadge'
|
||||||
|
export { Text } from './Text'
|
||||||
|
export { Heading } from './Heading'
|
||||||
|
export { List } from './List'
|
||||||
|
export { Grid } from './Grid'
|
||||||
|
export { DataSourceBadge } from './DataSourceBadge'
|
||||||
export { BindingIndicator } from './BindingIndicator'
|
export { BindingIndicator } from './BindingIndicator'
|
||||||
export { BreadcrumbNav as Breadcrumb, BreadcrumbNav } from './Breadcrumb'
|
export { StatCard } from './StatCard'
|
||||||
export { Button } from './Button'
|
export { LoadingState } from './LoadingState'
|
||||||
export { ButtonGroup } from './ButtonGroup'
|
export { EmptyState } from './EmptyState'
|
||||||
export { Calendar } from './Calendar'
|
export { DetailRow } from './DetailRow'
|
||||||
export { Card } from './Card'
|
|
||||||
export { Checkbox } from './Checkbox'
|
|
||||||
export { Chip } from './Chip'
|
|
||||||
export { CircularProgress } from './CircularProgress'
|
|
||||||
export { Code } from './Code'
|
|
||||||
export { ColorSwatch } from './ColorSwatch'
|
|
||||||
export { CommandPalette } from './CommandPalette'
|
|
||||||
export { CompletionCard } from './CompletionCard'
|
export { CompletionCard } from './CompletionCard'
|
||||||
|
export { TipsCard } from './TipsCard'
|
||||||
|
export { CountBadge } from './CountBadge'
|
||||||
export { ConfirmButton } from './ConfirmButton'
|
export { ConfirmButton } from './ConfirmButton'
|
||||||
|
export { FilterInput } from './FilterInput'
|
||||||
|
export { BasicPageHeader } from './PageHeader'
|
||||||
|
export { MetricCard } from './MetricCard'
|
||||||
|
|
||||||
|
export { Link } from './Link'
|
||||||
|
export { Divider } from './Divider'
|
||||||
|
export { Avatar } from './Avatar'
|
||||||
|
export { Chip } from './Chip'
|
||||||
|
export { Code } from './Code'
|
||||||
|
export { Kbd } from './Kbd'
|
||||||
|
export { ProgressBar } from './ProgressBar'
|
||||||
|
export { Skeleton } from './Skeleton'
|
||||||
|
export { Tooltip } from './Tooltip'
|
||||||
|
export { Alert } from './Alert'
|
||||||
|
export { Spinner } from './Spinner'
|
||||||
|
export { Dot } from './Dot'
|
||||||
|
export { Image } from './Image'
|
||||||
|
export { Label } from './Label'
|
||||||
|
export { HelperText } from './HelperText'
|
||||||
export { Container } from './Container'
|
export { Container } from './Container'
|
||||||
|
export { Section } from './Section'
|
||||||
|
export { Stack } from './Stack'
|
||||||
|
export { Spacer } from './Spacer'
|
||||||
|
export { Timestamp } from './Timestamp'
|
||||||
|
export { ScrollArea } from './ScrollArea'
|
||||||
|
|
||||||
|
export { Tag } from './Tag'
|
||||||
|
export { Breadcrumb, BreadcrumbNav } from './Breadcrumb'
|
||||||
|
export { IconText } from './IconText'
|
||||||
|
export { TextArea } from './TextArea'
|
||||||
|
export { Input } from './Input'
|
||||||
|
export { Toggle } from './Toggle'
|
||||||
|
export { RadioGroup } from './Radio'
|
||||||
|
export { Checkbox } from './Checkbox'
|
||||||
|
export { Slider } from './Slider'
|
||||||
|
export { ColorSwatch } from './ColorSwatch'
|
||||||
|
export { Stepper } from './Stepper'
|
||||||
|
export { Rating } from './Rating'
|
||||||
|
export { Timeline } from './Timeline'
|
||||||
|
export { FileUpload } from './FileUpload'
|
||||||
|
export { Popover } from './Popover'
|
||||||
|
export { Tabs } from './Tabs'
|
||||||
|
export { Menu } from './Menu'
|
||||||
|
export { Accordion } from './Accordion'
|
||||||
|
export { Card } from './Card'
|
||||||
|
export { Notification } from './Notification'
|
||||||
|
export { CopyButton } from './CopyButton'
|
||||||
|
export { PasswordInput } from './PasswordInput'
|
||||||
|
export { BasicSearchInput } from './SearchInput'
|
||||||
|
export { Select } from './Select'
|
||||||
|
export { Modal } from './Modal'
|
||||||
|
export { Drawer } from './Drawer'
|
||||||
|
export { Table } from './Table'
|
||||||
|
|
||||||
|
export { Button } from './Button'
|
||||||
|
export { Badge } from './Badge'
|
||||||
|
export { Switch } from './Switch'
|
||||||
|
export { Separator } from './Separator'
|
||||||
|
export { HoverCard } from './HoverCard'
|
||||||
|
export { Calendar } from './Calendar'
|
||||||
|
export { ButtonGroup } from './ButtonGroup'
|
||||||
|
export { CommandPalette } from './CommandPalette'
|
||||||
export { ContextMenu } from './ContextMenu'
|
export { ContextMenu } from './ContextMenu'
|
||||||
export type { ContextMenuItemType } from './ContextMenu'
|
export type { ContextMenuItemType } from './ContextMenu'
|
||||||
export { CountBadge } from './CountBadge'
|
|
||||||
export { DataList } from './DataList'
|
|
||||||
export { DataSourceBadge } from './DataSourceBadge'
|
|
||||||
export { DataTable } from './DataTable'
|
export { DataTable } from './DataTable'
|
||||||
export type { Column } from './DataTable'
|
export type { Column } from './DataTable'
|
||||||
export { DatePicker } from './DatePicker'
|
|
||||||
export { DetailRow } from './DetailRow'
|
|
||||||
export { Divider } from './Divider'
|
|
||||||
export { Dot } from './Dot'
|
|
||||||
export { Drawer } from './Drawer'
|
|
||||||
export { EmptyMessage } from './EmptyMessage'
|
|
||||||
export { EmptyState } from './EmptyState'
|
|
||||||
export { EmptyStateIcon } from './EmptyStateIcon'
|
|
||||||
export { ErrorBadge } from './ErrorBadge'
|
|
||||||
export { FileIcon } from './FileIcon'
|
|
||||||
export { Flex } from './Flex'
|
|
||||||
export { Form, FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage } from './Form'
|
export { Form, FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage } from './Form'
|
||||||
export { GlowCard } from './GlowCard'
|
export { DatePicker } from './DatePicker'
|
||||||
export { Grid } from './Grid'
|
export { RangeSlider } from './RangeSlider'
|
||||||
export { Heading } from './Heading'
|
|
||||||
export { HelperText } from './HelperText'
|
|
||||||
export { HoverCard } from './HoverCard'
|
|
||||||
export { IconButton } from './IconButton'
|
|
||||||
export { IconText } from './IconText'
|
|
||||||
export { IconWrapper } from './IconWrapper'
|
|
||||||
export { InfoBox } from './InfoBox'
|
|
||||||
export { InfoPanel } from './InfoPanel'
|
export { InfoPanel } from './InfoPanel'
|
||||||
export { Kbd } from './Kbd'
|
export { ResponsiveGrid } from './ResponsiveGrid'
|
||||||
export { KeyValue } from './KeyValue'
|
export { Flex } from './Flex'
|
||||||
export { Label } from './Label'
|
export { CircularProgress } from './CircularProgress'
|
||||||
export { Link } from './Link'
|
export { AvatarGroup } from './AvatarGroup'
|
||||||
export { List } from './List'
|
|
||||||
export { ListItem } from './ListItem'
|
|
||||||
export { LiveIndicator } from './LiveIndicator'
|
|
||||||
export { LoadingSpinner } from './LoadingSpinner'
|
|
||||||
export { LoadingState } from './LoadingState'
|
|
||||||
export { MetricCard } from './MetricCard'
|
|
||||||
export { MetricDisplay } from './MetricDisplay'
|
|
||||||
export { Modal } from './Modal'
|
|
||||||
export { Notification } from './Notification'
|
|
||||||
export { NumberInput } from './NumberInput'
|
export { NumberInput } from './NumberInput'
|
||||||
export { BasicPageHeader, BasicPageHeader as PageHeader } from './PageHeader'
|
export { TextGradient } from './TextGradient'
|
||||||
export { PanelHeader } from './PanelHeader'
|
|
||||||
export { ProgressBar } from './ProgressBar'
|
|
||||||
export { Pulse } from './Pulse'
|
export { Pulse } from './Pulse'
|
||||||
export { QuickActionButton } from './QuickActionButton'
|
export { QuickActionButton } from './QuickActionButton'
|
||||||
export { RadioGroup as Radio, RadioGroup } from './Radio'
|
export { PanelHeader } from './PanelHeader'
|
||||||
export { RangeSlider } from './RangeSlider'
|
export { LiveIndicator } from './LiveIndicator'
|
||||||
export { Rating } from './Rating'
|
|
||||||
export { ResponsiveGrid } from './ResponsiveGrid'
|
|
||||||
export { ScrollArea } from './ScrollArea'
|
|
||||||
export { BasicSearchInput as SearchInput, BasicSearchInput } from './SearchInput'
|
|
||||||
export { Section } from './Section'
|
|
||||||
export { SeedDataStatus } from './SeedDataStatus'
|
|
||||||
export { Select } from './Select'
|
|
||||||
export { Separator } from './Separator'
|
|
||||||
export { Skeleton } from './Skeleton'
|
|
||||||
export { Slider } from './Slider'
|
|
||||||
export { Spacer } from './Spacer'
|
|
||||||
export { Sparkle } from './Sparkle'
|
export { Sparkle } from './Sparkle'
|
||||||
export { Spinner } from './Spinner'
|
export { GlowCard } from './GlowCard'
|
||||||
export { Stack } from './Stack'
|
|
||||||
export { StatCard } from './StatCard'
|
|
||||||
export { StatusBadge } from './StatusBadge'
|
|
||||||
export { StatusIcon } from './StatusIcon'
|
|
||||||
export { StepIndicator } from './StepIndicator'
|
|
||||||
export { Stepper } from './Stepper'
|
|
||||||
export { Switch } from './Switch'
|
|
||||||
export { TabIcon } from './TabIcon'
|
|
||||||
export { Table } from './Table'
|
|
||||||
export { Tabs } from './Tabs'
|
|
||||||
export { Tag } from './Tag'
|
|
||||||
export { Text } from './Text'
|
|
||||||
export { TextArea } from './TextArea'
|
|
||||||
export { TextGradient } from './TextGradient'
|
|
||||||
export { TextHighlight } from './TextHighlight'
|
export { TextHighlight } from './TextHighlight'
|
||||||
export { Timeline } from './Timeline'
|
export { ActionCard } from './ActionCard'
|
||||||
export { Timestamp } from './Timestamp'
|
export { InfoBox } from './InfoBox'
|
||||||
export { TipsCard } from './TipsCard'
|
export { ListItem } from './ListItem'
|
||||||
export { Toggle } from './Toggle'
|
export { MetricDisplay } from './MetricDisplay'
|
||||||
export { Tooltip } from './Tooltip'
|
export { KeyValue } from './KeyValue'
|
||||||
export { TreeIcon } from './TreeIcon'
|
export { EmptyMessage } from './EmptyMessage'
|
||||||
|
export { StepIndicator } from './StepIndicator'
|
||||||
|
|||||||
@@ -1,120 +0,0 @@
|
|||||||
export { AppLogo } from './AppLogo'
|
|
||||||
export { TabIcon } from './TabIcon'
|
|
||||||
export { StatusIcon } from './StatusIcon'
|
|
||||||
export { ErrorBadge } from './ErrorBadge'
|
|
||||||
export { IconWrapper } from './IconWrapper'
|
|
||||||
export { LoadingSpinner } from './LoadingSpinner'
|
|
||||||
export { EmptyStateIcon } from './EmptyStateIcon'
|
|
||||||
export { TreeIcon } from './TreeIcon'
|
|
||||||
export { FileIcon } from './FileIcon'
|
|
||||||
export { ActionIcon } from './ActionIcon'
|
|
||||||
export { SeedDataStatus } from './SeedDataStatus'
|
|
||||||
export { ActionButton } from './ActionButton'
|
|
||||||
export { IconButton } from './IconButton'
|
|
||||||
export { DataList } from './DataList'
|
|
||||||
export { StatusBadge } from './StatusBadge'
|
|
||||||
export { Text } from './Text'
|
|
||||||
export { Heading } from './Heading'
|
|
||||||
export { List } from './List'
|
|
||||||
export { Grid } from './Grid'
|
|
||||||
export { DataSourceBadge } from './DataSourceBadge'
|
|
||||||
export { BindingIndicator } from './BindingIndicator'
|
|
||||||
export { StatCard } from './StatCard'
|
|
||||||
export { LoadingState } from './LoadingState'
|
|
||||||
export { EmptyState } from './EmptyState'
|
|
||||||
export { DetailRow } from './DetailRow'
|
|
||||||
export { CompletionCard } from './CompletionCard'
|
|
||||||
export { TipsCard } from './TipsCard'
|
|
||||||
export { CountBadge } from './CountBadge'
|
|
||||||
export { ConfirmButton } from './ConfirmButton'
|
|
||||||
export { FilterInput } from './FilterInput'
|
|
||||||
export { BasicPageHeader } from './PageHeader'
|
|
||||||
export { MetricCard } from './MetricCard'
|
|
||||||
|
|
||||||
export { Link } from './Link'
|
|
||||||
export { Divider } from './Divider'
|
|
||||||
export { Avatar } from './Avatar'
|
|
||||||
export { Chip } from './Chip'
|
|
||||||
export { Code } from './Code'
|
|
||||||
export { Kbd } from './Kbd'
|
|
||||||
export { ProgressBar } from './ProgressBar'
|
|
||||||
export { Skeleton } from './Skeleton'
|
|
||||||
export { Tooltip } from './Tooltip'
|
|
||||||
export { Alert } from './Alert'
|
|
||||||
export { Spinner } from './Spinner'
|
|
||||||
export { Dot } from './Dot'
|
|
||||||
export { Image } from './Image'
|
|
||||||
export { Label } from './Label'
|
|
||||||
export { HelperText } from './HelperText'
|
|
||||||
export { Container } from './Container'
|
|
||||||
export { Section } from './Section'
|
|
||||||
export { Stack } from './Stack'
|
|
||||||
export { Spacer } from './Spacer'
|
|
||||||
export { Timestamp } from './Timestamp'
|
|
||||||
export { ScrollArea } from './ScrollArea'
|
|
||||||
|
|
||||||
export { Tag } from './Tag'
|
|
||||||
export { Breadcrumb, BreadcrumbNav } from './Breadcrumb'
|
|
||||||
export { IconText } from './IconText'
|
|
||||||
export { TextArea } from './TextArea'
|
|
||||||
export { Input } from './Input'
|
|
||||||
export { Toggle } from './Toggle'
|
|
||||||
export { RadioGroup } from './Radio'
|
|
||||||
export { Checkbox } from './Checkbox'
|
|
||||||
export { Slider } from './Slider'
|
|
||||||
export { ColorSwatch } from './ColorSwatch'
|
|
||||||
export { Stepper } from './Stepper'
|
|
||||||
export { Rating } from './Rating'
|
|
||||||
export { Timeline } from './Timeline'
|
|
||||||
export { FileUpload } from './FileUpload'
|
|
||||||
export { Popover } from './Popover'
|
|
||||||
export { Tabs } from './Tabs'
|
|
||||||
export { Menu } from './Menu'
|
|
||||||
export { Accordion } from './Accordion'
|
|
||||||
export { Card } from './Card'
|
|
||||||
export { Notification } from './Notification'
|
|
||||||
export { CopyButton } from './CopyButton'
|
|
||||||
export { PasswordInput } from './PasswordInput'
|
|
||||||
export { BasicSearchInput } from './SearchInput'
|
|
||||||
export { Select } from './Select'
|
|
||||||
export { Modal } from './Modal'
|
|
||||||
export { Drawer } from './Drawer'
|
|
||||||
export { Table } from './Table'
|
|
||||||
|
|
||||||
export { Button } from './Button'
|
|
||||||
export { Badge } from './Badge'
|
|
||||||
export { Switch } from './Switch'
|
|
||||||
export { Separator } from './Separator'
|
|
||||||
export { HoverCard } from './HoverCard'
|
|
||||||
export { Calendar } from './Calendar'
|
|
||||||
export { ButtonGroup } from './ButtonGroup'
|
|
||||||
export { CommandPalette } from './CommandPalette'
|
|
||||||
export { ContextMenu } from './ContextMenu'
|
|
||||||
export type { ContextMenuItemType } from './ContextMenu'
|
|
||||||
export { DataTable } from './DataTable'
|
|
||||||
export type { Column } from './DataTable'
|
|
||||||
export { Form, FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage } from './Form'
|
|
||||||
export { DatePicker } from './DatePicker'
|
|
||||||
export { RangeSlider } from './RangeSlider'
|
|
||||||
export { InfoPanel } from './InfoPanel'
|
|
||||||
export { ResponsiveGrid } from './ResponsiveGrid'
|
|
||||||
export { Flex } from './Flex'
|
|
||||||
export { CircularProgress } from './CircularProgress'
|
|
||||||
export { AvatarGroup } from './AvatarGroup'
|
|
||||||
export { NumberInput } from './NumberInput'
|
|
||||||
export { TextGradient } from './TextGradient'
|
|
||||||
export { Pulse } from './Pulse'
|
|
||||||
export { QuickActionButton } from './QuickActionButton'
|
|
||||||
export { PanelHeader } from './PanelHeader'
|
|
||||||
export { LiveIndicator } from './LiveIndicator'
|
|
||||||
export { Sparkle } from './Sparkle'
|
|
||||||
export { GlowCard } from './GlowCard'
|
|
||||||
|
|
||||||
export { TextHighlight } from './TextHighlight'
|
|
||||||
export { ActionCard } from './ActionCard'
|
|
||||||
export { InfoBox } from './InfoBox'
|
|
||||||
export { ListItem } from './ListItem'
|
|
||||||
export { MetricDisplay } from './MetricDisplay'
|
|
||||||
export { KeyValue } from './KeyValue'
|
|
||||||
export { EmptyMessage } from './EmptyMessage'
|
|
||||||
export { StepIndicator } from './StepIndicator'
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "accordion-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "const base = 'space-y-2'; return data ? `${base} ${data}` : base"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "accordion-items-repeat",
|
|
||||||
"type": "_repeat",
|
|
||||||
"bindings": {
|
|
||||||
"_items": {
|
|
||||||
"source": "items",
|
|
||||||
"transform": "data || []"
|
|
||||||
},
|
|
||||||
"_renderItem": {
|
|
||||||
"source": ["accordionState.toggleItem", "accordionState.isOpen"],
|
|
||||||
"transform": "(item) => { const toggleItem = data[0]; const isOpen = data[1]; const itemIsOpen = isOpen(item.id); return { _element: 'div', _key: item.id, _props: { className: 'border border-border rounded-lg overflow-hidden' }, _children: [{ _element: 'button', _props: { onClick: () => !item.disabled && toggleItem(item.id), disabled: item.disabled, className: `w-full flex items-center justify-between p-4 bg-card text-card-foreground font-medium transition-colors hover:bg-accent hover:text-accent-foreground ${item.disabled ? 'opacity-50 cursor-not-allowed' : ''}` }, _children: [{ _element: 'span', _children: [item.title] }, { _element: 'CaretDown', _props: { className: `w-5 h-5 transition-transform ${itemIsOpen ? 'rotate-180' : ''}` } }] }, itemIsOpen ? { _element: 'div', _props: { className: 'p-4 bg-card border-t border-border animate-in slide-in-from-top-2' }, _children: [item.content] } : null].filter(Boolean) }; }"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,112 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "action-button-root",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "inline-flex"
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "tooltip",
|
|
||||||
"then": {
|
|
||||||
"id": "action-button-with-tooltip",
|
|
||||||
"type": "div",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-button-tooltip-provider",
|
|
||||||
"type": "TooltipProvider",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-button-tooltip",
|
|
||||||
"type": "Tooltip",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-button-trigger",
|
|
||||||
"type": "TooltipTrigger",
|
|
||||||
"props": {
|
|
||||||
"asChild": true
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-button-element",
|
|
||||||
"type": "Button",
|
|
||||||
"props": {
|
|
||||||
"disabled": "disabled",
|
|
||||||
"className": "className"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"variant": "variant",
|
|
||||||
"size": "size"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-button-icon",
|
|
||||||
"type": "span",
|
|
||||||
"props": {
|
|
||||||
"className": "mr-2"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "icon"
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "icon"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "action-button-label",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"children": "label"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "action-button-tooltip-content",
|
|
||||||
"type": "TooltipContent",
|
|
||||||
"bindings": {
|
|
||||||
"children": "tooltip"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"else": {
|
|
||||||
"id": "action-button-direct",
|
|
||||||
"type": "Button",
|
|
||||||
"props": {
|
|
||||||
"disabled": "disabled",
|
|
||||||
"className": "className"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"variant": "variant",
|
|
||||||
"size": "size"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-button-icon-direct",
|
|
||||||
"type": "span",
|
|
||||||
"props": {
|
|
||||||
"className": "mr-2"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "icon"
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "icon"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "action-button-label-direct",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"children": "label"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,89 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "action-card-root",
|
|
||||||
"type": "Card",
|
|
||||||
"props": {
|
|
||||||
"className": "cursor-pointer transition-all hover:shadow-md hover:border-primary/50"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "disabled",
|
|
||||||
"transform": "data ? 'opacity-50 cursor-not-allowed' : 'cursor-pointer transition-all hover:shadow-md hover:border-primary/50'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-card-content",
|
|
||||||
"type": "CardContent",
|
|
||||||
"props": {
|
|
||||||
"className": "p-4"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-card-flex",
|
|
||||||
"type": "Flex",
|
|
||||||
"props": {
|
|
||||||
"justify": "start",
|
|
||||||
"align": "start",
|
|
||||||
"gap": "md",
|
|
||||||
"className": "items-start gap-3"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-card-icon-wrapper",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "flex-shrink-0 p-2 rounded-lg bg-primary/10 text-primary"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "icon"
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "icon"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "action-card-text-container",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "flex-1 min-w-0"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "action-card-title",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "font-semibold text-sm mb-1"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "title"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "action-card-description",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "text-xs text-muted-foreground line-clamp-2"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "description"
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "description"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "action-card-caret",
|
|
||||||
"type": "CaretRight",
|
|
||||||
"props": {
|
|
||||||
"size": 16,
|
|
||||||
"className": "flex-shrink-0 text-muted-foreground"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "action-icon-root",
|
|
||||||
"type": "ActionIcon",
|
|
||||||
"bindings": {
|
|
||||||
"action": "action",
|
|
||||||
"size": "size",
|
|
||||||
"weight": "weight",
|
|
||||||
"className": "className"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "alert-root",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"role": "alert",
|
|
||||||
"className": "flex gap-3 p-4 rounded-lg border"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "variant",
|
|
||||||
"transform": "(() => { const config = { info: { classes: 'bg-blue-50 border-blue-200 text-blue-900' }, warning: { classes: 'bg-yellow-50 border-yellow-200 text-yellow-900' }, success: { classes: 'bg-green-50 border-green-200 text-green-900' }, error: { classes: 'bg-red-50 border-red-200 text-red-900' } }; return 'flex gap-3 p-4 rounded-lg border ' + (config[data]?.classes || config.info.classes); })()"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "alert-icon",
|
|
||||||
"type": "AlertIcon",
|
|
||||||
"bindings": {
|
|
||||||
"variant": "variant"
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"className": "flex-shrink-0 mt-0.5"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "alert-content",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "flex-1"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "alert-title",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "font-semibold mb-1"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "title"
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "title"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "alert-message",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "text-sm"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "children"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "app-dialogs",
|
|
||||||
"type": "div",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "global-search-suspense",
|
|
||||||
"type": "Suspense",
|
|
||||||
"props": {
|
|
||||||
"fallback": null
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "global-search",
|
|
||||||
"type": "GlobalSearch",
|
|
||||||
"bindings": {
|
|
||||||
"open": { "source": "props.searchOpen" },
|
|
||||||
"onOpenChange": { "source": "props.onSearchOpenChange" },
|
|
||||||
"files": { "source": "props.files" },
|
|
||||||
"models": { "source": "props.models" },
|
|
||||||
"components": { "source": "props.components" },
|
|
||||||
"componentTrees": { "source": "props.componentTrees" },
|
|
||||||
"workflows": { "source": "props.workflows" },
|
|
||||||
"lambdas": { "source": "props.lambdas" },
|
|
||||||
"playwrightTests": { "source": "props.playwrightTests" },
|
|
||||||
"storybookStories": { "source": "props.storybookStories" },
|
|
||||||
"unitTests": { "source": "props.unitTests" },
|
|
||||||
"onNavigate": { "source": "props.onNavigate" },
|
|
||||||
"onFileSelect": { "source": "props.onFileSelect" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "shortcuts-dialog-suspense",
|
|
||||||
"type": "Suspense",
|
|
||||||
"props": {
|
|
||||||
"fallback": null
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "shortcuts-dialog",
|
|
||||||
"type": "KeyboardShortcutsDialog",
|
|
||||||
"bindings": {
|
|
||||||
"open": { "source": "props.shortcutsOpen" },
|
|
||||||
"onOpenChange": { "source": "props.onShortcutsOpenChange" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "preview-dialog-suspense",
|
|
||||||
"type": "Suspense",
|
|
||||||
"props": {
|
|
||||||
"fallback": null
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "preview-dialog",
|
|
||||||
"type": "PreviewDialog",
|
|
||||||
"bindings": {
|
|
||||||
"open": { "source": "props.previewOpen" },
|
|
||||||
"onOpenChange": { "source": "props.onPreviewOpenChange" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "pwa-install-suspense",
|
|
||||||
"type": "Suspense",
|
|
||||||
"props": {
|
|
||||||
"fallback": null
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PWAInstallPrompt"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,76 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "app-layout",
|
|
||||||
"type": "SidebarProvider",
|
|
||||||
"props": {
|
|
||||||
"defaultOpen": true
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "nav-menu",
|
|
||||||
"type": "NavigationMenu",
|
|
||||||
"bindings": {
|
|
||||||
"activeTab": { "source": "hookData.currentPage" },
|
|
||||||
"onTabChange": { "source": "hookData.navigateToPage" },
|
|
||||||
"featureToggles": { "source": "hookData.featureToggles" },
|
|
||||||
"errorCount": { "source": "hookData.errorCount" }
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "sidebar-inset-wrapper",
|
|
||||||
"type": "SidebarInset",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "app-layout-main",
|
|
||||||
"type": "div",
|
|
||||||
"className": "h-screen flex flex-col bg-background",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "main-panel",
|
|
||||||
"type": "AppMainPanel",
|
|
||||||
"bindings": {
|
|
||||||
"currentPage": { "source": "hookData.currentPage" },
|
|
||||||
"navigateToPage": { "source": "hookData.navigateToPage" },
|
|
||||||
"featureToggles": { "source": "hookData.featureToggles" },
|
|
||||||
"errorCount": { "source": "hookData.errorCount" },
|
|
||||||
"lastSaved": { "source": "hookData.lastSaved" },
|
|
||||||
"currentProject": { "source": "hookData.currentProject" },
|
|
||||||
"onProjectLoad": { "source": "hookData.handleProjectLoad" },
|
|
||||||
"onSearch": { "source": "hookData.setSearchOpen", "transform": "() => setSearchOpen(true)" },
|
|
||||||
"onShowShortcuts": { "source": "hookData.setShortcutsOpen", "transform": "() => setShortcutsOpen(true)" },
|
|
||||||
"onGenerateAI": { "source": "hookData.onGenerateAI" },
|
|
||||||
"onExport": { "source": "hookData.onExport" },
|
|
||||||
"onPreview": { "source": "hookData.setPreviewOpen", "transform": "() => setPreviewOpen(true)" },
|
|
||||||
"onShowErrors": { "source": "hookData.navigateToPage", "transform": "() => navigateToPage('errors')" },
|
|
||||||
"stateContext": { "source": "hookData.stateContext" },
|
|
||||||
"actionContext": { "source": "hookData.actionContext" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dialogs-container",
|
|
||||||
"type": "AppDialogs",
|
|
||||||
"bindings": {
|
|
||||||
"searchOpen": { "source": "hookData.searchOpen" },
|
|
||||||
"onSearchOpenChange": { "source": "hookData.setSearchOpen" },
|
|
||||||
"shortcutsOpen": { "source": "hookData.shortcutsOpen" },
|
|
||||||
"onShortcutsOpenChange": { "source": "hookData.setShortcutsOpen" },
|
|
||||||
"previewOpen": { "source": "hookData.previewOpen" },
|
|
||||||
"onPreviewOpenChange": { "source": "hookData.setPreviewOpen" },
|
|
||||||
"files": { "source": "hookData.files" },
|
|
||||||
"models": { "source": "hookData.models" },
|
|
||||||
"components": { "source": "hookData.components" },
|
|
||||||
"componentTrees": { "source": "hookData.componentTrees" },
|
|
||||||
"workflows": { "source": "hookData.workflows" },
|
|
||||||
"lambdas": { "source": "hookData.lambdas" },
|
|
||||||
"playwrightTests": { "source": "hookData.playwrightTests" },
|
|
||||||
"storybookStories": { "source": "hookData.storybookStories" },
|
|
||||||
"unitTests": { "source": "hookData.unitTests" },
|
|
||||||
"onNavigate": { "source": "hookData.navigateToPage" },
|
|
||||||
"onFileSelect": { "source": "hookData.onFileSelect" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "app-logo-root",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "w-8 h-8 sm:w-10 sm:h-10 rounded-lg bg-gradient-to-br from-primary to-accent flex items-center justify-center shrink-0"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "app-logo-icon",
|
|
||||||
"type": "Code",
|
|
||||||
"props": {
|
|
||||||
"size": 20,
|
|
||||||
"weight": "duotone",
|
|
||||||
"className": "text-white sm:w-6 sm:h-6"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,99 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "app-main-panel",
|
|
||||||
"type": "div",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "pwa-status-bar-suspense",
|
|
||||||
"type": "Suspense",
|
|
||||||
"props": {
|
|
||||||
"fallback": {
|
|
||||||
"type": "div",
|
|
||||||
"className": "h-1 bg-primary animate-pulse"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PWAStatusBar"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "pwa-update-prompt-suspense",
|
|
||||||
"type": "Suspense",
|
|
||||||
"props": {
|
|
||||||
"fallback": null
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PWAUpdatePrompt"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "app-header",
|
|
||||||
"type": "AppHeader",
|
|
||||||
"bindings": {
|
|
||||||
"activeTab": {
|
|
||||||
"source": "props.currentPage"
|
|
||||||
},
|
|
||||||
"onTabChange": {
|
|
||||||
"source": "props.navigateToPage"
|
|
||||||
},
|
|
||||||
"featureToggles": {
|
|
||||||
"source": "props.featureToggles"
|
|
||||||
},
|
|
||||||
"errorCount": {
|
|
||||||
"source": "props.errorCount"
|
|
||||||
},
|
|
||||||
"lastSaved": {
|
|
||||||
"source": "props.lastSaved"
|
|
||||||
},
|
|
||||||
"currentProject": {
|
|
||||||
"source": "props.currentProject"
|
|
||||||
},
|
|
||||||
"onProjectLoad": {
|
|
||||||
"source": "props.onProjectLoad"
|
|
||||||
},
|
|
||||||
"onSearch": {
|
|
||||||
"source": "props.onSearch"
|
|
||||||
},
|
|
||||||
"onShowShortcuts": {
|
|
||||||
"source": "props.onShowShortcuts"
|
|
||||||
},
|
|
||||||
"onGenerateAI": {
|
|
||||||
"source": "props.onGenerateAI"
|
|
||||||
},
|
|
||||||
"onExport": {
|
|
||||||
"source": "props.onExport"
|
|
||||||
},
|
|
||||||
"onPreview": {
|
|
||||||
"source": "props.onPreview"
|
|
||||||
},
|
|
||||||
"onShowErrors": {
|
|
||||||
"source": "props.onShowErrors"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "main-content",
|
|
||||||
"type": "div",
|
|
||||||
"className": "flex-1 overflow-hidden",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "RouterProvider",
|
|
||||||
"bindings": {
|
|
||||||
"featureToggles": {
|
|
||||||
"source": "props.featureToggles"
|
|
||||||
},
|
|
||||||
"stateContext": {
|
|
||||||
"source": "props.stateContext"
|
|
||||||
},
|
|
||||||
"actionContext": {
|
|
||||||
"source": "props.actionContext"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,51 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "app-router-layout",
|
|
||||||
"type": "div",
|
|
||||||
"className": "h-screen flex flex-col bg-background",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "main-panel",
|
|
||||||
"type": "AppMainPanel",
|
|
||||||
"bindings": {
|
|
||||||
"currentPage": { "source": "hookData.currentPage" },
|
|
||||||
"navigateToPage": { "source": "hookData.navigateToPage" },
|
|
||||||
"featureToggles": { "source": "hookData.featureToggles" },
|
|
||||||
"errorCount": { "source": "hookData.errorCount" },
|
|
||||||
"lastSaved": { "source": "hookData.lastSaved" },
|
|
||||||
"currentProject": { "source": "hookData.currentProject" },
|
|
||||||
"onProjectLoad": { "source": "hookData.handleProjectLoad" },
|
|
||||||
"onSearch": { "source": "hookData.setSearchOpen", "transform": "() => setSearchOpen(true)" },
|
|
||||||
"onShowShortcuts": { "source": "hookData.setShortcutsOpen", "transform": "() => setShortcutsOpen(true)" },
|
|
||||||
"onGenerateAI": { "source": "hookData.onGenerateAI" },
|
|
||||||
"onExport": { "source": "hookData.onExport" },
|
|
||||||
"onPreview": { "source": "hookData.setPreviewOpen", "transform": "() => setPreviewOpen(true)" },
|
|
||||||
"onShowErrors": { "source": "hookData.navigateToPage", "transform": "() => navigateToPage('errors')" },
|
|
||||||
"stateContext": { "source": "hookData.stateContext" },
|
|
||||||
"actionContext": { "source": "hookData.actionContext" }
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dialogs-container",
|
|
||||||
"type": "AppDialogs",
|
|
||||||
"bindings": {
|
|
||||||
"searchOpen": { "source": "hookData.searchOpen" },
|
|
||||||
"onSearchOpenChange": { "source": "hookData.setSearchOpen" },
|
|
||||||
"shortcutsOpen": { "source": "hookData.shortcutsOpen" },
|
|
||||||
"onShortcutsOpenChange": { "source": "hookData.setShortcutsOpen" },
|
|
||||||
"previewOpen": { "source": "hookData.previewOpen" },
|
|
||||||
"onPreviewOpenChange": { "source": "hookData.setPreviewOpen" },
|
|
||||||
"files": { "source": "hookData.files" },
|
|
||||||
"models": { "source": "hookData.models" },
|
|
||||||
"components": { "source": "hookData.components" },
|
|
||||||
"componentTrees": { "source": "hookData.componentTrees" },
|
|
||||||
"workflows": { "source": "hookData.workflows" },
|
|
||||||
"lambdas": { "source": "hookData.lambdas" },
|
|
||||||
"playwrightTests": { "source": "hookData.playwrightTests" },
|
|
||||||
"storybookStories": { "source": "hookData.storybookStories" },
|
|
||||||
"unitTests": { "source": "hookData.unitTests" },
|
|
||||||
"onNavigate": { "source": "hookData.navigateToPage" },
|
|
||||||
"onFileSelect": { "source": "hookData.onFileSelect" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "avatar-group-root",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "data ? 'flex -space-x-2 ' + data : 'flex -space-x-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "avatar-group-list",
|
|
||||||
"type": "AvatarList",
|
|
||||||
"bindings": {
|
|
||||||
"avatars": "avatars",
|
|
||||||
"max": "max",
|
|
||||||
"size": "size"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "avatar-group-remainder",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "size",
|
|
||||||
"transform": "(() => { const sizeClasses = { xs: 'h-6 w-6 text-xs', sm: 'h-8 w-8 text-xs', md: 'h-10 w-10 text-sm', lg: 'h-12 w-12 text-base' }; return 'relative inline-flex items-center justify-center rounded-full border-2 border-background bg-muted ' + (sizeClasses[data] || sizeClasses.md); })()"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "remainingCount",
|
|
||||||
"transform": "avatars.length - (max || 5) > 0"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "avatar-group-count",
|
|
||||||
"type": "span",
|
|
||||||
"props": {
|
|
||||||
"className": "font-medium text-foreground"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": "avatars",
|
|
||||||
"transform": "`+${Math.max(data.length - (max || 5), 0)}`"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "avatar-root",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "size",
|
|
||||||
"transform": "(() => { const sizeClasses = { xs: 'w-6 h-6 text-xs', sm: 'w-8 h-8 text-sm', md: 'w-10 h-10 text-base', lg: 'w-12 h-12 text-lg', xl: 'w-16 h-16 text-xl' }; return 'relative inline-flex items-center justify-center rounded-full bg-muted overflow-hidden ' + (sizeClasses[data] || sizeClasses.md); })()"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "src",
|
|
||||||
"then": {
|
|
||||||
"id": "avatar-image",
|
|
||||||
"type": "img",
|
|
||||||
"bindings": {
|
|
||||||
"src": "src",
|
|
||||||
"alt": "alt"
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"className": "w-full h-full object-cover"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"else": {
|
|
||||||
"id": "avatar-fallback",
|
|
||||||
"type": "span",
|
|
||||||
"props": {
|
|
||||||
"className": "font-medium text-muted-foreground"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": "fallback",
|
|
||||||
"transform": "data || (props.alt?.slice(0, 2).toUpperCase()) || '?'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "badge-root",
|
|
||||||
"type": "Badge",
|
|
||||||
"bindings": {
|
|
||||||
"variant": "variant",
|
|
||||||
"className": {
|
|
||||||
"source": "size",
|
|
||||||
"transform": "(() => { const sizeClasses = { sm: 'text-xs px-2 py-0.5', md: 'text-sm px-2.5 py-0.5', lg: 'text-base px-3 py-1' }; return 'inline-flex items-center gap-1.5 ' + (sizeClasses[data] || sizeClasses.md); })()"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "badge-icon",
|
|
||||||
"type": "span",
|
|
||||||
"props": {
|
|
||||||
"className": "flex-shrink-0"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "icon"
|
|
||||||
},
|
|
||||||
"conditional": {
|
|
||||||
"if": "icon"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "badge-content",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"children": "children"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,290 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "binding-editor-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'space-y-4'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "bound-properties-section",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'space-y-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "bound-properties-label",
|
|
||||||
"type": "Label",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-sm font-medium'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'Bound Properties'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "no-bindings-message",
|
|
||||||
"type": "p",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-sm text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'No bindings yet'"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "bindings",
|
|
||||||
"transform": "Object.keys(data || {}).length === 0"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "bindings-list",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'space-y-2'"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "bindings",
|
|
||||||
"transform": "Object.keys(data || {}).length > 0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "bindings-repeat",
|
|
||||||
"type": "_repeat",
|
|
||||||
"bindings": {
|
|
||||||
"_items": {
|
|
||||||
"source": "bindings",
|
|
||||||
"transform": "Object.keys(data || {})"
|
|
||||||
},
|
|
||||||
"_renderItem": {
|
|
||||||
"source": ["bindings", "editorState.removeBinding"],
|
|
||||||
"transform": "(prop) => { const bindings = data[0] || {}; const removeBinding = data[1]; const binding = bindings[prop]; return { _element: 'div', _key: prop, _props: { className: 'flex items-center justify-between p-2 bg-muted/30 rounded border border-border' }, _children: [{ _element: 'div', _props: { className: 'flex items-center gap-2' }, _children: [{ _element: 'span', _props: { className: 'text-sm font-mono' }, _children: [prop] }, { _element: 'span', _props: { className: 'text-muted-foreground' }, _children: ['→'] }, { _element: 'BindingIndicator', _props: { sourceId: binding.source, path: binding.path } }] }, { _element: 'Button', _props: { size: 'sm', variant: 'ghost', onClick: () => removeBinding(prop), className: 'h-6 w-6 p-0' }, _children: [{ _element: 'X', _props: { className: 'w-3 h-3' } }] }] }; }"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "add-binding-section",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'space-y-3 pt-3 border-t border-border'"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": ["bindings", "availableProps"],
|
|
||||||
"transform": "const boundProps = Object.keys(data[0] || {}); const availableProps = data[1] || []; const unboundProps = availableProps.filter(p => !boundProps.includes(p)); return unboundProps.length > 0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "add-binding-label",
|
|
||||||
"type": "Label",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-sm font-medium'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'Add New Binding'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "property-source-grid",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'grid grid-cols-2 gap-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "property-select-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'space-y-1'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "property-label",
|
|
||||||
"type": "Label",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-xs text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'Property'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "property-select",
|
|
||||||
"type": "Select",
|
|
||||||
"bindings": {
|
|
||||||
"value": {
|
|
||||||
"source": "editorState.selectedProp",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onValueChange": {
|
|
||||||
"source": "editorState.setSelectedProp",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": ["bindings", "availableProps"],
|
|
||||||
"transform": "const boundProps = Object.keys(data[0] || {}); const availableProps = data[1] || []; const unboundProps = availableProps.filter(p => !boundProps.includes(p)); return [{ _element: 'SelectTrigger', _props: { className: 'h-9' }, _children: [{ _element: 'SelectValue', _props: { placeholder: 'Select property' } }] }, { _element: 'SelectContent', _children: unboundProps.map(prop => ({ _element: 'SelectItem', _key: prop, _props: { value: prop }, _children: [prop] })) }]"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "source-select-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'space-y-1'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "source-label",
|
|
||||||
"type": "Label",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-xs text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'Data Source'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "source-select",
|
|
||||||
"type": "Select",
|
|
||||||
"bindings": {
|
|
||||||
"value": {
|
|
||||||
"source": "editorState.selectedSource",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onValueChange": {
|
|
||||||
"source": "editorState.setSelectedSource",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "dataSources",
|
|
||||||
"transform": "[{ _element: 'SelectTrigger', _props: { className: 'h-9' }, _children: [{ _element: 'SelectValue', _props: { placeholder: 'Select source' } }] }, { _element: 'SelectContent', _children: (data || []).map(ds => ({ _element: 'SelectItem', _key: ds.id, _props: { value: ds.id }, _children: [ds.id] })) }]"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "path-input-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'space-y-1'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "path-label",
|
|
||||||
"type": "Label",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-xs text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'Path (optional)'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "path-input",
|
|
||||||
"type": "Input",
|
|
||||||
"bindings": {
|
|
||||||
"placeholder": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'e.g., user.name'"
|
|
||||||
},
|
|
||||||
"value": {
|
|
||||||
"source": "editorState.path",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onChange": {
|
|
||||||
"source": "editorState.setPath",
|
|
||||||
"transform": "(e) => data(e.target.value)"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'h-9 font-mono text-sm'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "add-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"size": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'sm'"
|
|
||||||
},
|
|
||||||
"onClick": {
|
|
||||||
"source": "editorState.addBinding",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"disabled": {
|
|
||||||
"source": ["editorState.selectedProp", "editorState.selectedSource"],
|
|
||||||
"transform": "!data[0] || !data[1]"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'w-full'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "[{ _element: 'Plus', _props: { className: 'w-4 h-4 mr-2' } }, 'Add Binding']"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,122 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "breadcrumb",
|
|
||||||
"type": "nav",
|
|
||||||
"bindings": {
|
|
||||||
"aria-label": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'Breadcrumb'"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["items", "className"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const className = data[1] || ''; return cn(['flex items-center gap-2', className])"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "breadcrumb-items",
|
|
||||||
"type": "Fragment",
|
|
||||||
"_map": {
|
|
||||||
"source": "items",
|
|
||||||
"itemVar": "item",
|
|
||||||
"indexVar": "index"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "breadcrumb-item",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex items-center gap-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "breadcrumb-link",
|
|
||||||
"type": "a",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "item.href",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"href": {
|
|
||||||
"source": "item.href",
|
|
||||||
"transform": "data || '#'"
|
|
||||||
},
|
|
||||||
"onClick": {
|
|
||||||
"source": "item.onClick",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["item", "items"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const item = data[0]; const items = data[1] || []; const isLast = items.indexOf(item) === items.length - 1; const baseClass = 'text-sm transition-colors'; const styleClass = isLast ? 'text-foreground font-medium' : 'text-muted-foreground hover:text-foreground'; return cn([baseClass, styleClass])"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "link-text",
|
|
||||||
"type": "span",
|
|
||||||
"children": [{"type": "text", "content": {"source": "item.label"}}]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "breadcrumb-button",
|
|
||||||
"type": "button",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": ["item.href", "item.onClick"],
|
|
||||||
"transform": "!data[0] && data[1]"
|
|
||||||
},
|
|
||||||
"onClick": {
|
|
||||||
"source": "item.onClick",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["item", "items"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const item = data[0]; const items = data[1] || []; const isLast = items.indexOf(item) === items.length - 1; const baseClass = 'text-sm transition-colors'; const styleClass = isLast ? 'text-foreground font-medium' : 'text-muted-foreground hover:text-foreground'; return cn([baseClass, styleClass])"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "button-text",
|
|
||||||
"type": "span",
|
|
||||||
"children": [{"type": "text", "content": {"source": "item.label"}}]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "breadcrumb-span",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": ["item.href", "item.onClick"],
|
|
||||||
"transform": "!data[0] && !data[1]"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["item", "items"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const item = data[0]; const items = data[1] || []; const isLast = items.indexOf(item) === items.length - 1; const baseClass = 'text-sm'; const styleClass = isLast ? 'text-foreground font-medium' : 'text-muted-foreground'; return cn([baseClass, styleClass])"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [{"type": "text", "content": {"source": "item.label"}}]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "separator",
|
|
||||||
"type": "CaretRight",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": ["item", "items"],
|
|
||||||
"transform": "const item = data[0]; const items = data[1] || []; return items.indexOf(item) < items.length - 1"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'w-4 h-4 text-muted-foreground'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,114 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "button",
|
|
||||||
"type": "button",
|
|
||||||
"bindings": {
|
|
||||||
"type": {
|
|
||||||
"source": "type",
|
|
||||||
"transform": "data || 'button'"
|
|
||||||
},
|
|
||||||
"disabled": {
|
|
||||||
"source": ["disabled", "loading"],
|
|
||||||
"transform": "data[0] || data[1]"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["fullWidth", "className"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const fullWidth = data[0]; const className = data[1] || ''; const widthClass = fullWidth ? 'w-full' : ''; return cn([widthClass, className])"
|
|
||||||
},
|
|
||||||
"onClick": {
|
|
||||||
"source": "onClick",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "button-loading-content",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "loading",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex items-center gap-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "spinner",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'h-4 w-4 border-2 border-current border-t-transparent rounded-full animate-spin'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "loading-text",
|
|
||||||
"type": "span",
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "children"}}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "button-normal-content",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "loading",
|
|
||||||
"transform": "!data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex items-center gap-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "left-icon",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "leftIcon",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex-shrink-0'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "slot", "source": "leftIcon"}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "button-text",
|
|
||||||
"type": "span",
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "children"}}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "right-icon",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "rightIcon",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex-shrink-0'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "slot", "source": "rightIcon"}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "calendar",
|
|
||||||
"type": "Calendar",
|
|
||||||
"bindings": {
|
|
||||||
"mode": {
|
|
||||||
"source": "mode",
|
|
||||||
"transform": "data || 'single'"
|
|
||||||
},
|
|
||||||
"selected": {
|
|
||||||
"source": "selected",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onSelect": {
|
|
||||||
"source": "onSelect",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"disabled": {
|
|
||||||
"source": "disabled",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); return cn(['rounded-md border', data || ''])"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "card",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"onClick": {
|
|
||||||
"source": "onClick",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["variant", "padding", "hover", "onClick", "className"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const variant = data[0] || 'default'; const padding = data[1] || 'md'; const hover = data[2]; const onClick = data[3]; const className = data[4] || ''; const variantStyles = { default: 'bg-card border border-border', bordered: 'bg-background border-2 border-border', elevated: 'bg-card shadow-lg border border-border', flat: 'bg-muted' }; const paddingStyles = { none: 'p-0', sm: 'p-3', md: 'p-6', lg: 'p-8' }; const hoverClass = (hover || onClick) ? 'hover:shadow-md hover:scale-[1.01] cursor-pointer' : ''; const cursorClass = onClick ? 'cursor-pointer' : ''; return cn(['rounded-lg transition-all', variantStyles[variant], paddingStyles[padding], hoverClass, cursorClass, className])"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "slot", "source": "children"}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,97 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "checkbox",
|
|
||||||
"type": "label",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": ["disabled", "className"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const disabled = data[0]; const className = data[1] || ''; const disabledClass = disabled ? 'opacity-50 cursor-not-allowed' : ''; return cn(['flex items-center gap-2 cursor-pointer', disabledClass, className])"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "checkbox-button",
|
|
||||||
"type": "button",
|
|
||||||
"bindings": {
|
|
||||||
"type": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'button'"
|
|
||||||
},
|
|
||||||
"role": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'checkbox'"
|
|
||||||
},
|
|
||||||
"aria-checked": {
|
|
||||||
"source": ["indeterminate", "checked"],
|
|
||||||
"transform": "data[0] ? 'mixed' : (data[1] ? 'true' : 'false')"
|
|
||||||
},
|
|
||||||
"disabled": {
|
|
||||||
"source": "disabled",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onClick": {
|
|
||||||
"source": ["disabled", "checked", "onChange"],
|
|
||||||
"transform": "const disabled = data[0]; const checked = data[1]; const onChange = data[2]; return !disabled ? () => onChange(!checked) : undefined"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["size", "checked", "indeterminate"],
|
|
||||||
"transform": "const cn = (classes) => classes.filter(Boolean).join(' '); const size = data[0] || 'md'; const checked = data[1]; const indeterminate = data[2]; const sizeStyles = { sm: 'w-4 h-4', md: 'w-5 h-5', lg: 'w-6 h-6' }; const stateClass = (checked || indeterminate) ? 'bg-primary border-primary text-primary-foreground' : 'bg-background border-input hover:border-ring'; return cn(['flex items-center justify-center rounded border-2 transition-colors', sizeStyles[size], stateClass])"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "indeterminate-icon",
|
|
||||||
"type": "Minus",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "indeterminate",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"size": {
|
|
||||||
"source": "size",
|
|
||||||
"transform": "const iconSize = { sm: 12, md: 16, lg: 20 }; return iconSize[data || 'md']"
|
|
||||||
},
|
|
||||||
"weight": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'bold'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "check-icon",
|
|
||||||
"type": "Check",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": ["checked", "indeterminate"],
|
|
||||||
"transform": "data[0] && !data[1]"
|
|
||||||
},
|
|
||||||
"size": {
|
|
||||||
"source": "size",
|
|
||||||
"transform": "const iconSize = { sm: 12, md: 16, lg: 20 }; return iconSize[data || 'md']"
|
|
||||||
},
|
|
||||||
"weight": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'bold'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "label-text",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "label",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-sm font-medium select-none'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "label"}}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "component-tree-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "className ? `space-y-2 ${className}` : 'space-y-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "empty-message",
|
|
||||||
"type": "p",
|
|
||||||
"props": { "className": "text-sm text-muted-foreground" },
|
|
||||||
"bindings": { "children": "emptyMessage" },
|
|
||||||
"conditional": { "if": "components.length === 0" }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "tree-nodes",
|
|
||||||
"type": "div",
|
|
||||||
"props": { "className": "space-y-1" },
|
|
||||||
"conditional": { "if": "components.length > 0" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "tree-node-list",
|
|
||||||
"type": "list",
|
|
||||||
"bindings": {
|
|
||||||
"items": "treeData",
|
|
||||||
"keyPath": "component.id"
|
|
||||||
},
|
|
||||||
"itemTemplate": {
|
|
||||||
"type": "button",
|
|
||||||
"props": { "type": "button" },
|
|
||||||
"bindings": {
|
|
||||||
"onClick": {
|
|
||||||
"source": "onSelect,item.component.id",
|
|
||||||
"transform": "() => onSelect?.(item.component.id)"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": "item.isSelected",
|
|
||||||
"transform": "item.isSelected ? 'flex w-full items-center gap-2 rounded-md px-2 py-1 text-left text-sm transition-colors bg-accent/40 text-foreground' : 'flex w-full items-center gap-2 rounded-md px-2 py-1 text-left text-sm transition-colors hover:bg-muted'"
|
|
||||||
},
|
|
||||||
"style": {
|
|
||||||
"source": "item.paddingLeft",
|
|
||||||
"transform": "{ paddingLeft: item.paddingLeft }"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "span",
|
|
||||||
"props": { "className": "font-medium" },
|
|
||||||
"bindings": { "children": "item.component.type" }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "span",
|
|
||||||
"props": { "className": "text-xs text-muted-foreground" },
|
|
||||||
"bindings": { "children": "item.component.id" }
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,236 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "context-menu",
|
|
||||||
"type": "ContextMenu",
|
|
||||||
"bindings": {},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "trigger",
|
|
||||||
"type": "ContextMenuTrigger",
|
|
||||||
"bindings": {
|
|
||||||
"asChild": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "true"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "slot", "source": "trigger"}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "content",
|
|
||||||
"type": "ContextMenuContent",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "menu-items",
|
|
||||||
"type": "Fragment",
|
|
||||||
"_map": {
|
|
||||||
"source": "items",
|
|
||||||
"itemVar": "item",
|
|
||||||
"indexVar": "index"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "menu-item",
|
|
||||||
"type": "ContextMenuSeparator",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "item.separator",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "submenu",
|
|
||||||
"type": "ContextMenuSub",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": ["item.submenu", "item.separator"],
|
|
||||||
"transform": "data[0] && data[0].length > 0 && !data[1]"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "submenu-trigger",
|
|
||||||
"type": "ContextMenuSubTrigger",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "submenu-icon",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "item.icon",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'mr-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "slot", "source": "item.icon"}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "submenu-label",
|
|
||||||
"type": "span",
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "item.label"}}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "submenu-content",
|
|
||||||
"type": "ContextMenuSubContent",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "nested-items",
|
|
||||||
"type": "Fragment",
|
|
||||||
"_map": {
|
|
||||||
"source": "item.submenu",
|
|
||||||
"itemVar": "subitem",
|
|
||||||
"indexVar": "subindex"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "nested-item",
|
|
||||||
"type": "ContextMenuItem",
|
|
||||||
"bindings": {
|
|
||||||
"onSelect": {
|
|
||||||
"source": "subitem.onSelect",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"disabled": {
|
|
||||||
"source": "subitem.disabled",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "nested-icon",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "subitem.icon",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'mr-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "slot", "source": "subitem.icon"}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "nested-label",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex-1'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "subitem.label"}}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "nested-shortcut",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "subitem.shortcut",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'ml-auto text-xs text-muted-foreground'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "subitem.shortcut"}}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "regular-item",
|
|
||||||
"type": "ContextMenuItem",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": ["item.submenu", "item.separator"],
|
|
||||||
"transform": "(!data[0] || data[0].length === 0) && !data[1]"
|
|
||||||
},
|
|
||||||
"onSelect": {
|
|
||||||
"source": "item.onSelect",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"disabled": {
|
|
||||||
"source": "item.disabled",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "item-icon",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "item.icon",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'mr-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "slot", "source": "item.icon"}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "item-label",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex-1'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "item.label"}}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "item-shortcut",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"_if": {
|
|
||||||
"source": "item.shortcut",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'ml-auto text-xs text-muted-foreground'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{"type": "text", "content": {"source": "item.shortcut"}}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "copy-button",
|
|
||||||
"type": "button",
|
|
||||||
"bindings": {
|
|
||||||
"onClick": {
|
|
||||||
"source": "copyState.handleCopy",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["copyState.copied", "size", "className"],
|
|
||||||
"transform": "const sizeStyles = { sm: 'p-1', md: 'p-2', lg: 'p-3' }; const size = data[1] || 'md'; const copied = data[0]; const className = data[2] || ''; const baseStyles = 'rounded-md transition-colors'; const stateStyles = copied ? 'bg-accent text-accent-foreground' : 'bg-muted text-muted-foreground hover:bg-accent hover:text-accent-foreground'; return `${baseStyles} ${stateStyles} ${sizeStyles[size]} ${className}`.trim()"
|
|
||||||
},
|
|
||||||
"aria-label": {
|
|
||||||
"source": "copyState.copied",
|
|
||||||
"transform": "data ? 'Copied' : 'Copy to clipboard'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "copy-icon",
|
|
||||||
"type": "Copy",
|
|
||||||
"bindings": {
|
|
||||||
"size": {
|
|
||||||
"source": "size",
|
|
||||||
"transform": "const iconSize = { sm: 12, md: 16, lg: 20 }; return iconSize[data || 'md']"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "copyState.copied",
|
|
||||||
"transform": "!data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "check-icon",
|
|
||||||
"type": "Check",
|
|
||||||
"bindings": {
|
|
||||||
"size": {
|
|
||||||
"source": "size",
|
|
||||||
"transform": "const iconSize = { sm: 12, md: 16, lg: 20 }; return iconSize[data || 'md']"
|
|
||||||
},
|
|
||||||
"weight": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'bold'"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "copyState.copied",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,265 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "data-source-manager",
|
|
||||||
"type": "div",
|
|
||||||
"props": { "className": "space-y-6" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "data-sources-card",
|
|
||||||
"type": "Card",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "card-header",
|
|
||||||
"type": "CardHeader",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "header-container",
|
|
||||||
"type": "div",
|
|
||||||
"props": { "className": "flex items-center justify-between" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "header-content",
|
|
||||||
"type": "Stack",
|
|
||||||
"props": { "direction": "vertical", "spacing": "xs" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "Heading",
|
|
||||||
"bindings": { "children": "headerCopy.title" },
|
|
||||||
"props": { "level": 2 }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "Text",
|
|
||||||
"bindings": { "children": "headerCopy.description" },
|
|
||||||
"props": { "variant": "muted" }
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "add-button-group",
|
|
||||||
"type": "div",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "DropdownMenu",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "DropdownMenuTrigger",
|
|
||||||
"props": { "asChild": true },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "div",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "ActionButton",
|
|
||||||
"bindings": {
|
|
||||||
"label": "headerCopy.addLabel",
|
|
||||||
"onClick": {
|
|
||||||
"source": "",
|
|
||||||
"transform": "() => {}"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"variant": "default"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PhosphorIcon",
|
|
||||||
"props": { "icon": "Plus", "className": "w-4 h-4" }
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "DropdownMenuContent",
|
|
||||||
"props": { "align": "end" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "DropdownMenuItem",
|
|
||||||
"bindings": {
|
|
||||||
"onClick": {
|
|
||||||
"source": "addDataSource",
|
|
||||||
"transform": "() => addDataSource('kv')"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PhosphorIcon",
|
|
||||||
"props": { "icon": "Database", "className": "w-4 h-4 mr-2" }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "text",
|
|
||||||
"bindings": { "children": "headerCopy.menu.kv" }
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "DropdownMenuItem",
|
|
||||||
"bindings": {
|
|
||||||
"onClick": {
|
|
||||||
"source": "addDataSource",
|
|
||||||
"transform": "() => addDataSource('static')"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PhosphorIcon",
|
|
||||||
"props": { "icon": "FileText", "className": "w-4 h-4 mr-2" }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "text",
|
|
||||||
"bindings": { "children": "headerCopy.menu.static" }
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "card-content",
|
|
||||||
"type": "CardContent",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "empty-or-content",
|
|
||||||
"type": "ConditionalRender",
|
|
||||||
"bindings": {
|
|
||||||
"condition": {
|
|
||||||
"source": "localSources",
|
|
||||||
"transform": "localSources.length === 0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "empty-state",
|
|
||||||
"type": "EmptyState",
|
|
||||||
"bindings": {
|
|
||||||
"title": "emptyStateCopy.title",
|
|
||||||
"description": "emptyStateCopy.description"
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"icon": {
|
|
||||||
"type": "PhosphorIcon",
|
|
||||||
"props": { "icon": "Database", "className": "w-12 h-12" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "sources-list",
|
|
||||||
"type": "Stack",
|
|
||||||
"props": { "direction": "vertical", "spacing": "xl" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "kv-section",
|
|
||||||
"type": "Section",
|
|
||||||
"bindings": {
|
|
||||||
"hidden": {
|
|
||||||
"source": "groupedSources",
|
|
||||||
"transform": "groupedSources.kv.length === 0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "IconText",
|
|
||||||
"props": { "className": "text-sm font-semibold mb-3" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PhosphorIcon",
|
|
||||||
"props": { "icon": "Database", "className": "w-4 h-4" }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "text",
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": "groupedSources",
|
|
||||||
"transform": "`${headerCopy.groups.kv} (${groupedSources.kv.length})`"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "Stack",
|
|
||||||
"props": { "direction": "vertical", "spacing": "sm" },
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": "groupedSources",
|
|
||||||
"transform": "groupedSources.kv.map((ds) => ({ id: ds.id, type: 'kv', item: ds }))"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "static-section",
|
|
||||||
"type": "Section",
|
|
||||||
"bindings": {
|
|
||||||
"hidden": {
|
|
||||||
"source": "groupedSources",
|
|
||||||
"transform": "groupedSources.static.length === 0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "IconText",
|
|
||||||
"props": { "className": "text-sm font-semibold mb-3" },
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "PhosphorIcon",
|
|
||||||
"props": { "icon": "FileText", "className": "w-4 h-4" }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "text",
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": "groupedSources",
|
|
||||||
"transform": "`${headerCopy.groups.static} (${groupedSources.static.length})`"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "Stack",
|
|
||||||
"props": { "direction": "vertical", "spacing": "sm" },
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": "groupedSources",
|
|
||||||
"transform": "groupedSources.static.map((ds) => ({ id: ds.id, type: 'static', item: ds }))"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "editor-dialog",
|
|
||||||
"type": "DataSourceEditorDialog",
|
|
||||||
"bindings": {
|
|
||||||
"open": "dialogOpen",
|
|
||||||
"dataSource": "editingSource",
|
|
||||||
"onOpenChange": {
|
|
||||||
"source": "setDialogOpen",
|
|
||||||
"transform": "setDialogOpen"
|
|
||||||
},
|
|
||||||
"onSave": {
|
|
||||||
"source": "updateDataSource",
|
|
||||||
"transform": "(source) => updateDataSource(source.id, source)"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,85 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "dialog-root",
|
|
||||||
"type": "Dialog",
|
|
||||||
"bindings": {
|
|
||||||
"open": "data.open",
|
|
||||||
"onOpenChange": "handlers.handleOpenChange"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dialog-trigger",
|
|
||||||
"type": "DialogTrigger",
|
|
||||||
"bindings": {
|
|
||||||
"asChild": true
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dialog-trigger-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "trigger",
|
|
||||||
"defaultContent": {
|
|
||||||
"id": "trigger-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.triggerLabel || 'Open'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dialog-content",
|
|
||||||
"type": "DialogContent",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dialog-header",
|
|
||||||
"type": "DialogHeader",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dialog-title",
|
|
||||||
"type": "DialogTitle",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.title"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dialog-description",
|
|
||||||
"type": "DialogDescription",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.description"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dialog-body",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "children"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dialog-footer",
|
|
||||||
"type": "DialogFooter",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dialog-close-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"variant": "outline",
|
|
||||||
"children": "data.cancelLabel || 'Cancel'",
|
|
||||||
"onClick": "handlers.handleCancel"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dialog-submit-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.submitLabel || 'Submit'",
|
|
||||||
"onClick": "handlers.handleSubmit"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,86 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "drawer-root",
|
|
||||||
"type": "Drawer",
|
|
||||||
"bindings": {
|
|
||||||
"open": "data.open",
|
|
||||||
"onOpenChange": "handlers.handleOpenChange",
|
|
||||||
"direction": "data.direction || 'right'"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "drawer-trigger",
|
|
||||||
"type": "DrawerTrigger",
|
|
||||||
"bindings": {
|
|
||||||
"asChild": true
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "drawer-trigger-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "trigger",
|
|
||||||
"defaultContent": {
|
|
||||||
"id": "trigger-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.triggerLabel || 'Open'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "drawer-content",
|
|
||||||
"type": "DrawerContent",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "drawer-header",
|
|
||||||
"type": "DrawerHeader",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "drawer-title",
|
|
||||||
"type": "DrawerTitle",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.title"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "drawer-description",
|
|
||||||
"type": "DrawerDescription",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.description"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "drawer-body",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "children"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "drawer-footer",
|
|
||||||
"type": "DrawerFooter",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "drawer-close-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"variant": "outline",
|
|
||||||
"children": "data.cancelLabel || 'Cancel'",
|
|
||||||
"onClick": "handlers.handleCancel"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "drawer-submit-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.submitLabel || 'Submit'",
|
|
||||||
"onClick": "handlers.handleSubmit"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "dropdown-menu-root",
|
|
||||||
"type": "DropdownMenu",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dropdown-menu-trigger",
|
|
||||||
"type": "DropdownMenuTrigger",
|
|
||||||
"bindings": {
|
|
||||||
"asChild": true
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dropdown-trigger-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "trigger",
|
|
||||||
"defaultContent": {
|
|
||||||
"id": "trigger-button",
|
|
||||||
"type": "Button",
|
|
||||||
"bindings": {
|
|
||||||
"variant": "ghost",
|
|
||||||
"size": "sm",
|
|
||||||
"children": "data.triggerLabel || 'Menu'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "dropdown-menu-content",
|
|
||||||
"type": "DropdownMenuContent",
|
|
||||||
"bindings": {
|
|
||||||
"align": "data.align || 'end'",
|
|
||||||
"side": "data.side || 'bottom'"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "dropdown-menu-items",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "children"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,190 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "file-upload-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "const base = 'w-full'; return data ? `${base} ${data}` : base"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "file-upload-label",
|
|
||||||
"type": "label",
|
|
||||||
"bindings": {
|
|
||||||
"onDrop": {
|
|
||||||
"source": "uploadState.handleDrop",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onDragOver": {
|
|
||||||
"source": "uploadState.handleDragOver",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onDragLeave": {
|
|
||||||
"source": "uploadState.handleDragLeave",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["uploadState.isDragging", "disabled"],
|
|
||||||
"transform": "const isDragging = data[0]; const disabled = data[1]; const base = 'flex flex-col items-center justify-center w-full h-32 border-2 border-dashed rounded-lg cursor-pointer transition-colors'; const dragStyle = isDragging ? 'border-primary bg-primary/5' : 'border-border bg-muted/30 hover:bg-muted/50'; const disabledStyle = disabled ? 'opacity-50 cursor-not-allowed' : ''; return `${base} ${dragStyle} ${disabledStyle}`.trim()"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "file-upload-content",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'flex flex-col items-center justify-center gap-2'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "upload-icon",
|
|
||||||
"type": "UploadSimple",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'w-8 h-8 text-muted-foreground'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "upload-text",
|
|
||||||
"type": "p",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-sm text-muted-foreground'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "upload-text-bold",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'font-medium'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'Click to upload'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "upload-text-or",
|
|
||||||
"type": "_text",
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "' or drag and drop'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "accept-text",
|
|
||||||
"type": "p",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-xs text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "accept",
|
|
||||||
"transform": "data ? data.split(',').join(', ') : ''"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "accept",
|
|
||||||
"transform": "!!data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "maxsize-text",
|
|
||||||
"type": "p",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-xs text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "maxSize",
|
|
||||||
"transform": "`Max size: ${(data / 1024 / 1024).toFixed(1)}MB`"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "maxSize",
|
|
||||||
"transform": "!!data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "file-input",
|
|
||||||
"type": "input",
|
|
||||||
"bindings": {
|
|
||||||
"type": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'file'"
|
|
||||||
},
|
|
||||||
"accept": {
|
|
||||||
"source": "accept",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"multiple": {
|
|
||||||
"source": "multiple",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onChange": {
|
|
||||||
"source": "uploadState.handleFiles",
|
|
||||||
"transform": "(e) => data(e.target.files)"
|
|
||||||
},
|
|
||||||
"disabled": {
|
|
||||||
"source": "disabled",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'hidden'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "file-list-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'mt-4 space-y-2'"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "uploadState.selectedFiles",
|
|
||||||
"transform": "data && data.length > 0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "file-list-repeat",
|
|
||||||
"type": "_repeat",
|
|
||||||
"bindings": {
|
|
||||||
"_items": {
|
|
||||||
"source": "uploadState.selectedFiles",
|
|
||||||
"transform": "data || []"
|
|
||||||
},
|
|
||||||
"_renderItem": {
|
|
||||||
"source": "uploadState.removeFile",
|
|
||||||
"transform": "(file, index) => ({ _element: 'div', _key: index, _props: { className: 'flex items-center justify-between p-3 bg-muted rounded-lg' }, _children: [{ _element: 'div', _props: { className: 'flex-1 min-w-0' }, _children: [{ _element: 'p', _props: { className: 'text-sm font-medium truncate' }, _children: [file.name] }, { _element: 'p', _props: { className: 'text-xs text-muted-foreground' }, _children: [`${(file.size / 1024).toFixed(1)} KB`] }] }, { _element: 'button', _props: { type: 'button', onClick: () => data(index), className: 'ml-2 p-1 hover:bg-background rounded transition-colors', 'aria-label': 'Remove file' }, _children: [{ _element: 'X', _props: { className: 'w-4 h-4' } }] }] })"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "filter-input-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "data ? `relative ${data}` : 'relative'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "filter-input-icon",
|
|
||||||
"type": "MagnifyingGlass",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "focusState.isFocused",
|
|
||||||
"transform": "data ? 'absolute left-3 top-1/2 -translate-y-1/2 transition-colors text-primary' : 'absolute left-3 top-1/2 -translate-y-1/2 transition-colors text-muted-foreground'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"size": 16
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "filter-input-field",
|
|
||||||
"type": "Input",
|
|
||||||
"bindings": {
|
|
||||||
"value": "value",
|
|
||||||
"placeholder": "placeholder",
|
|
||||||
"onFocus": {
|
|
||||||
"source": "focusState.setFocused",
|
|
||||||
"transform": "() => data(true)"
|
|
||||||
},
|
|
||||||
"onBlur": {
|
|
||||||
"source": "focusState.setFocused",
|
|
||||||
"transform": "() => data(false)"
|
|
||||||
},
|
|
||||||
"onChange": {
|
|
||||||
"source": "onChange",
|
|
||||||
"transform": "(e) => data(e.target.value)"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"className": "pl-9 pr-9"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "filter-input-clear",
|
|
||||||
"type": "button",
|
|
||||||
"conditional": {
|
|
||||||
"if": "value"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"onClick": {
|
|
||||||
"source": "onChange",
|
|
||||||
"transform": "() => data('')"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"className": "absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground transition-colors",
|
|
||||||
"type": "button"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "filter-input-clear-icon",
|
|
||||||
"type": "X",
|
|
||||||
"props": {
|
|
||||||
"size": 16
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "form-field-root",
|
|
||||||
"type": "FormField",
|
|
||||||
"bindings": {
|
|
||||||
"className": "data.className",
|
|
||||||
"control": "data.control",
|
|
||||||
"name": "data.name"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "form-field-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "children"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "form-root",
|
|
||||||
"type": "Form",
|
|
||||||
"bindings": {
|
|
||||||
"className": "data.className",
|
|
||||||
"onSubmit": "handlers.handleSubmit"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "form-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "children"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "heading-root",
|
|
||||||
"type": "Heading",
|
|
||||||
"bindings": {
|
|
||||||
"level": "data.level || 1",
|
|
||||||
"className": "data.className"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "heading-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "children"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "hover-card-root",
|
|
||||||
"type": "HoverCard",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "hover-card-trigger",
|
|
||||||
"type": "HoverCardTrigger",
|
|
||||||
"bindings": {
|
|
||||||
"asChild": true
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "hover-trigger-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "trigger",
|
|
||||||
"defaultContent": {
|
|
||||||
"id": "trigger-element",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"children": "data.triggerLabel || 'Hover'"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "hover-card-content",
|
|
||||||
"type": "HoverCardContent",
|
|
||||||
"bindings": {
|
|
||||||
"side": "data.side || 'bottom'",
|
|
||||||
"align": "data.align || 'center'",
|
|
||||||
"className": "data.className"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "hover-content",
|
|
||||||
"type": "slot",
|
|
||||||
"slot": "children"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "icon-root",
|
|
||||||
"type": "Icon",
|
|
||||||
"bindings": {
|
|
||||||
"name": "data.name",
|
|
||||||
"size": "data.size || 'md'",
|
|
||||||
"className": "data.className",
|
|
||||||
"color": "data.color",
|
|
||||||
"strokeWidth": "data.strokeWidth"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "image-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "const base = 'relative overflow-hidden'; return data ? `${base} ${data}` : base"
|
|
||||||
},
|
|
||||||
"style": {
|
|
||||||
"source": ["width", "height"],
|
|
||||||
"transform": "const width = data[0]; const height = data[1]; return { width: typeof width === 'number' ? `${width}px` : width, height: typeof height === 'number' ? `${height}px` : height }"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "image-loading",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'absolute inset-0 bg-muted animate-pulse'"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "imageState.loading",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "image-element",
|
|
||||||
"type": "img",
|
|
||||||
"bindings": {
|
|
||||||
"src": {
|
|
||||||
"source": ["imageState.error", "fallback", "src"],
|
|
||||||
"transform": "const error = data[0]; const fallback = data[1]; const src = data[2]; return (error && fallback) ? fallback : src"
|
|
||||||
},
|
|
||||||
"alt": {
|
|
||||||
"source": "alt",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onLoad": {
|
|
||||||
"source": "imageState.handleLoad",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onError": {
|
|
||||||
"source": "imageState.handleError",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["imageState.loading", "fit"],
|
|
||||||
"transform": "const loading = data[0]; const fit = data[1] || 'cover'; const base = 'w-full h-full transition-opacity'; const opacity = loading ? 'opacity-0' : 'opacity-100'; return `${base} ${opacity} object-${fit}`"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "input-otp-wrapper",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"data-slot": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'input-otp'"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["containerClassName"],
|
|
||||||
"transform": "const baseClass = 'flex items-center gap-2 has-disabled:opacity-50'; const containerClass = data[0] || ''; return containerClass ? `${baseClass} ${containerClass}`.trim() : baseClass"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "children",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"_spreadProps": {
|
|
||||||
"source": "_spreadProps",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,114 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "input-wrapper",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'w-full'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "input-label",
|
|
||||||
"type": "label",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'block text-sm font-medium mb-1.5 text-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "label",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "label",
|
|
||||||
"transform": "!!data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "input-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'relative'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "input-left-icon",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'absolute left-3 top-1/2 -translate-y-1/2 text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "leftIcon",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "leftIcon",
|
|
||||||
"transform": "!!data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "input-element",
|
|
||||||
"type": "input",
|
|
||||||
"bindings": {
|
|
||||||
"ref": {
|
|
||||||
"source": "_ref",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["error", "leftIcon", "rightIcon", "className"],
|
|
||||||
"transform": "const error = data[0]; const leftIcon = data[1]; const rightIcon = data[2]; const className = data[3] || ''; const baseClasses = 'flex h-10 w-full rounded-md border bg-background px-3 py-2 text-sm placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 transition-colors'; const errorClass = error ? 'border-destructive focus-visible:ring-destructive' : 'border-input'; const leftIconClass = leftIcon ? 'pl-10' : ''; const rightIconClass = rightIcon ? 'pr-10' : ''; return `${baseClasses} ${errorClass} ${leftIconClass} ${rightIconClass} ${className}`.trim()"
|
|
||||||
},
|
|
||||||
"_spreadProps": {
|
|
||||||
"source": "_spreadProps",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "input-right-icon",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "rightIcon",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "rightIcon",
|
|
||||||
"transform": "!!data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "input-helper-text",
|
|
||||||
"type": "p",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": "error",
|
|
||||||
"transform": "const baseClass = 'text-xs mt-1.5'; return data ? `${baseClass} text-destructive` : `${baseClass} text-muted-foreground`"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "helperText",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "helperText",
|
|
||||||
"transform": "!!data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "label-wrapper",
|
|
||||||
"type": "label",
|
|
||||||
"bindings": {
|
|
||||||
"htmlFor": {
|
|
||||||
"source": "htmlFor",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": ["required", "className"],
|
|
||||||
"transform": "const required = data[0]; const className = data[1] || ''; const baseClasses = 'text-sm font-medium text-foreground leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70'; return required ? `${baseClasses} ${className}`.trim() : `${baseClasses} ${className}`.trim()"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "label-content",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"children": {
|
|
||||||
"source": "children",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "label-required-indicator",
|
|
||||||
"type": "span",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'text-destructive ml-1'"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'*'"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "required",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,76 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "d3-bar-chart-svg",
|
|
||||||
"type": "svg",
|
|
||||||
"bindings": {
|
|
||||||
"width": "width",
|
|
||||||
"height": "height",
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "className ? `overflow-visible ${className}` : 'overflow-visible'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "chart-group",
|
|
||||||
"type": "g",
|
|
||||||
"bindings": {
|
|
||||||
"transform": {
|
|
||||||
"source": "chartData.translateX,chartData.translateY",
|
|
||||||
"transform": "`translate(${chartData.translateX},${chartData.translateY})`"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "bars-list",
|
|
||||||
"type": "list",
|
|
||||||
"bindings": {
|
|
||||||
"items": "chartData.bars",
|
|
||||||
"keyPath": "label"
|
|
||||||
},
|
|
||||||
"itemTemplate": {
|
|
||||||
"type": "g",
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"type": "rect",
|
|
||||||
"bindings": {
|
|
||||||
"x": "item.x",
|
|
||||||
"y": "item.y",
|
|
||||||
"width": "item.width",
|
|
||||||
"height": "item.height",
|
|
||||||
"fill": "color"
|
|
||||||
},
|
|
||||||
"props": { "rx": 2 }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "text",
|
|
||||||
"bindings": {
|
|
||||||
"x": "item.labelX",
|
|
||||||
"y": "item.labelY",
|
|
||||||
"children": "item.label"
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"textAnchor": "middle",
|
|
||||||
"fill": "currentColor",
|
|
||||||
"style": { "fontSize": 10 }
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "text",
|
|
||||||
"bindings": {
|
|
||||||
"x": "item.valueX",
|
|
||||||
"y": "item.valueY",
|
|
||||||
"children": "item.value"
|
|
||||||
},
|
|
||||||
"props": {
|
|
||||||
"textAnchor": "middle",
|
|
||||||
"fill": "currentColor",
|
|
||||||
"style": { "fontSize": 10 }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "loading-container",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "flex items-center justify-center h-full w-full"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "loading-content",
|
|
||||||
"type": "div",
|
|
||||||
"props": {
|
|
||||||
"className": "flex flex-col items-center gap-3"
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "loading-spinner",
|
|
||||||
"type": "LoadingSpinner"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "loading-message",
|
|
||||||
"type": "p",
|
|
||||||
"props": {
|
|
||||||
"className": "text-sm text-muted-foreground"
|
|
||||||
},
|
|
||||||
"bindings": {
|
|
||||||
"children": "message"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,76 +0,0 @@
|
|||||||
{
|
|
||||||
"id": "menu-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'relative inline-block'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "menu-trigger",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"ref": {
|
|
||||||
"source": "menuState.triggerRef",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"onClick": {
|
|
||||||
"source": "menuState.toggle",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"children": {
|
|
||||||
"source": "trigger",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "menu-content",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"ref": {
|
|
||||||
"source": "menuState.popoverRef",
|
|
||||||
"transform": "data"
|
|
||||||
},
|
|
||||||
"className": {
|
|
||||||
"source": "className",
|
|
||||||
"transform": "const base = 'absolute z-50 mt-2 w-56 bg-popover text-popover-foreground border border-border rounded-lg shadow-lg overflow-hidden animate-in fade-in-0 zoom-in-95'; return data ? `${base} ${data}` : base"
|
|
||||||
},
|
|
||||||
"_if": {
|
|
||||||
"source": "menuState.isOpen",
|
|
||||||
"transform": "data"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "menu-items-container",
|
|
||||||
"type": "div",
|
|
||||||
"bindings": {
|
|
||||||
"className": {
|
|
||||||
"source": null,
|
|
||||||
"transform": "'py-1'"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"children": [
|
|
||||||
{
|
|
||||||
"id": "menu-items-repeat",
|
|
||||||
"type": "_repeat",
|
|
||||||
"bindings": {
|
|
||||||
"_items": {
|
|
||||||
"source": "items",
|
|
||||||
"transform": "data || []"
|
|
||||||
},
|
|
||||||
"_renderItem": {
|
|
||||||
"source": ["menuState.handleItemClick"],
|
|
||||||
"transform": "(item, index) => { if (item.divider) { return { _element: 'div', _key: index, _props: { className: 'my-1 h-px bg-border' } }; } return { _element: 'button', _key: item.id, _props: { onClick: () => data[0](item), disabled: item.disabled, className: `w-full flex items-center justify-between px-3 py-2 text-sm transition-colors hover:bg-accent hover:text-accent-foreground ${item.disabled ? 'opacity-50 cursor-not-allowed' : ''} ${item.danger ? 'text-destructive hover:bg-destructive hover:text-destructive-foreground' : ''}` }, _children: [{ _element: 'div', _props: { className: 'flex items-center gap-2' }, _children: [item.icon ? { _element: 'span', _props: { className: 'w-4 h-4' }, _children: [item.icon] } : null, { _element: 'span', _children: [item.label] }].filter(Boolean) }, { _element: 'div', _props: { className: 'flex items-center gap-2' }, _children: [item.shortcut ? { _element: 'span', _props: { className: 'text-xs text-muted-foreground' }, _children: [item.shortcut] } : null, item.selected ? { _element: 'Check', _props: { className: 'w-4 h-4' } } : null].filter(Boolean) }] }; }"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user