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:
@@ -73,15 +73,21 @@ categories.forEach(category => {
|
||||
|
||||
categoryComps.forEach(comp => {
|
||||
const children = comp.canHaveChildren ? '👶 Can have children' : '➖ No children'
|
||||
const statusIcon = comp.status === 'supported' ? '✅' : '📋'
|
||||
const subComps = comp.subComponents ? ` (includes: ${comp.subComponents.join(', ')})` : ''
|
||||
let statusIcon = comp.status === 'supported' ? '✅' : '📋'
|
||||
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(` ${children}`)
|
||||
if (comp.subComponents) {
|
||||
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('')
|
||||
})
|
||||
})
|
||||
@@ -92,9 +98,24 @@ console.log(`\nTotal Components: ${componentsList.length}`)
|
||||
if (statusFilter === 'all') {
|
||||
const supported = componentsList.filter(c => c.status === 'supported').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(` ✅ Supported: ${supported}`)
|
||||
console.log(` 🔥 JSON-Compatible: ${jsonCompatible}`)
|
||||
console.log(` ⚠️ Maybe JSON-Compatible: ${maybeCompatible}`)
|
||||
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:`)
|
||||
|
||||
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