mirror of
https://github.com/johndoe6345789/low-code-react-app-b.git
synced 2026-04-24 13:44:54 +00:00
Fix component registry to include all atoms, molecules, and organisms with JSON compatibility analysis
Co-authored-by: johndoe6345789 <224850594+johndoe6345789@users.noreply.github.com>
This commit is contained in:
File diff suppressed because it is too large
Load Diff
@@ -21,7 +21,8 @@
|
|||||||
"pages:list": "node scripts/list-pages.js",
|
"pages:list": "node scripts/list-pages.js",
|
||||||
"pages:validate": "tsx src/config/validate-config.ts",
|
"pages:validate": "tsx src/config/validate-config.ts",
|
||||||
"pages:generate": "node scripts/generate-page.js",
|
"pages:generate": "node scripts/generate-page.js",
|
||||||
"components:list": "node scripts/list-json-components.cjs"
|
"components:list": "node scripts/list-json-components.cjs",
|
||||||
|
"components:scan": "node scripts/scan-and-update-registry.cjs"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@heroicons/react": "^2.2.0",
|
"@heroicons/react": "^2.2.0",
|
||||||
|
|||||||
@@ -73,15 +73,21 @@ categories.forEach(category => {
|
|||||||
|
|
||||||
categoryComps.forEach(comp => {
|
categoryComps.forEach(comp => {
|
||||||
const children = comp.canHaveChildren ? '👶 Can have children' : '➖ No children'
|
const children = comp.canHaveChildren ? '👶 Can have children' : '➖ No children'
|
||||||
const statusIcon = comp.status === 'supported' ? '✅' : '📋'
|
let statusIcon = comp.status === 'supported' ? '✅' : '📋'
|
||||||
const subComps = comp.subComponents ? ` (includes: ${comp.subComponents.join(', ')})` : ''
|
if (comp.status === 'json-compatible') statusIcon = '🔥'
|
||||||
|
if (comp.status === 'maybe-json-compatible') statusIcon = '⚠️ '
|
||||||
|
|
||||||
console.log(` ${statusIcon} ${comp.name} (${comp.type})`)
|
const source = comp.source ? ` [${comp.source}]` : ''
|
||||||
|
|
||||||
|
console.log(` ${statusIcon} ${comp.name} (${comp.type})${source}`)
|
||||||
console.log(` ${comp.description}`)
|
console.log(` ${comp.description}`)
|
||||||
console.log(` ${children}`)
|
console.log(` ${children}`)
|
||||||
if (comp.subComponents) {
|
if (comp.subComponents) {
|
||||||
console.log(` Sub-components: ${comp.subComponents.join(', ')}`)
|
console.log(` Sub-components: ${comp.subComponents.join(', ')}`)
|
||||||
}
|
}
|
||||||
|
if (comp.jsonCompatible !== undefined && !comp.jsonCompatible) {
|
||||||
|
console.log(` ⚠️ Not JSON-powered (${comp.jsonReason || 'complex state/logic'})`)
|
||||||
|
}
|
||||||
console.log('')
|
console.log('')
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@@ -92,9 +98,24 @@ console.log(`\nTotal Components: ${componentsList.length}`)
|
|||||||
if (statusFilter === 'all') {
|
if (statusFilter === 'all') {
|
||||||
const supported = componentsList.filter(c => c.status === 'supported').length
|
const supported = componentsList.filter(c => c.status === 'supported').length
|
||||||
const planned = componentsList.filter(c => c.status === 'planned').length
|
const planned = componentsList.filter(c => c.status === 'planned').length
|
||||||
|
const jsonCompatible = componentsList.filter(c => c.status === 'json-compatible').length
|
||||||
|
const maybeCompatible = componentsList.filter(c => c.status === 'maybe-json-compatible').length
|
||||||
|
const atoms = componentsList.filter(c => c.source === 'atoms').length
|
||||||
|
const molecules = componentsList.filter(c => c.source === 'molecules').length
|
||||||
|
const organisms = componentsList.filter(c => c.source === 'organisms').length
|
||||||
|
const ui = componentsList.filter(c => c.source === 'ui').length
|
||||||
|
|
||||||
console.log(`\nBy Status:`)
|
console.log(`\nBy Status:`)
|
||||||
console.log(` ✅ Supported: ${supported}`)
|
console.log(` ✅ Supported: ${supported}`)
|
||||||
|
console.log(` 🔥 JSON-Compatible: ${jsonCompatible}`)
|
||||||
|
console.log(` ⚠️ Maybe JSON-Compatible: ${maybeCompatible}`)
|
||||||
console.log(` 📋 Planned: ${planned}`)
|
console.log(` 📋 Planned: ${planned}`)
|
||||||
|
|
||||||
|
console.log(`\nBy Source:`)
|
||||||
|
if (atoms > 0) console.log(` 🧱 Atoms: ${atoms}`)
|
||||||
|
if (molecules > 0) console.log(` 🧪 Molecules: ${molecules}`)
|
||||||
|
if (organisms > 0) console.log(` 🦠 Organisms: ${organisms}`)
|
||||||
|
if (ui > 0) console.log(` 🎨 UI: ${ui}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(`\nBy Category:`)
|
console.log(`\nBy Category:`)
|
||||||
|
|||||||
368
scripts/scan-and-update-registry.cjs
Normal file
368
scripts/scan-and-update-registry.cjs
Normal file
@@ -0,0 +1,368 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scan and Update JSON Components Registry
|
||||||
|
*
|
||||||
|
* Scans the actual component files in src/components and updates
|
||||||
|
* json-components-registry.json to include all real components.
|
||||||
|
*
|
||||||
|
* Usage:
|
||||||
|
* node scripts/scan-and-update-registry.cjs
|
||||||
|
*/
|
||||||
|
|
||||||
|
const fs = require('fs')
|
||||||
|
const path = require('path')
|
||||||
|
|
||||||
|
// Scan a directory for .tsx files
|
||||||
|
function scanComponents(dir) {
|
||||||
|
const files = fs.readdirSync(dir)
|
||||||
|
return files
|
||||||
|
.filter(f => f.endsWith('.tsx') && !f.startsWith('index'))
|
||||||
|
.map(f => f.replace('.tsx', ''))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get all components
|
||||||
|
const atomsPath = path.join(process.cwd(), 'src/components/atoms')
|
||||||
|
const moleculesPath = path.join(process.cwd(), 'src/components/molecules')
|
||||||
|
const organismsPath = path.join(process.cwd(), 'src/components/organisms')
|
||||||
|
const uiPath = path.join(process.cwd(), 'src/components/ui')
|
||||||
|
|
||||||
|
const atoms = scanComponents(atomsPath)
|
||||||
|
const molecules = scanComponents(moleculesPath)
|
||||||
|
const organisms = fs.existsSync(organismsPath) ? scanComponents(organismsPath) : []
|
||||||
|
const ui = scanComponents(uiPath)
|
||||||
|
|
||||||
|
console.log(`Found ${atoms.length} atoms, ${molecules.length} molecules, ${organisms.length} organisms, ${ui.length} ui components`)
|
||||||
|
console.log(`Total: ${atoms.length + molecules.length + organisms.length + ui.length} components`)
|
||||||
|
|
||||||
|
// Read existing registry to preserve metadata
|
||||||
|
const registryPath = path.join(process.cwd(), 'json-components-registry.json')
|
||||||
|
let existingRegistry = { components: [] }
|
||||||
|
if (fs.existsSync(registryPath)) {
|
||||||
|
existingRegistry = JSON.parse(fs.readFileSync(registryPath, 'utf8'))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a map of existing components for quick lookup
|
||||||
|
const existingMap = new Map()
|
||||||
|
existingRegistry.components.forEach(c => {
|
||||||
|
existingMap.set(c.type, c)
|
||||||
|
})
|
||||||
|
|
||||||
|
// Category mapping heuristics
|
||||||
|
function guessCategory(name) {
|
||||||
|
const lower = name.toLowerCase()
|
||||||
|
|
||||||
|
// Layout
|
||||||
|
if (lower.includes('container') || lower.includes('grid') || lower.includes('flex') ||
|
||||||
|
lower.includes('stack') || lower.includes('card') || lower.includes('section') ||
|
||||||
|
lower.includes('drawer') || lower.includes('modal') || lower.includes('dialog')) {
|
||||||
|
return 'layout'
|
||||||
|
}
|
||||||
|
|
||||||
|
// Input
|
||||||
|
if (lower.includes('input') || lower.includes('button') || lower.includes('select') ||
|
||||||
|
lower.includes('checkbox') || lower.includes('radio') || lower.includes('switch') ||
|
||||||
|
lower.includes('slider') || lower.includes('form') || lower.includes('upload') ||
|
||||||
|
lower.includes('picker') || lower.includes('toggle')) {
|
||||||
|
return 'input'
|
||||||
|
}
|
||||||
|
|
||||||
|
// Navigation
|
||||||
|
if (lower.includes('link') || lower.includes('breadcrumb') || lower.includes('tab') ||
|
||||||
|
lower.includes('menu') || lower.includes('navigation')) {
|
||||||
|
return 'navigation'
|
||||||
|
}
|
||||||
|
|
||||||
|
// Feedback
|
||||||
|
if (lower.includes('alert') || lower.includes('notification') || lower.includes('badge') ||
|
||||||
|
lower.includes('status') || lower.includes('error') || lower.includes('empty') ||
|
||||||
|
lower.includes('loading') || lower.includes('spinner') || lower.includes('toast')) {
|
||||||
|
return 'feedback'
|
||||||
|
}
|
||||||
|
|
||||||
|
// Data
|
||||||
|
if (lower.includes('table') || lower.includes('list') || lower.includes('data') ||
|
||||||
|
lower.includes('metric') || lower.includes('stat') || lower.includes('chart') ||
|
||||||
|
lower.includes('timeline') || lower.includes('keyvalue')) {
|
||||||
|
return 'data'
|
||||||
|
}
|
||||||
|
|
||||||
|
// Display (default for text, images, icons, etc.)
|
||||||
|
if (lower.includes('text') || lower.includes('heading') || lower.includes('label') ||
|
||||||
|
lower.includes('image') || lower.includes('avatar') || lower.includes('icon') ||
|
||||||
|
lower.includes('code') || lower.includes('tag') || lower.includes('skeleton') ||
|
||||||
|
lower.includes('separator') || lower.includes('divider') || lower.includes('progress')) {
|
||||||
|
return 'display'
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'custom'
|
||||||
|
}
|
||||||
|
|
||||||
|
function canHaveChildren(name) {
|
||||||
|
const noChildren = [
|
||||||
|
'Input', 'TextArea', 'Select', 'Checkbox', 'Radio', 'Switch', 'Slider', 'NumberInput',
|
||||||
|
'Image', 'Avatar', 'Separator', 'Divider', 'Progress', 'ProgressBar', 'Skeleton',
|
||||||
|
'Spinner', 'Icon', 'FileUpload', 'DatePicker', 'CircularProgress', 'StatusIcon',
|
||||||
|
'StatusBadge', 'ErrorBadge', 'Table', 'DataTable', 'List', 'DataList', 'KeyValue',
|
||||||
|
'StatCard', 'MetricCard', 'DataCard', 'SearchInput', 'ActionBar', 'Timeline'
|
||||||
|
]
|
||||||
|
return !noChildren.includes(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDescription(name) {
|
||||||
|
// Try to generate a reasonable description
|
||||||
|
const descriptions = {
|
||||||
|
// Common patterns
|
||||||
|
'Accordion': 'Collapsible content sections',
|
||||||
|
'ActionButton': 'Button with action icon',
|
||||||
|
'ActionBar': 'Action button toolbar',
|
||||||
|
'Alert': 'Alert notification message',
|
||||||
|
'Avatar': 'User avatar image',
|
||||||
|
'AvatarGroup': 'Group of user avatars',
|
||||||
|
'Badge': 'Small status or count indicator',
|
||||||
|
'Breadcrumb': 'Navigation breadcrumb trail',
|
||||||
|
'Button': 'Interactive button element',
|
||||||
|
'ButtonGroup': 'Group of related buttons',
|
||||||
|
'Calendar': 'Calendar date selector',
|
||||||
|
'Card': 'Container card component',
|
||||||
|
'Checkbox': 'Checkbox toggle control',
|
||||||
|
'Chip': 'Compact element for tags or selections',
|
||||||
|
'CircularProgress': 'Circular progress indicator',
|
||||||
|
'Code': 'Inline or block code display',
|
||||||
|
'CommandPalette': 'Command search and execution',
|
||||||
|
'Container': 'Generic container element',
|
||||||
|
'ContextMenu': 'Right-click context menu',
|
||||||
|
'DataCard': 'Custom data display card',
|
||||||
|
'DataList': 'Styled data list',
|
||||||
|
'DataTable': 'Advanced data table with sorting and filtering',
|
||||||
|
'DatePicker': 'Date selection input',
|
||||||
|
'Divider': 'Visual section divider',
|
||||||
|
'Drawer': 'Sliding panel overlay',
|
||||||
|
'EmptyState': 'Empty state placeholder',
|
||||||
|
'ErrorBadge': 'Error state badge',
|
||||||
|
'FileUpload': 'File upload control',
|
||||||
|
'Flex': 'Flexible box layout container',
|
||||||
|
'Form': 'Form container component',
|
||||||
|
'Grid': 'Responsive grid layout',
|
||||||
|
'Heading': 'Heading text with level (h1-h6)',
|
||||||
|
'HoverCard': 'Card shown on hover',
|
||||||
|
'Icon': 'Icon from icon library',
|
||||||
|
'IconButton': 'Button with icon only',
|
||||||
|
'Image': 'Image element with loading states',
|
||||||
|
'InfoBox': 'Information box with icon',
|
||||||
|
'Input': 'Text input field',
|
||||||
|
'Kbd': 'Keyboard key display',
|
||||||
|
'KeyValue': 'Key-value pair display',
|
||||||
|
'Label': 'Form label element',
|
||||||
|
'Link': 'Hyperlink element',
|
||||||
|
'List': 'Generic list renderer with custom items',
|
||||||
|
'Menu': 'Menu component',
|
||||||
|
'MetricCard': 'Metric display card',
|
||||||
|
'Modal': 'Modal dialog overlay',
|
||||||
|
'Notification': 'Toast notification',
|
||||||
|
'NumberInput': 'Numeric input with increment/decrement',
|
||||||
|
'PasswordInput': 'Password input with visibility toggle',
|
||||||
|
'Popover': 'Popover overlay content',
|
||||||
|
'Progress': 'Progress bar indicator',
|
||||||
|
'ProgressBar': 'Linear progress bar',
|
||||||
|
'Radio': 'Radio button selection',
|
||||||
|
'Rating': 'Star rating component',
|
||||||
|
'ScrollArea': 'Scrollable container area',
|
||||||
|
'SearchInput': 'Search input with icon',
|
||||||
|
'Select': 'Dropdown select control',
|
||||||
|
'Separator': 'Visual divider line',
|
||||||
|
'Skeleton': 'Loading skeleton placeholder',
|
||||||
|
'Slider': 'Numeric range slider',
|
||||||
|
'Spinner': 'Loading spinner',
|
||||||
|
'Stack': 'Vertical or horizontal stack layout',
|
||||||
|
'StatCard': 'Statistic card display',
|
||||||
|
'StatusBadge': 'Status indicator badge',
|
||||||
|
'StatusIcon': 'Status indicator icon',
|
||||||
|
'Stepper': 'Step-by-step navigation',
|
||||||
|
'Switch': 'Toggle switch control',
|
||||||
|
'Table': 'Data table',
|
||||||
|
'Tabs': 'Tabbed interface container',
|
||||||
|
'Tag': 'Removable tag or chip',
|
||||||
|
'Text': 'Text content with typography variants',
|
||||||
|
'TextArea': 'Multi-line text input',
|
||||||
|
'Timeline': 'Timeline visualization',
|
||||||
|
'Toggle': 'Toggle button control',
|
||||||
|
'Tooltip': 'Tooltip overlay text'
|
||||||
|
}
|
||||||
|
|
||||||
|
return descriptions[name] || `${name} component`
|
||||||
|
}
|
||||||
|
|
||||||
|
// JSON compatibility lists based on analysis
|
||||||
|
const jsonCompatibleMolecules = [
|
||||||
|
'AppBranding', 'Breadcrumb', 'EmptyEditorState', 'LabelWithBadge',
|
||||||
|
'LazyBarChart', 'LazyD3BarChart', 'LazyLineChart', 'LoadingFallback',
|
||||||
|
'LoadingState', 'NavigationGroupHeader', 'SaveIndicator',
|
||||||
|
'SeedDataManager', 'StorageSettings'
|
||||||
|
]
|
||||||
|
|
||||||
|
const maybeJsonCompatibleMolecules = [
|
||||||
|
'ActionBar', 'BindingEditor', 'CanvasRenderer', 'CodeExplanationDialog',
|
||||||
|
'ComponentBindingDialog', 'ComponentPalette', 'ComponentTree', 'DataCard',
|
||||||
|
'DataSourceCard', 'DataSourceEditorDialog', 'EditorActions', 'EditorToolbar',
|
||||||
|
'EmptyState', 'FileTabs', 'LazyInlineMonacoEditor', 'LazyMonacoEditor',
|
||||||
|
'MonacoEditorPanel', 'NavigationItem', 'PageHeaderContent', 'PropertyEditor',
|
||||||
|
'SearchBar', 'SearchInput', 'StatCard', 'ToolbarButton', 'TreeCard',
|
||||||
|
'TreeFormDialog', 'TreeListHeader'
|
||||||
|
]
|
||||||
|
|
||||||
|
const jsonCompatibleOrganisms = ['PageHeader']
|
||||||
|
|
||||||
|
const maybeJsonCompatibleOrganisms = [
|
||||||
|
'AppHeader', 'DataSourceManager', 'EmptyCanvasState', 'JSONUIShowcase',
|
||||||
|
'NavigationMenu', 'SchemaCodeViewer', 'SchemaEditorCanvas',
|
||||||
|
'SchemaEditorLayout', 'SchemaEditorPropertiesPanel', 'SchemaEditorSidebar',
|
||||||
|
'SchemaEditorStatusBar', 'SchemaEditorToolbar', 'ToolbarActions', 'TreeListPanel'
|
||||||
|
]
|
||||||
|
|
||||||
|
// Build components array
|
||||||
|
const components = []
|
||||||
|
|
||||||
|
// Process atoms (all are foundational, mark as supported)
|
||||||
|
atoms.forEach(name => {
|
||||||
|
const existing = existingMap.get(name)
|
||||||
|
components.push({
|
||||||
|
type: name,
|
||||||
|
name: existing?.name || name,
|
||||||
|
category: existing?.category || guessCategory(name),
|
||||||
|
canHaveChildren: existing?.canHaveChildren !== undefined ? existing.canHaveChildren : canHaveChildren(name),
|
||||||
|
description: existing?.description || getDescription(name),
|
||||||
|
status: existing?.status || 'supported',
|
||||||
|
source: 'atoms'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
// Process molecules with JSON compatibility marking
|
||||||
|
molecules.forEach(name => {
|
||||||
|
const existing = existingMap.get(name)
|
||||||
|
let status = existing?.status || 'supported'
|
||||||
|
|
||||||
|
if (jsonCompatibleMolecules.includes(name)) {
|
||||||
|
status = 'json-compatible'
|
||||||
|
} else if (maybeJsonCompatibleMolecules.includes(name)) {
|
||||||
|
status = 'maybe-json-compatible'
|
||||||
|
}
|
||||||
|
|
||||||
|
components.push({
|
||||||
|
type: name,
|
||||||
|
name: existing?.name || name,
|
||||||
|
category: existing?.category || guessCategory(name),
|
||||||
|
canHaveChildren: existing?.canHaveChildren !== undefined ? existing.canHaveChildren : canHaveChildren(name),
|
||||||
|
description: existing?.description || getDescription(name),
|
||||||
|
status,
|
||||||
|
source: 'molecules',
|
||||||
|
jsonCompatible: jsonCompatibleMolecules.includes(name) || maybeJsonCompatibleMolecules.includes(name)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
// Process organisms with JSON compatibility marking
|
||||||
|
organisms.forEach(name => {
|
||||||
|
const existing = existingMap.get(name)
|
||||||
|
let status = existing?.status || 'supported'
|
||||||
|
|
||||||
|
if (jsonCompatibleOrganisms.includes(name)) {
|
||||||
|
status = 'json-compatible'
|
||||||
|
} else if (maybeJsonCompatibleOrganisms.includes(name)) {
|
||||||
|
status = 'maybe-json-compatible'
|
||||||
|
}
|
||||||
|
|
||||||
|
components.push({
|
||||||
|
type: name,
|
||||||
|
name: existing?.name || name,
|
||||||
|
category: existing?.category || guessCategory(name),
|
||||||
|
canHaveChildren: existing?.canHaveChildren !== undefined ? existing.canHaveChildren : true,
|
||||||
|
description: existing?.description || `${name} organism component`,
|
||||||
|
status,
|
||||||
|
source: 'organisms',
|
||||||
|
jsonCompatible: jsonCompatibleOrganisms.includes(name) || maybeJsonCompatibleOrganisms.includes(name)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
// Process ui components (convert kebab-case to PascalCase)
|
||||||
|
ui.forEach(name => {
|
||||||
|
// Convert kebab-case to PascalCase
|
||||||
|
const pascalName = name.split('-').map(word =>
|
||||||
|
word.charAt(0).toUpperCase() + word.slice(1)
|
||||||
|
).join('')
|
||||||
|
|
||||||
|
const existing = existingMap.get(pascalName) || existingMap.get(name)
|
||||||
|
components.push({
|
||||||
|
type: pascalName,
|
||||||
|
name: existing?.name || pascalName,
|
||||||
|
category: existing?.category || guessCategory(pascalName),
|
||||||
|
canHaveChildren: existing?.canHaveChildren !== undefined ? existing.canHaveChildren : canHaveChildren(pascalName),
|
||||||
|
description: existing?.description || getDescription(pascalName),
|
||||||
|
status: existing?.status || 'supported',
|
||||||
|
source: 'ui'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
// Sort by category then name
|
||||||
|
components.sort((a, b) => {
|
||||||
|
if (a.category !== b.category) {
|
||||||
|
const order = ['layout', 'input', 'display', 'navigation', 'feedback', 'data', 'custom']
|
||||||
|
return order.indexOf(a.category) - order.indexOf(b.category)
|
||||||
|
}
|
||||||
|
return a.name.localeCompare(b.name)
|
||||||
|
})
|
||||||
|
|
||||||
|
// Count by category
|
||||||
|
const byCategory = {}
|
||||||
|
components.forEach(c => {
|
||||||
|
byCategory[c.category] = (byCategory[c.category] || 0) + 1
|
||||||
|
})
|
||||||
|
|
||||||
|
// Build the registry
|
||||||
|
const registry = {
|
||||||
|
$schema: './schemas/json-components-registry-schema.json',
|
||||||
|
version: '2.0.0',
|
||||||
|
description: 'Registry of all components in the application',
|
||||||
|
lastUpdated: new Date().toISOString(),
|
||||||
|
categories: {
|
||||||
|
layout: 'Layout and container components',
|
||||||
|
input: 'Form inputs and interactive controls',
|
||||||
|
display: 'Display and presentation components',
|
||||||
|
navigation: 'Navigation and routing components',
|
||||||
|
feedback: 'Alerts, notifications, and status indicators',
|
||||||
|
data: 'Data display and visualization components',
|
||||||
|
custom: 'Custom domain-specific components'
|
||||||
|
},
|
||||||
|
components,
|
||||||
|
statistics: {
|
||||||
|
total: components.length,
|
||||||
|
supported: components.filter(c => c.status === 'supported').length,
|
||||||
|
planned: components.filter(c => c.status === 'planned').length,
|
||||||
|
jsonCompatible: components.filter(c => c.status === 'json-compatible').length,
|
||||||
|
maybeJsonCompatible: components.filter(c => c.status === 'maybe-json-compatible').length,
|
||||||
|
byCategory,
|
||||||
|
bySource: {
|
||||||
|
atoms: atoms.length,
|
||||||
|
molecules: molecules.length,
|
||||||
|
organisms: organisms.length,
|
||||||
|
ui: ui.length
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write to file
|
||||||
|
fs.writeFileSync(registryPath, JSON.stringify(registry, null, 2) + '\n', 'utf8')
|
||||||
|
|
||||||
|
console.log('\n✅ Updated json-components-registry.json')
|
||||||
|
console.log(` Total components: ${registry.statistics.total}`)
|
||||||
|
console.log(` By source:`)
|
||||||
|
console.log(` 🧱 atoms: ${registry.statistics.bySource.atoms}`)
|
||||||
|
console.log(` 🧪 molecules: ${registry.statistics.bySource.molecules}`)
|
||||||
|
console.log(` 🦠 organisms: ${registry.statistics.bySource.organisms}`)
|
||||||
|
console.log(` 🎨 ui: ${registry.statistics.bySource.ui}`)
|
||||||
|
console.log(` JSON compatibility:`)
|
||||||
|
console.log(` 🔥 Fully compatible: ${registry.statistics.jsonCompatible}`)
|
||||||
|
console.log(` ⚠️ Maybe compatible: ${registry.statistics.maybeJsonCompatible}`)
|
||||||
|
console.log(` By category:`)
|
||||||
|
Object.entries(byCategory).forEach(([cat, count]) => {
|
||||||
|
console.log(` ${cat}: ${count}`)
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user