Compare commits
6 Commits
pagelayout
...
frontendau
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1d610f0d2b | ||
|
|
fbfaf7bb9f | ||
|
|
2f0aeb948b | ||
|
|
0ad62cbf8d | ||
|
|
5a80f33078 | ||
|
|
57f27d28cd |
@@ -7,7 +7,7 @@ Use this checklist to ensure proper implementation of the field type system in y
|
|||||||
### Database
|
### Database
|
||||||
- [ ] Run migration: `npm run migrate:tenant` to add `ui_metadata` column
|
- [ ] Run migration: `npm run migrate:tenant` to add `ui_metadata` column
|
||||||
- [ ] Verify migration succeeded: Check `field_definitions` table has `ui_metadata` column
|
- [ ] Verify migration succeeded: Check `field_definitions` table has `ui_metadata` column
|
||||||
- [ ] (Optional) Run seed: `knex seed:run --specific=example_account_fields_with_ui_metadata.js`
|
- [ ] (Optional) Run seed: `knex seed:run --specific=example_contact_fields_with_ui_metadata.js`
|
||||||
- [ ] Test database access with sample queries
|
- [ ] Test database access with sample queries
|
||||||
|
|
||||||
### Services
|
### Services
|
||||||
|
|||||||
@@ -1,390 +0,0 @@
|
|||||||
# Page Layouts Architecture Diagram
|
|
||||||
|
|
||||||
## System Overview
|
|
||||||
|
|
||||||
```
|
|
||||||
┌─────────────────────────────────────────────────────────────────┐
|
|
||||||
│ FRONTEND (Vue 3 + Nuxt) │
|
|
||||||
├─────────────────────────────────────────────────────────────────┤
|
|
||||||
│ │
|
|
||||||
│ ┌───────────────────────────────────────────────────────────┐ │
|
|
||||||
│ │ Setup → Objects → [Object] → Layouts Tab │ │
|
|
||||||
│ ├───────────────────────────────────────────────────────────┤ │
|
|
||||||
│ │ │ │
|
|
||||||
│ │ ┌─────────────┐ ┌───────────────────────────────┐ │ │
|
|
||||||
│ │ │ Layouts │ │ PageLayoutEditor │ │ │
|
|
||||||
│ │ │ List │ --> │ ┌─────────────────────────┐ │ │ │
|
|
||||||
│ │ │ │ │ │ 6-Column Grid │ │ │ │
|
|
||||||
│ │ │ • Standard │ │ │ ┌───┬───┬───┬───┬───┐ │ │ │ │
|
|
||||||
│ │ │ • Compact │ │ │ │ F │ F │ F │ F │ F │ │ │ │ │
|
|
||||||
│ │ │ • Detailed │ │ │ ├───┴───┴───┴───┴───┤ │ │ │ │
|
|
||||||
│ │ │ │ │ │ │ Field 1 (w:5) │ │ │ │ │
|
|
||||||
│ │ │ [+ New] │ │ │ └─────────────────── │ │ │ │ │
|
|
||||||
│ │ └─────────────┘ │ └─────────────────────────┘ │ │ │
|
|
||||||
│ │ │ │ │ │
|
|
||||||
│ │ │ Sidebar: │ │ │
|
|
||||||
│ │ │ ┌─────────────────────────┐ │ │ │
|
|
||||||
│ │ │ │ Available Fields │ │ │ │
|
|
||||||
│ │ │ │ □ Email │ │ │ │
|
|
||||||
│ │ │ │ □ Phone │ │ │ │
|
|
||||||
│ │ │ │ □ Status │ │ │ │
|
|
||||||
│ │ │ └─────────────────────────┘ │ │ │
|
|
||||||
│ │ └───────────────────────────────┘ │ │
|
|
||||||
│ └───────────────────────────────────────────────────────┘ │
|
|
||||||
│ │
|
|
||||||
│ ┌───────────────────────────────────────────────────────┐ │
|
|
||||||
│ │ Record Detail/Edit Views │ │
|
|
||||||
│ ├───────────────────────────────────────────────────────┤ │
|
|
||||||
│ │ │ │
|
|
||||||
│ │ DetailViewEnhanced / EditViewEnhanced │ │
|
|
||||||
│ │ ↓ │ │
|
|
||||||
│ │ ┌─────────────────────────────────────────────────┐ │ │
|
|
||||||
│ │ │ PageLayoutRenderer │ │ │
|
|
||||||
│ │ │ │ │ │
|
|
||||||
│ │ │ Fetches default layout for object │ │ │
|
|
||||||
│ │ │ Renders fields in custom grid positions │ │ │
|
|
||||||
│ │ │ Fallback to 2-column if no layout │ │ │
|
|
||||||
│ │ └─────────────────────────────────────────────────┘ │ │
|
|
||||||
│ └───────────────────────────────────────────────────────┘ │
|
|
||||||
│ │
|
|
||||||
│ ┌───────────────────────────────────────────────────────┐ │
|
|
||||||
│ │ Composables (usePageLayouts) │ │
|
|
||||||
│ ├───────────────────────────────────────────────────────┤ │
|
|
||||||
│ │ • getPageLayouts() • createPageLayout() │ │
|
|
||||||
│ │ • getPageLayout() • updatePageLayout() │ │
|
|
||||||
│ │ • getDefaultPageLayout()• deletePageLayout() │ │
|
|
||||||
│ └───────────────────────────────────────────────────────┘ │
|
|
||||||
└─────────────────────────────────────────────────────────────┘
|
|
||||||
↕ HTTP REST API
|
|
||||||
┌─────────────────────────────────────────────────────────────┐
|
|
||||||
│ BACKEND (NestJS) │
|
|
||||||
├─────────────────────────────────────────────────────────────┤
|
|
||||||
│ │
|
|
||||||
│ ┌───────────────────────────────────────────────────────┐ │
|
|
||||||
│ │ PageLayoutController (API Layer) │ │
|
|
||||||
│ ├───────────────────────────────────────────────────────┤ │
|
|
||||||
│ │ POST /page-layouts │ │
|
|
||||||
│ │ GET /page-layouts?objectId={id} │ │
|
|
||||||
│ │ GET /page-layouts/:id │ │
|
|
||||||
│ │ GET /page-layouts/default/:objectId │ │
|
|
||||||
│ │ PATCH /page-layouts/:id │ │
|
|
||||||
│ │ DELETE /page-layouts/:id │ │
|
|
||||||
│ └───────────────────────────────────────────────────────┘ │
|
|
||||||
│ ↕ │
|
|
||||||
│ ┌───────────────────────────────────────────────────────┐ │
|
|
||||||
│ │ PageLayoutService (Business Logic) │ │
|
|
||||||
│ ├───────────────────────────────────────────────────────┤ │
|
|
||||||
│ │ • Tenant isolation │ │
|
|
||||||
│ │ • Default layout management │ │
|
|
||||||
│ │ • CRUD operations │ │
|
|
||||||
│ │ • Validation │ │
|
|
||||||
│ └───────────────────────────────────────────────────────┘ │
|
|
||||||
│ ↕ │
|
|
||||||
│ ┌───────────────────────────────────────────────────────┐ │
|
|
||||||
│ │ PrismaService (Data Layer) │ │
|
|
||||||
│ ├───────────────────────────────────────────────────────┤ │
|
|
||||||
│ │ • Raw SQL queries │ │
|
|
||||||
│ │ • Tenant database routing │ │
|
|
||||||
│ │ • Transaction management │ │
|
|
||||||
│ └───────────────────────────────────────────────────────┘ │
|
|
||||||
└─────────────────────────────────────────────────────────────┘
|
|
||||||
↕
|
|
||||||
┌─────────────────────────────────────────────────────────────┐
|
|
||||||
│ DATABASE (PostgreSQL) │
|
|
||||||
├─────────────────────────────────────────────────────────────┤
|
|
||||||
│ │
|
|
||||||
│ ┌───────────────────────────────────────────────────────┐ │
|
|
||||||
│ │ Table: page_layouts │ │
|
|
||||||
│ ├───────────────────────────────────────────────────────┤ │
|
|
||||||
│ │ id UUID PRIMARY KEY │ │
|
|
||||||
│ │ name VARCHAR(255) │ │
|
|
||||||
│ │ object_id UUID → object_definitions(id) │ │
|
|
||||||
│ │ is_default BOOLEAN │ │
|
|
||||||
│ │ layout_config JSONB │ │
|
|
||||||
│ │ description TEXT │ │
|
|
||||||
│ │ created_at TIMESTAMP │ │
|
|
||||||
│ │ updated_at TIMESTAMP │ │
|
|
||||||
│ └───────────────────────────────────────────────────────┘ │
|
|
||||||
│ │
|
|
||||||
│ Example layout_config JSONB: │
|
|
||||||
│ { │
|
|
||||||
│ "fields": [ │
|
|
||||||
│ { │
|
|
||||||
│ "fieldId": "uuid-123", │
|
|
||||||
│ "x": 0, // Column start (0-5) │
|
|
||||||
│ "y": 0, // Row start │
|
|
||||||
│ "w": 3, // Width (1-6 columns) │
|
|
||||||
│ "h": 1 // Height (fixed at 1) │
|
|
||||||
│ } │
|
|
||||||
│ ] │
|
|
||||||
│ } │
|
|
||||||
└─────────────────────────────────────────────────────────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
## Data Flow Diagrams
|
|
||||||
|
|
||||||
### Creating a Layout
|
|
||||||
|
|
||||||
```
|
|
||||||
Admin User
|
|
||||||
│
|
|
||||||
├─→ Navigates to Setup → Objects → [Object] → Page Layouts
|
|
||||||
│
|
|
||||||
├─→ Clicks "New Layout"
|
|
||||||
│
|
|
||||||
├─→ Enters layout name
|
|
||||||
│
|
|
||||||
├─→ PageLayoutEditor mounts
|
|
||||||
│ │
|
|
||||||
│ ├─→ Loads object fields
|
|
||||||
│ ├─→ Initializes GridStack with 6 columns
|
|
||||||
│ └─→ Shows available fields in sidebar
|
|
||||||
│
|
|
||||||
├─→ Drags fields from sidebar to grid
|
|
||||||
│ │
|
|
||||||
│ ├─→ GridStack handles positioning
|
|
||||||
│ ├─→ User resizes field width (1-6 columns)
|
|
||||||
│ └─→ User arranges fields
|
|
||||||
│
|
|
||||||
├─→ Clicks "Save Layout"
|
|
||||||
│
|
|
||||||
├─→ usePageLayouts.createPageLayout()
|
|
||||||
│ │
|
|
||||||
│ └─→ POST /page-layouts
|
|
||||||
│ │
|
|
||||||
│ └─→ PageLayoutController.create()
|
|
||||||
│ │
|
|
||||||
│ └─→ PageLayoutService.create()
|
|
||||||
│ │
|
|
||||||
│ ├─→ If is_default, unset others
|
|
||||||
│ └─→ INSERT INTO page_layouts
|
|
||||||
│
|
|
||||||
└─→ Layout saved ✓
|
|
||||||
```
|
|
||||||
|
|
||||||
### Rendering a Layout in Detail View
|
|
||||||
|
|
||||||
```
|
|
||||||
User Opens Record
|
|
||||||
│
|
|
||||||
├─→ Navigates to /[object]/[id]/detail
|
|
||||||
│
|
|
||||||
├─→ DetailViewEnhanced mounts
|
|
||||||
│ │
|
|
||||||
│ └─→ onMounted() hook
|
|
||||||
│ │
|
|
||||||
│ └─→ usePageLayouts.getDefaultPageLayout(objectId)
|
|
||||||
│ │
|
|
||||||
│ └─→ GET /page-layouts/default/:objectId
|
|
||||||
│ │
|
|
||||||
│ └─→ PageLayoutService.findDefaultByObject()
|
|
||||||
│ │
|
|
||||||
│ └─→ SELECT * FROM page_layouts
|
|
||||||
│ WHERE object_id = $1
|
|
||||||
│ AND is_default = true
|
|
||||||
│
|
|
||||||
├─→ Layout received
|
|
||||||
│ │
|
|
||||||
│ ├─→ If layout exists:
|
|
||||||
│ │ │
|
|
||||||
│ │ └─→ PageLayoutRenderer renders with layout
|
|
||||||
│ │ │
|
|
||||||
│ │ ├─→ Creates CSS Grid (6 columns)
|
|
||||||
│ │ ├─→ Positions fields based on x, y, w, h
|
|
||||||
│ │ └─→ Renders FieldRenderer for each field
|
|
||||||
│ │
|
|
||||||
│ └─→ If no layout:
|
|
||||||
│ │
|
|
||||||
│ └─→ Falls back to 2-column layout
|
|
||||||
│
|
|
||||||
└─→ Record displayed with custom layout ✓
|
|
||||||
```
|
|
||||||
|
|
||||||
## Grid Layout System
|
|
||||||
|
|
||||||
### 6-Column Grid Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
┌──────┬──────┬──────┬──────┬──────┬──────┐
|
|
||||||
│ 0 │ 1 │ 2 │ 3 │ 4 │ 5 │ ← Column indices
|
|
||||||
└──────┴──────┴──────┴──────┴──────┴──────┘
|
|
||||||
Each column = 16.67% of container width
|
|
||||||
```
|
|
||||||
|
|
||||||
### Example Layouts
|
|
||||||
|
|
||||||
#### Two-Column Layout (Default)
|
|
||||||
```
|
|
||||||
┌─────────────────────┬─────────────────────┐
|
|
||||||
│ Name (w:3) │ Email (w:3) │
|
|
||||||
├─────────────────────┼─────────────────────┤
|
|
||||||
│ Phone (w:3) │ Company (w:3) │
|
|
||||||
├─────────────────────┴─────────────────────┤
|
|
||||||
│ Description (w:6) │
|
|
||||||
└───────────────────────────────────────────┘
|
|
||||||
|
|
||||||
Field configs:
|
|
||||||
- Name: {x:0, y:0, w:3, h:1}
|
|
||||||
- Email: {x:3, y:0, w:3, h:1}
|
|
||||||
- Phone: {x:0, y:1, w:3, h:1}
|
|
||||||
- Company: {x:3, y:1, w:3, h:1}
|
|
||||||
- Description: {x:0, y:2, w:6, h:1}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Three-Column Layout
|
|
||||||
```
|
|
||||||
┌───────────┬───────────┬───────────┐
|
|
||||||
│ F1 (w:2) │ F2 (w:2) │ F3 (w:2) │
|
|
||||||
├───────────┴───────────┴───────────┤
|
|
||||||
│ F4 (w:6) │
|
|
||||||
└───────────────────────────────────┘
|
|
||||||
|
|
||||||
Field configs:
|
|
||||||
- F1: {x:0, y:0, w:2, h:1}
|
|
||||||
- F2: {x:2, y:0, w:2, h:1}
|
|
||||||
- F3: {x:4, y:0, w:2, h:1}
|
|
||||||
- F4: {x:0, y:1, w:6, h:1}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Mixed Width Layout
|
|
||||||
```
|
|
||||||
┌───────────────┬───────┬───────────┐
|
|
||||||
│ Title (w:3) │ ID(1) │ Type (w:2)│
|
|
||||||
├───────────────┴───────┴───────────┤
|
|
||||||
│ Address (w:6) │
|
|
||||||
├──────────┬────────────────────────┤
|
|
||||||
│ City(2) │ State/ZIP (w:4) │
|
|
||||||
└──────────┴────────────────────────┘
|
|
||||||
|
|
||||||
Field configs:
|
|
||||||
- Title: {x:0, y:0, w:3, h:1}
|
|
||||||
- ID: {x:3, y:0, w:1, h:1}
|
|
||||||
- Type: {x:4, y:0, w:2, h:1}
|
|
||||||
- Address: {x:0, y:1, w:6, h:1}
|
|
||||||
- City: {x:0, y:2, w:2, h:1}
|
|
||||||
- State: {x:2, y:2, w:4, h:1}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Component Hierarchy
|
|
||||||
|
|
||||||
```
|
|
||||||
App.vue
|
|
||||||
│
|
|
||||||
└─→ NuxtLayout (default)
|
|
||||||
│
|
|
||||||
├─→ Setup Pages
|
|
||||||
│ │
|
|
||||||
│ └─→ pages/setup/objects/[apiName].vue
|
|
||||||
│ │
|
|
||||||
│ └─→ Tabs Component
|
|
||||||
│ │
|
|
||||||
│ ├─→ Tab: Fields (existing)
|
|
||||||
│ │
|
|
||||||
│ └─→ Tab: Page Layouts
|
|
||||||
│ │
|
|
||||||
│ ├─→ Layout List View
|
|
||||||
│ │ └─→ Card per layout
|
|
||||||
│ │
|
|
||||||
│ └─→ Layout Editor View
|
|
||||||
│ │
|
|
||||||
│ └─→ PageLayoutEditor
|
|
||||||
│ │
|
|
||||||
│ ├─→ GridStack (6 columns)
|
|
||||||
│ │ └─→ Field items
|
|
||||||
│ │
|
|
||||||
│ └─→ Sidebar
|
|
||||||
│ └─→ Available fields
|
|
||||||
│
|
|
||||||
└─→ Record Pages
|
|
||||||
│
|
|
||||||
└─→ pages/[objectName]/[[recordId]]/[[view]].vue
|
|
||||||
│
|
|
||||||
├─→ DetailViewEnhanced
|
|
||||||
│ │
|
|
||||||
│ └─→ PageLayoutRenderer
|
|
||||||
│ └─→ FieldRenderer (per field)
|
|
||||||
│
|
|
||||||
└─→ EditViewEnhanced
|
|
||||||
│
|
|
||||||
└─→ PageLayoutRenderer
|
|
||||||
└─→ FieldRenderer (per field)
|
|
||||||
```
|
|
||||||
|
|
||||||
## State Management
|
|
||||||
|
|
||||||
```
|
|
||||||
┌─────────────────────────────────────┐
|
|
||||||
│ Component State (ref/reactive) │
|
|
||||||
├─────────────────────────────────────┤
|
|
||||||
│ • selectedLayout │
|
|
||||||
│ • layouts[] │
|
|
||||||
│ • loadingLayouts │
|
|
||||||
│ • pageLayout (current) │
|
|
||||||
│ • formData │
|
|
||||||
│ • gridItems │
|
|
||||||
│ • placedFieldIds │
|
|
||||||
└─────────────────────────────────────┘
|
|
||||||
↕
|
|
||||||
┌─────────────────────────────────────┐
|
|
||||||
│ Composables (Reactive) │
|
|
||||||
├─────────────────────────────────────┤
|
|
||||||
│ • usePageLayouts() - API calls │
|
|
||||||
│ • useApi() - HTTP client │
|
|
||||||
│ • useAuth() - Authentication │
|
|
||||||
│ • useToast() - Notifications │
|
|
||||||
└─────────────────────────────────────┘
|
|
||||||
↕
|
|
||||||
┌─────────────────────────────────────┐
|
|
||||||
│ Browser Storage │
|
|
||||||
├─────────────────────────────────────┤
|
|
||||||
│ • localStorage: token, tenantId │
|
|
||||||
│ • SessionStorage: (none yet) │
|
|
||||||
│ • Cookies: (managed by server) │
|
|
||||||
└─────────────────────────────────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
## Security Flow
|
|
||||||
|
|
||||||
```
|
|
||||||
┌────────────────────────────────────────────────┐
|
|
||||||
│ 1. User Login │
|
|
||||||
│ → Receives JWT token │
|
|
||||||
│ → Token stored in localStorage │
|
|
||||||
└────────────────────────────────────────────────┘
|
|
||||||
↓
|
|
||||||
┌────────────────────────────────────────────────┐
|
|
||||||
│ 2. API Request │
|
|
||||||
│ → useApi() adds Authorization header │
|
|
||||||
│ → useApi() adds x-tenant-id header │
|
|
||||||
└────────────────────────────────────────────────┘
|
|
||||||
↓
|
|
||||||
┌────────────────────────────────────────────────┐
|
|
||||||
│ 3. Backend Validation │
|
|
||||||
│ → JwtAuthGuard validates token │
|
|
||||||
│ → Extracts user info (userId, tenantId) │
|
|
||||||
│ → Attaches to request object │
|
|
||||||
└────────────────────────────────────────────────┘
|
|
||||||
↓
|
|
||||||
┌────────────────────────────────────────────────┐
|
|
||||||
│ 4. Service Layer │
|
|
||||||
│ → Receives tenantId from request │
|
|
||||||
│ → All queries scoped to tenant │
|
|
||||||
│ → Tenant isolation enforced │
|
|
||||||
└────────────────────────────────────────────────┘
|
|
||||||
↓
|
|
||||||
┌────────────────────────────────────────────────┐
|
|
||||||
│ 5. Database │
|
|
||||||
│ → Tenant-specific database selected │
|
|
||||||
│ → Query executed in tenant context │
|
|
||||||
│ → Results returned │
|
|
||||||
└────────────────────────────────────────────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Legend:**
|
|
||||||
- `→` : Data flow direction
|
|
||||||
- `↕` : Bidirectional communication
|
|
||||||
- `├─→` : Hierarchical relationship
|
|
||||||
- `└─→` : Terminal branch
|
|
||||||
- `✓` : Successful operation
|
|
||||||
@@ -1,356 +0,0 @@
|
|||||||
# Page Layouts Feature - Implementation Complete ✅
|
|
||||||
|
|
||||||
## Summary
|
|
||||||
|
|
||||||
Successfully implemented a comprehensive page layouts feature for customizing field display in detail and edit views using a 6-column drag-and-drop grid system powered by GridStack.js.
|
|
||||||
|
|
||||||
## What Was Built
|
|
||||||
|
|
||||||
### Backend (NestJS + PostgreSQL)
|
|
||||||
- ✅ Database migration for `page_layouts` table
|
|
||||||
- ✅ Complete CRUD API with 6 endpoints
|
|
||||||
- ✅ Service layer with tenant isolation
|
|
||||||
- ✅ DTO validation
|
|
||||||
- ✅ JWT authentication integration
|
|
||||||
|
|
||||||
### Frontend (Vue 3 + Nuxt)
|
|
||||||
- ✅ **PageLayoutEditor** - Visual drag-and-drop layout builder
|
|
||||||
- ✅ **PageLayoutRenderer** - Dynamic field rendering based on layouts
|
|
||||||
- ✅ **DetailViewEnhanced** - Enhanced detail view with layout support
|
|
||||||
- ✅ **EditViewEnhanced** - Enhanced edit view with layout support
|
|
||||||
- ✅ **usePageLayouts** - Composable for API interactions
|
|
||||||
- ✅ Setup page integration with tabs (Fields | Page Layouts)
|
|
||||||
|
|
||||||
## Key Features
|
|
||||||
|
|
||||||
### Layout Editor
|
|
||||||
- 6-column responsive grid
|
|
||||||
- Drag fields from sidebar to grid
|
|
||||||
- Reposition fields via drag-and-drop
|
|
||||||
- Horizontal resizing (1-6 columns width)
|
|
||||||
- Default 3-column width (2-column appearance)
|
|
||||||
- Fixed 80px height for consistency
|
|
||||||
- Remove fields from layout
|
|
||||||
- Clear all functionality
|
|
||||||
- Save/load layout state
|
|
||||||
|
|
||||||
### Layout Renderer
|
|
||||||
- CSS Grid-based rendering
|
|
||||||
- Position-aware field placement
|
|
||||||
- Size-aware field scaling
|
|
||||||
- All field types supported
|
|
||||||
- Readonly mode (detail view)
|
|
||||||
- Edit mode (form view)
|
|
||||||
- Automatic fallback to 2-column layout
|
|
||||||
|
|
||||||
### API Endpoints
|
|
||||||
```
|
|
||||||
POST /page-layouts Create new layout
|
|
||||||
GET /page-layouts?objectId={id} List layouts for object
|
|
||||||
GET /page-layouts/:id Get specific layout
|
|
||||||
GET /page-layouts/default/:objectId Get default layout
|
|
||||||
PATCH /page-layouts/:id Update layout (changed from PUT)
|
|
||||||
DELETE /page-layouts/:id Delete layout
|
|
||||||
```
|
|
||||||
|
|
||||||
## Files Created
|
|
||||||
|
|
||||||
### Backend
|
|
||||||
```
|
|
||||||
backend/
|
|
||||||
├── migrations/tenant/
|
|
||||||
│ └── 20250126000008_create_page_layouts.js
|
|
||||||
└── src/
|
|
||||||
├── app.module.ts (updated)
|
|
||||||
└── page-layout/
|
|
||||||
├── dto/
|
|
||||||
│ └── page-layout.dto.ts
|
|
||||||
├── page-layout.controller.ts
|
|
||||||
├── page-layout.service.ts
|
|
||||||
└── page-layout.module.ts
|
|
||||||
```
|
|
||||||
|
|
||||||
### Frontend
|
|
||||||
```
|
|
||||||
frontend/
|
|
||||||
├── components/
|
|
||||||
│ ├── PageLayoutEditor.vue
|
|
||||||
│ ├── PageLayoutRenderer.vue
|
|
||||||
│ └── views/
|
|
||||||
│ ├── DetailViewEnhanced.vue
|
|
||||||
│ └── EditViewEnhanced.vue
|
|
||||||
├── composables/
|
|
||||||
│ └── usePageLayouts.ts
|
|
||||||
├── pages/
|
|
||||||
│ └── setup/
|
|
||||||
│ └── objects/
|
|
||||||
│ └── [apiName].vue (updated)
|
|
||||||
└── types/
|
|
||||||
└── page-layout.ts
|
|
||||||
```
|
|
||||||
|
|
||||||
### Documentation
|
|
||||||
```
|
|
||||||
/root/neo/
|
|
||||||
├── PAGE_LAYOUTS_GUIDE.md
|
|
||||||
├── PAGE_LAYOUTS_IMPLEMENTATION_SUMMARY.md
|
|
||||||
├── PAGE_LAYOUTS_COMPLETE.md (this file)
|
|
||||||
└── setup-page-layouts.sh
|
|
||||||
```
|
|
||||||
|
|
||||||
## Quick Start
|
|
||||||
|
|
||||||
### 1. Run Database Migration
|
|
||||||
```bash
|
|
||||||
cd backend
|
|
||||||
npm run migrate:tenant
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Start Services
|
|
||||||
```bash
|
|
||||||
# Terminal 1
|
|
||||||
cd backend && npm run start:dev
|
|
||||||
|
|
||||||
# Terminal 2
|
|
||||||
cd frontend && npm run dev
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. Create Your First Layout
|
|
||||||
1. Login to application
|
|
||||||
2. Navigate to **Setup → Objects → [Select Object]**
|
|
||||||
3. Click **Page Layouts** tab
|
|
||||||
4. Click **New Layout**
|
|
||||||
5. Name your layout
|
|
||||||
6. Drag fields from sidebar onto grid
|
|
||||||
7. Resize and arrange as needed
|
|
||||||
8. Click **Save Layout**
|
|
||||||
|
|
||||||
### 4. See It In Action
|
|
||||||
Visit any record detail or edit page for that object to see your custom layout!
|
|
||||||
|
|
||||||
## Technical Highlights
|
|
||||||
|
|
||||||
### Grid System
|
|
||||||
- **6 columns** for flexible layouts
|
|
||||||
- **Default 3-column width** (creates 2-column appearance)
|
|
||||||
- **Fixed 80px height** for visual consistency
|
|
||||||
- **CSS Grid** for performant rendering
|
|
||||||
- **Responsive** design
|
|
||||||
|
|
||||||
### Data Storage
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"fields": [
|
|
||||||
{
|
|
||||||
"fieldId": "field-uuid-here",
|
|
||||||
"x": 0, // Start column (0-5)
|
|
||||||
"y": 0, // Start row (0-based)
|
|
||||||
"w": 3, // Width in columns (1-6)
|
|
||||||
"h": 1 // Height in rows (always 1)
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Type Safety
|
|
||||||
- Full TypeScript support
|
|
||||||
- Validated DTOs on backend
|
|
||||||
- Type-safe composables
|
|
||||||
- Strongly-typed components
|
|
||||||
|
|
||||||
### Performance
|
|
||||||
- Layouts cached after first load
|
|
||||||
- JSONB column for efficient queries
|
|
||||||
- CSS Grid for fast rendering
|
|
||||||
- Optimized drag-and-drop
|
|
||||||
|
|
||||||
## Integration Examples
|
|
||||||
|
|
||||||
### Use Enhanced Views
|
|
||||||
```vue
|
|
||||||
<script setup>
|
|
||||||
import DetailViewEnhanced from '@/components/views/DetailViewEnhanced.vue'
|
|
||||||
import EditViewEnhanced from '@/components/views/EditViewEnhanced.vue'
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<DetailViewEnhanced
|
|
||||||
:config="detailConfig"
|
|
||||||
:data="record"
|
|
||||||
:object-id="objectId"
|
|
||||||
@edit="handleEdit"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
```
|
|
||||||
|
|
||||||
### Use Renderer Directly
|
|
||||||
```vue
|
|
||||||
<script setup>
|
|
||||||
import PageLayoutRenderer from '@/components/PageLayoutRenderer.vue'
|
|
||||||
|
|
||||||
const { getDefaultPageLayout } = usePageLayouts()
|
|
||||||
const layout = ref(null)
|
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
layout.value = await getDefaultPageLayout(objectId)
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<PageLayoutRenderer
|
|
||||||
:fields="fields"
|
|
||||||
:layout="layout?.layoutConfig"
|
|
||||||
v-model="formData"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
```
|
|
||||||
|
|
||||||
## Backward Compatibility
|
|
||||||
|
|
||||||
✅ Fully backward compatible:
|
|
||||||
- Objects without layouts use traditional views
|
|
||||||
- Existing components unaffected
|
|
||||||
- Enhanced views auto-detect layouts
|
|
||||||
- Graceful fallback to 2-column layout
|
|
||||||
|
|
||||||
## Testing Checklist
|
|
||||||
|
|
||||||
- [x] Migration runs without errors
|
|
||||||
- [x] API endpoints accessible
|
|
||||||
- [x] Can create page layout
|
|
||||||
- [x] Fields draggable from sidebar
|
|
||||||
- [x] Fields repositionable on grid
|
|
||||||
- [x] Fields resizable (width)
|
|
||||||
- [x] Layout saves successfully
|
|
||||||
- [x] Layout loads in detail view
|
|
||||||
- [x] Layout works in edit view
|
|
||||||
- [x] Multiple layouts per object
|
|
||||||
- [x] Default layout auto-loads
|
|
||||||
- [x] Can delete layout
|
|
||||||
- [x] Fallback works when no layout
|
|
||||||
|
|
||||||
## Known Limitations
|
|
||||||
|
|
||||||
1. **Height not resizable** - All fields have uniform 80px height
|
|
||||||
2. **No vertical sizing** - Only horizontal width is adjustable
|
|
||||||
3. **Single default layout** - Only one layout can be default per object
|
|
||||||
4. **No layout cloning** - Must create from scratch (future enhancement)
|
|
||||||
|
|
||||||
## Future Enhancements
|
|
||||||
|
|
||||||
- [ ] Variable field heights
|
|
||||||
- [ ] Multi-row field spanning
|
|
||||||
- [ ] Layout templates
|
|
||||||
- [ ] Clone/duplicate layouts
|
|
||||||
- [ ] Layout permissions
|
|
||||||
- [ ] Related list sections
|
|
||||||
- [ ] Responsive breakpoints
|
|
||||||
- [ ] Custom components
|
|
||||||
- [ ] Layout preview mode
|
|
||||||
- [ ] A/B testing support
|
|
||||||
|
|
||||||
## Troubleshooting
|
|
||||||
|
|
||||||
### Layout Not Appearing
|
|
||||||
**Check:**
|
|
||||||
- Migration ran successfully
|
|
||||||
- Default layout is set
|
|
||||||
- objectId prop passed to enhanced views
|
|
||||||
- Browser console for errors
|
|
||||||
|
|
||||||
### Fields Not Draggable
|
|
||||||
**Check:**
|
|
||||||
- GridStack CSS loaded
|
|
||||||
- `draggable="true"` on sidebar items
|
|
||||||
- Browser JavaScript enabled
|
|
||||||
- No console errors
|
|
||||||
|
|
||||||
### Layout Not Saving
|
|
||||||
**Check:**
|
|
||||||
- API endpoint accessible
|
|
||||||
- JWT token valid
|
|
||||||
- Network tab for failed requests
|
|
||||||
- Backend logs for errors
|
|
||||||
|
|
||||||
## Performance Notes
|
|
||||||
|
|
||||||
- Initial layout fetch: ~50-100ms
|
|
||||||
- Drag operation: <16ms (60fps)
|
|
||||||
- Save operation: ~100-200ms
|
|
||||||
- Render time: ~50ms for 20 fields
|
|
||||||
|
|
||||||
## Security
|
|
||||||
|
|
||||||
- ✅ JWT authentication required
|
|
||||||
- ✅ Tenant isolation enforced
|
|
||||||
- ✅ Input validation on DTOs
|
|
||||||
- ✅ RBAC compatible (admin only for editing)
|
|
||||||
- ✅ SQL injection prevented (parameterized queries)
|
|
||||||
|
|
||||||
## Browser Support
|
|
||||||
|
|
||||||
- ✅ Chrome 90+
|
|
||||||
- ✅ Firefox 88+
|
|
||||||
- ✅ Safari 14+
|
|
||||||
- ✅ Edge 90+
|
|
||||||
|
|
||||||
## Dependencies
|
|
||||||
|
|
||||||
### Backend
|
|
||||||
- @nestjs/common: ^10.3.0
|
|
||||||
- class-validator: (existing)
|
|
||||||
- knex: (existing)
|
|
||||||
|
|
||||||
### Frontend
|
|
||||||
- gridstack: ^10.x (newly added)
|
|
||||||
- vue: ^3.4.15
|
|
||||||
- nuxt: ^3.10.0
|
|
||||||
|
|
||||||
## Maintenance
|
|
||||||
|
|
||||||
### Adding New Field Types
|
|
||||||
1. Add type to field component mapping in PageLayoutRenderer
|
|
||||||
2. Ensure field component follows FieldRenderer interface
|
|
||||||
3. Test in both detail and edit modes
|
|
||||||
|
|
||||||
### Modifying Grid Settings
|
|
||||||
Edit PageLayoutEditor.vue:
|
|
||||||
```typescript
|
|
||||||
GridStack.init({
|
|
||||||
column: 6, // Number of columns
|
|
||||||
cellHeight: 80, // Cell height in px
|
|
||||||
// ...other options
|
|
||||||
})
|
|
||||||
```
|
|
||||||
|
|
||||||
## Success Metrics
|
|
||||||
|
|
||||||
✅ **Implementation**: 100% complete
|
|
||||||
✅ **Type Safety**: Full TypeScript coverage
|
|
||||||
✅ **Testing**: All core functionality verified
|
|
||||||
✅ **Documentation**: Comprehensive guides created
|
|
||||||
✅ **Performance**: Meets 60fps drag operations
|
|
||||||
✅ **Compatibility**: Backward compatible
|
|
||||||
|
|
||||||
## Support
|
|
||||||
|
|
||||||
For questions or issues:
|
|
||||||
1. Check [PAGE_LAYOUTS_GUIDE.md](./PAGE_LAYOUTS_GUIDE.md) for detailed usage
|
|
||||||
2. Review [PAGE_LAYOUTS_IMPLEMENTATION_SUMMARY.md](./PAGE_LAYOUTS_IMPLEMENTATION_SUMMARY.md) for technical details
|
|
||||||
3. Check browser console for client-side errors
|
|
||||||
4. Review backend logs for server-side issues
|
|
||||||
|
|
||||||
## Credits
|
|
||||||
|
|
||||||
- **GridStack.js** - Drag-and-drop grid library
|
|
||||||
- **shadcn/ui** - UI component library
|
|
||||||
- **NestJS** - Backend framework
|
|
||||||
- **Nuxt 3** - Frontend framework
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Status**: ✅ PRODUCTION READY
|
|
||||||
|
|
||||||
**Last Updated**: December 22, 2025
|
|
||||||
|
|
||||||
**Version**: 1.0.0
|
|
||||||
@@ -1,304 +0,0 @@
|
|||||||
# Page Layouts Feature
|
|
||||||
|
|
||||||
## Overview
|
|
||||||
|
|
||||||
The Page Layouts feature allows administrators to customize how fields are displayed in detail and edit views using a visual drag-and-drop interface based on GridStack.js with a 6-column grid system.
|
|
||||||
|
|
||||||
## Architecture
|
|
||||||
|
|
||||||
### Backend Components
|
|
||||||
|
|
||||||
1. **Database Schema** (`migrations/tenant/20250126000008_create_page_layouts.js`)
|
|
||||||
- `page_layouts` table stores layout configurations
|
|
||||||
- Fields: `id`, `name`, `object_id`, `is_default`, `layout_config`, `description`
|
|
||||||
- JSON-based `layout_config` stores field positions and sizes
|
|
||||||
|
|
||||||
2. **API Endpoints** (`src/page-layout/`)
|
|
||||||
- `POST /page-layouts` - Create a new page layout
|
|
||||||
- `GET /page-layouts?objectId={id}` - Get all layouts for an object
|
|
||||||
- `GET /page-layouts/:id` - Get a specific layout
|
|
||||||
- `GET /page-layouts/default/:objectId` - Get the default layout for an object
|
|
||||||
- `PATCH /page-layouts/:id` - Update a layout
|
|
||||||
- `DELETE /page-layouts/:id` - Delete a layout
|
|
||||||
|
|
||||||
### Frontend Components
|
|
||||||
|
|
||||||
1. **PageLayoutEditor.vue** - Visual editor for creating/editing layouts
|
|
||||||
- 6-column grid system using GridStack.js
|
|
||||||
- Drag and drop fields from sidebar
|
|
||||||
- Resize fields horizontally (1-6 columns width)
|
|
||||||
- Default width: 3 columns (2-column template effect)
|
|
||||||
- Save layout configuration
|
|
||||||
|
|
||||||
2. **PageLayoutRenderer.vue** - Renders fields based on saved layouts
|
|
||||||
- Used in detail and edit views
|
|
||||||
- Falls back to traditional 2-column layout if no layout configured
|
|
||||||
- Supports all field types
|
|
||||||
|
|
||||||
3. **DetailViewEnhanced.vue** & **EditViewEnhanced.vue**
|
|
||||||
- Enhanced versions of views with page layout support
|
|
||||||
- Automatically fetch and use default page layout
|
|
||||||
- Maintain backward compatibility with section-based layouts
|
|
||||||
|
|
||||||
### Types
|
|
||||||
|
|
||||||
- **PageLayout** (`types/page-layout.ts`)
|
|
||||||
- Layout metadata and configuration
|
|
||||||
- Field position and size definitions
|
|
||||||
- Grid configuration options
|
|
||||||
|
|
||||||
## Usage
|
|
||||||
|
|
||||||
### 1. Run Database Migration
|
|
||||||
|
|
||||||
```bash
|
|
||||||
cd backend
|
|
||||||
npm run migrate:tenant
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Configure Page Layouts
|
|
||||||
|
|
||||||
Navigate to **Setup → Objects → [Object Name] → Page Layouts** tab:
|
|
||||||
|
|
||||||
1. Click "New Layout" to create a layout
|
|
||||||
2. Enter a layout name
|
|
||||||
3. Drag fields from the right sidebar onto the 6-column grid
|
|
||||||
4. Resize fields by dragging their edges (width only)
|
|
||||||
5. Rearrange fields by dragging them to new positions
|
|
||||||
6. Click "Save Layout" to persist changes
|
|
||||||
|
|
||||||
### 3. Use in Views
|
|
||||||
|
|
||||||
#### Option A: Use Enhanced Views (Recommended)
|
|
||||||
|
|
||||||
Replace existing views in your page:
|
|
||||||
|
|
||||||
```vue
|
|
||||||
<script setup>
|
|
||||||
import DetailViewEnhanced from '@/components/views/DetailViewEnhanced.vue'
|
|
||||||
import EditViewEnhanced from '@/components/views/EditViewEnhanced.vue'
|
|
||||||
|
|
||||||
const objectDefinition = ref(null)
|
|
||||||
|
|
||||||
// Fetch object definition...
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<!-- Detail View -->
|
|
||||||
<DetailViewEnhanced
|
|
||||||
:config="detailConfig"
|
|
||||||
:data="currentRecord"
|
|
||||||
:object-id="objectDefinition.id"
|
|
||||||
@edit="handleEdit"
|
|
||||||
@delete="handleDelete"
|
|
||||||
@back="handleBack"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<!-- Edit View -->
|
|
||||||
<EditViewEnhanced
|
|
||||||
:config="editConfig"
|
|
||||||
:data="currentRecord"
|
|
||||||
:object-id="objectDefinition.id"
|
|
||||||
@save="handleSave"
|
|
||||||
@cancel="handleCancel"
|
|
||||||
@back="handleBack"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Option B: Use PageLayoutRenderer Directly
|
|
||||||
|
|
||||||
```vue
|
|
||||||
<script setup>
|
|
||||||
import PageLayoutRenderer from '@/components/PageLayoutRenderer.vue'
|
|
||||||
import { usePageLayouts } from '~/composables/usePageLayouts'
|
|
||||||
|
|
||||||
const { getDefaultPageLayout } = usePageLayouts()
|
|
||||||
const pageLayout = ref(null)
|
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
const layout = await getDefaultPageLayout(objectId)
|
|
||||||
if (layout) {
|
|
||||||
pageLayout.value = layout.layoutConfig
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<PageLayoutRenderer
|
|
||||||
:fields="fields"
|
|
||||||
:layout="pageLayout"
|
|
||||||
:model-value="formData"
|
|
||||||
:readonly="false"
|
|
||||||
@update:model-value="formData = $event"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4. Composable API
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
const {
|
|
||||||
getPageLayouts, // Get all layouts for an object
|
|
||||||
getPageLayout, // Get a specific layout
|
|
||||||
getDefaultPageLayout, // Get default layout for an object
|
|
||||||
createPageLayout, // Create new layout
|
|
||||||
updatePageLayout, // Update existing layout
|
|
||||||
deletePageLayout // Delete layout
|
|
||||||
} = usePageLayouts()
|
|
||||||
|
|
||||||
// Example: Create a layout
|
|
||||||
await createPageLayout({
|
|
||||||
name: 'Sales Layout',
|
|
||||||
objectId: 'uuid-here',
|
|
||||||
isDefault: true,
|
|
||||||
layoutConfig: {
|
|
||||||
fields: [
|
|
||||||
{ fieldId: 'field-1', x: 0, y: 0, w: 3, h: 1 },
|
|
||||||
{ fieldId: 'field-2', x: 3, y: 0, w: 3, h: 1 },
|
|
||||||
]
|
|
||||||
}
|
|
||||||
})
|
|
||||||
```
|
|
||||||
|
|
||||||
## Grid System
|
|
||||||
|
|
||||||
### Configuration
|
|
||||||
- **Columns**: 6
|
|
||||||
- **Default field width**: 3 columns (50% width)
|
|
||||||
- **Min width**: 1 column (16.67%)
|
|
||||||
- **Max width**: 6 columns (100%)
|
|
||||||
- **Height**: Fixed at 1 unit (80px), uniform across all fields
|
|
||||||
|
|
||||||
### Layout Example
|
|
||||||
|
|
||||||
```
|
|
||||||
┌─────────────────────┬─────────────────────┐
|
|
||||||
│ Field 1 (w:3) │ Field 2 (w:3) │ ← Two 3-column fields
|
|
||||||
├─────────────────────┴─────────────────────┤
|
|
||||||
│ Field 3 (w:6) │ ← One full-width field
|
|
||||||
├──────────┬──────────┬──────────┬──────────┤
|
|
||||||
│ F4 (w:2) │ F5 (w:2) │ F6 (w:2) │ (empty) │ ← Three 2-column fields
|
|
||||||
└──────────┴──────────┴──────────┴──────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
## Features
|
|
||||||
|
|
||||||
### Editor
|
|
||||||
- ✅ 6-column responsive grid
|
|
||||||
- ✅ Drag fields from sidebar to grid
|
|
||||||
- ✅ Drag to reposition fields on grid
|
|
||||||
- ✅ Resize fields horizontally (1-6 columns)
|
|
||||||
- ✅ Remove fields from layout
|
|
||||||
- ✅ Save layout configuration
|
|
||||||
- ✅ Clear all fields
|
|
||||||
|
|
||||||
### Renderer
|
|
||||||
- ✅ Renders fields based on saved layout
|
|
||||||
- ✅ Respects field positioning and sizing
|
|
||||||
- ✅ Supports all field types
|
|
||||||
- ✅ Falls back to 2-column layout if no layout configured
|
|
||||||
- ✅ Works in both readonly (detail) and edit modes
|
|
||||||
|
|
||||||
### Layout Management
|
|
||||||
- ✅ Multiple layouts per object
|
|
||||||
- ✅ Default layout designation
|
|
||||||
- ✅ Create, read, update, delete layouts
|
|
||||||
- ✅ Tab-based interface in object setup
|
|
||||||
|
|
||||||
## Backward Compatibility
|
|
||||||
|
|
||||||
The system maintains full backward compatibility:
|
|
||||||
- Objects without page layouts use traditional section-based views
|
|
||||||
- Existing DetailView and EditView components continue to work
|
|
||||||
- Enhanced views automatically detect and use page layouts when available
|
|
||||||
|
|
||||||
## Technical Details
|
|
||||||
|
|
||||||
### Layout Storage Format
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"fields": [
|
|
||||||
{
|
|
||||||
"fieldId": "uuid-of-field",
|
|
||||||
"x": 0, // Column start (0-5)
|
|
||||||
"y": 0, // Row start (0-based)
|
|
||||||
"w": 3, // Width in columns (1-6)
|
|
||||||
"h": 1 // Height in rows (fixed at 1)
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Field Component Mapping
|
|
||||||
|
|
||||||
The renderer automatically maps field types to appropriate components:
|
|
||||||
- TEXT → TextFieldView
|
|
||||||
- NUMBER → NumberFieldView
|
|
||||||
- DATE/DATETIME → DateFieldView
|
|
||||||
- BOOLEAN → BooleanFieldView
|
|
||||||
- PICKLIST → SelectFieldView
|
|
||||||
- EMAIL → EmailFieldView
|
|
||||||
- PHONE → PhoneFieldView
|
|
||||||
- URL → UrlFieldView
|
|
||||||
- CURRENCY → CurrencyFieldView
|
|
||||||
- PERCENT → PercentFieldView
|
|
||||||
- TEXTAREA → TextareaFieldView
|
|
||||||
|
|
||||||
## Development
|
|
||||||
|
|
||||||
### Adding New Field Types
|
|
||||||
|
|
||||||
1. Create field view component in `components/fields/`
|
|
||||||
2. Add mapping in `PageLayoutRenderer.vue`:
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
const componentMap: Record<string, any> = {
|
|
||||||
// ... existing mappings
|
|
||||||
NEW_TYPE: NewFieldView,
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Customizing Grid Settings
|
|
||||||
|
|
||||||
Edit `PageLayoutEditor.vue`:
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
grid = GridStack.init({
|
|
||||||
column: 6, // Change column count
|
|
||||||
cellHeight: 80, // Change cell height
|
|
||||||
minRow: 1, // Minimum rows
|
|
||||||
float: true, // Allow floating
|
|
||||||
acceptWidgets: true,
|
|
||||||
animate: true,
|
|
||||||
})
|
|
||||||
```
|
|
||||||
|
|
||||||
## Troubleshooting
|
|
||||||
|
|
||||||
### Layout not appearing
|
|
||||||
- Ensure migration has been run
|
|
||||||
- Check that a default layout is set
|
|
||||||
- Verify objectId is passed to enhanced views
|
|
||||||
|
|
||||||
### Fields not draggable
|
|
||||||
- Check GridStack CSS is loaded
|
|
||||||
- Verify draggable attribute on sidebar fields
|
|
||||||
- Check browser console for errors
|
|
||||||
|
|
||||||
### Layout not saving
|
|
||||||
- Verify API endpoints are accessible
|
|
||||||
- Check authentication token
|
|
||||||
- Review backend logs for errors
|
|
||||||
|
|
||||||
## Future Enhancements
|
|
||||||
|
|
||||||
- [ ] Variable field heights
|
|
||||||
- [ ] Field-level permissions in layouts
|
|
||||||
- [ ] Clone/duplicate layouts
|
|
||||||
- [ ] Layout templates
|
|
||||||
- [ ] Layout preview mode
|
|
||||||
- [ ] Responsive breakpoints
|
|
||||||
- [ ] Related list sections
|
|
||||||
- [ ] Custom components in layouts
|
|
||||||
@@ -1,286 +0,0 @@
|
|||||||
# Page Layouts Implementation Summary
|
|
||||||
|
|
||||||
## ✅ Completed Components
|
|
||||||
|
|
||||||
### Backend (100%)
|
|
||||||
|
|
||||||
1. **Database Schema** ✓
|
|
||||||
- Migration file: `backend/migrations/tenant/20250126000008_create_page_layouts.js`
|
|
||||||
- Table: `page_layouts` with JSONB layout configuration storage
|
|
||||||
|
|
||||||
2. **API Layer** ✓
|
|
||||||
- Service: `backend/src/page-layout/page-layout.service.ts`
|
|
||||||
- Controller: `backend/src/page-layout/page-layout.controller.ts`
|
|
||||||
- DTOs: `backend/src/page-layout/dto/page-layout.dto.ts`
|
|
||||||
- Module: `backend/src/page-layout/page-layout.module.ts`
|
|
||||||
- Registered in: `backend/src/app.module.ts`
|
|
||||||
|
|
||||||
### Frontend (100%)
|
|
||||||
|
|
||||||
1. **Core Components** ✓
|
|
||||||
- **PageLayoutEditor.vue** - Drag-and-drop layout editor with 6-column grid
|
|
||||||
- **PageLayoutRenderer.vue** - Renders fields based on saved layouts
|
|
||||||
- **DetailViewEnhanced.vue** - Detail view with page layout support
|
|
||||||
- **EditViewEnhanced.vue** - Edit view with page layout support
|
|
||||||
|
|
||||||
2. **Types & Interfaces** ✓
|
|
||||||
- `frontend/types/page-layout.ts` - TypeScript definitions
|
|
||||||
|
|
||||||
3. **Composables** ✓
|
|
||||||
- `frontend/composables/usePageLayouts.ts` - API interaction layer
|
|
||||||
|
|
||||||
4. **Page Integration** ✓
|
|
||||||
- Updated: `frontend/pages/setup/objects/[apiName].vue` with tabs
|
|
||||||
- Tab 1: Fields list
|
|
||||||
- Tab 2: Page layouts management and editor
|
|
||||||
|
|
||||||
### Dependencies ✓
|
|
||||||
- GridStack.js installed in frontend
|
|
||||||
- All required UI components available (Tabs, Button, Card, etc.)
|
|
||||||
|
|
||||||
## 🎯 Key Features Implemented
|
|
||||||
|
|
||||||
### Layout Editor
|
|
||||||
- [x] 6-column grid system
|
|
||||||
- [x] Drag fields from sidebar to grid
|
|
||||||
- [x] Reposition fields via drag-and-drop
|
|
||||||
- [x] Resize fields horizontally (1-6 columns)
|
|
||||||
- [x] Default 3-column width per field
|
|
||||||
- [x] Uniform height (80px)
|
|
||||||
- [x] Remove fields from layout
|
|
||||||
- [x] Clear all functionality
|
|
||||||
- [x] Save layout state
|
|
||||||
|
|
||||||
### Layout Renderer
|
|
||||||
- [x] Grid-based field rendering
|
|
||||||
- [x] Respects saved positions and sizes
|
|
||||||
- [x] All field types supported
|
|
||||||
- [x] Readonly mode (detail view)
|
|
||||||
- [x] Edit mode (form view)
|
|
||||||
- [x] Fallback to 2-column layout
|
|
||||||
|
|
||||||
### Layout Management
|
|
||||||
- [x] Create multiple layouts per object
|
|
||||||
- [x] Set default layout
|
|
||||||
- [x] Edit existing layouts
|
|
||||||
- [x] Delete layouts
|
|
||||||
- [x] List all layouts for object
|
|
||||||
|
|
||||||
### Integration
|
|
||||||
- [x] Setup page with tabs
|
|
||||||
- [x] Enhanced detail/edit views
|
|
||||||
- [x] Automatic default layout loading
|
|
||||||
- [x] Backward compatibility maintained
|
|
||||||
|
|
||||||
## 📦 File Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
backend/
|
|
||||||
├── migrations/tenant/
|
|
||||||
│ └── 20250126000008_create_page_layouts.js
|
|
||||||
└── src/
|
|
||||||
└── page-layout/
|
|
||||||
├── dto/
|
|
||||||
│ └── page-layout.dto.ts
|
|
||||||
├── page-layout.controller.ts
|
|
||||||
├── page-layout.service.ts
|
|
||||||
└── page-layout.module.ts
|
|
||||||
|
|
||||||
frontend/
|
|
||||||
├── components/
|
|
||||||
│ ├── PageLayoutEditor.vue
|
|
||||||
│ ├── PageLayoutRenderer.vue
|
|
||||||
│ └── views/
|
|
||||||
│ ├── DetailViewEnhanced.vue
|
|
||||||
│ └── EditViewEnhanced.vue
|
|
||||||
├── composables/
|
|
||||||
│ └── usePageLayouts.ts
|
|
||||||
├── pages/
|
|
||||||
│ └── setup/
|
|
||||||
│ └── objects/
|
|
||||||
│ └── [apiName].vue (updated)
|
|
||||||
└── types/
|
|
||||||
└── page-layout.ts
|
|
||||||
|
|
||||||
Documentation/
|
|
||||||
├── PAGE_LAYOUTS_GUIDE.md
|
|
||||||
├── PAGE_LAYOUTS_IMPLEMENTATION_SUMMARY.md
|
|
||||||
└── setup-page-layouts.sh
|
|
||||||
```
|
|
||||||
|
|
||||||
## 🚀 Quick Start
|
|
||||||
|
|
||||||
### 1. Run Setup Script
|
|
||||||
```bash
|
|
||||||
./setup-page-layouts.sh
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Manual Setup (Alternative)
|
|
||||||
```bash
|
|
||||||
# Backend migration
|
|
||||||
cd backend
|
|
||||||
npm run migrate:tenant
|
|
||||||
|
|
||||||
# Frontend dependencies (already installed)
|
|
||||||
cd frontend
|
|
||||||
npm install gridstack
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. Start Services
|
|
||||||
```bash
|
|
||||||
# Terminal 1: Backend
|
|
||||||
cd backend
|
|
||||||
npm run start:dev
|
|
||||||
|
|
||||||
# Terminal 2: Frontend
|
|
||||||
cd frontend
|
|
||||||
npm run dev
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4. Create Your First Layout
|
|
||||||
|
|
||||||
1. Login to your application
|
|
||||||
2. Navigate to **Setup → Objects**
|
|
||||||
3. Select an object (e.g., Account, Contact)
|
|
||||||
4. Click the **Page Layouts** tab
|
|
||||||
5. Click **New Layout**
|
|
||||||
6. Name your layout (e.g., "Standard Layout")
|
|
||||||
7. Drag fields from the right sidebar onto the grid
|
|
||||||
8. Resize and arrange as desired
|
|
||||||
9. Click **Save Layout**
|
|
||||||
|
|
||||||
### 5. View Results
|
|
||||||
|
|
||||||
Navigate to a record detail or edit page for that object to see your layout in action!
|
|
||||||
|
|
||||||
## 🔧 Testing Checklist
|
|
||||||
|
|
||||||
- [ ] Migration runs successfully
|
|
||||||
- [ ] Can create a new page layout
|
|
||||||
- [ ] Fields appear in sidebar
|
|
||||||
- [ ] Can drag field from sidebar to grid
|
|
||||||
- [ ] Can reposition field on grid
|
|
||||||
- [ ] Can resize field width
|
|
||||||
- [ ] Can remove field from grid
|
|
||||||
- [ ] Layout saves successfully
|
|
||||||
- [ ] Layout loads on detail view
|
|
||||||
- [ ] Layout works on edit view
|
|
||||||
- [ ] Multiple layouts can coexist
|
|
||||||
- [ ] Default layout is used automatically
|
|
||||||
- [ ] Can delete a layout
|
|
||||||
- [ ] Fallback works when no layout exists
|
|
||||||
|
|
||||||
## 📊 API Endpoints
|
|
||||||
|
|
||||||
```
|
|
||||||
POST /page-layouts - Create layout
|
|
||||||
GET /page-layouts?objectId={id} - List layouts
|
|
||||||
GET /page-layouts/:id - Get specific layout
|
|
||||||
GET /page-layouts/default/:objectId - Get default layout
|
|
||||||
PATCH /page-layouts/:id - Update layout
|
|
||||||
DELETE /page-layouts/:id - Delete layout
|
|
||||||
```
|
|
||||||
|
|
||||||
## 🎨 Grid System Specs
|
|
||||||
|
|
||||||
- **Columns**: 6
|
|
||||||
- **Cell Height**: 80px
|
|
||||||
- **Default Width**: 3 columns (50%)
|
|
||||||
- **Min Width**: 1 column (16.67%)
|
|
||||||
- **Max Width**: 6 columns (100%)
|
|
||||||
- **Height**: 1 row (fixed, not resizable)
|
|
||||||
|
|
||||||
## 🔄 Integration Examples
|
|
||||||
|
|
||||||
### Using Enhanced Views
|
|
||||||
|
|
||||||
```vue
|
|
||||||
<script setup>
|
|
||||||
import DetailViewEnhanced from '@/components/views/DetailViewEnhanced.vue'
|
|
||||||
import EditViewEnhanced from '@/components/views/EditViewEnhanced.vue'
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<DetailViewEnhanced
|
|
||||||
:config="detailConfig"
|
|
||||||
:data="currentRecord"
|
|
||||||
:object-id="objectDefinition.id"
|
|
||||||
@edit="handleEdit"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
```
|
|
||||||
|
|
||||||
### Using Renderer Directly
|
|
||||||
|
|
||||||
```vue
|
|
||||||
<script setup>
|
|
||||||
import PageLayoutRenderer from '@/components/PageLayoutRenderer.vue'
|
|
||||||
|
|
||||||
const { getDefaultPageLayout } = usePageLayouts()
|
|
||||||
const layout = ref(null)
|
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
const result = await getDefaultPageLayout(objectId)
|
|
||||||
layout.value = result?.layoutConfig
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<PageLayoutRenderer
|
|
||||||
:fields="fields"
|
|
||||||
:layout="layout"
|
|
||||||
:model-value="formData"
|
|
||||||
@update:model-value="formData = $event"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
```
|
|
||||||
|
|
||||||
## 🐛 Common Issues & Solutions
|
|
||||||
|
|
||||||
### Issue: GridStack CSS not loading
|
|
||||||
**Solution**: Add to your main layout or nuxt.config.ts:
|
|
||||||
```javascript
|
|
||||||
css: ['gridstack/dist/gridstack.min.css']
|
|
||||||
```
|
|
||||||
|
|
||||||
### Issue: Fields not draggable
|
|
||||||
**Solution**: Ensure the field elements have `draggable="true"` attribute
|
|
||||||
|
|
||||||
### Issue: Layout not appearing in views
|
|
||||||
**Solution**: Pass `objectId` prop to enhanced views
|
|
||||||
|
|
||||||
### Issue: Migration fails
|
|
||||||
**Solution**: Check database connection and ensure migrations directory is correct
|
|
||||||
|
|
||||||
## 📈 Performance Considerations
|
|
||||||
|
|
||||||
- Layouts are cached on the frontend after first fetch
|
|
||||||
- JSONB column in PostgreSQL provides efficient storage and querying
|
|
||||||
- GridStack uses CSS Grid for performant rendering
|
|
||||||
- Only default layout is auto-loaded (other layouts loaded on-demand)
|
|
||||||
|
|
||||||
## 🔐 Security
|
|
||||||
|
|
||||||
- All endpoints protected by JWT authentication
|
|
||||||
- Tenant isolation maintained through service layer
|
|
||||||
- Layout operations scoped to authenticated user's tenant
|
|
||||||
- Input validation on all DTOs
|
|
||||||
|
|
||||||
## 🎓 Learning Resources
|
|
||||||
|
|
||||||
- [GridStack.js Documentation](https://gridstackjs.com)
|
|
||||||
- [PAGE_LAYOUTS_GUIDE.md](./PAGE_LAYOUTS_GUIDE.md) - Comprehensive usage guide
|
|
||||||
- Backend API follows NestJS best practices
|
|
||||||
- Frontend follows Vue 3 Composition API patterns
|
|
||||||
|
|
||||||
## 🚦 Status: Production Ready ✅
|
|
||||||
|
|
||||||
All core functionality is implemented and tested. The feature is backward compatible and ready for production use.
|
|
||||||
|
|
||||||
## 📝 Notes
|
|
||||||
|
|
||||||
- Height resizing intentionally disabled for consistent UI
|
|
||||||
- Default width of 3 columns provides good starting point (2-column effect)
|
|
||||||
- Sidebar shows only fields not yet on the layout
|
|
||||||
- Multiple layouts per object supported (admin can switch between them)
|
|
||||||
- Enhanced views maintain full compatibility with existing views
|
|
||||||
@@ -1,302 +0,0 @@
|
|||||||
# Tenant Migration Guide
|
|
||||||
|
|
||||||
## Quick Start
|
|
||||||
|
|
||||||
### Create a New Migration
|
|
||||||
```bash
|
|
||||||
cd backend
|
|
||||||
npm run migrate:make add_your_feature_name
|
|
||||||
```
|
|
||||||
|
|
||||||
Edit the generated file in `backend/migrations/tenant/`
|
|
||||||
|
|
||||||
### Test on Single Tenant
|
|
||||||
```bash
|
|
||||||
npm run migrate:tenant acme-corp
|
|
||||||
```
|
|
||||||
|
|
||||||
### Apply to All Tenants
|
|
||||||
```bash
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
## Available Commands
|
|
||||||
|
|
||||||
| Command | Description |
|
|
||||||
|---------|-------------|
|
|
||||||
| `npm run migrate:make <name>` | Create a new migration file |
|
|
||||||
| `npm run migrate:tenant <slug>` | Run migrations for a specific tenant |
|
|
||||||
| `npm run migrate:all-tenants` | Run migrations for all active tenants |
|
|
||||||
| `npm run migrate:latest` | Run migrations (default DB - rarely used) |
|
|
||||||
| `npm run migrate:rollback` | Rollback last migration (default DB) |
|
|
||||||
|
|
||||||
## Architecture
|
|
||||||
|
|
||||||
### Multi-Tenant Database Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
┌─────────────────────────┐
|
|
||||||
│ Central Database │
|
|
||||||
│ │
|
|
||||||
│ - tenants table │
|
|
||||||
│ - users table │
|
|
||||||
│ - (encrypted creds) │
|
|
||||||
└─────────────────────────┘
|
|
||||||
│
|
|
||||||
│ manages
|
|
||||||
│
|
|
||||||
┌───────┴────────┐
|
|
||||||
│ │
|
|
||||||
┌───▼────┐ ┌───▼────┐
|
|
||||||
│ Tenant │ │ Tenant │
|
|
||||||
│ DB1 │ ... │ DBN │
|
|
||||||
│ │ │ │
|
|
||||||
│ - users│ │ - users│
|
|
||||||
│ - roles│ │ - roles│
|
|
||||||
│ - apps │ │ - apps │
|
|
||||||
│ - ... │ │ - ... │
|
|
||||||
└────────┘ └────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
### How Migrations Work
|
|
||||||
|
|
||||||
1. **New Tenant Provisioning** (Automatic)
|
|
||||||
- User creates tenant via API
|
|
||||||
- `TenantProvisioningService.provisionTenant()` is called
|
|
||||||
- Database is created
|
|
||||||
- All migrations in `migrations/tenant/` are automatically run
|
|
||||||
- Tenant status set to ACTIVE
|
|
||||||
|
|
||||||
2. **Existing Tenants** (Manual)
|
|
||||||
- Developer creates new migration file
|
|
||||||
- Tests on single tenant: `npm run migrate:tenant test-tenant`
|
|
||||||
- Applies to all: `npm run migrate:all-tenants`
|
|
||||||
- Each tenant database is updated independently
|
|
||||||
|
|
||||||
### Migration Scripts
|
|
||||||
|
|
||||||
#### `migrate-tenant.ts`
|
|
||||||
- Accepts tenant slug or ID as argument
|
|
||||||
- Fetches tenant from central database
|
|
||||||
- Decrypts database password
|
|
||||||
- Creates Knex connection to tenant DB
|
|
||||||
- Runs pending migrations
|
|
||||||
- Reports success/failure
|
|
||||||
|
|
||||||
#### `migrate-all-tenants.ts`
|
|
||||||
- Fetches all ACTIVE tenants from central DB
|
|
||||||
- Iterates through each tenant
|
|
||||||
- Runs migrations sequentially
|
|
||||||
- Collects success/failure results
|
|
||||||
- Provides comprehensive summary
|
|
||||||
- Exits with error if any tenant fails
|
|
||||||
|
|
||||||
## Security
|
|
||||||
|
|
||||||
### Password Encryption
|
|
||||||
|
|
||||||
Tenant database passwords are encrypted using **AES-256-CBC** and stored in the central database.
|
|
||||||
|
|
||||||
**Required Environment Variable:**
|
|
||||||
```bash
|
|
||||||
DB_ENCRYPTION_KEY=your-32-character-secret-key!!
|
|
||||||
```
|
|
||||||
|
|
||||||
This key must:
|
|
||||||
- Be exactly 32 characters (256 bits)
|
|
||||||
- Match the key used by backend services
|
|
||||||
- Be kept secure (never commit to git)
|
|
||||||
- Be the same across all environments accessing tenant DBs
|
|
||||||
|
|
||||||
### Encryption Flow
|
|
||||||
|
|
||||||
```
|
|
||||||
Tenant Creation:
|
|
||||||
Plain Password → Encrypt → Store in Central DB
|
|
||||||
|
|
||||||
Migration Time:
|
|
||||||
Encrypted Password → Decrypt → Connect to Tenant DB → Run Migrations
|
|
||||||
```
|
|
||||||
|
|
||||||
## Example Workflow
|
|
||||||
|
|
||||||
### Adding a New Field to All Tenants
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# 1. Create migration
|
|
||||||
cd backend
|
|
||||||
npm run migrate:make add_priority_to_tasks
|
|
||||||
|
|
||||||
# 2. Edit the migration file
|
|
||||||
# migrations/tenant/20250127120000_add_priority_to_tasks.js
|
|
||||||
|
|
||||||
# 3. Test on staging tenant
|
|
||||||
npm run migrate:tenant staging-company
|
|
||||||
|
|
||||||
# 4. Verify it worked
|
|
||||||
# Connect to staging DB and check schema
|
|
||||||
|
|
||||||
# 5. Apply to all tenants
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
Expected output:
|
|
||||||
```
|
|
||||||
🚀 Starting migration for all tenants...
|
|
||||||
|
|
||||||
📋 Found 5 active tenant(s)
|
|
||||||
|
|
||||||
🔄 Migrating tenant: Acme Corp (acme_corp_db)
|
|
||||||
✅ Acme Corp: Ran 1 migrations:
|
|
||||||
- 20250127120000_add_priority_to_tasks.js
|
|
||||||
|
|
||||||
🔄 Migrating tenant: TechStart (techstart_db)
|
|
||||||
✅ TechStart: Ran 1 migrations:
|
|
||||||
- 20250127120000_add_priority_to_tasks.js
|
|
||||||
|
|
||||||
...
|
|
||||||
|
|
||||||
============================================================
|
|
||||||
📊 Migration Summary
|
|
||||||
============================================================
|
|
||||||
✅ Successful: 5
|
|
||||||
❌ Failed: 0
|
|
||||||
|
|
||||||
🎉 All tenant migrations completed successfully!
|
|
||||||
```
|
|
||||||
|
|
||||||
## Troubleshooting
|
|
||||||
|
|
||||||
### Error: "Cannot find module '../prisma/generated-central/client'"
|
|
||||||
|
|
||||||
**Solution:** Generate Prisma client
|
|
||||||
```bash
|
|
||||||
cd backend
|
|
||||||
npx prisma generate --schema=prisma/schema-central.prisma
|
|
||||||
```
|
|
||||||
|
|
||||||
### Error: "Invalid encrypted password format"
|
|
||||||
|
|
||||||
**Solution:** Check `DB_ENCRYPTION_KEY` environment variable matches the one used for encryption.
|
|
||||||
|
|
||||||
### Error: "Migration failed: Table already exists"
|
|
||||||
|
|
||||||
**Cause:** Migration was partially applied or run manually
|
|
||||||
|
|
||||||
**Solution:**
|
|
||||||
```bash
|
|
||||||
# Check migration status in tenant DB
|
|
||||||
mysql -h <host> -u <user> -p<pass> <dbname> -e "SELECT * FROM knex_migrations"
|
|
||||||
|
|
||||||
# If migration is listed, it's already applied
|
|
||||||
# If not, investigate why table exists and fix manually
|
|
||||||
```
|
|
||||||
|
|
||||||
### Migration Hangs
|
|
||||||
|
|
||||||
**Possible causes:**
|
|
||||||
- Network connection to database lost
|
|
||||||
- Database server down
|
|
||||||
- Migration has long-running query
|
|
||||||
|
|
||||||
**Solution:** Add timeout to migration and check database connectivity
|
|
||||||
|
|
||||||
## Best Practices
|
|
||||||
|
|
||||||
1. ✅ **Test first**: Always test migrations on a single tenant before applying to all
|
|
||||||
2. ✅ **Rollback ready**: Write `down()` functions for every migration
|
|
||||||
3. ✅ **Idempotent**: Use `IF NOT EXISTS` clauses where possible
|
|
||||||
4. ✅ **Backup**: Take database backups before major migrations
|
|
||||||
5. ✅ **Monitor**: Watch the output of `migrate:all-tenants` carefully
|
|
||||||
6. ✅ **Version control**: Commit migration files to git
|
|
||||||
7. ✅ **Document**: Add comments explaining complex migrations
|
|
||||||
|
|
||||||
8. ❌ **Don't skip testing**: Never run untested migrations on production
|
|
||||||
9. ❌ **Don't modify**: Never modify existing migration files after they're deployed
|
|
||||||
10. ❌ **Don't forget down()**: Always implement rollback logic
|
|
||||||
|
|
||||||
## Integration with TenantProvisioningService
|
|
||||||
|
|
||||||
The migrations are also used during tenant provisioning:
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
// src/tenant/tenant-provisioning.service.ts
|
|
||||||
|
|
||||||
async provisionTenant(tenantId: string): Promise<void> {
|
|
||||||
// ... create database ...
|
|
||||||
|
|
||||||
// Run migrations automatically
|
|
||||||
await this.runTenantMigrations(tenant);
|
|
||||||
|
|
||||||
// ... update tenant status ...
|
|
||||||
}
|
|
||||||
|
|
||||||
async runTenantMigrations(tenant: any): Promise<void> {
|
|
||||||
const knexConfig = {
|
|
||||||
client: 'mysql2',
|
|
||||||
connection: {
|
|
||||||
host: tenant.dbHost,
|
|
||||||
port: tenant.dbPort,
|
|
||||||
user: tenant.dbUser,
|
|
||||||
password: decryptedPassword,
|
|
||||||
database: tenant.dbName,
|
|
||||||
},
|
|
||||||
migrations: {
|
|
||||||
directory: './migrations/tenant',
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const knexInstance = knex(knexConfig);
|
|
||||||
await knexInstance.migrate.latest();
|
|
||||||
await knexInstance.destroy();
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
This ensures every new tenant starts with the complete schema.
|
|
||||||
|
|
||||||
## CI/CD Integration
|
|
||||||
|
|
||||||
### Docker Compose
|
|
||||||
```yaml
|
|
||||||
services:
|
|
||||||
backend:
|
|
||||||
image: your-backend:latest
|
|
||||||
command: sh -c "npm run migrate:all-tenants && npm run start:prod"
|
|
||||||
environment:
|
|
||||||
- DB_ENCRYPTION_KEY=${DB_ENCRYPTION_KEY}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Kubernetes Job
|
|
||||||
```yaml
|
|
||||||
apiVersion: batch/v1
|
|
||||||
kind: Job
|
|
||||||
metadata:
|
|
||||||
name: tenant-migrations
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: migrate
|
|
||||||
image: your-backend:latest
|
|
||||||
command: ["npm", "run", "migrate:all-tenants"]
|
|
||||||
env:
|
|
||||||
- name: DB_ENCRYPTION_KEY
|
|
||||||
valueFrom:
|
|
||||||
secretKeyRef:
|
|
||||||
name: db-secrets
|
|
||||||
key: encryption-key
|
|
||||||
restartPolicy: OnFailure
|
|
||||||
```
|
|
||||||
|
|
||||||
## Further Documentation
|
|
||||||
|
|
||||||
- [Backend Scripts README](backend/scripts/README.md) - Detailed script documentation
|
|
||||||
- [Multi-Tenant Implementation](MULTI_TENANT_IMPLEMENTATION.md) - Architecture overview
|
|
||||||
- [Multi-Tenant Migration](MULTI_TENANT_MIGRATION.md) - Migration strategy
|
|
||||||
|
|
||||||
## Support
|
|
||||||
|
|
||||||
For questions or issues:
|
|
||||||
1. Check the [Backend Scripts README](backend/scripts/README.md)
|
|
||||||
2. Review existing migration files in `backend/migrations/tenant/`
|
|
||||||
3. Check Knex documentation: https://knexjs.org/guide/migrations.html
|
|
||||||
@@ -1,374 +0,0 @@
|
|||||||
# Tenant Migration Implementation - Complete
|
|
||||||
|
|
||||||
## ✅ Implementation Summary
|
|
||||||
|
|
||||||
All tenant migration functionality has been successfully added to the backend. This implementation provides comprehensive tools for managing database schema changes across all tenants in the multi-tenant platform.
|
|
||||||
|
|
||||||
## 📁 Files Created
|
|
||||||
|
|
||||||
### Scripts Directory: `/root/neo/backend/scripts/`
|
|
||||||
|
|
||||||
1. **`migrate-tenant.ts`** (167 lines)
|
|
||||||
- Migrates a single tenant by slug or ID
|
|
||||||
- Handles password decryption
|
|
||||||
- Provides detailed progress output
|
|
||||||
- Usage: `npm run migrate:tenant <slug-or-id>`
|
|
||||||
|
|
||||||
2. **`migrate-all-tenants.ts`** (170 lines)
|
|
||||||
- Migrates all active tenants in sequence
|
|
||||||
- Collects success/failure statistics
|
|
||||||
- Provides comprehensive summary
|
|
||||||
- Exits with error code if any tenant fails
|
|
||||||
- Usage: `npm run migrate:all-tenants`
|
|
||||||
|
|
||||||
3. **`check-migration-status.ts`** (181 lines)
|
|
||||||
- Checks migration status across all tenants
|
|
||||||
- Shows completed and pending migrations
|
|
||||||
- Identifies which tenants need updates
|
|
||||||
- Usage: `npm run migrate:status`
|
|
||||||
|
|
||||||
4. **`README.md`** (Comprehensive documentation)
|
|
||||||
- Detailed usage instructions
|
|
||||||
- Security notes on password encryption
|
|
||||||
- Troubleshooting guide
|
|
||||||
- Best practices
|
|
||||||
- Example workflows
|
|
||||||
|
|
||||||
### Documentation Files
|
|
||||||
|
|
||||||
5. **`/root/neo/TENANT_MIGRATION_GUIDE.md`** (Root level guide)
|
|
||||||
- Quick start guide
|
|
||||||
- Architecture diagrams
|
|
||||||
- Complete workflow examples
|
|
||||||
- CI/CD integration examples
|
|
||||||
- Security documentation
|
|
||||||
|
|
||||||
### Updated Files
|
|
||||||
|
|
||||||
6. **`/root/neo/backend/package.json`**
|
|
||||||
- Added 6 new migration scripts to the `scripts` section
|
|
||||||
|
|
||||||
## 🚀 Available Commands
|
|
||||||
|
|
||||||
| Command | Description |
|
|
||||||
|---------|-------------|
|
|
||||||
| `npm run migrate:make <name>` | Create a new migration file in `migrations/tenant/` |
|
|
||||||
| `npm run migrate:status` | Check migration status for all tenants |
|
|
||||||
| `npm run migrate:tenant <slug>` | Run pending migrations for a specific tenant |
|
|
||||||
| `npm run migrate:all-tenants` | Run pending migrations for all active tenants |
|
|
||||||
| `npm run migrate:latest` | Run migrations on default database (rarely used) |
|
|
||||||
| `npm run migrate:rollback` | Rollback last migration on default database |
|
|
||||||
|
|
||||||
## 🔧 How It Works
|
|
||||||
|
|
||||||
### Architecture
|
|
||||||
|
|
||||||
```
|
|
||||||
┌─────────────────────────────────────────────────────────┐
|
|
||||||
│ Central Database │
|
|
||||||
│ │
|
|
||||||
│ ┌──────────┐ ┌──────────┐ ┌──────────┐ │
|
|
||||||
│ │ Tenant │ │ Tenant │ │ Tenant │ │
|
|
||||||
│ │ 1 │ │ 2 │ │ N │ │
|
|
||||||
│ └──────────┘ └──────────┘ └──────────┘ │
|
|
||||||
│ │ │ │ │
|
|
||||||
│ │ (encrypted │ │ │
|
|
||||||
│ │ password) │ │ │
|
|
||||||
└───────┼──────────────┼──────────────┼───────────────────┘
|
|
||||||
│ │ │
|
|
||||||
▼ ▼ ▼
|
|
||||||
┌───────────┐ ┌───────────┐ ┌───────────┐
|
|
||||||
│ Tenant │ │ Tenant │ │ Tenant │
|
|
||||||
│ DB 1 │ │ DB 2 │ │ DB N │
|
|
||||||
│ │ │ │ │ │
|
|
||||||
│ Migrations│ │ Migrations│ │ Migrations│
|
|
||||||
│ Applied │ │ Applied │ │ Applied │
|
|
||||||
└───────────┘ └───────────┘ └───────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
### Migration Flow
|
|
||||||
|
|
||||||
1. **Creating a Migration**
|
|
||||||
```bash
|
|
||||||
npm run migrate:make add_custom_fields
|
|
||||||
# Creates: migrations/tenant/20250127123456_add_custom_fields.js
|
|
||||||
```
|
|
||||||
|
|
||||||
2. **Testing on Single Tenant**
|
|
||||||
```bash
|
|
||||||
npm run migrate:tenant acme-corp
|
|
||||||
# Output:
|
|
||||||
# 📋 Tenant: Acme Corp (acme-corp)
|
|
||||||
# 📊 Database: acme_corp_db
|
|
||||||
# 🔄 Running migrations...
|
|
||||||
# ✅ Ran 1 migration(s):
|
|
||||||
# - 20250127123456_add_custom_fields.js
|
|
||||||
```
|
|
||||||
|
|
||||||
3. **Checking Status**
|
|
||||||
```bash
|
|
||||||
npm run migrate:status
|
|
||||||
# Shows which tenants have pending migrations
|
|
||||||
```
|
|
||||||
|
|
||||||
4. **Applying to All Tenants**
|
|
||||||
```bash
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
# Migrates all active tenants sequentially
|
|
||||||
# Provides summary of successes/failures
|
|
||||||
```
|
|
||||||
|
|
||||||
## 🔐 Security Features
|
|
||||||
|
|
||||||
### Password Encryption
|
|
||||||
- Tenant database passwords are encrypted using **AES-256-CBC**
|
|
||||||
- Stored encrypted in central database
|
|
||||||
- Automatically decrypted during migration
|
|
||||||
- Requires `DB_ENCRYPTION_KEY` environment variable
|
|
||||||
|
|
||||||
### Environment Setup
|
|
||||||
```bash
|
|
||||||
# Required for migration scripts
|
|
||||||
export DB_ENCRYPTION_KEY="your-32-character-secret-key!!"
|
|
||||||
```
|
|
||||||
|
|
||||||
This key must match the key used by `TenantService` for encryption/decryption.
|
|
||||||
|
|
||||||
## 📋 Example Workflows
|
|
||||||
|
|
||||||
### Scenario 1: Adding a Field to All Tenants
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# 1. Create migration
|
|
||||||
npm run migrate:make add_priority_field
|
|
||||||
|
|
||||||
# 2. Edit the generated file
|
|
||||||
# migrations/tenant/20250127120000_add_priority_field.js
|
|
||||||
|
|
||||||
# 3. Test on one tenant
|
|
||||||
npm run migrate:tenant test-company
|
|
||||||
|
|
||||||
# 4. Check status
|
|
||||||
npm run migrate:status
|
|
||||||
|
|
||||||
# 5. Apply to all
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
### Scenario 2: Checking Migration Status
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run migrate:status
|
|
||||||
|
|
||||||
# Output:
|
|
||||||
# 📋 Found 3 active tenant(s)
|
|
||||||
#
|
|
||||||
# 📦 Acme Corp (acme-corp)
|
|
||||||
# Database: acme_corp_db
|
|
||||||
# Completed: 5 migration(s)
|
|
||||||
# ✅ Up to date
|
|
||||||
#
|
|
||||||
# 📦 TechStart (techstart)
|
|
||||||
# Database: techstart_db
|
|
||||||
# Completed: 4 migration(s)
|
|
||||||
# ⚠️ Pending: 1 migration(s)
|
|
||||||
# - 20250127120000_add_priority_field.js
|
|
||||||
#
|
|
||||||
# 💡 Run: npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
### Scenario 3: New Tenant Provisioning (Automatic)
|
|
||||||
|
|
||||||
When a new tenant is created via the API:
|
|
||||||
```typescript
|
|
||||||
// Happens automatically in TenantProvisioningService
|
|
||||||
POST /tenants
|
|
||||||
{
|
|
||||||
"name": "New Company",
|
|
||||||
"slug": "new-company"
|
|
||||||
}
|
|
||||||
|
|
||||||
// Backend automatically:
|
|
||||||
// 1. Creates database
|
|
||||||
// 2. Runs all migrations
|
|
||||||
// 3. Sets tenant status to ACTIVE
|
|
||||||
```
|
|
||||||
|
|
||||||
## 🛠️ Technical Implementation
|
|
||||||
|
|
||||||
### Script Structure
|
|
||||||
|
|
||||||
All scripts follow this pattern:
|
|
||||||
|
|
||||||
1. **Import Dependencies**
|
|
||||||
```typescript
|
|
||||||
import { PrismaClient as CentralPrismaClient } from '../prisma/generated-central/client';
|
|
||||||
import knex, { Knex } from 'knex';
|
|
||||||
import { createDecipheriv } from 'crypto';
|
|
||||||
```
|
|
||||||
|
|
||||||
2. **Decrypt Password**
|
|
||||||
```typescript
|
|
||||||
function decryptPassword(encryptedPassword: string): string {
|
|
||||||
// AES-256-CBC decryption
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
3. **Create Tenant Connection**
|
|
||||||
```typescript
|
|
||||||
function createTenantKnexConnection(tenant: any): Knex {
|
|
||||||
const decryptedPassword = decryptPassword(tenant.dbPassword);
|
|
||||||
return knex({ /* config */ });
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
4. **Run Migrations**
|
|
||||||
```typescript
|
|
||||||
const [batchNo, log] = await tenantKnex.migrate.latest();
|
|
||||||
```
|
|
||||||
|
|
||||||
5. **Report Results**
|
|
||||||
```typescript
|
|
||||||
console.log(`✅ Ran ${log.length} migrations`);
|
|
||||||
```
|
|
||||||
|
|
||||||
## 🧪 Testing the Implementation
|
|
||||||
|
|
||||||
### 1. Verify Scripts Are Available
|
|
||||||
```bash
|
|
||||||
cd /root/neo/backend
|
|
||||||
npm run | grep migrate
|
|
||||||
```
|
|
||||||
|
|
||||||
Expected output:
|
|
||||||
```
|
|
||||||
migrate:make
|
|
||||||
migrate:latest
|
|
||||||
migrate:rollback
|
|
||||||
migrate:status
|
|
||||||
migrate:tenant
|
|
||||||
migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Test Creating a Migration
|
|
||||||
```bash
|
|
||||||
npm run migrate:make test_migration
|
|
||||||
```
|
|
||||||
|
|
||||||
Should create a file in `migrations/tenant/`
|
|
||||||
|
|
||||||
### 3. Check Status (if tenants exist)
|
|
||||||
```bash
|
|
||||||
npm run migrate:status
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4. Test Single Tenant Migration (if tenants exist)
|
|
||||||
```bash
|
|
||||||
npm run migrate:tenant <your-tenant-slug>
|
|
||||||
```
|
|
||||||
|
|
||||||
## 📚 Documentation Locations
|
|
||||||
|
|
||||||
- **Quick Reference**: `/root/neo/TENANT_MIGRATION_GUIDE.md`
|
|
||||||
- **Detailed Scripts Docs**: `/root/neo/backend/scripts/README.md`
|
|
||||||
- **Architecture Overview**: `/root/neo/MULTI_TENANT_IMPLEMENTATION.md`
|
|
||||||
|
|
||||||
## 🎯 Key Features
|
|
||||||
|
|
||||||
✅ **Single Tenant Migration** - Target specific tenants for testing
|
|
||||||
✅ **Bulk Migration** - Update all tenants at once
|
|
||||||
✅ **Status Checking** - See which tenants need updates
|
|
||||||
✅ **Progress Tracking** - Detailed output for each operation
|
|
||||||
✅ **Error Handling** - Graceful failure with detailed error messages
|
|
||||||
✅ **Security** - Encrypted password storage and decryption
|
|
||||||
✅ **Comprehensive Docs** - Multiple levels of documentation
|
|
||||||
|
|
||||||
## 🔄 Integration Points
|
|
||||||
|
|
||||||
### With Existing Code
|
|
||||||
|
|
||||||
1. **TenantProvisioningService**
|
|
||||||
- Already uses `runTenantMigrations()` method
|
|
||||||
- New scripts complement automatic provisioning
|
|
||||||
- Same migration directory: `migrations/tenant/`
|
|
||||||
|
|
||||||
2. **Knex Configuration**
|
|
||||||
- Uses existing `knexfile.js`
|
|
||||||
- Same migration table: `knex_migrations`
|
|
||||||
- Compatible with existing migrations
|
|
||||||
|
|
||||||
3. **Prisma Central Client**
|
|
||||||
- Scripts use central DB to fetch tenant list
|
|
||||||
- Same encryption/decryption logic as backend services
|
|
||||||
|
|
||||||
## 🚦 Next Steps
|
|
||||||
|
|
||||||
### To Use This Implementation:
|
|
||||||
|
|
||||||
1. **Ensure Environment Variables**
|
|
||||||
```bash
|
|
||||||
export DB_ENCRYPTION_KEY="your-32-character-secret-key!!"
|
|
||||||
```
|
|
||||||
|
|
||||||
2. **Generate Prisma Client** (if not already done)
|
|
||||||
```bash
|
|
||||||
cd /root/neo/backend
|
|
||||||
npx prisma generate --schema=prisma/schema-central.prisma
|
|
||||||
```
|
|
||||||
|
|
||||||
3. **Check Current Status**
|
|
||||||
```bash
|
|
||||||
npm run migrate:status
|
|
||||||
```
|
|
||||||
|
|
||||||
4. **Create Your First Migration**
|
|
||||||
```bash
|
|
||||||
npm run migrate:make add_my_feature
|
|
||||||
```
|
|
||||||
|
|
||||||
5. **Test and Apply**
|
|
||||||
```bash
|
|
||||||
# Test on one tenant
|
|
||||||
npm run migrate:tenant <slug>
|
|
||||||
|
|
||||||
# Apply to all
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
## 📊 Complete File List
|
|
||||||
|
|
||||||
```
|
|
||||||
/root/neo/
|
|
||||||
├── TENANT_MIGRATION_GUIDE.md (new)
|
|
||||||
└── backend/
|
|
||||||
├── package.json (updated - 6 new scripts)
|
|
||||||
├── knexfile.js (existing)
|
|
||||||
├── migrations/
|
|
||||||
│ └── tenant/ (existing)
|
|
||||||
│ ├── 20250126000001_create_users_and_rbac.js
|
|
||||||
│ ├── 20250126000002_create_object_definitions.js
|
|
||||||
│ ├── 20250126000003_create_apps.js
|
|
||||||
│ ├── 20250126000004_create_standard_objects.js
|
|
||||||
│ └── 20250126000005_add_ui_metadata_to_fields.js
|
|
||||||
├── scripts/ (new directory)
|
|
||||||
│ ├── README.md (new)
|
|
||||||
│ ├── migrate-tenant.ts (new)
|
|
||||||
│ ├── migrate-all-tenants.ts (new)
|
|
||||||
│ └── check-migration-status.ts (new)
|
|
||||||
└── src/
|
|
||||||
└── tenant/
|
|
||||||
└── tenant-provisioning.service.ts (existing - uses migrations)
|
|
||||||
```
|
|
||||||
|
|
||||||
## ✨ Summary
|
|
||||||
|
|
||||||
The tenant migration system is now fully implemented with:
|
|
||||||
- ✅ 3 TypeScript migration scripts
|
|
||||||
- ✅ 6 npm commands
|
|
||||||
- ✅ 2 comprehensive documentation files
|
|
||||||
- ✅ Full integration with existing architecture
|
|
||||||
- ✅ Security features (password encryption)
|
|
||||||
- ✅ Error handling and progress reporting
|
|
||||||
- ✅ Status checking capabilities
|
|
||||||
|
|
||||||
You can now manage database migrations across all tenants efficiently and safely! 🎉
|
|
||||||
@@ -1,91 +0,0 @@
|
|||||||
╔══════════════════════════════════════════════════════════════════════╗
|
|
||||||
║ TENANT MIGRATION - QUICK REFERENCE ║
|
|
||||||
╚══════════════════════════════════════════════════════════════════════╝
|
|
||||||
|
|
||||||
📍 LOCATION: /root/neo/backend
|
|
||||||
|
|
||||||
┌─────────────────────────────────────────────────────────────────────┐
|
|
||||||
│ COMMON COMMANDS │
|
|
||||||
└─────────────────────────────────────────────────────────────────────┘
|
|
||||||
|
|
||||||
Create Migration:
|
|
||||||
$ npm run migrate:make add_my_feature
|
|
||||||
|
|
||||||
Check Status:
|
|
||||||
$ npm run migrate:status
|
|
||||||
|
|
||||||
Test on One Tenant:
|
|
||||||
$ npm run migrate:tenant acme-corp
|
|
||||||
|
|
||||||
Apply to All Tenants:
|
|
||||||
$ npm run migrate:all-tenants
|
|
||||||
|
|
||||||
|
|
||||||
┌─────────────────────────────────────────────────────────────────────┐
|
|
||||||
│ ALL AVAILABLE COMMANDS │
|
|
||||||
└─────────────────────────────────────────────────────────────────────┘
|
|
||||||
|
|
||||||
npm run migrate:make <name> Create new migration file
|
|
||||||
npm run migrate:status Check status across all tenants
|
|
||||||
npm run migrate:tenant <slug> Migrate specific tenant
|
|
||||||
npm run migrate:all-tenants Migrate all active tenants
|
|
||||||
npm run migrate:latest Migrate default DB (rarely used)
|
|
||||||
npm run migrate:rollback Rollback default DB (rarely used)
|
|
||||||
|
|
||||||
|
|
||||||
┌─────────────────────────────────────────────────────────────────────┐
|
|
||||||
│ TYPICAL WORKFLOW │
|
|
||||||
└─────────────────────────────────────────────────────────────────────┘
|
|
||||||
|
|
||||||
1. Create: npm run migrate:make add_priority_field
|
|
||||||
2. Edit: vim migrations/tenant/20250127_*.js
|
|
||||||
3. Test: npm run migrate:tenant test-company
|
|
||||||
4. Status: npm run migrate:status
|
|
||||||
5. Deploy: npm run migrate:all-tenants
|
|
||||||
|
|
||||||
|
|
||||||
┌─────────────────────────────────────────────────────────────────────┐
|
|
||||||
│ ENVIRONMENT REQUIRED │
|
|
||||||
└─────────────────────────────────────────────────────────────────────┘
|
|
||||||
|
|
||||||
export DB_ENCRYPTION_KEY="your-32-character-secret-key!!"
|
|
||||||
|
|
||||||
|
|
||||||
┌─────────────────────────────────────────────────────────────────────┐
|
|
||||||
│ FILE LOCATIONS │
|
|
||||||
└─────────────────────────────────────────────────────────────────────┘
|
|
||||||
|
|
||||||
Scripts: backend/scripts/migrate-*.ts
|
|
||||||
Migrations: backend/migrations/tenant/
|
|
||||||
Config: backend/knexfile.js
|
|
||||||
Docs: TENANT_MIGRATION_GUIDE.md
|
|
||||||
|
|
||||||
|
|
||||||
┌─────────────────────────────────────────────────────────────────────┐
|
|
||||||
│ DOCUMENTATION │
|
|
||||||
└─────────────────────────────────────────────────────────────────────┘
|
|
||||||
|
|
||||||
Quick Guide: cat TENANT_MIGRATION_GUIDE.md
|
|
||||||
Script Docs: cat backend/scripts/README.md
|
|
||||||
Complete: cat TENANT_MIGRATION_IMPLEMENTATION_COMPLETE.md
|
|
||||||
|
|
||||||
|
|
||||||
┌─────────────────────────────────────────────────────────────────────┐
|
|
||||||
│ TROUBLESHOOTING │
|
|
||||||
└─────────────────────────────────────────────────────────────────────┘
|
|
||||||
|
|
||||||
Missing Prisma Client:
|
|
||||||
$ npx prisma generate --schema=prisma/schema-central.prisma
|
|
||||||
|
|
||||||
Check Scripts Available:
|
|
||||||
$ npm run | grep migrate
|
|
||||||
|
|
||||||
Connection Error:
|
|
||||||
- Check DB_ENCRYPTION_KEY matches encryption key
|
|
||||||
- Verify central database is accessible
|
|
||||||
- Ensure tenant databases are online
|
|
||||||
|
|
||||||
|
|
||||||
╔══════════════════════════════════════════════════════════════════════╗
|
|
||||||
║ For detailed help: cat TENANT_MIGRATION_GUIDE.md ║
|
|
||||||
╚══════════════════════════════════════════════════════════════════════╝
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
exports.up = function (knex) {
|
|
||||||
return knex.schema.table('object_definitions', (table) => {
|
|
||||||
table.string('nameField', 255).comment('API name of the field to use as record display name');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.down = function (knex) {
|
|
||||||
return knex.schema.table('object_definitions', (table) => {
|
|
||||||
table.dropColumn('nameField');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
exports.up = function (knex) {
|
|
||||||
return knex.schema.table('object_definitions', (table) => {
|
|
||||||
table.uuid('app_id').nullable()
|
|
||||||
.comment('Optional: App that this object belongs to');
|
|
||||||
|
|
||||||
table
|
|
||||||
.foreign('app_id')
|
|
||||||
.references('id')
|
|
||||||
.inTable('apps')
|
|
||||||
.onDelete('SET NULL');
|
|
||||||
|
|
||||||
table.index(['app_id']);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.down = function (knex) {
|
|
||||||
return knex.schema.table('object_definitions', (table) => {
|
|
||||||
table.dropForeign('app_id');
|
|
||||||
table.dropIndex('app_id');
|
|
||||||
table.dropColumn('app_id');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
/**
|
|
||||||
* @param { import("knex").Knex } knex
|
|
||||||
* @returns { Promise<void> }
|
|
||||||
*/
|
|
||||||
exports.up = function(knex) {
|
|
||||||
return knex.schema.createTable('page_layouts', (table) => {
|
|
||||||
table.uuid('id').primary().defaultTo(knex.raw('(UUID())'));
|
|
||||||
table.string('name').notNullable();
|
|
||||||
table.uuid('object_id').notNullable();
|
|
||||||
table.boolean('is_default').defaultTo(false);
|
|
||||||
table.json('layout_config').notNullable();
|
|
||||||
table.text('description');
|
|
||||||
table.timestamps(true, true);
|
|
||||||
|
|
||||||
// Foreign key to object_definitions
|
|
||||||
table.foreign('object_id').references('id').inTable('object_definitions').onDelete('CASCADE');
|
|
||||||
|
|
||||||
// Index for faster lookups
|
|
||||||
table.index(['object_id', 'is_default']);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param { import("knex").Knex } knex
|
|
||||||
* @returns { Promise<void> }
|
|
||||||
*/
|
|
||||||
exports.down = function(knex) {
|
|
||||||
return knex.schema.dropTable('page_layouts');
|
|
||||||
};
|
|
||||||
@@ -17,13 +17,7 @@
|
|||||||
"test:watch": "jest --watch",
|
"test:watch": "jest --watch",
|
||||||
"test:cov": "jest --coverage",
|
"test:cov": "jest --coverage",
|
||||||
"test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand",
|
"test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand",
|
||||||
"test:e2e": "jest --config ./test/jest-e2e.json",
|
"test:e2e": "jest --config ./test/jest-e2e.json"
|
||||||
"migrate:make": "knex migrate:make --knexfile=knexfile.js",
|
|
||||||
"migrate:latest": "knex migrate:latest --knexfile=knexfile.js",
|
|
||||||
"migrate:rollback": "knex migrate:rollback --knexfile=knexfile.js",
|
|
||||||
"migrate:status": "ts-node -r tsconfig-paths/register scripts/check-migration-status.ts",
|
|
||||||
"migrate:tenant": "ts-node -r tsconfig-paths/register scripts/migrate-tenant.ts",
|
|
||||||
"migrate:all-tenants": "ts-node -r tsconfig-paths/register scripts/migrate-all-tenants.ts"
|
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@nestjs/bullmq": "^10.1.0",
|
"@nestjs/bullmq": "^10.1.0",
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
generator client {
|
generator client {
|
||||||
provider = "prisma-client-js"
|
provider = "prisma-client-js"
|
||||||
output = "../node_modules/.prisma/central"
|
output = "../node_modules/.prisma/central"
|
||||||
binaryTargets = ["native", "debian-openssl-3.0.x"]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
datasource db {
|
datasource db {
|
||||||
|
|||||||
@@ -1,11 +1,9 @@
|
|||||||
// Tenant-specific database schema
|
// Tenant-specific database schema
|
||||||
// This schema is applied to each tenant's database
|
// This schema is applied to each tenant's database
|
||||||
// NOTE: Each tenant has its own database, so there is NO tenantId column in these tables
|
|
||||||
|
|
||||||
generator client {
|
generator client {
|
||||||
provider = "prisma-client-js"
|
provider = "prisma-client-js"
|
||||||
output = "../node_modules/.prisma/tenant"
|
output = "../node_modules/.prisma/tenant"
|
||||||
binaryTargets = ["native", "debian-openssl-3.0.x"]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
datasource db {
|
datasource db {
|
||||||
@@ -13,10 +11,30 @@ datasource db {
|
|||||||
url = env("TENANT_DATABASE_URL")
|
url = env("TENANT_DATABASE_URL")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Multi-tenancy
|
||||||
|
model Tenant {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
name String
|
||||||
|
slug String @unique
|
||||||
|
isActive Boolean @default(true)
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
users User[]
|
||||||
|
objectDefinitions ObjectDefinition[]
|
||||||
|
accounts Account[]
|
||||||
|
apps App[]
|
||||||
|
roles Role[]
|
||||||
|
permissions Permission[]
|
||||||
|
|
||||||
|
@@map("tenants")
|
||||||
|
}
|
||||||
|
|
||||||
// User & Auth
|
// User & Auth
|
||||||
model User {
|
model User {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
email String @unique
|
tenantId String
|
||||||
|
email String
|
||||||
password String
|
password String
|
||||||
firstName String?
|
firstName String?
|
||||||
lastName String?
|
lastName String?
|
||||||
@@ -24,39 +42,48 @@ model User {
|
|||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
tenant Tenant @relation(fields: [tenantId], references: [id], onDelete: Cascade)
|
||||||
userRoles UserRole[]
|
userRoles UserRole[]
|
||||||
accounts Account[]
|
accounts Account[]
|
||||||
|
|
||||||
|
@@unique([tenantId, email])
|
||||||
|
@@index([tenantId])
|
||||||
@@map("users")
|
@@map("users")
|
||||||
}
|
}
|
||||||
|
|
||||||
// RBAC - Spatie-like
|
// RBAC - Spatie-like
|
||||||
model Role {
|
model Role {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
|
tenantId String
|
||||||
name String
|
name String
|
||||||
guardName String @default("api")
|
guardName String @default("api")
|
||||||
description String?
|
description String?
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
tenant Tenant @relation(fields: [tenantId], references: [id], onDelete: Cascade)
|
||||||
userRoles UserRole[]
|
userRoles UserRole[]
|
||||||
rolePermissions RolePermission[]
|
rolePermissions RolePermission[]
|
||||||
|
|
||||||
@@unique([name, guardName])
|
@@unique([tenantId, name, guardName])
|
||||||
|
@@index([tenantId])
|
||||||
@@map("roles")
|
@@map("roles")
|
||||||
}
|
}
|
||||||
|
|
||||||
model Permission {
|
model Permission {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
|
tenantId String
|
||||||
name String
|
name String
|
||||||
guardName String @default("api")
|
guardName String @default("api")
|
||||||
description String?
|
description String?
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
tenant Tenant @relation(fields: [tenantId], references: [id], onDelete: Cascade)
|
||||||
rolePermissions RolePermission[]
|
rolePermissions RolePermission[]
|
||||||
|
|
||||||
@@unique([name, guardName])
|
@@unique([tenantId, name, guardName])
|
||||||
|
@@index([tenantId])
|
||||||
@@map("permissions")
|
@@map("permissions")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -93,59 +120,66 @@ model RolePermission {
|
|||||||
// Object Definition (Metadata)
|
// Object Definition (Metadata)
|
||||||
model ObjectDefinition {
|
model ObjectDefinition {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
apiName String @unique
|
tenantId String
|
||||||
|
apiName String
|
||||||
label String
|
label String
|
||||||
pluralLabel String?
|
pluralLabel String?
|
||||||
description String? @db.Text
|
description String? @db.Text
|
||||||
isSystem Boolean @default(false)
|
isSystem Boolean @default(false)
|
||||||
isCustom Boolean @default(true)
|
tableName String?
|
||||||
createdAt DateTime @default(now()) @map("created_at")
|
isActive Boolean @default(true)
|
||||||
updatedAt DateTime @updatedAt @map("updated_at")
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
tenant Tenant @relation(fields: [tenantId], references: [id], onDelete: Cascade)
|
||||||
fields FieldDefinition[]
|
fields FieldDefinition[]
|
||||||
pages AppPage[]
|
pages AppPage[]
|
||||||
|
|
||||||
|
@@unique([tenantId, apiName])
|
||||||
|
@@index([tenantId])
|
||||||
@@map("object_definitions")
|
@@map("object_definitions")
|
||||||
}
|
}
|
||||||
|
|
||||||
model FieldDefinition {
|
model FieldDefinition {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
objectDefinitionId String
|
objectId String
|
||||||
apiName String
|
apiName String
|
||||||
label String
|
label String
|
||||||
type String // String, Number, Date, Boolean, Reference, etc.
|
type String // text, number, boolean, date, datetime, lookup, picklist, etc.
|
||||||
length Int?
|
description String? @db.Text
|
||||||
precision Int?
|
isRequired Boolean @default(false)
|
||||||
scale Int?
|
isUnique Boolean @default(false)
|
||||||
referenceObject String?
|
isReadonly Boolean @default(false)
|
||||||
defaultValue String? @db.Text
|
isLookup Boolean @default(false)
|
||||||
description String? @db.Text
|
referenceTo String? // objectApiName for lookup fields
|
||||||
isRequired Boolean @default(false)
|
defaultValue String?
|
||||||
isUnique Boolean @default(false)
|
options Json? // for picklist fields
|
||||||
isSystem Boolean @default(false)
|
validationRules Json? // custom validation rules
|
||||||
isCustom Boolean @default(true)
|
isActive Boolean @default(true)
|
||||||
displayOrder Int @default(0)
|
createdAt DateTime @default(now())
|
||||||
createdAt DateTime @default(now()) @map("created_at")
|
updatedAt DateTime @updatedAt
|
||||||
updatedAt DateTime @updatedAt @map("updated_at")
|
|
||||||
|
|
||||||
object ObjectDefinition @relation(fields: [objectDefinitionId], references: [id], onDelete: Cascade)
|
object ObjectDefinition @relation(fields: [objectId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
@@unique([objectDefinitionId, apiName])
|
@@unique([objectId, apiName])
|
||||||
@@index([objectDefinitionId])
|
@@index([objectId])
|
||||||
@@map("field_definitions")
|
@@map("field_definitions")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Example static object: Account
|
// Example static object: Account
|
||||||
model Account {
|
model Account {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
|
tenantId String
|
||||||
name String
|
name String
|
||||||
status String @default("active")
|
status String @default("active")
|
||||||
ownerId String
|
ownerId String
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
tenant Tenant @relation(fields: [tenantId], references: [id], onDelete: Cascade)
|
||||||
owner User @relation(fields: [ownerId], references: [id])
|
owner User @relation(fields: [ownerId], references: [id])
|
||||||
|
|
||||||
|
@@index([tenantId])
|
||||||
@@index([ownerId])
|
@@index([ownerId])
|
||||||
@@map("accounts")
|
@@map("accounts")
|
||||||
}
|
}
|
||||||
@@ -153,7 +187,8 @@ model Account {
|
|||||||
// Application Builder
|
// Application Builder
|
||||||
model App {
|
model App {
|
||||||
id String @id @default(uuid())
|
id String @id @default(uuid())
|
||||||
slug String @unique
|
tenantId String
|
||||||
|
slug String
|
||||||
label String
|
label String
|
||||||
description String? @db.Text
|
description String? @db.Text
|
||||||
icon String?
|
icon String?
|
||||||
@@ -161,8 +196,11 @@ model App {
|
|||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
tenant Tenant @relation(fields: [tenantId], references: [id], onDelete: Cascade)
|
||||||
pages AppPage[]
|
pages AppPage[]
|
||||||
|
|
||||||
|
@@unique([tenantId, slug])
|
||||||
|
@@index([tenantId])
|
||||||
@@map("apps")
|
@@map("apps")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,194 +0,0 @@
|
|||||||
# Tenant Migration Scripts
|
|
||||||
|
|
||||||
This directory contains scripts for managing database migrations across all tenants in the multi-tenant platform.
|
|
||||||
|
|
||||||
## Available Scripts
|
|
||||||
|
|
||||||
### 1. Create a New Migration
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run migrate:make <migration_name>
|
|
||||||
```
|
|
||||||
|
|
||||||
Creates a new migration file in `migrations/tenant/` directory.
|
|
||||||
|
|
||||||
**Example:**
|
|
||||||
```bash
|
|
||||||
npm run migrate:make add_status_field_to_contacts
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Migrate a Single Tenant
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run migrate:tenant <tenant-slug-or-id>
|
|
||||||
```
|
|
||||||
|
|
||||||
Runs all pending migrations for a specific tenant. You can identify the tenant by its slug or ID.
|
|
||||||
|
|
||||||
**Example:**
|
|
||||||
```bash
|
|
||||||
npm run migrate:tenant acme-corp
|
|
||||||
npm run migrate:tenant cm5a1b2c3d4e5f6g7h8i9j0k
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. Migrate All Tenants
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
Runs all pending migrations for **all active tenants** in the system. This is useful when:
|
|
||||||
- You've created a new migration that needs to be applied to all tenants
|
|
||||||
- You're updating the schema across the entire platform
|
|
||||||
- You need to ensure all tenants are up to date
|
|
||||||
|
|
||||||
**Output:**
|
|
||||||
- Shows progress for each tenant
|
|
||||||
- Lists which migrations were applied
|
|
||||||
- Provides a summary at the end
|
|
||||||
- Exits with error code if any tenant fails
|
|
||||||
|
|
||||||
### 4. Rollback Migration (Manual)
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run migrate:rollback
|
|
||||||
```
|
|
||||||
|
|
||||||
⚠️ **Warning:** This runs a rollback on the **default database** configured in `knexfile.js`. For tenant-specific rollbacks, you'll need to manually configure the connection.
|
|
||||||
|
|
||||||
## Migration Flow
|
|
||||||
|
|
||||||
### During New Tenant Provisioning
|
|
||||||
|
|
||||||
When a new tenant is created via the API, migrations are automatically run as part of the provisioning process:
|
|
||||||
|
|
||||||
1. Tenant database is created
|
|
||||||
2. `TenantProvisioningService.runTenantMigrations()` is called
|
|
||||||
3. All migrations in `migrations/tenant/` are executed
|
|
||||||
|
|
||||||
### For Existing Tenants
|
|
||||||
|
|
||||||
When you add a new migration file and need to apply it to existing tenants:
|
|
||||||
|
|
||||||
1. Create the migration:
|
|
||||||
```bash
|
|
||||||
npm run migrate:make add_new_feature
|
|
||||||
```
|
|
||||||
|
|
||||||
2. Edit the generated migration file in `migrations/tenant/`
|
|
||||||
|
|
||||||
3. Test on a single tenant first:
|
|
||||||
```bash
|
|
||||||
npm run migrate:tenant test-tenant
|
|
||||||
```
|
|
||||||
|
|
||||||
4. If successful, apply to all tenants:
|
|
||||||
```bash
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
## Migration Directory Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
backend/
|
|
||||||
├── migrations/
|
|
||||||
│ └── tenant/ # Tenant-specific migrations
|
|
||||||
│ ├── 20250126000001_create_users_and_rbac.js
|
|
||||||
│ ├── 20250126000002_create_object_definitions.js
|
|
||||||
│ └── ...
|
|
||||||
├── scripts/
|
|
||||||
│ ├── migrate-tenant.ts # Single tenant migration
|
|
||||||
│ └── migrate-all-tenants.ts # All tenants migration
|
|
||||||
└── knexfile.js # Knex configuration
|
|
||||||
```
|
|
||||||
|
|
||||||
## Security Notes
|
|
||||||
|
|
||||||
### Database Password Encryption
|
|
||||||
|
|
||||||
Tenant database passwords are encrypted in the central database using AES-256-CBC encryption. The migration scripts automatically:
|
|
||||||
|
|
||||||
1. Fetch tenant connection details from the central database
|
|
||||||
2. Decrypt the database password using the `DB_ENCRYPTION_KEY` environment variable
|
|
||||||
3. Connect to the tenant database
|
|
||||||
4. Run migrations
|
|
||||||
5. Close the connection
|
|
||||||
|
|
||||||
**Required Environment Variable:**
|
|
||||||
```bash
|
|
||||||
DB_ENCRYPTION_KEY=your-32-character-secret-key!!
|
|
||||||
```
|
|
||||||
|
|
||||||
This key must match the key used by `TenantService` for encryption.
|
|
||||||
|
|
||||||
## Troubleshooting
|
|
||||||
|
|
||||||
### Migration Fails for One Tenant
|
|
||||||
|
|
||||||
If `migrate:all-tenants` fails for a specific tenant:
|
|
||||||
|
|
||||||
1. Check the error message in the output
|
|
||||||
2. Investigate the tenant's database directly
|
|
||||||
3. Fix the issue (manual SQL, data cleanup, etc.)
|
|
||||||
4. Re-run migrations for that tenant: `npm run migrate:tenant <slug>`
|
|
||||||
5. Once fixed, run `migrate:all-tenants` again to ensure others are updated
|
|
||||||
|
|
||||||
### Migration Already Exists
|
|
||||||
|
|
||||||
Knex tracks which migrations have been run in the `knex_migrations` table in each tenant database. If a migration was already applied, it will be skipped automatically.
|
|
||||||
|
|
||||||
### Connection Issues
|
|
||||||
|
|
||||||
If you see connection errors:
|
|
||||||
|
|
||||||
1. Verify the central database is accessible
|
|
||||||
2. Check that tenant database credentials are correct
|
|
||||||
3. Ensure `DB_ENCRYPTION_KEY` matches the one used for encryption
|
|
||||||
4. Verify the tenant's database server is running and accessible
|
|
||||||
|
|
||||||
## Example Migration File
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
// migrations/tenant/20250126000006_add_custom_fields.js
|
|
||||||
|
|
||||||
exports.up = async function(knex) {
|
|
||||||
await knex.schema.table('field_definitions', (table) => {
|
|
||||||
table.boolean('is_custom').defaultTo(false);
|
|
||||||
table.string('custom_type', 50).nullable();
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.down = async function(knex) {
|
|
||||||
await knex.schema.table('field_definitions', (table) => {
|
|
||||||
table.dropColumn('is_custom');
|
|
||||||
table.dropColumn('custom_type');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
```
|
|
||||||
|
|
||||||
## Best Practices
|
|
||||||
|
|
||||||
1. **Always test on a single tenant first** before running migrations on all tenants
|
|
||||||
2. **Include rollback logic** in your `down()` function
|
|
||||||
3. **Use transactions** for complex multi-step migrations
|
|
||||||
4. **Backup production databases** before running migrations
|
|
||||||
5. **Monitor the output** when running `migrate:all-tenants` to catch any failures
|
|
||||||
6. **Version control** your migration files
|
|
||||||
7. **Document breaking changes** in migration comments
|
|
||||||
8. **Consider data migrations** separately from schema migrations when dealing with large datasets
|
|
||||||
|
|
||||||
## CI/CD Integration
|
|
||||||
|
|
||||||
In your deployment pipeline, you can automatically migrate all tenants:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# After deploying new code
|
|
||||||
npm run migrate:all-tenants
|
|
||||||
```
|
|
||||||
|
|
||||||
Or integrate it into your Docker deployment:
|
|
||||||
|
|
||||||
```dockerfile
|
|
||||||
# In your Dockerfile or docker-compose.yml
|
|
||||||
CMD npm run migrate:all-tenants && npm run start:prod
|
|
||||||
```
|
|
||||||
@@ -1,181 +0,0 @@
|
|||||||
import { PrismaClient as CentralPrismaClient } from '.prisma/central';
|
|
||||||
import knex, { Knex } from 'knex';
|
|
||||||
import { createDecipheriv } from 'crypto';
|
|
||||||
|
|
||||||
// Encryption configuration
|
|
||||||
const ALGORITHM = 'aes-256-cbc';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Decrypt a tenant's database password
|
|
||||||
*/
|
|
||||||
function decryptPassword(encryptedPassword: string): string {
|
|
||||||
try {
|
|
||||||
// Check if password is already plaintext (for legacy/development)
|
|
||||||
if (!encryptedPassword.includes(':')) {
|
|
||||||
return encryptedPassword;
|
|
||||||
}
|
|
||||||
|
|
||||||
const key = Buffer.from(process.env.ENCRYPTION_KEY, 'hex');
|
|
||||||
const parts = encryptedPassword.split(':');
|
|
||||||
if (parts.length !== 2) {
|
|
||||||
throw new Error('Invalid encrypted password format');
|
|
||||||
}
|
|
||||||
|
|
||||||
const iv = Buffer.from(parts[0], 'hex');
|
|
||||||
const encrypted = parts[1];
|
|
||||||
|
|
||||||
const decipher = createDecipheriv(ALGORITHM, key, iv);
|
|
||||||
|
|
||||||
let decrypted = decipher.update(encrypted, 'hex', 'utf8');
|
|
||||||
decrypted += decipher.final('utf8');
|
|
||||||
|
|
||||||
return decrypted;
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error decrypting password:', error);
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a Knex connection for a specific tenant
|
|
||||||
*/
|
|
||||||
function createTenantKnexConnection(tenant: any): Knex {
|
|
||||||
const decryptedPassword = decryptPassword(tenant.dbPassword);
|
|
||||||
|
|
||||||
return knex({
|
|
||||||
client: 'mysql2',
|
|
||||||
connection: {
|
|
||||||
host: tenant.dbHost,
|
|
||||||
port: tenant.dbPort,
|
|
||||||
user: tenant.dbUsername,
|
|
||||||
password: decryptedPassword,
|
|
||||||
database: tenant.dbName,
|
|
||||||
},
|
|
||||||
migrations: {
|
|
||||||
tableName: 'knex_migrations',
|
|
||||||
directory: './migrations/tenant',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get migration status for a specific tenant
|
|
||||||
*/
|
|
||||||
async function getTenantMigrationStatus(tenant: any): Promise<{
|
|
||||||
completed: string[];
|
|
||||||
pending: string[];
|
|
||||||
}> {
|
|
||||||
const tenantKnex = createTenantKnexConnection(tenant);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const [completed, pending] = await tenantKnex.migrate.list();
|
|
||||||
return {
|
|
||||||
completed: completed[1] || [],
|
|
||||||
pending: pending || [],
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
} finally {
|
|
||||||
await tenantKnex.destroy();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check migration status across all tenants
|
|
||||||
*/
|
|
||||||
async function checkMigrationStatus() {
|
|
||||||
console.log('🔍 Checking migration status for all tenants...\n');
|
|
||||||
|
|
||||||
const centralPrisma = new CentralPrismaClient();
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Fetch all active tenants
|
|
||||||
const tenants = await centralPrisma.tenant.findMany({
|
|
||||||
where: {
|
|
||||||
status: 'ACTIVE',
|
|
||||||
},
|
|
||||||
orderBy: {
|
|
||||||
name: 'asc',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (tenants.length === 0) {
|
|
||||||
console.log('⚠️ No active tenants found.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`📋 Found ${tenants.length} active tenant(s)\n`);
|
|
||||||
console.log('='.repeat(80));
|
|
||||||
|
|
||||||
let allUpToDate = true;
|
|
||||||
const tenantsWithPending: { name: string; pending: string[] }[] = [];
|
|
||||||
|
|
||||||
// Check each tenant
|
|
||||||
for (const tenant of tenants) {
|
|
||||||
try {
|
|
||||||
const status = await getTenantMigrationStatus(tenant);
|
|
||||||
|
|
||||||
console.log(`\n📦 ${tenant.name} (${tenant.slug})`);
|
|
||||||
console.log(` Database: ${tenant.dbName}`);
|
|
||||||
console.log(` Completed: ${status.completed.length} migration(s)`);
|
|
||||||
|
|
||||||
if (status.pending.length > 0) {
|
|
||||||
allUpToDate = false;
|
|
||||||
console.log(` ⚠️ Pending: ${status.pending.length} migration(s)`);
|
|
||||||
status.pending.forEach((migration) => {
|
|
||||||
console.log(` - ${migration}`);
|
|
||||||
});
|
|
||||||
tenantsWithPending.push({
|
|
||||||
name: tenant.name,
|
|
||||||
pending: status.pending,
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
console.log(` ✅ Up to date`);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show last 3 completed migrations
|
|
||||||
if (status.completed.length > 0) {
|
|
||||||
const recent = status.completed.slice(-3);
|
|
||||||
console.log(` Recent migrations:`);
|
|
||||||
recent.forEach((migration) => {
|
|
||||||
console.log(` - ${migration}`);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.log(`\n❌ ${tenant.name}: Failed to check status`);
|
|
||||||
console.log(` Error: ${error.message}`);
|
|
||||||
allUpToDate = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print summary
|
|
||||||
console.log('\n' + '='.repeat(80));
|
|
||||||
console.log('📊 Summary');
|
|
||||||
console.log('='.repeat(80));
|
|
||||||
|
|
||||||
if (allUpToDate) {
|
|
||||||
console.log('✅ All tenants are up to date!');
|
|
||||||
} else {
|
|
||||||
console.log(`⚠️ ${tenantsWithPending.length} tenant(s) have pending migrations:\n`);
|
|
||||||
tenantsWithPending.forEach(({ name, pending }) => {
|
|
||||||
console.log(` ${name}: ${pending.length} pending`);
|
|
||||||
});
|
|
||||||
console.log('\n💡 Run: npm run migrate:all-tenants');
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('❌ Fatal error:', error);
|
|
||||||
process.exit(1);
|
|
||||||
} finally {
|
|
||||||
await centralPrisma.$disconnect();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run the status check
|
|
||||||
checkMigrationStatus()
|
|
||||||
.then(() => {
|
|
||||||
process.exit(0);
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('Unhandled error:', error);
|
|
||||||
process.exit(1);
|
|
||||||
});
|
|
||||||
@@ -1,168 +0,0 @@
|
|||||||
import { PrismaClient as CentralPrismaClient } from '.prisma/central';
|
|
||||||
import knex, { Knex } from 'knex';
|
|
||||||
import { createDecipheriv } from 'crypto';
|
|
||||||
|
|
||||||
// Encryption configuration - must match the one used in tenant service
|
|
||||||
const ALGORITHM = 'aes-256-cbc';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Decrypt a tenant's database password
|
|
||||||
*/
|
|
||||||
function decryptPassword(encryptedPassword: string): string {
|
|
||||||
try {
|
|
||||||
// Check if password is already plaintext (for legacy/development)
|
|
||||||
if (!encryptedPassword.includes(':')) {
|
|
||||||
console.warn('⚠️ Password appears to be unencrypted, using as-is');
|
|
||||||
return encryptedPassword;
|
|
||||||
}
|
|
||||||
|
|
||||||
const key = Buffer.from(process.env.ENCRYPTION_KEY, 'hex');
|
|
||||||
const parts = encryptedPassword.split(':');
|
|
||||||
if (parts.length !== 2) {
|
|
||||||
throw new Error('Invalid encrypted password format');
|
|
||||||
}
|
|
||||||
|
|
||||||
const iv = Buffer.from(parts[0], 'hex');
|
|
||||||
const encrypted = parts[1];
|
|
||||||
|
|
||||||
const decipher = createDecipheriv(ALGORITHM, key, iv);
|
|
||||||
|
|
||||||
let decrypted = decipher.update(encrypted, 'hex', 'utf8');
|
|
||||||
decrypted += decipher.final('utf8');
|
|
||||||
|
|
||||||
return decrypted;
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error decrypting password:', error);
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a Knex connection for a specific tenant
|
|
||||||
*/
|
|
||||||
function createTenantKnexConnection(tenant: any): Knex {
|
|
||||||
const decryptedPassword = decryptPassword(tenant.dbPassword);
|
|
||||||
|
|
||||||
// Replace 'db' hostname with 'localhost' when running outside Docker
|
|
||||||
const dbHost = tenant.dbHost === 'db' ? 'localhost' : tenant.dbHost;
|
|
||||||
|
|
||||||
return knex({
|
|
||||||
client: 'mysql2',
|
|
||||||
connection: {
|
|
||||||
host: dbHost,
|
|
||||||
port: tenant.dbPort,
|
|
||||||
user: tenant.dbUsername,
|
|
||||||
password: decryptedPassword,
|
|
||||||
database: tenant.dbName,
|
|
||||||
},
|
|
||||||
migrations: {
|
|
||||||
tableName: 'knex_migrations',
|
|
||||||
directory: './migrations/tenant',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Run migrations for a specific tenant
|
|
||||||
*/
|
|
||||||
async function migrateTenant(tenant: any): Promise<void> {
|
|
||||||
console.log(`\n🔄 Migrating tenant: ${tenant.name} (${tenant.dbName})`);
|
|
||||||
|
|
||||||
const tenantKnex = createTenantKnexConnection(tenant);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const [batchNo, log] = await tenantKnex.migrate.latest();
|
|
||||||
|
|
||||||
if (log.length === 0) {
|
|
||||||
console.log(`✅ ${tenant.name}: Already up to date`);
|
|
||||||
} else {
|
|
||||||
console.log(`✅ ${tenant.name}: Ran ${log.length} migrations:`);
|
|
||||||
log.forEach((migration) => {
|
|
||||||
console.log(` - ${migration}`);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`❌ ${tenant.name}: Migration failed:`, error.message);
|
|
||||||
throw error;
|
|
||||||
} finally {
|
|
||||||
await tenantKnex.destroy();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Main function to migrate all active tenants
|
|
||||||
*/
|
|
||||||
async function migrateAllTenants() {
|
|
||||||
console.log('🚀 Starting migration for all tenants...\n');
|
|
||||||
|
|
||||||
const centralPrisma = new CentralPrismaClient();
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Fetch all active tenants
|
|
||||||
const tenants = await centralPrisma.tenant.findMany({
|
|
||||||
where: {
|
|
||||||
status: 'ACTIVE',
|
|
||||||
},
|
|
||||||
orderBy: {
|
|
||||||
name: 'asc',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (tenants.length === 0) {
|
|
||||||
console.log('⚠️ No active tenants found.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`📋 Found ${tenants.length} active tenant(s)\n`);
|
|
||||||
|
|
||||||
let successCount = 0;
|
|
||||||
let failureCount = 0;
|
|
||||||
const failures: { tenant: string; error: string }[] = [];
|
|
||||||
|
|
||||||
// Migrate each tenant sequentially
|
|
||||||
for (const tenant of tenants) {
|
|
||||||
try {
|
|
||||||
await migrateTenant(tenant);
|
|
||||||
successCount++;
|
|
||||||
} catch (error) {
|
|
||||||
failureCount++;
|
|
||||||
failures.push({
|
|
||||||
tenant: tenant.name,
|
|
||||||
error: error.message,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print summary
|
|
||||||
console.log('\n' + '='.repeat(60));
|
|
||||||
console.log('📊 Migration Summary');
|
|
||||||
console.log('='.repeat(60));
|
|
||||||
console.log(`✅ Successful: ${successCount}`);
|
|
||||||
console.log(`❌ Failed: ${failureCount}`);
|
|
||||||
|
|
||||||
if (failures.length > 0) {
|
|
||||||
console.log('\n❌ Failed Tenants:');
|
|
||||||
failures.forEach(({ tenant, error }) => {
|
|
||||||
console.log(` - ${tenant}: ${error}`);
|
|
||||||
});
|
|
||||||
process.exit(1);
|
|
||||||
} else {
|
|
||||||
console.log('\n🎉 All tenant migrations completed successfully!');
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('❌ Fatal error:', error);
|
|
||||||
process.exit(1);
|
|
||||||
} finally {
|
|
||||||
await centralPrisma.$disconnect();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run the migration
|
|
||||||
migrateAllTenants()
|
|
||||||
.then(() => {
|
|
||||||
process.exit(0);
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('Unhandled error:', error);
|
|
||||||
process.exit(1);
|
|
||||||
});
|
|
||||||
@@ -1,134 +0,0 @@
|
|||||||
import { PrismaClient as CentralPrismaClient } from '.prisma/central';
|
|
||||||
import knex, { Knex } from 'knex';
|
|
||||||
import { createDecipheriv } from 'crypto';
|
|
||||||
|
|
||||||
// Encryption configuration
|
|
||||||
const ALGORITHM = 'aes-256-cbc';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Decrypt a tenant's database password
|
|
||||||
*/
|
|
||||||
function decryptPassword(encryptedPassword: string): string {
|
|
||||||
try {
|
|
||||||
// Check if password is already plaintext (for legacy/development)
|
|
||||||
if (!encryptedPassword.includes(':')) {
|
|
||||||
console.warn('⚠️ Password appears to be unencrypted, using as-is');
|
|
||||||
return encryptedPassword;
|
|
||||||
}
|
|
||||||
|
|
||||||
const key = Buffer.from(process.env.ENCRYPTION_KEY, 'hex');
|
|
||||||
const parts = encryptedPassword.split(':');
|
|
||||||
if (parts.length !== 2) {
|
|
||||||
throw new Error('Invalid encrypted password format');
|
|
||||||
}
|
|
||||||
|
|
||||||
const iv = Buffer.from(parts[0], 'hex');
|
|
||||||
const encrypted = parts[1];
|
|
||||||
|
|
||||||
const decipher = createDecipheriv(ALGORITHM, key, iv);
|
|
||||||
|
|
||||||
let decrypted = decipher.update(encrypted, 'hex', 'utf8');
|
|
||||||
decrypted += decipher.final('utf8');
|
|
||||||
|
|
||||||
return decrypted;
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error decrypting password:', error);
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a Knex connection for a specific tenant
|
|
||||||
*/
|
|
||||||
function createTenantKnexConnection(tenant: any): Knex {
|
|
||||||
const decryptedPassword = decryptPassword(tenant.dbPassword);
|
|
||||||
|
|
||||||
return knex({
|
|
||||||
client: 'mysql2',
|
|
||||||
connection: {
|
|
||||||
host: tenant.dbHost,
|
|
||||||
port: tenant.dbPort,
|
|
||||||
user: tenant.dbUsername,
|
|
||||||
password: decryptedPassword,
|
|
||||||
database: tenant.dbName,
|
|
||||||
},
|
|
||||||
migrations: {
|
|
||||||
tableName: 'knex_migrations',
|
|
||||||
directory: './migrations/tenant',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Migrate a specific tenant by slug or ID
|
|
||||||
*/
|
|
||||||
async function migrateTenant() {
|
|
||||||
const tenantIdentifier = process.argv[2];
|
|
||||||
|
|
||||||
if (!tenantIdentifier) {
|
|
||||||
console.error('❌ Usage: npm run migrate:tenant <tenant-slug-or-id>');
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`🔍 Looking for tenant: ${tenantIdentifier}\n`);
|
|
||||||
|
|
||||||
const centralPrisma = new CentralPrismaClient();
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Find tenant by slug or ID
|
|
||||||
const tenant = await centralPrisma.tenant.findFirst({
|
|
||||||
where: {
|
|
||||||
OR: [
|
|
||||||
{ slug: tenantIdentifier },
|
|
||||||
{ id: tenantIdentifier },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!tenant) {
|
|
||||||
console.error(`❌ Tenant not found: ${tenantIdentifier}`);
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`📋 Tenant: ${tenant.name} (${tenant.slug})`);
|
|
||||||
console.log(`📊 Database: ${tenant.dbName}`);
|
|
||||||
console.log(`🔄 Running migrations...\n`);
|
|
||||||
|
|
||||||
const tenantKnex = createTenantKnexConnection(tenant);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const [batchNo, log] = await tenantKnex.migrate.latest();
|
|
||||||
|
|
||||||
if (log.length === 0) {
|
|
||||||
console.log(`✅ Already up to date (batch ${batchNo})`);
|
|
||||||
} else {
|
|
||||||
console.log(`✅ Ran ${log.length} migration(s) (batch ${batchNo}):`);
|
|
||||||
log.forEach((migration) => {
|
|
||||||
console.log(` - ${migration}`);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n🎉 Migration completed successfully!');
|
|
||||||
} catch (error) {
|
|
||||||
console.error('❌ Migration failed:', error.message);
|
|
||||||
throw error;
|
|
||||||
} finally {
|
|
||||||
await tenantKnex.destroy();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('❌ Fatal error:', error);
|
|
||||||
process.exit(1);
|
|
||||||
} finally {
|
|
||||||
await centralPrisma.$disconnect();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run the migration
|
|
||||||
migrateTenant()
|
|
||||||
.then(() => {
|
|
||||||
process.exit(0);
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('Unhandled error:', error);
|
|
||||||
process.exit(1);
|
|
||||||
});
|
|
||||||
@@ -1,72 +0,0 @@
|
|||||||
import { getCentralPrisma } from '../src/prisma/central-prisma.service';
|
|
||||||
import * as knex from 'knex';
|
|
||||||
import * as crypto from 'crypto';
|
|
||||||
|
|
||||||
function decrypt(text: string): string {
|
|
||||||
const parts = text.split(':');
|
|
||||||
const iv = Buffer.from(parts.shift()!, 'hex');
|
|
||||||
const encryptedText = Buffer.from(parts.join(':'), 'hex');
|
|
||||||
const key = Buffer.from(process.env.ENCRYPTION_KEY, 'hex');
|
|
||||||
const decipher = crypto.createDecipheriv(
|
|
||||||
'aes-256-cbc',
|
|
||||||
key,
|
|
||||||
iv,
|
|
||||||
);
|
|
||||||
let decrypted = decipher.update(encryptedText);
|
|
||||||
decrypted = Buffer.concat([decrypted, decipher.final()]);
|
|
||||||
return decrypted.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
async function updateNameField() {
|
|
||||||
const centralPrisma = getCentralPrisma();
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Find tenant1
|
|
||||||
const tenant = await centralPrisma.tenant.findFirst({
|
|
||||||
where: {
|
|
||||||
OR: [
|
|
||||||
{ id: 'tenant1' },
|
|
||||||
{ slug: 'tenant1' },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!tenant) {
|
|
||||||
console.error('❌ Tenant tenant1 not found');
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`📋 Tenant: ${tenant.name} (${tenant.slug})`);
|
|
||||||
console.log(`📊 Database: ${tenant.dbName}`);
|
|
||||||
|
|
||||||
// Decrypt password
|
|
||||||
const password = decrypt(tenant.dbPassword);
|
|
||||||
|
|
||||||
// Create connection
|
|
||||||
const tenantKnex = knex.default({
|
|
||||||
client: 'mysql2',
|
|
||||||
connection: {
|
|
||||||
host: tenant.dbHost,
|
|
||||||
port: tenant.dbPort,
|
|
||||||
user: tenant.dbUsername,
|
|
||||||
password: password,
|
|
||||||
database: tenant.dbName,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
// Update Account object
|
|
||||||
await tenantKnex('object_definitions')
|
|
||||||
.where({ apiName: 'Account' })
|
|
||||||
.update({ nameField: 'name' });
|
|
||||||
|
|
||||||
console.log('✅ Updated Account object nameField to "name"');
|
|
||||||
|
|
||||||
await tenantKnex.destroy();
|
|
||||||
await centralPrisma.$disconnect();
|
|
||||||
} catch (error) {
|
|
||||||
console.error('❌ Error:', error);
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
updateNameField();
|
|
||||||
@@ -1,147 +0,0 @@
|
|||||||
/**
|
|
||||||
* Example seed data for Account object with UI metadata
|
|
||||||
* Run this after migrations to add UI metadata to existing Account fields
|
|
||||||
*/
|
|
||||||
|
|
||||||
exports.seed = async function(knex) {
|
|
||||||
// Get the Account object
|
|
||||||
const accountObj = await knex('object_definitions')
|
|
||||||
.where({ apiName: 'Account' })
|
|
||||||
.first();
|
|
||||||
|
|
||||||
if (!accountObj) {
|
|
||||||
console.log('Account object not found. Please run migrations first.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`Found Account object with ID: ${accountObj.id}`);
|
|
||||||
|
|
||||||
// Update existing Account fields with UI metadata
|
|
||||||
const fieldsToUpdate = [
|
|
||||||
{
|
|
||||||
apiName: 'name',
|
|
||||||
ui_metadata: JSON.stringify({
|
|
||||||
fieldType: 'TEXT',
|
|
||||||
placeholder: 'Enter account name',
|
|
||||||
helpText: 'The name of the organization or company',
|
|
||||||
showOnList: true,
|
|
||||||
showOnDetail: true,
|
|
||||||
showOnEdit: true,
|
|
||||||
sortable: true,
|
|
||||||
section: 'basic',
|
|
||||||
sectionLabel: 'Basic Information',
|
|
||||||
sectionOrder: 1,
|
|
||||||
validationRules: [
|
|
||||||
{ type: 'required', message: 'Account name is required' },
|
|
||||||
{ type: 'minLength', value: 2, message: 'Account name must be at least 2 characters' },
|
|
||||||
{ type: 'maxLength', value: 255, message: 'Account name cannot exceed 255 characters' }
|
|
||||||
]
|
|
||||||
})
|
|
||||||
},
|
|
||||||
{
|
|
||||||
apiName: 'website',
|
|
||||||
ui_metadata: JSON.stringify({
|
|
||||||
fieldType: 'URL',
|
|
||||||
placeholder: 'https://www.example.com',
|
|
||||||
helpText: 'Company website URL',
|
|
||||||
showOnList: true,
|
|
||||||
showOnDetail: true,
|
|
||||||
showOnEdit: true,
|
|
||||||
sortable: true,
|
|
||||||
section: 'basic',
|
|
||||||
sectionLabel: 'Basic Information',
|
|
||||||
sectionOrder: 1,
|
|
||||||
validationRules: [
|
|
||||||
{ type: 'url', message: 'Please enter a valid URL' }
|
|
||||||
]
|
|
||||||
})
|
|
||||||
},
|
|
||||||
{
|
|
||||||
apiName: 'phone',
|
|
||||||
ui_metadata: JSON.stringify({
|
|
||||||
fieldType: 'TEXT',
|
|
||||||
placeholder: '+1 (555) 000-0000',
|
|
||||||
helpText: 'Primary phone number',
|
|
||||||
showOnList: true,
|
|
||||||
showOnDetail: true,
|
|
||||||
showOnEdit: true,
|
|
||||||
sortable: false,
|
|
||||||
section: 'contact',
|
|
||||||
sectionLabel: 'Contact Information',
|
|
||||||
sectionOrder: 2,
|
|
||||||
validationRules: [
|
|
||||||
{ type: 'pattern', value: '^\\+?[0-9\\s\\-\\(\\)]+$', message: 'Please enter a valid phone number' }
|
|
||||||
]
|
|
||||||
})
|
|
||||||
},
|
|
||||||
{
|
|
||||||
apiName: 'industry',
|
|
||||||
ui_metadata: JSON.stringify({
|
|
||||||
fieldType: 'SELECT',
|
|
||||||
placeholder: 'Select industry',
|
|
||||||
helpText: 'The primary industry this account operates in',
|
|
||||||
showOnList: true,
|
|
||||||
showOnDetail: true,
|
|
||||||
showOnEdit: true,
|
|
||||||
sortable: true,
|
|
||||||
section: 'details',
|
|
||||||
sectionLabel: 'Account Details',
|
|
||||||
sectionOrder: 3,
|
|
||||||
options: [
|
|
||||||
{ value: 'technology', label: 'Technology' },
|
|
||||||
{ value: 'finance', label: 'Finance' },
|
|
||||||
{ value: 'healthcare', label: 'Healthcare' },
|
|
||||||
{ value: 'manufacturing', label: 'Manufacturing' },
|
|
||||||
{ value: 'retail', label: 'Retail' },
|
|
||||||
{ value: 'education', label: 'Education' },
|
|
||||||
{ value: 'government', label: 'Government' },
|
|
||||||
{ value: 'nonprofit', label: 'Non-Profit' },
|
|
||||||
{ value: 'other', label: 'Other' }
|
|
||||||
]
|
|
||||||
})
|
|
||||||
},
|
|
||||||
{
|
|
||||||
apiName: 'ownerId',
|
|
||||||
ui_metadata: JSON.stringify({
|
|
||||||
fieldType: 'SELECT',
|
|
||||||
placeholder: 'Select owner',
|
|
||||||
helpText: 'The user who owns this account',
|
|
||||||
showOnList: true,
|
|
||||||
showOnDetail: true,
|
|
||||||
showOnEdit: true,
|
|
||||||
sortable: true,
|
|
||||||
section: 'system',
|
|
||||||
sectionLabel: 'System Information',
|
|
||||||
sectionOrder: 4,
|
|
||||||
// This would be dynamically populated from the users table
|
|
||||||
// For now, providing static structure
|
|
||||||
isReference: true,
|
|
||||||
referenceObject: 'User',
|
|
||||||
referenceDisplayField: 'name'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
// Update each field with UI metadata
|
|
||||||
for (const fieldUpdate of fieldsToUpdate) {
|
|
||||||
const result = await knex('field_definitions')
|
|
||||||
.where({
|
|
||||||
objectDefinitionId: accountObj.id,
|
|
||||||
apiName: fieldUpdate.apiName
|
|
||||||
})
|
|
||||||
.update({
|
|
||||||
ui_metadata: fieldUpdate.ui_metadata,
|
|
||||||
updated_at: knex.fn.now()
|
|
||||||
});
|
|
||||||
|
|
||||||
if (result > 0) {
|
|
||||||
console.log(`✓ Updated ${fieldUpdate.apiName} with UI metadata`);
|
|
||||||
} else {
|
|
||||||
console.log(`✗ Field ${fieldUpdate.apiName} not found`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\n✅ Account fields UI metadata seed completed successfully!');
|
|
||||||
console.log('You can now fetch the Account object UI config via:');
|
|
||||||
console.log('GET /api/setup/objects/Account/ui-config');
|
|
||||||
};
|
|
||||||
@@ -12,7 +12,7 @@ import { JwtAuthGuard } from '../auth/jwt-auth.guard';
|
|||||||
import { TenantId } from '../tenant/tenant.decorator';
|
import { TenantId } from '../tenant/tenant.decorator';
|
||||||
|
|
||||||
@Controller('setup/apps')
|
@Controller('setup/apps')
|
||||||
@UseGuards(JwtAuthGuard)
|
//@UseGuards(JwtAuthGuard)
|
||||||
export class SetupAppController {
|
export class SetupAppController {
|
||||||
constructor(private appBuilderService: AppBuilderService) {}
|
constructor(private appBuilderService: AppBuilderService) {}
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import { AuthModule } from './auth/auth.module';
|
|||||||
import { RbacModule } from './rbac/rbac.module';
|
import { RbacModule } from './rbac/rbac.module';
|
||||||
import { ObjectModule } from './object/object.module';
|
import { ObjectModule } from './object/object.module';
|
||||||
import { AppBuilderModule } from './app-builder/app-builder.module';
|
import { AppBuilderModule } from './app-builder/app-builder.module';
|
||||||
import { PageLayoutModule } from './page-layout/page-layout.module';
|
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [
|
imports: [
|
||||||
@@ -19,7 +18,6 @@ import { PageLayoutModule } from './page-layout/page-layout.module';
|
|||||||
RbacModule,
|
RbacModule,
|
||||||
ObjectModule,
|
ObjectModule,
|
||||||
AppBuilderModule,
|
AppBuilderModule,
|
||||||
PageLayoutModule,
|
|
||||||
],
|
],
|
||||||
})
|
})
|
||||||
export class AppModule {}
|
export class AppModule {}
|
||||||
|
|||||||
@@ -79,12 +79,4 @@ export class AuthController {
|
|||||||
|
|
||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
||||||
@HttpCode(HttpStatus.OK)
|
|
||||||
@Post('logout')
|
|
||||||
async logout() {
|
|
||||||
// For stateless JWT, logout is handled on client-side
|
|
||||||
// This endpoint exists for consistency and potential future enhancements
|
|
||||||
return { message: 'Logged out successfully' };
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -47,6 +47,7 @@ export class AuthService {
|
|||||||
const payload = {
|
const payload = {
|
||||||
sub: user.id,
|
sub: user.id,
|
||||||
email: user.email,
|
email: user.email,
|
||||||
|
tenantId: user.tenantId,
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@@ -56,6 +57,7 @@ export class AuthService {
|
|||||||
email: user.email,
|
email: user.email,
|
||||||
firstName: user.firstName,
|
firstName: user.firstName,
|
||||||
lastName: user.lastName,
|
lastName: user.lastName,
|
||||||
|
tenantId: user.tenantId,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,62 +1,42 @@
|
|||||||
import { Injectable, NotFoundException } from '@nestjs/common';
|
import { Injectable, NotFoundException } from '@nestjs/common';
|
||||||
import { TenantDatabaseService } from '../tenant/tenant-database.service';
|
import { PrismaService } from '../prisma/prisma.service';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ObjectService {
|
export class ObjectService {
|
||||||
constructor(private tenantDbService: TenantDatabaseService) {}
|
constructor(private prisma: PrismaService) {}
|
||||||
|
|
||||||
// Setup endpoints - Object metadata management
|
// Setup endpoints - Object metadata management
|
||||||
async getObjectDefinitions(tenantId: string) {
|
async getObjectDefinitions(tenantId: string) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
return this.prisma.objectDefinition.findMany({
|
||||||
|
where: { tenantId },
|
||||||
const objects = await knex('object_definitions')
|
include: {
|
||||||
.select('object_definitions.*')
|
fields: true,
|
||||||
.orderBy('label', 'asc');
|
},
|
||||||
|
orderBy: { label: 'asc' },
|
||||||
// Fetch app information for objects that have app_id
|
});
|
||||||
for (const obj of objects) {
|
|
||||||
if (obj.app_id) {
|
|
||||||
const app = await knex('apps')
|
|
||||||
.where({ id: obj.app_id })
|
|
||||||
.select('id', 'slug', 'label', 'description')
|
|
||||||
.first();
|
|
||||||
obj.app = app;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return objects;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async getObjectDefinition(tenantId: string, apiName: string) {
|
async getObjectDefinition(tenantId: string, apiName: string) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
const obj = await this.prisma.objectDefinition.findUnique({
|
||||||
|
where: {
|
||||||
const obj = await knex('object_definitions')
|
tenantId_apiName: {
|
||||||
.where({ apiName })
|
tenantId,
|
||||||
.first();
|
apiName,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
fields: {
|
||||||
|
where: { isActive: true },
|
||||||
|
orderBy: { label: 'asc' },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
if (!obj) {
|
if (!obj) {
|
||||||
throw new NotFoundException(`Object ${apiName} not found`);
|
throw new NotFoundException(`Object ${apiName} not found`);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get fields for this object
|
return obj;
|
||||||
const fields = await knex('field_definitions')
|
|
||||||
.where({ objectDefinitionId: obj.id })
|
|
||||||
.orderBy('label', 'asc');
|
|
||||||
|
|
||||||
// Get app information if object belongs to an app
|
|
||||||
let app = null;
|
|
||||||
if (obj.app_id) {
|
|
||||||
app = await knex('apps')
|
|
||||||
.where({ id: obj.app_id })
|
|
||||||
.select('id', 'slug', 'label', 'description')
|
|
||||||
.first();
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
...obj,
|
|
||||||
fields,
|
|
||||||
app,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async createObjectDefinition(
|
async createObjectDefinition(
|
||||||
@@ -69,15 +49,13 @@ export class ObjectService {
|
|||||||
isSystem?: boolean;
|
isSystem?: boolean;
|
||||||
},
|
},
|
||||||
) {
|
) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
return this.prisma.objectDefinition.create({
|
||||||
const [id] = await knex('object_definitions').insert({
|
data: {
|
||||||
id: knex.raw('(UUID())'),
|
tenantId,
|
||||||
...data,
|
...data,
|
||||||
created_at: knex.fn.now(),
|
tableName: `custom_${data.apiName.toLowerCase()}`,
|
||||||
updated_at: knex.fn.now(),
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return knex('object_definitions').where({ id }).first();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async createFieldDefinition(
|
async createFieldDefinition(
|
||||||
@@ -90,41 +68,20 @@ export class ObjectService {
|
|||||||
description?: string;
|
description?: string;
|
||||||
isRequired?: boolean;
|
isRequired?: boolean;
|
||||||
isUnique?: boolean;
|
isUnique?: boolean;
|
||||||
referenceObject?: string;
|
isLookup?: boolean;
|
||||||
|
referenceTo?: string;
|
||||||
defaultValue?: string;
|
defaultValue?: string;
|
||||||
|
options?: any;
|
||||||
},
|
},
|
||||||
) {
|
) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
|
||||||
const obj = await this.getObjectDefinition(tenantId, objectApiName);
|
const obj = await this.getObjectDefinition(tenantId, objectApiName);
|
||||||
|
|
||||||
const [id] = await knex('field_definitions').insert({
|
return this.prisma.fieldDefinition.create({
|
||||||
id: knex.raw('(UUID())'),
|
data: {
|
||||||
objectDefinitionId: obj.id,
|
objectId: obj.id,
|
||||||
...data,
|
...data,
|
||||||
created_at: knex.fn.now(),
|
},
|
||||||
updated_at: knex.fn.now(),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return knex('field_definitions').where({ id }).first();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Helper to get table name from object definition
|
|
||||||
private getTableName(objectApiName: string): string {
|
|
||||||
// Convert CamelCase to snake_case and pluralize
|
|
||||||
// Account -> accounts, ContactPerson -> contact_persons
|
|
||||||
const snakeCase = objectApiName
|
|
||||||
.replace(/([A-Z])/g, '_$1')
|
|
||||||
.toLowerCase()
|
|
||||||
.replace(/^_/, '');
|
|
||||||
|
|
||||||
// Simple pluralization (can be enhanced)
|
|
||||||
if (snakeCase.endsWith('y')) {
|
|
||||||
return snakeCase.slice(0, -1) + 'ies';
|
|
||||||
} else if (snakeCase.endsWith('s')) {
|
|
||||||
return snakeCase;
|
|
||||||
} else {
|
|
||||||
return snakeCase + 's';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Runtime endpoints - CRUD operations
|
// Runtime endpoints - CRUD operations
|
||||||
@@ -134,27 +91,20 @@ export class ObjectService {
|
|||||||
userId: string,
|
userId: string,
|
||||||
filters?: any,
|
filters?: any,
|
||||||
) {
|
) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
// For demonstration, using Account as example static object
|
||||||
|
if (objectApiName === 'Account') {
|
||||||
// Verify object exists
|
return this.prisma.account.findMany({
|
||||||
await this.getObjectDefinition(tenantId, objectApiName);
|
where: {
|
||||||
|
tenantId,
|
||||||
const tableName = this.getTableName(objectApiName);
|
ownerId: userId, // Basic sharing rule
|
||||||
|
...filters,
|
||||||
let query = knex(tableName);
|
},
|
||||||
|
});
|
||||||
// Add ownership filter if ownerId field exists
|
|
||||||
const hasOwner = await knex.schema.hasColumn(tableName, 'ownerId');
|
|
||||||
if (hasOwner) {
|
|
||||||
query = query.where({ ownerId: userId });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Apply additional filters
|
// For custom objects, you'd need dynamic query building
|
||||||
if (filters) {
|
// This is a simplified version
|
||||||
query = query.where(filters);
|
throw new Error(`Runtime queries for ${objectApiName} not yet implemented`);
|
||||||
}
|
|
||||||
|
|
||||||
return query.select('*');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async getRecord(
|
async getRecord(
|
||||||
@@ -163,28 +113,23 @@ export class ObjectService {
|
|||||||
recordId: string,
|
recordId: string,
|
||||||
userId: string,
|
userId: string,
|
||||||
) {
|
) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
if (objectApiName === 'Account') {
|
||||||
|
const record = await this.prisma.account.findFirst({
|
||||||
|
where: {
|
||||||
|
id: recordId,
|
||||||
|
tenantId,
|
||||||
|
ownerId: userId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
// Verify object exists
|
if (!record) {
|
||||||
await this.getObjectDefinition(tenantId, objectApiName);
|
throw new NotFoundException('Record not found');
|
||||||
|
}
|
||||||
|
|
||||||
const tableName = this.getTableName(objectApiName);
|
return record;
|
||||||
|
|
||||||
let query = knex(tableName).where({ id: recordId });
|
|
||||||
|
|
||||||
// Add ownership filter if ownerId field exists
|
|
||||||
const hasOwner = await knex.schema.hasColumn(tableName, 'ownerId');
|
|
||||||
if (hasOwner) {
|
|
||||||
query = query.where({ ownerId: userId });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const record = await query.first();
|
throw new Error(`Runtime queries for ${objectApiName} not yet implemented`);
|
||||||
|
|
||||||
if (!record) {
|
|
||||||
throw new NotFoundException('Record not found');
|
|
||||||
}
|
|
||||||
|
|
||||||
return record;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async createRecord(
|
async createRecord(
|
||||||
@@ -193,30 +138,17 @@ export class ObjectService {
|
|||||||
data: any,
|
data: any,
|
||||||
userId: string,
|
userId: string,
|
||||||
) {
|
) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
if (objectApiName === 'Account') {
|
||||||
|
return this.prisma.account.create({
|
||||||
// Verify object exists
|
data: {
|
||||||
await this.getObjectDefinition(tenantId, objectApiName);
|
tenantId,
|
||||||
|
ownerId: userId,
|
||||||
const tableName = this.getTableName(objectApiName);
|
...data,
|
||||||
|
},
|
||||||
// Check if table has ownerId column
|
});
|
||||||
const hasOwner = await knex.schema.hasColumn(tableName, 'ownerId');
|
|
||||||
|
|
||||||
const recordData: any = {
|
|
||||||
id: knex.raw('(UUID())'),
|
|
||||||
...data,
|
|
||||||
created_at: knex.fn.now(),
|
|
||||||
updated_at: knex.fn.now(),
|
|
||||||
};
|
|
||||||
|
|
||||||
if (hasOwner) {
|
|
||||||
recordData.ownerId = userId;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const [id] = await knex(tableName).insert(recordData);
|
throw new Error(`Runtime queries for ${objectApiName} not yet implemented`);
|
||||||
|
|
||||||
return knex(tableName).where({ id }).first();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateRecord(
|
async updateRecord(
|
||||||
@@ -226,18 +158,17 @@ export class ObjectService {
|
|||||||
data: any,
|
data: any,
|
||||||
userId: string,
|
userId: string,
|
||||||
) {
|
) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
if (objectApiName === 'Account') {
|
||||||
|
// Verify ownership
|
||||||
|
await this.getRecord(tenantId, objectApiName, recordId, userId);
|
||||||
|
|
||||||
// Verify object exists and user has access
|
return this.prisma.account.update({
|
||||||
await this.getRecord(tenantId, objectApiName, recordId, userId);
|
where: { id: recordId },
|
||||||
|
data,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const tableName = this.getTableName(objectApiName);
|
throw new Error(`Runtime queries for ${objectApiName} not yet implemented`);
|
||||||
|
|
||||||
await knex(tableName)
|
|
||||||
.where({ id: recordId })
|
|
||||||
.update({ ...data, updated_at: knex.fn.now() });
|
|
||||||
|
|
||||||
return knex(tableName).where({ id: recordId }).first();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async deleteRecord(
|
async deleteRecord(
|
||||||
@@ -246,15 +177,15 @@ export class ObjectService {
|
|||||||
recordId: string,
|
recordId: string,
|
||||||
userId: string,
|
userId: string,
|
||||||
) {
|
) {
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
if (objectApiName === 'Account') {
|
||||||
|
// Verify ownership
|
||||||
|
await this.getRecord(tenantId, objectApiName, recordId, userId);
|
||||||
|
|
||||||
// Verify object exists and user has access
|
return this.prisma.account.delete({
|
||||||
await this.getRecord(tenantId, objectApiName, recordId, userId);
|
where: { id: recordId },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const tableName = this.getTableName(objectApiName);
|
throw new Error(`Runtime queries for ${objectApiName} not yet implemented`);
|
||||||
|
|
||||||
await knex(tableName).where({ id: recordId }).delete();
|
|
||||||
|
|
||||||
return { success: true };
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,54 +0,0 @@
|
|||||||
import { IsString, IsUUID, IsBoolean, IsOptional, IsObject } from 'class-validator';
|
|
||||||
|
|
||||||
export class CreatePageLayoutDto {
|
|
||||||
@IsString()
|
|
||||||
name: string;
|
|
||||||
|
|
||||||
@IsUUID()
|
|
||||||
objectId: string;
|
|
||||||
|
|
||||||
@IsBoolean()
|
|
||||||
@IsOptional()
|
|
||||||
isDefault?: boolean;
|
|
||||||
|
|
||||||
@IsObject()
|
|
||||||
layoutConfig: {
|
|
||||||
fields: Array<{
|
|
||||||
fieldId: string;
|
|
||||||
x: number;
|
|
||||||
y: number;
|
|
||||||
w: number;
|
|
||||||
h: number;
|
|
||||||
}>;
|
|
||||||
};
|
|
||||||
|
|
||||||
@IsString()
|
|
||||||
@IsOptional()
|
|
||||||
description?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export class UpdatePageLayoutDto {
|
|
||||||
@IsString()
|
|
||||||
@IsOptional()
|
|
||||||
name?: string;
|
|
||||||
|
|
||||||
@IsBoolean()
|
|
||||||
@IsOptional()
|
|
||||||
isDefault?: boolean;
|
|
||||||
|
|
||||||
@IsObject()
|
|
||||||
@IsOptional()
|
|
||||||
layoutConfig?: {
|
|
||||||
fields: Array<{
|
|
||||||
fieldId: string;
|
|
||||||
x: number;
|
|
||||||
y: number;
|
|
||||||
w: number;
|
|
||||||
h: number;
|
|
||||||
}>;
|
|
||||||
};
|
|
||||||
|
|
||||||
@IsString()
|
|
||||||
@IsOptional()
|
|
||||||
description?: string;
|
|
||||||
}
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
import {
|
|
||||||
Controller,
|
|
||||||
Get,
|
|
||||||
Post,
|
|
||||||
Body,
|
|
||||||
Patch,
|
|
||||||
Param,
|
|
||||||
Delete,
|
|
||||||
UseGuards,
|
|
||||||
Query,
|
|
||||||
} from '@nestjs/common';
|
|
||||||
import { PageLayoutService } from './page-layout.service';
|
|
||||||
import { CreatePageLayoutDto, UpdatePageLayoutDto } from './dto/page-layout.dto';
|
|
||||||
import { JwtAuthGuard } from '../auth/jwt-auth.guard';
|
|
||||||
import { TenantId } from '../tenant/tenant.decorator';
|
|
||||||
|
|
||||||
@Controller('page-layouts')
|
|
||||||
@UseGuards(JwtAuthGuard)
|
|
||||||
export class PageLayoutController {
|
|
||||||
constructor(private readonly pageLayoutService: PageLayoutService) {}
|
|
||||||
|
|
||||||
@Post()
|
|
||||||
create(@TenantId() tenantId: string, @Body() createPageLayoutDto: CreatePageLayoutDto) {
|
|
||||||
return this.pageLayoutService.create(tenantId, createPageLayoutDto);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Get()
|
|
||||||
findAll(@TenantId() tenantId: string, @Query('objectId') objectId?: string) {
|
|
||||||
return this.pageLayoutService.findAll(tenantId, objectId);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Get('default/:objectId')
|
|
||||||
findDefaultByObject(@TenantId() tenantId: string, @Param('objectId') objectId: string) {
|
|
||||||
return this.pageLayoutService.findDefaultByObject(tenantId, objectId);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Get(':id')
|
|
||||||
findOne(@TenantId() tenantId: string, @Param('id') id: string) {
|
|
||||||
return this.pageLayoutService.findOne(tenantId, id);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Patch(':id')
|
|
||||||
update(
|
|
||||||
@TenantId() tenantId: string,
|
|
||||||
@Param('id') id: string,
|
|
||||||
@Body() updatePageLayoutDto: UpdatePageLayoutDto,
|
|
||||||
) {
|
|
||||||
return this.pageLayoutService.update(tenantId, id, updatePageLayoutDto);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Delete(':id')
|
|
||||||
remove(@TenantId() tenantId: string, @Param('id') id: string) {
|
|
||||||
return this.pageLayoutService.remove(tenantId, id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
import { Module } from '@nestjs/common';
|
|
||||||
import { PageLayoutService } from './page-layout.service';
|
|
||||||
import { PageLayoutController } from './page-layout.controller';
|
|
||||||
import { TenantModule } from '../tenant/tenant.module';
|
|
||||||
|
|
||||||
@Module({
|
|
||||||
imports: [TenantModule],
|
|
||||||
controllers: [PageLayoutController],
|
|
||||||
providers: [PageLayoutService],
|
|
||||||
exports: [PageLayoutService],
|
|
||||||
})
|
|
||||||
export class PageLayoutModule {}
|
|
||||||
@@ -1,118 +0,0 @@
|
|||||||
import { Injectable, NotFoundException } from '@nestjs/common';
|
|
||||||
import { TenantDatabaseService } from '../tenant/tenant-database.service';
|
|
||||||
import { CreatePageLayoutDto, UpdatePageLayoutDto } from './dto/page-layout.dto';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class PageLayoutService {
|
|
||||||
constructor(private tenantDbService: TenantDatabaseService) {}
|
|
||||||
|
|
||||||
async create(tenantId: string, createDto: CreatePageLayoutDto) {
|
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
|
||||||
|
|
||||||
// If this layout is set as default, unset other defaults for the same object
|
|
||||||
if (createDto.isDefault) {
|
|
||||||
await knex('page_layouts')
|
|
||||||
.where({ object_id: createDto.objectId })
|
|
||||||
.update({ is_default: false });
|
|
||||||
}
|
|
||||||
|
|
||||||
const [id] = await knex('page_layouts').insert({
|
|
||||||
name: createDto.name,
|
|
||||||
object_id: createDto.objectId,
|
|
||||||
is_default: createDto.isDefault || false,
|
|
||||||
layout_config: JSON.stringify(createDto.layoutConfig),
|
|
||||||
description: createDto.description || null,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Get the inserted record
|
|
||||||
const result = await knex('page_layouts').where({ id }).first();
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
async findAll(tenantId: string, objectId?: string) {
|
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
|
||||||
|
|
||||||
let query = knex('page_layouts');
|
|
||||||
|
|
||||||
if (objectId) {
|
|
||||||
query = query.where({ object_id: objectId });
|
|
||||||
}
|
|
||||||
|
|
||||||
const layouts = await query.orderByRaw('is_default DESC, name ASC');
|
|
||||||
return layouts;
|
|
||||||
}
|
|
||||||
|
|
||||||
async findOne(tenantId: string, id: string) {
|
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
|
||||||
|
|
||||||
const layout = await knex('page_layouts').where({ id }).first();
|
|
||||||
|
|
||||||
if (!layout) {
|
|
||||||
throw new NotFoundException(`Page layout with ID ${id} not found`);
|
|
||||||
}
|
|
||||||
|
|
||||||
return layout;
|
|
||||||
}
|
|
||||||
|
|
||||||
async findDefaultByObject(tenantId: string, objectId: string) {
|
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
|
||||||
|
|
||||||
const layout = await knex('page_layouts')
|
|
||||||
.where({ object_id: objectId, is_default: true })
|
|
||||||
.first();
|
|
||||||
|
|
||||||
return layout || null;
|
|
||||||
}
|
|
||||||
|
|
||||||
async update(tenantId: string, id: string, updateDto: UpdatePageLayoutDto) {
|
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
|
||||||
|
|
||||||
// Check if layout exists
|
|
||||||
await this.findOne(tenantId, id);
|
|
||||||
|
|
||||||
// If setting as default, unset other defaults for the same object
|
|
||||||
if (updateDto.isDefault) {
|
|
||||||
const layout = await this.findOne(tenantId, id);
|
|
||||||
await knex('page_layouts')
|
|
||||||
.where({ object_id: layout.object_id })
|
|
||||||
.whereNot({ id })
|
|
||||||
.update({ is_default: false });
|
|
||||||
}
|
|
||||||
|
|
||||||
const updates: any = {};
|
|
||||||
|
|
||||||
if (updateDto.name !== undefined) {
|
|
||||||
updates.name = updateDto.name;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (updateDto.isDefault !== undefined) {
|
|
||||||
updates.is_default = updateDto.isDefault;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (updateDto.layoutConfig !== undefined) {
|
|
||||||
updates.layout_config = JSON.stringify(updateDto.layoutConfig);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (updateDto.description !== undefined) {
|
|
||||||
updates.description = updateDto.description;
|
|
||||||
}
|
|
||||||
|
|
||||||
updates.updated_at = knex.fn.now();
|
|
||||||
|
|
||||||
await knex('page_layouts').where({ id }).update(updates);
|
|
||||||
|
|
||||||
// Get the updated record
|
|
||||||
const result = await knex('page_layouts').where({ id }).first();
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
async remove(tenantId: string, id: string) {
|
|
||||||
const knex = await this.tenantDbService.getTenantKnex(tenantId);
|
|
||||||
|
|
||||||
await this.findOne(tenantId, id);
|
|
||||||
|
|
||||||
await knex('page_layouts').where({ id }).delete();
|
|
||||||
|
|
||||||
return { message: 'Page layout deleted successfully' };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Injectable, OnModuleInit, OnModuleDestroy } from '@nestjs/common';
|
import { Injectable, OnModuleInit, OnModuleDestroy } from '@nestjs/common';
|
||||||
import { PrismaClient } from '.prisma/tenant';
|
import { PrismaClient } from '@prisma/client';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class PrismaService
|
export class PrismaService
|
||||||
|
|||||||
@@ -22,8 +22,6 @@
|
|||||||
--input: 214.3 31.8% 91.4%;
|
--input: 214.3 31.8% 91.4%;
|
||||||
--ring: 222.2 84% 4.9%;
|
--ring: 222.2 84% 4.9%;
|
||||||
--radius: 0.5rem;
|
--radius: 0.5rem;
|
||||||
--popover: 0 0% 100%;
|
|
||||||
--popover-foreground: 222.2 84% 4.9%;
|
|
||||||
--sidebar-background: 0 0% 98%;
|
--sidebar-background: 0 0% 98%;
|
||||||
--sidebar-foreground: 240 5.3% 26.1%;
|
--sidebar-foreground: 240 5.3% 26.1%;
|
||||||
--sidebar-primary: 240 5.9% 10%;
|
--sidebar-primary: 240 5.9% 10%;
|
||||||
@@ -52,8 +50,6 @@
|
|||||||
--border: 217.2 32.6% 17.5%;
|
--border: 217.2 32.6% 17.5%;
|
||||||
--input: 217.2 32.6% 17.5%;
|
--input: 217.2 32.6% 17.5%;
|
||||||
--ring: 212.7 26.8% 83.9%;
|
--ring: 212.7 26.8% 83.9%;
|
||||||
--popover: 222.2 84% 4.9%;
|
|
||||||
--popover-foreground: 210 40% 98%;
|
|
||||||
--sidebar-background: 240 5.9% 10%;
|
--sidebar-background: 240 5.9% 10%;
|
||||||
--sidebar-foreground: 240 4.8% 95.9%;
|
--sidebar-foreground: 240 4.8% 95.9%;
|
||||||
--sidebar-primary: 224.3 76.3% 48%;
|
--sidebar-primary: 224.3 76.3% 48%;
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, onMounted } from 'vue'
|
|
||||||
import {
|
import {
|
||||||
Sidebar,
|
Sidebar,
|
||||||
SidebarContent,
|
SidebarContent,
|
||||||
@@ -17,56 +16,9 @@ import {
|
|||||||
SidebarRail,
|
SidebarRail,
|
||||||
} from '@/components/ui/sidebar'
|
} from '@/components/ui/sidebar'
|
||||||
import { Collapsible, CollapsibleContent, CollapsibleTrigger } from '@/components/ui/collapsible'
|
import { Collapsible, CollapsibleContent, CollapsibleTrigger } from '@/components/ui/collapsible'
|
||||||
import { LayoutGrid, Boxes, Settings, Home, ChevronRight, Database, Layers, LogOut } from 'lucide-vue-next'
|
import { LayoutGrid, Boxes, Settings, Home, ChevronRight, Database, Layers } from 'lucide-vue-next'
|
||||||
|
|
||||||
const { logout } = useAuth()
|
const menuItems = [
|
||||||
const { api } = useApi()
|
|
||||||
|
|
||||||
const handleLogout = async () => {
|
|
||||||
await logout()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fetch objects and group by app
|
|
||||||
const apps = ref<any[]>([])
|
|
||||||
const topLevelObjects = ref<any[]>([])
|
|
||||||
const loading = ref(true)
|
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
try {
|
|
||||||
const response = await api.get('/setup/objects')
|
|
||||||
const allObjects = response.data || response || []
|
|
||||||
|
|
||||||
// Group objects by app
|
|
||||||
const appMap = new Map<string, any>()
|
|
||||||
const noAppObjects: any[] = []
|
|
||||||
|
|
||||||
allObjects.forEach((obj: any) => {
|
|
||||||
const appId = obj.app_id || obj.appId
|
|
||||||
if (appId) {
|
|
||||||
if (!appMap.has(appId)) {
|
|
||||||
appMap.set(appId, {
|
|
||||||
id: appId,
|
|
||||||
name: obj.app?.name || obj.app?.label || 'Unknown App',
|
|
||||||
label: obj.app?.label || obj.app?.name || 'Unknown App',
|
|
||||||
objects: []
|
|
||||||
})
|
|
||||||
}
|
|
||||||
appMap.get(appId)!.objects.push(obj)
|
|
||||||
} else {
|
|
||||||
noAppObjects.push(obj)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
apps.value = Array.from(appMap.values())
|
|
||||||
topLevelObjects.value = noAppObjects
|
|
||||||
} catch (e) {
|
|
||||||
console.error('Failed to load objects:', e)
|
|
||||||
} finally {
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
const staticMenuItems = [
|
|
||||||
{
|
{
|
||||||
title: 'Home',
|
title: 'Home',
|
||||||
url: '/',
|
url: '/',
|
||||||
@@ -88,6 +40,17 @@ const staticMenuItems = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
title: 'Runtime',
|
||||||
|
icon: Database,
|
||||||
|
items: [
|
||||||
|
{
|
||||||
|
title: 'My Apps',
|
||||||
|
url: '/app',
|
||||||
|
icon: Layers,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
]
|
]
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -113,12 +76,11 @@ const staticMenuItems = [
|
|||||||
</SidebarMenu>
|
</SidebarMenu>
|
||||||
</SidebarHeader>
|
</SidebarHeader>
|
||||||
<SidebarContent>
|
<SidebarContent>
|
||||||
<!-- Static Menu Items -->
|
|
||||||
<SidebarGroup>
|
<SidebarGroup>
|
||||||
<SidebarGroupLabel>Navigation</SidebarGroupLabel>
|
<SidebarGroupLabel>Application</SidebarGroupLabel>
|
||||||
<SidebarGroupContent>
|
<SidebarGroupContent>
|
||||||
<SidebarMenu>
|
<SidebarMenu>
|
||||||
<template v-for="item in staticMenuItems" :key="item.title">
|
<template v-for="item in menuItems" :key="item.title">
|
||||||
<!-- Simple menu item -->
|
<!-- Simple menu item -->
|
||||||
<SidebarMenuItem v-if="!item.items">
|
<SidebarMenuItem v-if="!item.items">
|
||||||
<SidebarMenuButton as-child>
|
<SidebarMenuButton as-child>
|
||||||
@@ -159,70 +121,12 @@ const staticMenuItems = [
|
|||||||
</SidebarMenu>
|
</SidebarMenu>
|
||||||
</SidebarGroupContent>
|
</SidebarGroupContent>
|
||||||
</SidebarGroup>
|
</SidebarGroup>
|
||||||
|
|
||||||
<!-- Top-level Objects (no app) -->
|
|
||||||
<SidebarGroup v-if="!loading && topLevelObjects.length > 0">
|
|
||||||
<SidebarGroupLabel>Objects</SidebarGroupLabel>
|
|
||||||
<SidebarGroupContent>
|
|
||||||
<SidebarMenu>
|
|
||||||
<SidebarMenuItem v-for="obj in topLevelObjects" :key="obj.id">
|
|
||||||
<SidebarMenuButton as-child>
|
|
||||||
<NuxtLink :to="`/${obj.apiName.toLowerCase()}`">
|
|
||||||
<Database class="h-4 w-4" />
|
|
||||||
<span>{{ obj.label || obj.apiName }}</span>
|
|
||||||
</NuxtLink>
|
|
||||||
</SidebarMenuButton>
|
|
||||||
</SidebarMenuItem>
|
|
||||||
</SidebarMenu>
|
|
||||||
</SidebarGroupContent>
|
|
||||||
</SidebarGroup>
|
|
||||||
|
|
||||||
<!-- App-grouped Objects -->
|
|
||||||
<SidebarGroup v-if="!loading && apps.length > 0">
|
|
||||||
<SidebarGroupLabel>Apps</SidebarGroupLabel>
|
|
||||||
<SidebarGroupContent>
|
|
||||||
<SidebarMenu>
|
|
||||||
<Collapsible
|
|
||||||
v-for="app in apps"
|
|
||||||
:key="app.id"
|
|
||||||
as-child
|
|
||||||
:default-open="true"
|
|
||||||
class="group/collapsible"
|
|
||||||
>
|
|
||||||
<SidebarMenuItem>
|
|
||||||
<CollapsibleTrigger as-child>
|
|
||||||
<SidebarMenuButton :tooltip="app.label">
|
|
||||||
<LayoutGrid class="h-4 w-4" />
|
|
||||||
<span>{{ app.label }}</span>
|
|
||||||
<ChevronRight
|
|
||||||
class="ml-auto transition-transform duration-200 group-data-[state=open]/collapsible:rotate-90"
|
|
||||||
/>
|
|
||||||
</SidebarMenuButton>
|
|
||||||
</CollapsibleTrigger>
|
|
||||||
<CollapsibleContent>
|
|
||||||
<SidebarMenuSub>
|
|
||||||
<SidebarMenuSubItem v-for="obj in app.objects" :key="obj.id">
|
|
||||||
<SidebarMenuSubButton as-child>
|
|
||||||
<NuxtLink :to="`/${obj.apiName.toLowerCase()}`">
|
|
||||||
<Database class="h-4 w-4" />
|
|
||||||
<span>{{ obj.label || obj.apiName }}</span>
|
|
||||||
</NuxtLink>
|
|
||||||
</SidebarMenuSubButton>
|
|
||||||
</SidebarMenuSubItem>
|
|
||||||
</SidebarMenuSub>
|
|
||||||
</CollapsibleContent>
|
|
||||||
</SidebarMenuItem>
|
|
||||||
</Collapsible>
|
|
||||||
</SidebarMenu>
|
|
||||||
</SidebarGroupContent>
|
|
||||||
</SidebarGroup>
|
|
||||||
</SidebarContent>
|
</SidebarContent>
|
||||||
<SidebarFooter>
|
<SidebarFooter>
|
||||||
<SidebarMenu>
|
<SidebarMenu>
|
||||||
<SidebarMenuItem>
|
<SidebarMenuItem>
|
||||||
<SidebarMenuButton @click="handleLogout" class="cursor-pointer hover:bg-accent">
|
<SidebarMenuButton>
|
||||||
<LogOut class="h-4 w-4" />
|
<span class="text-sm text-muted-foreground">Logged in as user</span>
|
||||||
<span>Logout</span>
|
|
||||||
</SidebarMenuButton>
|
</SidebarMenuButton>
|
||||||
</SidebarMenuItem>
|
</SidebarMenuItem>
|
||||||
</SidebarMenu>
|
</SidebarMenu>
|
||||||
|
|||||||
@@ -1,334 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="page-layout-editor">
|
|
||||||
<div class="flex h-full">
|
|
||||||
<!-- Main Grid Area -->
|
|
||||||
<div class="flex-1 p-4 overflow-auto">
|
|
||||||
<div class="mb-4 flex justify-between items-center">
|
|
||||||
<h3 class="text-lg font-semibold">{{ layoutName || 'Page Layout' }}</h3>
|
|
||||||
<div class="flex gap-2">
|
|
||||||
<Button variant="outline" size="sm" @click="handleClear">
|
|
||||||
Clear All
|
|
||||||
</Button>
|
|
||||||
<Button size="sm" @click="handleSave">
|
|
||||||
Save Layout
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="border rounded-lg bg-slate-50 dark:bg-slate-900 p-4 min-h-[600px]">
|
|
||||||
<div
|
|
||||||
ref="gridContainer"
|
|
||||||
class="grid-stack"
|
|
||||||
@dragover.prevent="handleDragOver"
|
|
||||||
@drop="handleDrop"
|
|
||||||
>
|
|
||||||
<!-- Grid items will be dynamically added here -->
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Available Fields Sidebar -->
|
|
||||||
<div class="w-80 border-l bg-white dark:bg-slate-950 p-4 overflow-auto">
|
|
||||||
<h3 class="text-lg font-semibold mb-4">Available Fields</h3>
|
|
||||||
<p class="text-xs text-muted-foreground mb-4">Click or drag to add field to grid</p>
|
|
||||||
<div class="space-y-2" id="sidebar-fields">
|
|
||||||
<div
|
|
||||||
v-for="field in availableFields"
|
|
||||||
:key="field.id"
|
|
||||||
class="p-3 border rounded cursor-move bg-white dark:bg-slate-900 hover:border-primary hover:bg-slate-50 dark:hover:bg-slate-800 transition-colors"
|
|
||||||
:data-field-id="field.id"
|
|
||||||
draggable="true"
|
|
||||||
@dragstart="handleDragStart($event, field)"
|
|
||||||
@click="addFieldToGrid(field)"
|
|
||||||
>
|
|
||||||
<div class="font-medium text-sm">{{ field.label }}</div>
|
|
||||||
<div class="text-xs text-muted-foreground">{{ field.apiName }}</div>
|
|
||||||
<div class="text-xs text-muted-foreground">Type: {{ field.type }}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref, onMounted, onBeforeUnmount, watch, nextTick } from 'vue'
|
|
||||||
import { GridStack } from 'gridstack'
|
|
||||||
import 'gridstack/dist/gridstack.min.css'
|
|
||||||
import type { FieldLayoutItem } from '~/types/page-layout'
|
|
||||||
import type { FieldConfig } from '~/types/field-types'
|
|
||||||
import { Button } from '@/components/ui/button'
|
|
||||||
|
|
||||||
const props = defineProps<{
|
|
||||||
fields: FieldConfig[]
|
|
||||||
initialLayout?: FieldLayoutItem[]
|
|
||||||
layoutName?: string
|
|
||||||
}>()
|
|
||||||
|
|
||||||
const emit = defineEmits<{
|
|
||||||
save: [layout: FieldLayoutItem[]]
|
|
||||||
}>()
|
|
||||||
|
|
||||||
const gridContainer = ref<HTMLElement | null>(null)
|
|
||||||
let grid: GridStack | null = null
|
|
||||||
const gridItems = ref<Map<string, any>>(new Map())
|
|
||||||
|
|
||||||
// Fields that are already on the grid
|
|
||||||
const placedFieldIds = ref<Set<string>>(new Set())
|
|
||||||
|
|
||||||
// Fields available to be added
|
|
||||||
const availableFields = computed(() => {
|
|
||||||
return props.fields.filter(field => !placedFieldIds.value.has(field.id))
|
|
||||||
})
|
|
||||||
|
|
||||||
const initGrid = () => {
|
|
||||||
if (!gridContainer.value) return
|
|
||||||
|
|
||||||
grid = GridStack.init({
|
|
||||||
column: 6,
|
|
||||||
cellHeight: 80,
|
|
||||||
minRow: 10,
|
|
||||||
float: true,
|
|
||||||
animate: true,
|
|
||||||
acceptWidgets: true,
|
|
||||||
disableOneColumnMode: true,
|
|
||||||
resizable: {
|
|
||||||
handles: 'e, w'
|
|
||||||
}
|
|
||||||
}, gridContainer.value)
|
|
||||||
|
|
||||||
// Listen for changes
|
|
||||||
grid.on('change', () => {
|
|
||||||
updatePlacedFields()
|
|
||||||
})
|
|
||||||
|
|
||||||
// Listen for item removal
|
|
||||||
grid.on('removed', (event, items) => {
|
|
||||||
items.forEach(item => {
|
|
||||||
const contentEl = item.el?.querySelector('.grid-stack-item-content')
|
|
||||||
const fieldId = contentEl?.getAttribute('data-field-id')
|
|
||||||
if (fieldId) {
|
|
||||||
placedFieldIds.value.delete(fieldId)
|
|
||||||
gridItems.value.delete(fieldId)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
// Load initial layout if provided
|
|
||||||
if (props.initialLayout && props.initialLayout.length > 0) {
|
|
||||||
loadLayout(props.initialLayout)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const loadLayout = (layout: FieldLayoutItem[]) => {
|
|
||||||
if (!grid) return
|
|
||||||
|
|
||||||
layout.forEach(item => {
|
|
||||||
const field = props.fields.find(f => f.id === item.fieldId)
|
|
||||||
if (field) {
|
|
||||||
addFieldToGrid(field, item.x, item.y, item.w, item.h)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleDragStart = (event: DragEvent, field: FieldConfig) => {
|
|
||||||
if (event.dataTransfer) {
|
|
||||||
event.dataTransfer.effectAllowed = 'copy';
|
|
||||||
event.dataTransfer.setData('application/json', JSON.stringify(field));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleDragOver = (event: DragEvent) => {
|
|
||||||
event.preventDefault();
|
|
||||||
if (event.dataTransfer) {
|
|
||||||
event.dataTransfer.dropEffect = 'copy';
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDrop = (event: DragEvent) => {
|
|
||||||
event.preventDefault();
|
|
||||||
|
|
||||||
const fieldData = event.dataTransfer?.getData('application/json');
|
|
||||||
if (!fieldData || !grid) return;
|
|
||||||
|
|
||||||
const field = JSON.parse(fieldData);
|
|
||||||
|
|
||||||
// Get the grid bounding rect
|
|
||||||
const gridRect = gridContainer.value?.getBoundingClientRect();
|
|
||||||
if (!gridRect) return;
|
|
||||||
|
|
||||||
// Calculate grid position from drop coordinates
|
|
||||||
const x = event.clientX - gridRect.left;
|
|
||||||
const y = event.clientY - gridRect.top;
|
|
||||||
|
|
||||||
// Convert pixels to grid coordinates (approx)
|
|
||||||
const cellWidth = gridRect.width / 6; // 6 columns
|
|
||||||
const cellHeight = 80; // from our config
|
|
||||||
|
|
||||||
const gridX = Math.floor(x / cellWidth);
|
|
||||||
const gridY = Math.floor(y / cellHeight);
|
|
||||||
|
|
||||||
// Add the field at the calculated position
|
|
||||||
addFieldToGrid(field, gridX, gridY);
|
|
||||||
};
|
|
||||||
|
|
||||||
const addFieldToGrid = (field: FieldConfig, x?: number, y?: number, w: number = 3, h: number = 1) => {
|
|
||||||
if (!grid || placedFieldIds.value.has(field.id)) return
|
|
||||||
|
|
||||||
// Create the widget element manually
|
|
||||||
const widgetEl = document.createElement('div')
|
|
||||||
widgetEl.className = 'grid-stack-item'
|
|
||||||
|
|
||||||
const contentEl = document.createElement('div')
|
|
||||||
contentEl.className = 'grid-stack-item-content bg-white dark:bg-slate-900 border rounded p-3 shadow-sm'
|
|
||||||
contentEl.setAttribute('data-field-id', field.id)
|
|
||||||
|
|
||||||
contentEl.innerHTML = `
|
|
||||||
<div class="flex justify-between items-start">
|
|
||||||
<div class="flex-1">
|
|
||||||
<div class="font-medium text-sm">${field.label}</div>
|
|
||||||
<div class="text-xs text-muted-foreground">${field.apiName}</div>
|
|
||||||
<div class="text-xs text-muted-foreground">Type: ${field.type}</div>
|
|
||||||
</div>
|
|
||||||
<button class="remove-btn text-destructive hover:text-destructive/80 text-xl leading-none ml-2" type="button">
|
|
||||||
×
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
`
|
|
||||||
|
|
||||||
// Add click handler for remove button
|
|
||||||
const removeBtn = contentEl.querySelector('.remove-btn')
|
|
||||||
if (removeBtn) {
|
|
||||||
removeBtn.addEventListener('click', (e) => {
|
|
||||||
e.stopPropagation()
|
|
||||||
if (grid) {
|
|
||||||
grid.removeWidget(widgetEl)
|
|
||||||
placedFieldIds.value.delete(field.id)
|
|
||||||
gridItems.value.delete(field.id)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
widgetEl.appendChild(contentEl)
|
|
||||||
|
|
||||||
// Use makeWidget for GridStack v11+
|
|
||||||
grid.makeWidget(widgetEl)
|
|
||||||
|
|
||||||
// Set grid position after making it a widget
|
|
||||||
grid.update(widgetEl, {
|
|
||||||
x: x,
|
|
||||||
y: y,
|
|
||||||
w: w,
|
|
||||||
h: h,
|
|
||||||
minW: 1,
|
|
||||||
maxW: 6,
|
|
||||||
})
|
|
||||||
|
|
||||||
placedFieldIds.value.add(field.id)
|
|
||||||
gridItems.value.set(field.id, widgetEl)
|
|
||||||
}
|
|
||||||
|
|
||||||
const updatePlacedFields = () => {
|
|
||||||
if (!grid) return
|
|
||||||
|
|
||||||
const items = grid.getGridItems()
|
|
||||||
const newPlacedIds = new Set<string>()
|
|
||||||
|
|
||||||
items.forEach(item => {
|
|
||||||
const contentEl = item.querySelector('.grid-stack-item-content')
|
|
||||||
const fieldId = contentEl?.getAttribute('data-field-id')
|
|
||||||
if (fieldId) {
|
|
||||||
newPlacedIds.add(fieldId)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
placedFieldIds.value = newPlacedIds
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleClear = () => {
|
|
||||||
if (!grid) return
|
|
||||||
|
|
||||||
if (confirm('Are you sure you want to clear all fields from the layout?')) {
|
|
||||||
grid.removeAll()
|
|
||||||
placedFieldIds.value.clear()
|
|
||||||
gridItems.value.clear()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleSave = () => {
|
|
||||||
if (!grid) return
|
|
||||||
|
|
||||||
const items = grid.getGridItems()
|
|
||||||
const layout: FieldLayoutItem[] = []
|
|
||||||
|
|
||||||
items.forEach(item => {
|
|
||||||
// Look for data-field-id in the content element
|
|
||||||
const contentEl = item.querySelector('.grid-stack-item-content')
|
|
||||||
const fieldId = contentEl?.getAttribute('data-field-id')
|
|
||||||
const node = (item as any).gridstackNode
|
|
||||||
|
|
||||||
if (fieldId && node) {
|
|
||||||
layout.push({
|
|
||||||
fieldId,
|
|
||||||
x: node.x,
|
|
||||||
y: node.y,
|
|
||||||
w: node.w,
|
|
||||||
h: node.h,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
emit('save', layout)
|
|
||||||
}
|
|
||||||
|
|
||||||
onMounted(() => {
|
|
||||||
initGrid()
|
|
||||||
})
|
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
|
||||||
if (grid) {
|
|
||||||
grid.destroy(false)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
// Watch for fields changes
|
|
||||||
watch(() => props.fields, () => {
|
|
||||||
updatePlacedFields()
|
|
||||||
}, { deep: true })
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.grid-stack {
|
|
||||||
background: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
.grid-stack-item {
|
|
||||||
cursor: move;
|
|
||||||
}
|
|
||||||
|
|
||||||
.grid-stack-item-content {
|
|
||||||
cursor: move;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.grid-stack-item .remove-btn {
|
|
||||||
cursor: pointer;
|
|
||||||
font-size: 20px;
|
|
||||||
line-height: 1;
|
|
||||||
padding: 4px 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Customize grid appearance */
|
|
||||||
.grid-stack > .grid-stack-item > .ui-resizable-se {
|
|
||||||
background: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.grid-stack > .grid-stack-item > .ui-resizable-handle {
|
|
||||||
background-color: rgba(0, 0, 0, 0.1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Dark mode adjustments */
|
|
||||||
.dark .grid-stack > .grid-stack-item > .ui-resizable-handle {
|
|
||||||
background-color: rgba(255, 255, 255, 0.1);
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1,101 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="page-layout-renderer w-full">
|
|
||||||
<div
|
|
||||||
v-if="layout && layout.fields.length > 0"
|
|
||||||
class="grid grid-cols-6 gap-4 auto-rows-[80px]"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
v-for="fieldItem in sortedFields"
|
|
||||||
:key="fieldItem.fieldId"
|
|
||||||
:style="getFieldStyle(fieldItem)"
|
|
||||||
class="flex flex-col min-h-[60px]"
|
|
||||||
>
|
|
||||||
<FieldRenderer
|
|
||||||
v-if="fieldItem.field"
|
|
||||||
:field="fieldItem.field"
|
|
||||||
:model-value="modelValue?.[fieldItem.field.apiName]"
|
|
||||||
:mode="readonly ? VM.DETAIL : VM.EDIT"
|
|
||||||
@update:model-value="handleFieldUpdate(fieldItem.field.apiName, $event)"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Fallback: Simple two-column layout if no page layout is configured -->
|
|
||||||
<div v-else class="grid grid-cols-1 md:grid-cols-2 gap-4">
|
|
||||||
<div
|
|
||||||
v-for="field in fields"
|
|
||||||
:key="field.id"
|
|
||||||
class="flex flex-col min-h-[60px]"
|
|
||||||
>
|
|
||||||
<FieldRenderer
|
|
||||||
:field="field"
|
|
||||||
:model-value="modelValue?.[field.apiName]"
|
|
||||||
:mode="readonly ? VM.DETAIL : VM.EDIT"
|
|
||||||
@update:model-value="handleFieldUpdate(field.apiName, $event)"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { computed, watch } from 'vue'
|
|
||||||
import type { FieldConfig, ViewMode } from '~/types/field-types'
|
|
||||||
import type { PageLayoutConfig, FieldLayoutItem } from '~/types/page-layout'
|
|
||||||
import FieldRenderer from '@/components/fields/FieldRenderer.vue'
|
|
||||||
import { ViewMode as VM } from '~/types/field-types'
|
|
||||||
|
|
||||||
const props = defineProps<{
|
|
||||||
fields: FieldConfig[]
|
|
||||||
layout?: PageLayoutConfig | null
|
|
||||||
modelValue?: Record<string, any>
|
|
||||||
readonly?: boolean
|
|
||||||
}>()
|
|
||||||
|
|
||||||
const emit = defineEmits<{
|
|
||||||
'update:modelValue': [value: Record<string, any>]
|
|
||||||
}>()
|
|
||||||
|
|
||||||
// Map field IDs to field objects and sort by position
|
|
||||||
const sortedFields = computed(() => {
|
|
||||||
if (!props.layout || !props.layout.fields) return []
|
|
||||||
|
|
||||||
const fieldsMap = new Map(props.fields.map(f => [f.id, f]))
|
|
||||||
|
|
||||||
return props.layout.fields
|
|
||||||
.map(item => ({
|
|
||||||
...item,
|
|
||||||
field: fieldsMap.get(item.fieldId),
|
|
||||||
}))
|
|
||||||
.filter(item => item.field)
|
|
||||||
.sort((a, b) => {
|
|
||||||
// Sort by y position first, then x position
|
|
||||||
if (a.y !== b.y) return a.y - b.y
|
|
||||||
return a.x - b.x
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
const getFieldStyle = (item: FieldLayoutItem) => {
|
|
||||||
return {
|
|
||||||
gridColumnStart: item.x + 1,
|
|
||||||
gridColumnEnd: `span ${item.w}`,
|
|
||||||
gridRowStart: item.y + 1,
|
|
||||||
gridRowEnd: `span ${item.h}`,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleFieldUpdate = (fieldName: string, value: any) => {
|
|
||||||
if (props.readonly) return
|
|
||||||
|
|
||||||
const updated = {
|
|
||||||
...props.modelValue,
|
|
||||||
[fieldName]: value,
|
|
||||||
}
|
|
||||||
|
|
||||||
emit('update:modelValue', updated)
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
/* Additional styles if needed */
|
|
||||||
</style>
|
|
||||||
@@ -1,19 +1,19 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import type { PrimitiveProps } from "reka-ui"
|
import type { PrimitiveProps } from 'reka-ui'
|
||||||
import type { HTMLAttributes } from "vue"
|
import type { HTMLAttributes } from 'vue'
|
||||||
import type { ButtonVariants } from "."
|
import type { ButtonVariants } from '.'
|
||||||
import { Primitive } from "reka-ui"
|
import { Primitive } from 'reka-ui'
|
||||||
import { cn } from "@/lib/utils"
|
import { cn } from '@/lib/utils'
|
||||||
import { buttonVariants } from "."
|
import { buttonVariants } from '.'
|
||||||
|
|
||||||
interface Props extends PrimitiveProps {
|
interface Props extends PrimitiveProps {
|
||||||
variant?: ButtonVariants["variant"]
|
variant?: ButtonVariants['variant']
|
||||||
size?: ButtonVariants["size"]
|
size?: ButtonVariants['size']
|
||||||
class?: HTMLAttributes["class"]
|
class?: HTMLAttributes['class']
|
||||||
}
|
}
|
||||||
|
|
||||||
const props = withDefaults(defineProps<Props>(), {
|
const props = withDefaults(defineProps<Props>(), {
|
||||||
as: "button",
|
as: 'button',
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -1,38 +1,36 @@
|
|||||||
import type { VariantProps } from "class-variance-authority"
|
import type { VariantProps } from 'class-variance-authority'
|
||||||
import { cva } from "class-variance-authority"
|
import { cva } from 'class-variance-authority'
|
||||||
|
|
||||||
export { default as Button } from "./Button.vue"
|
export { default as Button } from './Button.vue'
|
||||||
|
|
||||||
export const buttonVariants = cva(
|
export const buttonVariants = cva(
|
||||||
"inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0",
|
'inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0',
|
||||||
{
|
{
|
||||||
variants: {
|
variants: {
|
||||||
variant: {
|
variant: {
|
||||||
default: "bg-primary text-primary-foreground shadow hover:bg-primary/90",
|
default: 'bg-primary text-primary-foreground shadow hover:bg-primary/90',
|
||||||
destructive:
|
destructive: 'bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90',
|
||||||
"bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90",
|
|
||||||
outline:
|
outline:
|
||||||
"border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground",
|
'border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground',
|
||||||
secondary:
|
secondary: 'bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80',
|
||||||
"bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80",
|
ghost: 'hover:bg-accent hover:text-accent-foreground',
|
||||||
ghost: "hover:bg-accent hover:text-accent-foreground",
|
link: 'text-primary underline-offset-4 hover:underline',
|
||||||
link: "text-primary underline-offset-4 hover:underline",
|
|
||||||
},
|
},
|
||||||
size: {
|
size: {
|
||||||
"default": "h-9 px-4 py-2",
|
default: 'h-9 px-4 py-2',
|
||||||
"xs": "h-7 rounded px-2",
|
xs: 'h-7 rounded px-2',
|
||||||
"sm": "h-8 rounded-md px-3 text-xs",
|
sm: 'h-8 rounded-md px-3 text-xs',
|
||||||
"lg": "h-10 rounded-md px-8",
|
lg: 'h-10 rounded-md px-8',
|
||||||
"icon": "h-9 w-9",
|
icon: 'h-9 w-9',
|
||||||
"icon-sm": "size-8",
|
'icon-sm': 'size-8',
|
||||||
"icon-lg": "size-10",
|
'icon-lg': 'size-10',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
defaultVariants: {
|
defaultVariants: {
|
||||||
variant: "default",
|
variant: 'default',
|
||||||
size: "default",
|
size: 'default',
|
||||||
},
|
},
|
||||||
},
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
export type ButtonVariants = VariantProps<typeof buttonVariants>
|
export type ButtonVariants = VariantProps<typeof buttonVariants>
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import { Button } from '@/components/ui/button'
|
|||||||
import { Popover, PopoverContent, PopoverTrigger } from '@/components/ui/popover'
|
import { Popover, PopoverContent, PopoverTrigger } from '@/components/ui/popover'
|
||||||
import { CalendarIcon } from 'lucide-vue-next'
|
import { CalendarIcon } from 'lucide-vue-next'
|
||||||
import { cn } from '@/lib/utils'
|
import { cn } from '@/lib/utils'
|
||||||
import { CalendarDate, type DateValue } from '@internationalized/date'
|
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
modelValue?: Date | string | null
|
modelValue?: Date | string | null
|
||||||
@@ -23,27 +22,18 @@ const emit = defineEmits<{
|
|||||||
'update:modelValue': [value: Date | null]
|
'update:modelValue': [value: Date | null]
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const placeholder = ref<DateValue>(new CalendarDate(new Date().getFullYear(), new Date().getMonth() + 1, new Date().getDate()))
|
const value = computed({
|
||||||
|
|
||||||
const value = computed<DateValue | undefined>({
|
|
||||||
get: () => {
|
get: () => {
|
||||||
if (!props.modelValue) return undefined
|
if (!props.modelValue) return undefined
|
||||||
const date = props.modelValue instanceof Date ? props.modelValue : new Date(props.modelValue)
|
return props.modelValue instanceof Date ? props.modelValue : new Date(props.modelValue)
|
||||||
return new CalendarDate(date.getFullYear(), date.getMonth() + 1, date.getDate())
|
|
||||||
},
|
},
|
||||||
set: (dateValue) => {
|
set: (date) => {
|
||||||
if (!dateValue) {
|
emit('update:modelValue', date || null)
|
||||||
emit('update:modelValue', null)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const jsDate = new Date(dateValue.year, dateValue.month - 1, dateValue.day)
|
|
||||||
emit('update:modelValue', jsDate)
|
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const formatDate = (dateValue: DateValue | undefined) => {
|
const formatDate = (date: Date | undefined) => {
|
||||||
if (!dateValue) return props.placeholder
|
if (!date) return props.placeholder
|
||||||
const date = new Date(dateValue.year, dateValue.month - 1, dateValue.day)
|
|
||||||
return date.toLocaleDateString('en-US', {
|
return date.toLocaleDateString('en-US', {
|
||||||
year: 'numeric',
|
year: 'numeric',
|
||||||
month: 'long',
|
month: 'long',
|
||||||
@@ -68,7 +58,7 @@ const formatDate = (dateValue: DateValue | undefined) => {
|
|||||||
</Button>
|
</Button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent class="w-auto p-0">
|
<PopoverContent class="w-auto p-0">
|
||||||
<Calendar v-model="value" :placeholder="placeholder" />
|
<Calendar v-model="value" initial-focus />
|
||||||
</PopoverContent>
|
</PopoverContent>
|
||||||
</Popover>
|
</Popover>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import type { HTMLAttributes } from "vue"
|
|||||||
import { reactiveOmit } from "@vueuse/core"
|
import { reactiveOmit } from "@vueuse/core"
|
||||||
import {
|
import {
|
||||||
PopoverContent,
|
PopoverContent,
|
||||||
|
PopoverPortal,
|
||||||
useForwardPropsEmits,
|
useForwardPropsEmits,
|
||||||
} from "reka-ui"
|
} from "reka-ui"
|
||||||
import { cn } from "@/lib/utils"
|
import { cn } from "@/lib/utils"
|
||||||
@@ -27,15 +28,17 @@ const forwarded = useForwardPropsEmits(delegatedProps, emits)
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<PopoverContent
|
<PopoverPortal>
|
||||||
v-bind="{ ...forwarded, ...$attrs }"
|
<PopoverContent
|
||||||
:class="
|
v-bind="{ ...forwarded, ...$attrs }"
|
||||||
cn(
|
:class="
|
||||||
'z-50 w-72 rounded-md border bg-popover p-4 text-popover-foreground shadow-md outline-none data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2',
|
cn(
|
||||||
props.class,
|
'z-50 w-72 rounded-md border bg-popover p-4 text-popover-foreground shadow-md outline-none data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2',
|
||||||
)
|
props.class,
|
||||||
"
|
)
|
||||||
>
|
"
|
||||||
<slot />
|
>
|
||||||
</PopoverContent>
|
<slot />
|
||||||
|
</PopoverContent>
|
||||||
|
</PopoverPortal>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { TabsRoot, type TabsRootProps } from 'radix-vue'
|
|
||||||
import { computed, type HTMLAttributes } from 'vue'
|
|
||||||
import { cn } from '@/lib/utils'
|
|
||||||
|
|
||||||
const props = defineProps<TabsRootProps & { class?: HTMLAttributes['class'] }>()
|
|
||||||
const emit = defineEmits(['update:modelValue'])
|
|
||||||
|
|
||||||
const delegatedProps = computed(() => {
|
|
||||||
const { class: _, ...delegated } = props
|
|
||||||
return delegated
|
|
||||||
})
|
|
||||||
|
|
||||||
const handleUpdate = (value: string) => {
|
|
||||||
emit('update:modelValue', value)
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<TabsRoot v-bind="delegatedProps" :class="cn('', props.class)" @update:model-value="handleUpdate">
|
|
||||||
<slot />
|
|
||||||
</TabsRoot>
|
|
||||||
</template>
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { TabsContent, type TabsContentProps } from 'radix-vue'
|
|
||||||
import { computed, type HTMLAttributes } from 'vue'
|
|
||||||
import { cn } from '@/lib/utils'
|
|
||||||
|
|
||||||
const props = defineProps<TabsContentProps & { class?: HTMLAttributes['class'] }>()
|
|
||||||
|
|
||||||
const delegatedProps = computed(() => {
|
|
||||||
const { class: _, ...delegated } = props
|
|
||||||
return delegated
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<TabsContent
|
|
||||||
v-bind="delegatedProps"
|
|
||||||
:class="
|
|
||||||
cn(
|
|
||||||
'mt-2 ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2',
|
|
||||||
props.class
|
|
||||||
)
|
|
||||||
"
|
|
||||||
>
|
|
||||||
<slot />
|
|
||||||
</TabsContent>
|
|
||||||
</template>
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { TabsList, type TabsListProps } from 'radix-vue'
|
|
||||||
import { computed, type HTMLAttributes } from 'vue'
|
|
||||||
import { cn } from '@/lib/utils'
|
|
||||||
|
|
||||||
const props = defineProps<TabsListProps & { class?: HTMLAttributes['class'] }>()
|
|
||||||
|
|
||||||
const delegatedProps = computed(() => {
|
|
||||||
const { class: _, ...delegated } = props
|
|
||||||
return delegated
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<TabsList
|
|
||||||
v-bind="delegatedProps"
|
|
||||||
:class="
|
|
||||||
cn(
|
|
||||||
'inline-flex h-9 items-center justify-center rounded-lg bg-muted p-1 text-muted-foreground',
|
|
||||||
props.class
|
|
||||||
)
|
|
||||||
"
|
|
||||||
>
|
|
||||||
<slot />
|
|
||||||
</TabsList>
|
|
||||||
</template>
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { TabsTrigger, type TabsTriggerProps } from 'radix-vue'
|
|
||||||
import { computed, type HTMLAttributes } from 'vue'
|
|
||||||
import { cn } from '@/lib/utils'
|
|
||||||
|
|
||||||
const props = defineProps<TabsTriggerProps & { class?: HTMLAttributes['class'] }>()
|
|
||||||
|
|
||||||
const delegatedProps = computed(() => {
|
|
||||||
const { class: _, ...delegated } = props
|
|
||||||
return delegated
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<TabsTrigger
|
|
||||||
v-bind="delegatedProps"
|
|
||||||
:class="
|
|
||||||
cn(
|
|
||||||
'inline-flex items-center justify-center whitespace-nowrap rounded-md px-3 py-1 text-sm font-medium ring-offset-background transition-all focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:shadow',
|
|
||||||
props.class
|
|
||||||
)
|
|
||||||
"
|
|
||||||
>
|
|
||||||
<slot />
|
|
||||||
</TabsTrigger>
|
|
||||||
</template>
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
export { default as Tabs } from './Tabs.vue'
|
|
||||||
export { default as TabsContent } from './TabsContent.vue'
|
|
||||||
export { default as TabsList } from './TabsList.vue'
|
|
||||||
export { default as TabsTrigger } from './TabsTrigger.vue'
|
|
||||||
@@ -139,7 +139,7 @@ const getFieldsBySection = (section: FieldSection) => {
|
|||||||
<div class="grid gap-6 md:grid-cols-2">
|
<div class="grid gap-6 md:grid-cols-2">
|
||||||
<FieldRenderer
|
<FieldRenderer
|
||||||
v-for="field in getFieldsBySection(section)"
|
v-for="field in getFieldsBySection(section)"
|
||||||
:key="field?.id"
|
:key="field.id"
|
||||||
:field="field"
|
:field="field"
|
||||||
:model-value="data[field.apiName]"
|
:model-value="data[field.apiName]"
|
||||||
:mode="ViewMode.DETAIL"
|
:mode="ViewMode.DETAIL"
|
||||||
|
|||||||
@@ -1,203 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { computed, ref, onMounted } from 'vue'
|
|
||||||
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@/components/ui/card'
|
|
||||||
import { Button } from '@/components/ui/button'
|
|
||||||
import FieldRenderer from '@/components/fields/FieldRenderer.vue'
|
|
||||||
import PageLayoutRenderer from '@/components/PageLayoutRenderer.vue'
|
|
||||||
import { DetailViewConfig, ViewMode, FieldSection, FieldConfig } from '@/types/field-types'
|
|
||||||
import { Edit, Trash2, ArrowLeft } from 'lucide-vue-next'
|
|
||||||
import {
|
|
||||||
Collapsible,
|
|
||||||
CollapsibleContent,
|
|
||||||
CollapsibleTrigger,
|
|
||||||
} from '@/components/ui/collapsible'
|
|
||||||
import type { PageLayoutConfig } from '~/types/page-layout'
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
config: DetailViewConfig
|
|
||||||
data: any
|
|
||||||
loading?: boolean
|
|
||||||
objectId?: string // For fetching page layout
|
|
||||||
}
|
|
||||||
|
|
||||||
const props = withDefaults(defineProps<Props>(), {
|
|
||||||
loading: false,
|
|
||||||
})
|
|
||||||
|
|
||||||
const emit = defineEmits<{
|
|
||||||
'edit': []
|
|
||||||
'delete': []
|
|
||||||
'back': []
|
|
||||||
'action': [actionId: string]
|
|
||||||
}>()
|
|
||||||
|
|
||||||
const { getDefaultPageLayout } = usePageLayouts()
|
|
||||||
const pageLayout = ref<PageLayoutConfig | null>(null)
|
|
||||||
const loadingLayout = ref(false)
|
|
||||||
|
|
||||||
// Fetch page layout if objectId is provided
|
|
||||||
onMounted(async () => {
|
|
||||||
if (props.objectId) {
|
|
||||||
try {
|
|
||||||
loadingLayout.value = true
|
|
||||||
const layout = await getDefaultPageLayout(props.objectId)
|
|
||||||
if (layout) {
|
|
||||||
// Handle both camelCase and snake_case
|
|
||||||
pageLayout.value = layout.layoutConfig || layout.layout_config
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error loading page layout:', error)
|
|
||||||
} finally {
|
|
||||||
loadingLayout.value = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
// Organize fields into sections (for traditional view)
|
|
||||||
const sections = computed<FieldSection[]>(() => {
|
|
||||||
if (props.config.sections && props.config.sections.length > 0) {
|
|
||||||
return props.config.sections
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default section with all visible fields
|
|
||||||
return [{
|
|
||||||
title: 'Details',
|
|
||||||
fields: props.config.fields
|
|
||||||
.filter(f => f.showOnDetail !== false)
|
|
||||||
.map(f => f.apiName),
|
|
||||||
}]
|
|
||||||
})
|
|
||||||
|
|
||||||
const getFieldsBySection = (section: FieldSection) => {
|
|
||||||
return section.fields
|
|
||||||
.map(apiName => props.config.fields.find(f => f.apiName === apiName))
|
|
||||||
.filter((field): field is FieldConfig => field !== undefined)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use page layout if available, otherwise fall back to sections
|
|
||||||
const usePageLayout = computed(() => {
|
|
||||||
return pageLayout.value && pageLayout.value.fields && pageLayout.value.fields.length > 0
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<div class="detail-view-enhanced space-y-6">
|
|
||||||
<!-- Header -->
|
|
||||||
<div class="flex items-center justify-between">
|
|
||||||
<div class="flex items-center gap-4">
|
|
||||||
<Button variant="ghost" size="sm" @click="emit('back')">
|
|
||||||
<ArrowLeft class="h-4 w-4 mr-2" />
|
|
||||||
Back
|
|
||||||
</Button>
|
|
||||||
<div>
|
|
||||||
<h2 class="text-2xl font-bold tracking-tight">
|
|
||||||
{{ data?.name || data?.title || config.objectApiName }}
|
|
||||||
</h2>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex items-center gap-2">
|
|
||||||
<!-- Custom Actions -->
|
|
||||||
<Button
|
|
||||||
v-for="action in config.actions"
|
|
||||||
:key="action.id"
|
|
||||||
:variant="action.variant || 'outline'"
|
|
||||||
size="sm"
|
|
||||||
@click="emit('action', action.id)"
|
|
||||||
>
|
|
||||||
{{ action.label }}
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<!-- Default Actions -->
|
|
||||||
<Button variant="outline" size="sm" @click="emit('edit')">
|
|
||||||
<Edit class="h-4 w-4 mr-2" />
|
|
||||||
Edit
|
|
||||||
</Button>
|
|
||||||
<Button variant="destructive" size="sm" @click="emit('delete')">
|
|
||||||
<Trash2 class="h-4 w-4 mr-2" />
|
|
||||||
Delete
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Loading State -->
|
|
||||||
<div v-if="loading || loadingLayout" class="flex items-center justify-center py-12">
|
|
||||||
<div class="animate-spin rounded-full h-12 w-12 border-b-2 border-primary"></div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Content with Page Layout -->
|
|
||||||
<Card v-else-if="usePageLayout">
|
|
||||||
<CardHeader>
|
|
||||||
<CardTitle>Details</CardTitle>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<PageLayoutRenderer
|
|
||||||
:fields="config.fields"
|
|
||||||
:layout="pageLayout"
|
|
||||||
:model-value="data"
|
|
||||||
:readonly="true"
|
|
||||||
/>
|
|
||||||
</CardContent>
|
|
||||||
</Card>
|
|
||||||
|
|
||||||
<!-- Traditional Section-based Layout -->
|
|
||||||
<div v-else class="space-y-6">
|
|
||||||
<Card v-for="(section, idx) in sections" :key="idx">
|
|
||||||
<Collapsible
|
|
||||||
v-if="section.collapsible"
|
|
||||||
:default-open="!section.defaultCollapsed"
|
|
||||||
>
|
|
||||||
<CardHeader>
|
|
||||||
<CollapsibleTrigger class="flex items-center justify-between w-full hover:bg-muted/50 -m-2 p-2 rounded">
|
|
||||||
<div>
|
|
||||||
<CardTitle v-if="section.title">{{ section.title }}</CardTitle>
|
|
||||||
<CardDescription v-if="section.description">
|
|
||||||
{{ section.description }}
|
|
||||||
</CardDescription>
|
|
||||||
</div>
|
|
||||||
</CollapsibleTrigger>
|
|
||||||
</CardHeader>
|
|
||||||
<CollapsibleContent>
|
|
||||||
<CardContent>
|
|
||||||
<div class="grid gap-6 md:grid-cols-2">
|
|
||||||
<FieldRenderer
|
|
||||||
v-for="field in getFieldsBySection(section)"
|
|
||||||
:key="field.id"
|
|
||||||
:field="field"
|
|
||||||
:model-value="data[field.apiName]"
|
|
||||||
:mode="ViewMode.DETAIL"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</CollapsibleContent>
|
|
||||||
</Collapsible>
|
|
||||||
|
|
||||||
<template v-else>
|
|
||||||
<CardHeader v-if="section.title || section.description">
|
|
||||||
<CardTitle v-if="section.title">{{ section.title }}</CardTitle>
|
|
||||||
<CardDescription v-if="section.description">
|
|
||||||
{{ section.description }}
|
|
||||||
</CardDescription>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<div class="grid gap-6 md:grid-cols-2">
|
|
||||||
<FieldRenderer
|
|
||||||
v-for="field in getFieldsBySection(section)"
|
|
||||||
:key="field?.id"
|
|
||||||
:field="field"
|
|
||||||
:model-value="data[field.apiName]"
|
|
||||||
:mode="ViewMode.DETAIL"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</template>
|
|
||||||
</Card>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.detail-view-enhanced {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -47,22 +47,18 @@ const sections = computed<FieldSection[]>(() => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Default section with all visible fields
|
// Default section with all visible fields
|
||||||
const visibleFields = props.config.fields
|
|
||||||
.filter(f => f.showOnEdit !== false)
|
|
||||||
.map(f => f.apiName)
|
|
||||||
|
|
||||||
return [{
|
return [{
|
||||||
title: 'Details',
|
title: 'Details',
|
||||||
fields: visibleFields,
|
fields: props.config.fields
|
||||||
|
.filter(f => f.showOnEdit !== false)
|
||||||
|
.map(f => f.apiName),
|
||||||
}]
|
}]
|
||||||
})
|
})
|
||||||
|
|
||||||
const getFieldsBySection = (section: FieldSection) => {
|
const getFieldsBySection = (section: FieldSection) => {
|
||||||
const fields = section.fields
|
return section.fields
|
||||||
.map(apiName => props.config.fields.find(f => f.apiName === apiName))
|
.map(apiName => props.config.fields.find(f => f.apiName === apiName))
|
||||||
.filter(Boolean)
|
.filter(Boolean)
|
||||||
|
|
||||||
return fields
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const validateField = (field: any): string | null => {
|
const validateField = (field: any): string | null => {
|
||||||
|
|||||||
@@ -1,303 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { ref, computed, watch, onMounted } from 'vue'
|
|
||||||
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@/components/ui/card'
|
|
||||||
import { Button } from '@/components/ui/button'
|
|
||||||
import FieldRenderer from '@/components/fields/FieldRenderer.vue'
|
|
||||||
import PageLayoutRenderer from '@/components/PageLayoutRenderer.vue'
|
|
||||||
import { EditViewConfig, ViewMode, FieldSection, FieldConfig } from '@/types/field-types'
|
|
||||||
import { Save, X, ArrowLeft } from 'lucide-vue-next'
|
|
||||||
import {
|
|
||||||
Collapsible,
|
|
||||||
CollapsibleContent,
|
|
||||||
CollapsibleTrigger,
|
|
||||||
} from '@/components/ui/collapsible'
|
|
||||||
import type { PageLayoutConfig } from '~/types/page-layout'
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
config: EditViewConfig
|
|
||||||
data?: any
|
|
||||||
loading?: boolean
|
|
||||||
saving?: boolean
|
|
||||||
objectId?: string // For fetching page layout
|
|
||||||
}
|
|
||||||
|
|
||||||
const props = withDefaults(defineProps<Props>(), {
|
|
||||||
data: () => ({}),
|
|
||||||
loading: false,
|
|
||||||
saving: false,
|
|
||||||
})
|
|
||||||
|
|
||||||
const emit = defineEmits<{
|
|
||||||
'save': [data: any]
|
|
||||||
'cancel': []
|
|
||||||
'back': []
|
|
||||||
}>()
|
|
||||||
|
|
||||||
const { getDefaultPageLayout } = usePageLayouts()
|
|
||||||
const pageLayout = ref<PageLayoutConfig | null>(null)
|
|
||||||
const loadingLayout = ref(false)
|
|
||||||
|
|
||||||
// Form data
|
|
||||||
const formData = ref<Record<string, any>>({ ...props.data })
|
|
||||||
const errors = ref<Record<string, string>>({})
|
|
||||||
|
|
||||||
// Watch for data changes (useful for edit mode)
|
|
||||||
watch(() => props.data, (newData) => {
|
|
||||||
formData.value = { ...newData }
|
|
||||||
}, { deep: true })
|
|
||||||
|
|
||||||
// Fetch page layout if objectId is provided
|
|
||||||
onMounted(async () => {
|
|
||||||
if (props.objectId) {
|
|
||||||
try {
|
|
||||||
loadingLayout.value = true
|
|
||||||
const layout = await getDefaultPageLayout(props.objectId)
|
|
||||||
if (layout) {
|
|
||||||
// Handle both camelCase and snake_case
|
|
||||||
pageLayout.value = layout.layoutConfig || layout.layout_config
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error loading page layout:', error)
|
|
||||||
} finally {
|
|
||||||
loadingLayout.value = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
// Organize fields into sections
|
|
||||||
const sections = computed<FieldSection[]>(() => {
|
|
||||||
if (props.config.sections && props.config.sections.length > 0) {
|
|
||||||
return props.config.sections
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default section with all visible fields
|
|
||||||
const visibleFields = props.config.fields
|
|
||||||
.filter(f => f.showOnEdit !== false)
|
|
||||||
.map(f => f.apiName)
|
|
||||||
|
|
||||||
return [{
|
|
||||||
title: 'Details',
|
|
||||||
fields: visibleFields,
|
|
||||||
}]
|
|
||||||
})
|
|
||||||
|
|
||||||
const getFieldsBySection = (section: FieldSection) => {
|
|
||||||
const fields = section.fields
|
|
||||||
.map(apiName => props.config.fields.find(f => f.apiName === apiName))
|
|
||||||
.filter((field): field is FieldConfig => field !== undefined)
|
|
||||||
|
|
||||||
return fields
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use page layout if available, otherwise fall back to sections
|
|
||||||
const usePageLayout = computed(() => {
|
|
||||||
return pageLayout.value && pageLayout.value.fields && pageLayout.value.fields.length > 0
|
|
||||||
})
|
|
||||||
|
|
||||||
const validateField = (field: any): string | null => {
|
|
||||||
const value = formData.value[field.apiName]
|
|
||||||
|
|
||||||
// Required validation
|
|
||||||
if (field.isRequired && (value === null || value === undefined || value === '')) {
|
|
||||||
return `${field.label} is required`
|
|
||||||
}
|
|
||||||
|
|
||||||
// Custom validation rules
|
|
||||||
if (field.validationRules) {
|
|
||||||
for (const rule of field.validationRules) {
|
|
||||||
switch (rule.type) {
|
|
||||||
case 'required':
|
|
||||||
if (value === null || value === undefined || value === '') {
|
|
||||||
return rule.message || `${field.label} is required`
|
|
||||||
}
|
|
||||||
break
|
|
||||||
case 'min':
|
|
||||||
if (typeof value === 'number' && value < rule.value) {
|
|
||||||
return rule.message || `${field.label} must be at least ${rule.value}`
|
|
||||||
}
|
|
||||||
if (typeof value === 'string' && value.length < rule.value) {
|
|
||||||
return rule.message || `${field.label} must be at least ${rule.value} characters`
|
|
||||||
}
|
|
||||||
break
|
|
||||||
case 'max':
|
|
||||||
if (typeof value === 'number' && value > rule.value) {
|
|
||||||
return rule.message || `${field.label} must be at most ${rule.value}`
|
|
||||||
}
|
|
||||||
if (typeof value === 'string' && value.length > rule.value) {
|
|
||||||
return rule.message || `${field.label} must be at most ${rule.value} characters`
|
|
||||||
}
|
|
||||||
break
|
|
||||||
case 'pattern':
|
|
||||||
if (value && !new RegExp(rule.value).test(value)) {
|
|
||||||
return rule.message || `${field.label} format is invalid`
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateForm = (): boolean => {
|
|
||||||
errors.value = {}
|
|
||||||
let isValid = true
|
|
||||||
|
|
||||||
for (const field of props.config.fields) {
|
|
||||||
if (field.showOnEdit === false) continue
|
|
||||||
|
|
||||||
const error = validateField(field)
|
|
||||||
if (error) {
|
|
||||||
errors.value[field.apiName] = error
|
|
||||||
isValid = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return isValid
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleSave = () => {
|
|
||||||
if (validateForm()) {
|
|
||||||
emit('save', formData.value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleFieldUpdate = (fieldName: string, value: any) => {
|
|
||||||
formData.value[fieldName] = value
|
|
||||||
// Clear error for this field when user makes changes
|
|
||||||
if (errors.value[fieldName]) {
|
|
||||||
delete errors.value[fieldName]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<div class="edit-view-enhanced space-y-6">
|
|
||||||
<!-- Header -->
|
|
||||||
<div class="flex items-center justify-between">
|
|
||||||
<div class="flex items-center gap-4">
|
|
||||||
<Button variant="ghost" size="sm" @click="emit('back')">
|
|
||||||
<ArrowLeft class="h-4 w-4 mr-2" />
|
|
||||||
Back
|
|
||||||
</Button>
|
|
||||||
<div>
|
|
||||||
<h2 class="text-2xl font-bold tracking-tight">
|
|
||||||
{{ data?.id ? `Edit ${config.objectApiName}` : `New ${config.objectApiName}` }}
|
|
||||||
</h2>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex items-center gap-2">
|
|
||||||
<Button variant="outline" @click="emit('cancel')" :disabled="saving">
|
|
||||||
<X class="h-4 w-4 mr-2" />
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
<Button @click="handleSave" :disabled="saving || loading || loadingLayout">
|
|
||||||
<Save class="h-4 w-4 mr-2" />
|
|
||||||
{{ saving ? 'Saving...' : 'Save' }}
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Loading State -->
|
|
||||||
<div v-if="loading || loadingLayout" class="flex items-center justify-center py-12">
|
|
||||||
<div class="animate-spin rounded-full h-12 w-12 border-b-2 border-primary"></div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Content with Page Layout -->
|
|
||||||
<Card v-else-if="usePageLayout">
|
|
||||||
<CardHeader>
|
|
||||||
<CardTitle>{{ data?.id ? 'Edit Details' : 'New Record' }}</CardTitle>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<PageLayoutRenderer
|
|
||||||
:fields="config.fields"
|
|
||||||
:layout="pageLayout"
|
|
||||||
:model-value="formData"
|
|
||||||
:readonly="false"
|
|
||||||
@update:model-value="formData = $event"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<!-- Display validation errors -->
|
|
||||||
<div v-if="Object.keys(errors).length > 0" class="mt-4 p-4 bg-destructive/10 text-destructive rounded-md">
|
|
||||||
<p class="font-semibold mb-2">Please fix the following errors:</p>
|
|
||||||
<ul class="list-disc list-inside space-y-1">
|
|
||||||
<li v-for="(error, field) in errors" :key="field">{{ error }}</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</Card>
|
|
||||||
|
|
||||||
<!-- Traditional Section-based Layout -->
|
|
||||||
<form v-else @submit.prevent="handleSave" class="space-y-6">
|
|
||||||
<Card v-for="(section, idx) in sections" :key="idx">
|
|
||||||
<Collapsible
|
|
||||||
v-if="section.collapsible"
|
|
||||||
:default-open="!section.defaultCollapsed"
|
|
||||||
>
|
|
||||||
<CardHeader>
|
|
||||||
<CollapsibleTrigger class="flex items-center justify-between w-full hover:bg-muted/50 -m-2 p-2 rounded">
|
|
||||||
<div>
|
|
||||||
<CardTitle v-if="section.title">{{ section.title }}</CardTitle>
|
|
||||||
<CardDescription v-if="section.description">
|
|
||||||
{{ section.description }}
|
|
||||||
</CardDescription>
|
|
||||||
</div>
|
|
||||||
</CollapsibleTrigger>
|
|
||||||
</CardHeader>
|
|
||||||
<CollapsibleContent>
|
|
||||||
<CardContent>
|
|
||||||
<div class="grid gap-6 md:grid-cols-2">
|
|
||||||
<div v-for="field in getFieldsBySection(section)" :key="field.id">
|
|
||||||
<FieldRenderer
|
|
||||||
:field="field"
|
|
||||||
:model-value="formData[field.apiName]"
|
|
||||||
:mode="ViewMode.EDIT"
|
|
||||||
:error="errors[field.apiName]"
|
|
||||||
@update:model-value="handleFieldUpdate(field.apiName, $event)"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</CollapsibleContent>
|
|
||||||
</Collapsible>
|
|
||||||
|
|
||||||
<template v-else>
|
|
||||||
<CardHeader v-if="section.title || section.description">
|
|
||||||
<CardTitle v-if="section.title">{{ section.title }}</CardTitle>
|
|
||||||
<CardDescription v-if="section.description">
|
|
||||||
{{ section.description }}
|
|
||||||
</CardDescription>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<div class="grid gap-6 md:grid-cols-2">
|
|
||||||
<div v-for="field in getFieldsBySection(section)" :key="field?.id">
|
|
||||||
<FieldRenderer
|
|
||||||
:field="field"
|
|
||||||
:model-value="formData[field.apiName]"
|
|
||||||
:mode="ViewMode.EDIT"
|
|
||||||
:error="errors[field.apiName]"
|
|
||||||
@update:model-value="handleFieldUpdate(field.apiName, $event)"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</template>
|
|
||||||
</Card>
|
|
||||||
|
|
||||||
<!-- Display validation errors -->
|
|
||||||
<div v-if="Object.keys(errors).length > 0" class="p-4 bg-destructive/10 text-destructive rounded-md">
|
|
||||||
<p class="font-semibold mb-2">Please fix the following errors:</p>
|
|
||||||
<ul class="list-disc list-inside space-y-1">
|
|
||||||
<li v-for="(error, field) in errors" :key="field">{{ error }}</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.edit-view-enhanced {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -231,12 +231,4 @@ const handleAction = (actionId: string) => {
|
|||||||
.list-view {
|
.list-view {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-view :deep(.border) {
|
|
||||||
background-color: hsl(var(--card));
|
|
||||||
}
|
|
||||||
|
|
||||||
.list-view :deep(input) {
|
|
||||||
background-color: hsl(var(--background));
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -63,63 +63,15 @@ export const useApi = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
// Try to get error details from response
|
throw new Error(`HTTP error! status: ${response.status}`)
|
||||||
const text = await response.text()
|
|
||||||
console.error('API Error Response:', {
|
|
||||||
status: response.status,
|
|
||||||
statusText: response.statusText,
|
|
||||||
body: text
|
|
||||||
})
|
|
||||||
|
|
||||||
let errorMessage = `HTTP error! status: ${response.status}`
|
|
||||||
if (text) {
|
|
||||||
try {
|
|
||||||
const errorData = JSON.parse(text)
|
|
||||||
errorMessage = errorData.message || errorData.error || errorMessage
|
|
||||||
} catch (e) {
|
|
||||||
// If not JSON, use the text directly if it's not too long
|
|
||||||
if (text.length < 200) {
|
|
||||||
errorMessage = text
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new Error(errorMessage)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle empty responses
|
return response.json()
|
||||||
const text = await response.text()
|
|
||||||
if (!text) {
|
|
||||||
return {}
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
return JSON.parse(text)
|
|
||||||
} catch (e) {
|
|
||||||
console.error('Failed to parse JSON response:', text)
|
|
||||||
throw new Error('Invalid JSON response from server')
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const api = {
|
const api = {
|
||||||
async get(path: string, options?: { params?: Record<string, any> }) {
|
async get(path: string) {
|
||||||
let url = `${getApiBaseUrl()}/api${path}`
|
const response = await fetch(`${getApiBaseUrl()}/api${path}`, {
|
||||||
|
|
||||||
// Add query parameters if provided
|
|
||||||
if (options?.params) {
|
|
||||||
const searchParams = new URLSearchParams()
|
|
||||||
Object.entries(options.params).forEach(([key, value]) => {
|
|
||||||
if (value !== undefined && value !== null) {
|
|
||||||
searchParams.append(key, String(value))
|
|
||||||
}
|
|
||||||
})
|
|
||||||
const queryString = searchParams.toString()
|
|
||||||
if (queryString) {
|
|
||||||
url += `?${queryString}`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const response = await fetch(url, {
|
|
||||||
headers: getHeaders(),
|
headers: getHeaders(),
|
||||||
})
|
})
|
||||||
return handleResponse(response)
|
return handleResponse(response)
|
||||||
@@ -143,15 +95,6 @@ export const useApi = () => {
|
|||||||
return handleResponse(response)
|
return handleResponse(response)
|
||||||
},
|
},
|
||||||
|
|
||||||
async patch(path: string, data: any) {
|
|
||||||
const response = await fetch(`${getApiBaseUrl()}/api${path}`, {
|
|
||||||
method: 'PATCH',
|
|
||||||
headers: getHeaders(),
|
|
||||||
body: JSON.stringify(data),
|
|
||||||
})
|
|
||||||
return handleResponse(response)
|
|
||||||
},
|
|
||||||
|
|
||||||
async delete(path: string) {
|
async delete(path: string) {
|
||||||
const response = await fetch(`${getApiBaseUrl()}/api${path}`, {
|
const response = await fetch(`${getApiBaseUrl()}/api${path}`, {
|
||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
|
|||||||
@@ -1,8 +1,5 @@
|
|||||||
export const useAuth = () => {
|
export const useAuth = () => {
|
||||||
const tokenCookie = useCookie('token')
|
const tokenCookie = useCookie('token')
|
||||||
const authMessageCookie = useCookie('authMessage')
|
|
||||||
const router = useRouter()
|
|
||||||
const config = useRuntimeConfig()
|
|
||||||
|
|
||||||
const isLoggedIn = () => {
|
const isLoggedIn = () => {
|
||||||
if (!import.meta.client) return false
|
if (!import.meta.client) return false
|
||||||
@@ -11,40 +8,14 @@ export const useAuth = () => {
|
|||||||
return !!(token && tenantId)
|
return !!(token && tenantId)
|
||||||
}
|
}
|
||||||
|
|
||||||
const logout = async () => {
|
const logout = () => {
|
||||||
if (import.meta.client) {
|
if (import.meta.client) {
|
||||||
// Call backend logout endpoint
|
|
||||||
try {
|
|
||||||
const token = localStorage.getItem('token')
|
|
||||||
const tenantId = localStorage.getItem('tenantId')
|
|
||||||
|
|
||||||
if (token) {
|
|
||||||
await fetch(`${config.public.apiBaseUrl}/api/auth/logout`, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
'Authorization': `Bearer ${token}`,
|
|
||||||
...(tenantId && { 'x-tenant-id': tenantId }),
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Logout error:', error)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clear local storage
|
|
||||||
localStorage.removeItem('token')
|
localStorage.removeItem('token')
|
||||||
localStorage.removeItem('tenantId')
|
localStorage.removeItem('tenantId')
|
||||||
localStorage.removeItem('user')
|
localStorage.removeItem('user')
|
||||||
|
|
||||||
// Clear cookie for server-side check
|
|
||||||
tokenCookie.value = null
|
|
||||||
|
|
||||||
// Set flash message for login page
|
|
||||||
authMessageCookie.value = 'Logged out successfully'
|
|
||||||
|
|
||||||
// Redirect to login page
|
|
||||||
router.push('/login')
|
|
||||||
}
|
}
|
||||||
|
// Clear cookie for server-side check
|
||||||
|
tokenCookie.value = null
|
||||||
}
|
}
|
||||||
|
|
||||||
const getUser = () => {
|
const getUser = () => {
|
||||||
|
|||||||
@@ -1,20 +0,0 @@
|
|||||||
import { ref } from 'vue'
|
|
||||||
|
|
||||||
// Shared state for breadcrumbs
|
|
||||||
const customBreadcrumbs = ref<Array<{ name: string; path?: string; isLast?: boolean }>>([])
|
|
||||||
|
|
||||||
export function useBreadcrumbs() {
|
|
||||||
const setBreadcrumbs = (crumbs: Array<{ name: string; path?: string; isLast?: boolean }>) => {
|
|
||||||
customBreadcrumbs.value = crumbs
|
|
||||||
}
|
|
||||||
|
|
||||||
const clearBreadcrumbs = () => {
|
|
||||||
customBreadcrumbs.value = []
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
breadcrumbs: customBreadcrumbs,
|
|
||||||
setBreadcrumbs,
|
|
||||||
clearBreadcrumbs
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -10,12 +10,6 @@ export const useFields = () => {
|
|||||||
* Convert backend field definition to frontend FieldConfig
|
* Convert backend field definition to frontend FieldConfig
|
||||||
*/
|
*/
|
||||||
const mapFieldDefinitionToConfig = (fieldDef: any): FieldConfig => {
|
const mapFieldDefinitionToConfig = (fieldDef: any): FieldConfig => {
|
||||||
// Convert isSystem to boolean (handle 0/1 from database)
|
|
||||||
const isSystemField = Boolean(fieldDef.isSystem)
|
|
||||||
|
|
||||||
// Only truly system fields (id, createdAt, updatedAt, etc.) should be hidden on edit
|
|
||||||
const isAutoGeneratedField = ['id', 'createdAt', 'updatedAt', 'createdBy', 'updatedBy'].includes(fieldDef.apiName)
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
id: fieldDef.id,
|
id: fieldDef.id,
|
||||||
apiName: fieldDef.apiName,
|
apiName: fieldDef.apiName,
|
||||||
@@ -29,13 +23,13 @@ export const useFields = () => {
|
|||||||
|
|
||||||
// Validation
|
// Validation
|
||||||
isRequired: fieldDef.isRequired,
|
isRequired: fieldDef.isRequired,
|
||||||
isReadOnly: isAutoGeneratedField || fieldDef.uiMetadata?.isReadOnly,
|
isReadOnly: fieldDef.isSystem || fieldDef.uiMetadata?.isReadOnly,
|
||||||
validationRules: fieldDef.uiMetadata?.validationRules || [],
|
validationRules: fieldDef.uiMetadata?.validationRules || [],
|
||||||
|
|
||||||
// View options - only hide auto-generated fields by default
|
// View options
|
||||||
showOnList: fieldDef.uiMetadata?.showOnList ?? true,
|
showOnList: fieldDef.uiMetadata?.showOnList ?? true,
|
||||||
showOnDetail: fieldDef.uiMetadata?.showOnDetail ?? true,
|
showOnDetail: fieldDef.uiMetadata?.showOnDetail ?? true,
|
||||||
showOnEdit: fieldDef.uiMetadata?.showOnEdit ?? !isAutoGeneratedField,
|
showOnEdit: fieldDef.uiMetadata?.showOnEdit ?? !fieldDef.isSystem,
|
||||||
sortable: fieldDef.uiMetadata?.sortable ?? true,
|
sortable: fieldDef.uiMetadata?.sortable ?? true,
|
||||||
|
|
||||||
// Field type specific
|
// Field type specific
|
||||||
@@ -182,15 +176,14 @@ export const useViewState = <T extends { id?: string }>(
|
|||||||
const saving = ref(false)
|
const saving = ref(false)
|
||||||
const error = ref<string | null>(null)
|
const error = ref<string | null>(null)
|
||||||
|
|
||||||
const { api } = useApi()
|
const api = useApi()
|
||||||
|
|
||||||
const fetchRecords = async (params?: Record<string, any>) => {
|
const fetchRecords = async (params?: Record<string, any>) => {
|
||||||
loading.value = true
|
loading.value = true
|
||||||
error.value = null
|
error.value = null
|
||||||
try {
|
try {
|
||||||
const response = await api.get(apiEndpoint, { params })
|
const response = await api.get(apiEndpoint, { params })
|
||||||
// Handle response - data might be directly in response or in response.data
|
records.value = response.data
|
||||||
records.value = response.data || response || []
|
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
error.value = e.message
|
error.value = e.message
|
||||||
console.error('Failed to fetch records:', e)
|
console.error('Failed to fetch records:', e)
|
||||||
@@ -204,8 +197,7 @@ export const useViewState = <T extends { id?: string }>(
|
|||||||
error.value = null
|
error.value = null
|
||||||
try {
|
try {
|
||||||
const response = await api.get(`${apiEndpoint}/${id}`)
|
const response = await api.get(`${apiEndpoint}/${id}`)
|
||||||
// Handle response - data might be directly in response or in response.data
|
currentRecord.value = response.data
|
||||||
currentRecord.value = response.data || response
|
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
error.value = e.message
|
error.value = e.message
|
||||||
console.error('Failed to fetch record:', e)
|
console.error('Failed to fetch record:', e)
|
||||||
@@ -219,12 +211,9 @@ export const useViewState = <T extends { id?: string }>(
|
|||||||
error.value = null
|
error.value = null
|
||||||
try {
|
try {
|
||||||
const response = await api.post(apiEndpoint, data)
|
const response = await api.post(apiEndpoint, data)
|
||||||
|
records.value.push(response.data)
|
||||||
// Handle response - it might be the data directly or wrapped in { data: ... }
|
currentRecord.value = response.data
|
||||||
const recordData = response.data || response
|
return response.data
|
||||||
records.value.push(recordData)
|
|
||||||
currentRecord.value = recordData
|
|
||||||
return recordData
|
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
error.value = e.message
|
error.value = e.message
|
||||||
console.error('Failed to create record:', e)
|
console.error('Failed to create record:', e)
|
||||||
@@ -238,18 +227,13 @@ export const useViewState = <T extends { id?: string }>(
|
|||||||
saving.value = true
|
saving.value = true
|
||||||
error.value = null
|
error.value = null
|
||||||
try {
|
try {
|
||||||
// Remove auto-generated fields that shouldn't be updated
|
const response = await api.put(`${apiEndpoint}/${id}`, data)
|
||||||
const { id: _id, createdAt, created_at, updatedAt, updated_at, createdBy, updatedBy, ...updateData } = data as any
|
|
||||||
|
|
||||||
const response = await api.put(`${apiEndpoint}/${id}`, updateData)
|
|
||||||
// Handle response - data might be directly in response or in response.data
|
|
||||||
const recordData = response.data || response
|
|
||||||
const idx = records.value.findIndex(r => r.id === id)
|
const idx = records.value.findIndex(r => r.id === id)
|
||||||
if (idx !== -1) {
|
if (idx !== -1) {
|
||||||
records.value[idx] = recordData
|
records.value[idx] = response.data
|
||||||
}
|
}
|
||||||
currentRecord.value = recordData
|
currentRecord.value = response.data
|
||||||
return recordData
|
return response.data
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
error.value = e.message
|
error.value = e.message
|
||||||
console.error('Failed to update record:', e)
|
console.error('Failed to update record:', e)
|
||||||
@@ -308,13 +292,12 @@ export const useViewState = <T extends { id?: string }>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const handleSave = async (data: T) => {
|
const handleSave = async (data: T) => {
|
||||||
let savedRecord
|
|
||||||
if (data.id) {
|
if (data.id) {
|
||||||
savedRecord = await updateRecord(data.id, data)
|
await updateRecord(data.id, data)
|
||||||
} else {
|
} else {
|
||||||
savedRecord = await createRecord(data)
|
await createRecord(data)
|
||||||
}
|
}
|
||||||
return savedRecord
|
showDetail(currentRecord.value!)
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -1,75 +0,0 @@
|
|||||||
import type { PageLayout, CreatePageLayoutRequest, UpdatePageLayoutRequest } from '~/types/page-layout'
|
|
||||||
|
|
||||||
export const usePageLayouts = () => {
|
|
||||||
const { api } = useApi()
|
|
||||||
|
|
||||||
const getPageLayouts = async (objectId?: string) => {
|
|
||||||
try {
|
|
||||||
const params = objectId ? { objectId } : {}
|
|
||||||
const response = await api.get('/page-layouts', { params })
|
|
||||||
return response
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error fetching page layouts:', error)
|
|
||||||
throw error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const getPageLayout = async (id: string) => {
|
|
||||||
try {
|
|
||||||
const response = await api.get(`/page-layouts/${id}`)
|
|
||||||
return response
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error fetching page layout:', error)
|
|
||||||
throw error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const getDefaultPageLayout = async (objectId: string) => {
|
|
||||||
try {
|
|
||||||
const response = await api.get(`/page-layouts/default/${objectId}`)
|
|
||||||
return response
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error fetching default page layout:', error)
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const createPageLayout = async (data: CreatePageLayoutRequest) => {
|
|
||||||
try {
|
|
||||||
const response = await api.post('/page-layouts', data)
|
|
||||||
return response
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error creating page layout:', error)
|
|
||||||
throw error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const updatePageLayout = async (id: string, data: UpdatePageLayoutRequest) => {
|
|
||||||
try {
|
|
||||||
const response = await api.patch(`/page-layouts/${id}`, data)
|
|
||||||
return response
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error updating page layout:', error)
|
|
||||||
throw error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const deletePageLayout = async (id: string) => {
|
|
||||||
try {
|
|
||||||
const response = await api.delete(`/page-layouts/${id}`)
|
|
||||||
return response
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error deleting page layout:', error)
|
|
||||||
throw error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
getPageLayouts,
|
|
||||||
getPageLayout,
|
|
||||||
getDefaultPageLayout,
|
|
||||||
createPageLayout,
|
|
||||||
updatePageLayout,
|
|
||||||
deletePageLayout,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref } from 'vue'
|
|
||||||
import AppSidebar from '@/components/AppSidebar.vue'
|
import AppSidebar from '@/components/AppSidebar.vue'
|
||||||
import AIChatBar from '@/components/AIChatBar.vue'
|
import AIChatBar from '@/components/AIChatBar.vue'
|
||||||
import {
|
import {
|
||||||
@@ -14,15 +13,8 @@ import { Separator } from '@/components/ui/separator'
|
|||||||
import { SidebarInset, SidebarProvider, SidebarTrigger } from '@/components/ui/sidebar'
|
import { SidebarInset, SidebarProvider, SidebarTrigger } from '@/components/ui/sidebar'
|
||||||
|
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const { breadcrumbs: customBreadcrumbs } = useBreadcrumbs()
|
|
||||||
|
|
||||||
const breadcrumbs = computed(() => {
|
const breadcrumbs = computed(() => {
|
||||||
// If custom breadcrumbs are set by the page, use those
|
|
||||||
if (customBreadcrumbs.value.length > 0) {
|
|
||||||
return customBreadcrumbs.value
|
|
||||||
}
|
|
||||||
|
|
||||||
// Otherwise, fall back to URL-based breadcrumbs
|
|
||||||
const paths = route.path.split('/').filter(Boolean)
|
const paths = route.path.split('/').filter(Boolean)
|
||||||
return paths.map((path, index) => ({
|
return paths.map((path, index) => ({
|
||||||
name: path.charAt(0).toUpperCase() + path.slice(1),
|
name: path.charAt(0).toUpperCase() + path.slice(1),
|
||||||
|
|||||||
@@ -41,11 +41,6 @@ export default defineNuxtConfig({
|
|||||||
|
|
||||||
typescript: {
|
typescript: {
|
||||||
strict: true,
|
strict: true,
|
||||||
tsConfig: {
|
|
||||||
compilerOptions: {
|
|
||||||
verbatimModuleSyntax: false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
|
||||||
features: {
|
features: {
|
||||||
@@ -53,14 +48,11 @@ export default defineNuxtConfig({
|
|||||||
},
|
},
|
||||||
|
|
||||||
vite: {
|
vite: {
|
||||||
optimizeDeps: {
|
|
||||||
include: ['@internationalized/date'],
|
|
||||||
},
|
|
||||||
server: {
|
server: {
|
||||||
hmr: {
|
hmr: {
|
||||||
clientPort: 3001,
|
clientPort: 3001,
|
||||||
},
|
},
|
||||||
allowedHosts: ['.routebox.co', 'localhost', '127.0.0.1'],
|
allowedHosts: ['.routebox.co', 'localhost', '127.0.0.1',],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
24
frontend/package-lock.json
generated
24
frontend/package-lock.json
generated
@@ -9,12 +9,10 @@
|
|||||||
"version": "0.0.1",
|
"version": "0.0.1",
|
||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@internationalized/date": "^3.10.1",
|
|
||||||
"@nuxtjs/tailwindcss": "^6.11.4",
|
"@nuxtjs/tailwindcss": "^6.11.4",
|
||||||
"@vueuse/core": "^10.11.1",
|
"@vueuse/core": "^10.11.1",
|
||||||
"class-variance-authority": "^0.7.0",
|
"class-variance-authority": "^0.7.0",
|
||||||
"clsx": "^2.1.0",
|
"clsx": "^2.1.0",
|
||||||
"gridstack": "^12.4.1",
|
|
||||||
"lucide-vue-next": "^0.309.0",
|
"lucide-vue-next": "^0.309.0",
|
||||||
"nuxt": "^3.10.0",
|
"nuxt": "^3.10.0",
|
||||||
"radix-vue": "^1.4.1",
|
"radix-vue": "^1.4.1",
|
||||||
@@ -1261,9 +1259,9 @@
|
|||||||
"peer": true
|
"peer": true
|
||||||
},
|
},
|
||||||
"node_modules/@internationalized/date": {
|
"node_modules/@internationalized/date": {
|
||||||
"version": "3.10.1",
|
"version": "3.10.0",
|
||||||
"resolved": "https://registry.npmjs.org/@internationalized/date/-/date-3.10.1.tgz",
|
"resolved": "https://registry.npmjs.org/@internationalized/date/-/date-3.10.0.tgz",
|
||||||
"integrity": "sha512-oJrXtQiAXLvT9clCf1K4kxp3eKsQhIaZqxEyowkBcsvZDdZkbWrVmnGknxs5flTD0VGsxrxKgBCZty1EzoiMzA==",
|
"integrity": "sha512-oxDR/NTEJ1k+UFVQElaNIk65E/Z83HK1z1WI3lQyhTtnNg4R5oVXaPzK3jcpKG8UHKDVuDQHzn+wsxSz8RP3aw==",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@swc/helpers": "^0.5.0"
|
"@swc/helpers": "^0.5.0"
|
||||||
@@ -8668,22 +8666,6 @@
|
|||||||
"devOptional": true,
|
"devOptional": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/gridstack": {
|
|
||||||
"version": "12.4.1",
|
|
||||||
"resolved": "https://registry.npmjs.org/gridstack/-/gridstack-12.4.1.tgz",
|
|
||||||
"integrity": "sha512-dYBNVEDw2zwnz0bCDouHk8rMclrMoMn4r6rtNyyWSeYsV3RF8QV2KFRTj4c86T2FsZPr3iQv+/LD/ae29FcpHQ==",
|
|
||||||
"funding": [
|
|
||||||
{
|
|
||||||
"type": "paypal",
|
|
||||||
"url": "https://www.paypal.me/alaind831"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "venmo",
|
|
||||||
"url": "https://www.venmo.com/adumesny"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"node_modules/gzip-size": {
|
"node_modules/gzip-size": {
|
||||||
"version": "7.0.0",
|
"version": "7.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/gzip-size/-/gzip-size-7.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/gzip-size/-/gzip-size-7.0.0.tgz",
|
||||||
|
|||||||
@@ -15,12 +15,10 @@
|
|||||||
"format": "prettier --write \"**/*.{js,ts,vue,json,css,scss,md}\""
|
"format": "prettier --write \"**/*.{js,ts,vue,json,css,scss,md}\""
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@internationalized/date": "^3.10.1",
|
|
||||||
"@nuxtjs/tailwindcss": "^6.11.4",
|
"@nuxtjs/tailwindcss": "^6.11.4",
|
||||||
"@vueuse/core": "^10.11.1",
|
"@vueuse/core": "^10.11.1",
|
||||||
"class-variance-authority": "^0.7.0",
|
"class-variance-authority": "^0.7.0",
|
||||||
"clsx": "^2.1.0",
|
"clsx": "^2.1.0",
|
||||||
"gridstack": "^12.4.1",
|
|
||||||
"lucide-vue-next": "^0.309.0",
|
"lucide-vue-next": "^0.309.0",
|
||||||
"nuxt": "^3.10.0",
|
"nuxt": "^3.10.0",
|
||||||
"radix-vue": "^1.4.1",
|
"radix-vue": "^1.4.1",
|
||||||
|
|||||||
@@ -1,302 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { ref, computed, onMounted, watch, nextTick } from 'vue'
|
|
||||||
import { useRoute, useRouter } from 'vue-router'
|
|
||||||
import { useApi } from '@/composables/useApi'
|
|
||||||
import { useFields, useViewState } from '@/composables/useFieldViews'
|
|
||||||
import ListView from '@/components/views/ListView.vue'
|
|
||||||
import DetailView from '@/components/views/DetailViewEnhanced.vue'
|
|
||||||
import EditView from '@/components/views/EditViewEnhanced.vue'
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
const router = useRouter()
|
|
||||||
const { api } = useApi()
|
|
||||||
const { buildListViewConfig, buildDetailViewConfig, buildEditViewConfig } = useFields()
|
|
||||||
|
|
||||||
// Use breadcrumbs composable
|
|
||||||
const { setBreadcrumbs } = useBreadcrumbs()
|
|
||||||
|
|
||||||
// Get object API name from route (case-insensitive)
|
|
||||||
const objectApiName = computed(() => {
|
|
||||||
const name = route.params.objectName as string
|
|
||||||
// We'll look up the actual case-sensitive name from the backend
|
|
||||||
return name
|
|
||||||
})
|
|
||||||
const recordId = computed(() => route.params.recordId as string)
|
|
||||||
const view = computed(() => {
|
|
||||||
// If recordId is 'new', default to 'edit' view
|
|
||||||
if (route.params.recordId === 'new' && !route.params.view) {
|
|
||||||
return 'edit'
|
|
||||||
}
|
|
||||||
return (route.params.view as 'list' | 'detail' | 'edit') || 'list'
|
|
||||||
})
|
|
||||||
|
|
||||||
// State
|
|
||||||
const objectDefinition = ref<any>(null)
|
|
||||||
const loading = ref(true)
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
// Use view state composable
|
|
||||||
const {
|
|
||||||
records,
|
|
||||||
currentRecord,
|
|
||||||
loading: dataLoading,
|
|
||||||
saving,
|
|
||||||
fetchRecords,
|
|
||||||
fetchRecord,
|
|
||||||
deleteRecord,
|
|
||||||
deleteRecords,
|
|
||||||
handleSave,
|
|
||||||
} = useViewState(`/runtime/objects/${objectApiName.value}/records`)
|
|
||||||
|
|
||||||
// Compute breadcrumbs based on the current route and object data
|
|
||||||
const updateBreadcrumbs = () => {
|
|
||||||
if (!objectDefinition.value) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const crumbs: Array<{ name: string; path?: string; isLast?: boolean }> = []
|
|
||||||
|
|
||||||
// Add app breadcrumb if object belongs to an app
|
|
||||||
if (objectDefinition.value?.app) {
|
|
||||||
crumbs.push({
|
|
||||||
name: objectDefinition.value.app.label || objectDefinition.value.app.name,
|
|
||||||
path: undefined, // No path for app grouping
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add object breadcrumb - always use plural
|
|
||||||
const objectLabel = objectDefinition.value?.pluralLabel || objectDefinition.value?.label || objectApiName.value
|
|
||||||
|
|
||||||
crumbs.push({
|
|
||||||
name: objectLabel,
|
|
||||||
path: `/${objectApiName.value.toLowerCase()}`,
|
|
||||||
})
|
|
||||||
|
|
||||||
// Add record name if viewing/editing a specific record
|
|
||||||
if (recordId.value && recordId.value !== 'new' && currentRecord.value) {
|
|
||||||
const nameField = objectDefinition.value?.nameField
|
|
||||||
let recordName = recordId.value // fallback to ID
|
|
||||||
|
|
||||||
// Try to get the display name from the nameField
|
|
||||||
if (nameField && currentRecord.value[nameField]) {
|
|
||||||
recordName = currentRecord.value[nameField]
|
|
||||||
}
|
|
||||||
|
|
||||||
crumbs.push({
|
|
||||||
name: recordName,
|
|
||||||
isLast: true,
|
|
||||||
})
|
|
||||||
} else if (recordId.value === 'new') {
|
|
||||||
crumbs.push({
|
|
||||||
name: 'New',
|
|
||||||
isLast: true,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
setBreadcrumbs(crumbs)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Watch for changes that affect breadcrumbs
|
|
||||||
watch([objectDefinition, currentRecord, recordId], () => {
|
|
||||||
updateBreadcrumbs()
|
|
||||||
}, { deep: true })
|
|
||||||
|
|
||||||
// View configs
|
|
||||||
const listConfig = computed(() => {
|
|
||||||
if (!objectDefinition.value) return null
|
|
||||||
return buildListViewConfig(objectDefinition.value, {
|
|
||||||
searchable: true,
|
|
||||||
exportable: true,
|
|
||||||
filterable: true,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
const detailConfig = computed(() => {
|
|
||||||
if (!objectDefinition.value) return null
|
|
||||||
return buildDetailViewConfig(objectDefinition.value)
|
|
||||||
})
|
|
||||||
|
|
||||||
const editConfig = computed(() => {
|
|
||||||
if (!objectDefinition.value) return null
|
|
||||||
return buildEditViewConfig(objectDefinition.value)
|
|
||||||
})
|
|
||||||
|
|
||||||
// Fetch object definition
|
|
||||||
const fetchObjectDefinition = async () => {
|
|
||||||
try {
|
|
||||||
loading.value = true
|
|
||||||
error.value = null
|
|
||||||
const response = await api.get(`/setup/objects/${objectApiName.value}`)
|
|
||||||
objectDefinition.value = response
|
|
||||||
} catch (e: any) {
|
|
||||||
error.value = e.message || 'Failed to load object definition'
|
|
||||||
console.error('Error fetching object definition:', e)
|
|
||||||
} finally {
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Navigation handlers - use lowercase URLs
|
|
||||||
const handleRowClick = (row: any) => {
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/${row.id}/detail`)
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleCreate = () => {
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/new`)
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleEdit = (row?: any) => {
|
|
||||||
const id = row?.id || recordId.value
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/${id}/edit`)
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleBack = () => {
|
|
||||||
// Navigate to list view explicitly
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/`)
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleDelete = async (rows: any[]) => {
|
|
||||||
if (confirm(`Delete ${rows.length} record(s)? This action cannot be undone.`)) {
|
|
||||||
try {
|
|
||||||
const ids = rows.map(r => r.id)
|
|
||||||
await deleteRecords(ids)
|
|
||||||
if (view.value !== 'list') {
|
|
||||||
await router.push(`/${objectApiName.value.toLowerCase()}/`)
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
error.value = e.message || 'Failed to delete records'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleSaveRecord = async (data: any) => {
|
|
||||||
try {
|
|
||||||
const savedRecord = await handleSave(data)
|
|
||||||
if (savedRecord?.id) {
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/${savedRecord.id}/detail`)
|
|
||||||
} else {
|
|
||||||
// Fallback to list if no ID available
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/`)
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
|
||||||
error.value = e.message || 'Failed to save record'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleCancel = () => {
|
|
||||||
if (recordId.value && recordId.value !== 'new') {
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/${recordId.value}/detail`)
|
|
||||||
} else {
|
|
||||||
router.push(`/${objectApiName.value.toLowerCase()}/`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Watch for route changes
|
|
||||||
watch(() => route.params, async (newParams, oldParams) => {
|
|
||||||
// Reset current record when navigating to 'new'
|
|
||||||
if (newParams.recordId === 'new') {
|
|
||||||
currentRecord.value = null
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fetch record if navigating to existing record
|
|
||||||
if (newParams.recordId && newParams.recordId !== 'new' && newParams.recordId !== oldParams.recordId) {
|
|
||||||
await fetchRecord(newParams.recordId as string)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fetch records if navigating back to list
|
|
||||||
if (!newParams.recordId && !newParams.view) {
|
|
||||||
await fetchRecords()
|
|
||||||
}
|
|
||||||
}, { deep: true })
|
|
||||||
|
|
||||||
// Initialize
|
|
||||||
onMounted(async () => {
|
|
||||||
await fetchObjectDefinition()
|
|
||||||
|
|
||||||
if (view.value === 'list') {
|
|
||||||
await fetchRecords()
|
|
||||||
} else if (recordId.value && recordId.value !== 'new') {
|
|
||||||
await fetchRecord(recordId.value)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update breadcrumbs after data is loaded
|
|
||||||
updateBreadcrumbs()
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<NuxtLayout name="default">
|
|
||||||
<div class="object-view-container">
|
|
||||||
|
|
||||||
<!-- Page Header -->
|
|
||||||
<div v-if="!loading && !error && view === 'list'" class="mb-6">
|
|
||||||
<h1 class="text-3xl font-bold">{{ objectDefinition?.label || objectApiName }}</h1>
|
|
||||||
<p v-if="objectDefinition?.description" class="text-muted-foreground mt-2">
|
|
||||||
{{ objectDefinition.description }}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Loading State -->
|
|
||||||
<div v-if="loading" class="flex items-center justify-center min-h-screen">
|
|
||||||
<div class="text-center space-y-4">
|
|
||||||
<div class="animate-spin rounded-full h-12 w-12 border-b-2 border-primary mx-auto"></div>
|
|
||||||
<p class="text-muted-foreground">Loading {{ objectApiName }}...</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Error State -->
|
|
||||||
<div v-else-if="error" class="flex items-center justify-center min-h-screen">
|
|
||||||
<div class="text-center space-y-4 max-w-md">
|
|
||||||
<div class="text-destructive text-5xl">⚠️</div>
|
|
||||||
<h2 class="text-2xl font-bold">Error</h2>
|
|
||||||
<p class="text-muted-foreground">{{ error }}</p>
|
|
||||||
<Button @click="router.back()">Go Back</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- List View -->
|
|
||||||
<ListView
|
|
||||||
v-else-if="view === 'list' && listConfig"
|
|
||||||
:config="listConfig"
|
|
||||||
:data="records"
|
|
||||||
:loading="dataLoading"
|
|
||||||
selectable
|
|
||||||
@row-click="handleRowClick"
|
|
||||||
@create="handleCreate"
|
|
||||||
@edit="handleEdit"
|
|
||||||
@delete="handleDelete"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<!-- Detail View -->
|
|
||||||
<DetailView
|
|
||||||
v-else-if="view === 'detail' && detailConfig && currentRecord"
|
|
||||||
:config="detailConfig"
|
|
||||||
:data="currentRecord"
|
|
||||||
:loading="dataLoading"
|
|
||||||
:object-id="objectDefinition?.id"
|
|
||||||
@edit="handleEdit"
|
|
||||||
@delete="() => handleDelete([currentRecord])"
|
|
||||||
@back="handleBack"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<!-- Edit View -->
|
|
||||||
<EditView
|
|
||||||
v-else-if="(view === 'edit' || recordId === 'new') && editConfig"
|
|
||||||
:config="editConfig"
|
|
||||||
:data="currentRecord || {}"
|
|
||||||
:loading="dataLoading"
|
|
||||||
:saving="saving"
|
|
||||||
:object-id="objectDefinition?.id"
|
|
||||||
@save="handleSaveRecord"
|
|
||||||
@cancel="handleCancel"
|
|
||||||
@back="handleBack"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</NuxtLayout>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.object-view-container {
|
|
||||||
padding: 2rem;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
// Redirect to a default page or show dashboard
|
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
// You can redirect to a dashboard or objects list
|
|
||||||
// For now, just show a simple message
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<NuxtLayout name="default">
|
|
||||||
<div class="container mx-auto p-8">
|
|
||||||
<h1 class="text-3xl font-bold mb-4">Welcome to Neo Platform</h1>
|
|
||||||
<p class="text-muted-foreground">Select an object from the sidebar to get started.</p>
|
|
||||||
</div>
|
|
||||||
</NuxtLayout>
|
|
||||||
</template>
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, computed, onMounted, watch, nextTick } from 'vue'
|
import { ref, computed, onMounted } from 'vue'
|
||||||
import { useRoute, useRouter } from 'vue-router'
|
import { useRoute, useRouter } from 'vue-router'
|
||||||
import { useApi } from '@/composables/useApi'
|
import { useApi } from '@/composables/useApi'
|
||||||
import { useFields, useViewState } from '@/composables/useFieldViews'
|
import { useFields, useViewState } from '@/composables/useFieldViews'
|
||||||
@@ -9,19 +9,13 @@ import EditView from '@/components/views/EditView.vue'
|
|||||||
|
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const { api } = useApi()
|
const api = useApi()
|
||||||
const { buildListViewConfig, buildDetailViewConfig, buildEditViewConfig } = useFields()
|
const { buildListViewConfig, buildDetailViewConfig, buildEditViewConfig } = useFields()
|
||||||
|
|
||||||
// Get object API name from route
|
// Get object API name from route
|
||||||
const objectApiName = computed(() => route.params.objectName as string)
|
const objectApiName = computed(() => route.params.objectName as string)
|
||||||
const recordId = computed(() => route.params.recordId as string)
|
const recordId = computed(() => route.params.recordId as string)
|
||||||
const view = computed(() => {
|
const view = computed(() => route.params.view as 'list' | 'detail' | 'edit' || 'list')
|
||||||
// If recordId is 'new', default to 'edit' view
|
|
||||||
if (route.params.recordId === 'new' && !route.params.view) {
|
|
||||||
return 'edit'
|
|
||||||
}
|
|
||||||
return (route.params.view as 'list' | 'detail' | 'edit') || 'list'
|
|
||||||
})
|
|
||||||
|
|
||||||
// State
|
// State
|
||||||
const objectDefinition = ref<any>(null)
|
const objectDefinition = ref<any>(null)
|
||||||
@@ -39,7 +33,7 @@ const {
|
|||||||
deleteRecord,
|
deleteRecord,
|
||||||
deleteRecords,
|
deleteRecords,
|
||||||
handleSave,
|
handleSave,
|
||||||
} = useViewState(`/runtime/objects/${objectApiName.value}/records`)
|
} = useViewState(`/api/runtime/objects/${objectApiName.value}`)
|
||||||
|
|
||||||
// View configs
|
// View configs
|
||||||
const listConfig = computed(() => {
|
const listConfig = computed(() => {
|
||||||
@@ -66,8 +60,8 @@ const fetchObjectDefinition = async () => {
|
|||||||
try {
|
try {
|
||||||
loading.value = true
|
loading.value = true
|
||||||
error.value = null
|
error.value = null
|
||||||
const response = await api.get(`/setup/objects/${objectApiName.value}`)
|
const response = await api.get(`/api/runtime/objects/${objectApiName.value}/definition`)
|
||||||
objectDefinition.value = response
|
objectDefinition.value = response.data
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
error.value = e.message || 'Failed to load object definition'
|
error.value = e.message || 'Failed to load object definition'
|
||||||
console.error('Error fetching object definition:', e)
|
console.error('Error fetching object definition:', e)
|
||||||
@@ -78,7 +72,7 @@ const fetchObjectDefinition = async () => {
|
|||||||
|
|
||||||
// Navigation handlers
|
// Navigation handlers
|
||||||
const handleRowClick = (row: any) => {
|
const handleRowClick = (row: any) => {
|
||||||
router.push(`/app/objects/${objectApiName.value}/${row.id}/detail`)
|
router.push(`/app/objects/${objectApiName.value}/${row.id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleCreate = () => {
|
const handleCreate = () => {
|
||||||
@@ -91,8 +85,7 @@ const handleEdit = (row?: any) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const handleBack = () => {
|
const handleBack = () => {
|
||||||
// Navigate to list view explicitly
|
router.push(`/app/objects/${objectApiName.value}`)
|
||||||
router.push(`/app/objects/${objectApiName.value}/`)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleDelete = async (rows: any[]) => {
|
const handleDelete = async (rows: any[]) => {
|
||||||
@@ -101,7 +94,7 @@ const handleDelete = async (rows: any[]) => {
|
|||||||
const ids = rows.map(r => r.id)
|
const ids = rows.map(r => r.id)
|
||||||
await deleteRecords(ids)
|
await deleteRecords(ids)
|
||||||
if (view.value !== 'list') {
|
if (view.value !== 'list') {
|
||||||
await router.push(`/app/objects/${objectApiName.value}/`)
|
await router.push(`/app/objects/${objectApiName.value}`)
|
||||||
}
|
}
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
error.value = e.message || 'Failed to delete records'
|
error.value = e.message || 'Failed to delete records'
|
||||||
@@ -111,44 +104,21 @@ const handleDelete = async (rows: any[]) => {
|
|||||||
|
|
||||||
const handleSaveRecord = async (data: any) => {
|
const handleSaveRecord = async (data: any) => {
|
||||||
try {
|
try {
|
||||||
const savedRecord = await handleSave(data)
|
await handleSave(data)
|
||||||
if (savedRecord?.id) {
|
router.push(`/app/objects/${objectApiName.value}/${currentRecord.value?.id || data.id}`)
|
||||||
router.push(`/app/objects/${objectApiName.value}/${savedRecord.id}/detail`)
|
|
||||||
} else {
|
|
||||||
// Fallback to list if no ID available
|
|
||||||
router.push(`/app/objects/${objectApiName.value}/`)
|
|
||||||
}
|
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
error.value = e.message || 'Failed to save record'
|
error.value = e.message || 'Failed to save record'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleCancel = () => {
|
const handleCancel = () => {
|
||||||
if (recordId.value && recordId.value !== 'new') {
|
if (recordId.value) {
|
||||||
router.push(`/app/objects/${objectApiName.value}/${recordId.value}/detail`)
|
router.push(`/app/objects/${objectApiName.value}/${recordId.value}`)
|
||||||
} else {
|
} else {
|
||||||
router.push(`/app/objects/${objectApiName.value}/`)
|
router.push(`/app/objects/${objectApiName.value}`)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Watch for route changes
|
|
||||||
watch(() => route.params, async (newParams, oldParams) => {
|
|
||||||
// Reset current record when navigating to 'new'
|
|
||||||
if (newParams.recordId === 'new') {
|
|
||||||
currentRecord.value = null
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fetch record if navigating to existing record
|
|
||||||
if (newParams.recordId && newParams.recordId !== 'new' && newParams.recordId !== oldParams.recordId) {
|
|
||||||
await fetchRecord(newParams.recordId as string)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fetch records if navigating back to list
|
|
||||||
if (!newParams.recordId && !newParams.view) {
|
|
||||||
await fetchRecords()
|
|
||||||
}
|
|
||||||
}, { deep: true })
|
|
||||||
|
|
||||||
// Initialize
|
// Initialize
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
await fetchObjectDefinition()
|
await fetchObjectDefinition()
|
||||||
@@ -162,71 +132,61 @@ onMounted(async () => {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<NuxtLayout name="default">
|
<div class="object-view-container">
|
||||||
<div class="object-view-container">
|
<!-- Loading State -->
|
||||||
<!-- Page Header -->
|
<div v-if="loading" class="flex items-center justify-center min-h-screen">
|
||||||
<div v-if="!loading && !error" class="mb-6">
|
<div class="text-center space-y-4">
|
||||||
<h1 class="text-3xl font-bold">{{ objectDefinition?.label || objectApiName }}</h1>
|
<div class="animate-spin rounded-full h-12 w-12 border-b-2 border-primary mx-auto"></div>
|
||||||
<p v-if="objectDefinition?.description" class="text-muted-foreground mt-2">
|
<p class="text-muted-foreground">Loading {{ objectApiName }}...</p>
|
||||||
{{ objectDefinition.description }}
|
|
||||||
</p>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Loading State -->
|
|
||||||
<div v-if="loading" class="flex items-center justify-center min-h-screen">
|
|
||||||
<div class="text-center space-y-4">
|
|
||||||
<div class="animate-spin rounded-full h-12 w-12 border-b-2 border-primary mx-auto"></div>
|
|
||||||
<p class="text-muted-foreground">Loading {{ objectApiName }}...</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Error State -->
|
|
||||||
<div v-else-if="error" class="flex items-center justify-center min-h-screen">
|
|
||||||
<div class="text-center space-y-4 max-w-md">
|
|
||||||
<div class="text-destructive text-5xl">⚠️</div>
|
|
||||||
<h2 class="text-2xl font-bold">Error</h2>
|
|
||||||
<p class="text-muted-foreground">{{ error }}</p>
|
|
||||||
<Button @click="router.back()">Go Back</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- List View -->
|
|
||||||
<ListView
|
|
||||||
v-else-if="view === 'list' && listConfig"
|
|
||||||
:config="listConfig"
|
|
||||||
:data="records"
|
|
||||||
:loading="dataLoading"
|
|
||||||
selectable
|
|
||||||
@row-click="handleRowClick"
|
|
||||||
@create="handleCreate"
|
|
||||||
@edit="handleEdit"
|
|
||||||
@delete="handleDelete"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<!-- Detail View -->
|
|
||||||
<DetailView
|
|
||||||
v-else-if="view === 'detail' && detailConfig && currentRecord"
|
|
||||||
:config="detailConfig"
|
|
||||||
:data="currentRecord"
|
|
||||||
:loading="dataLoading"
|
|
||||||
@edit="handleEdit"
|
|
||||||
@delete="() => handleDelete([currentRecord])"
|
|
||||||
@back="handleBack"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<!-- Edit View -->
|
|
||||||
<EditView
|
|
||||||
v-else-if="(view === 'edit' || recordId === 'new') && editConfig"
|
|
||||||
:config="editConfig"
|
|
||||||
:data="currentRecord || {}"
|
|
||||||
:loading="dataLoading"
|
|
||||||
:saving="saving"
|
|
||||||
@save="handleSaveRecord"
|
|
||||||
@cancel="handleCancel"
|
|
||||||
@back="handleBack"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</NuxtLayout>
|
|
||||||
|
<!-- Error State -->
|
||||||
|
<div v-else-if="error" class="flex items-center justify-center min-h-screen">
|
||||||
|
<div class="text-center space-y-4 max-w-md">
|
||||||
|
<div class="text-destructive text-5xl">⚠️</div>
|
||||||
|
<h2 class="text-2xl font-bold">Error</h2>
|
||||||
|
<p class="text-muted-foreground">{{ error }}</p>
|
||||||
|
<Button @click="router.back()">Go Back</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- List View -->
|
||||||
|
<ListView
|
||||||
|
v-else-if="view === 'list' && listConfig"
|
||||||
|
:config="listConfig"
|
||||||
|
:data="records"
|
||||||
|
:loading="dataLoading"
|
||||||
|
selectable
|
||||||
|
@row-click="handleRowClick"
|
||||||
|
@create="handleCreate"
|
||||||
|
@edit="handleEdit"
|
||||||
|
@delete="handleDelete"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- Detail View -->
|
||||||
|
<DetailView
|
||||||
|
v-else-if="view === 'detail' && detailConfig && currentRecord"
|
||||||
|
:config="detailConfig"
|
||||||
|
:data="currentRecord"
|
||||||
|
:loading="dataLoading"
|
||||||
|
@edit="handleEdit"
|
||||||
|
@delete="() => handleDelete([currentRecord])"
|
||||||
|
@back="handleBack"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- Edit View -->
|
||||||
|
<EditView
|
||||||
|
v-else-if="(view === 'edit' || recordId === 'new') && editConfig"
|
||||||
|
:config="editConfig"
|
||||||
|
:data="currentRecord || {}"
|
||||||
|
:loading="dataLoading"
|
||||||
|
:saving="saving"
|
||||||
|
@save="handleSaveRecord"
|
||||||
|
@cancel="handleCancel"
|
||||||
|
@back="handleBack"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|||||||
@@ -1,43 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
// List all available objects
|
|
||||||
const { api } = useApi()
|
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
const objects = ref<any[]>([])
|
|
||||||
const loading = ref(true)
|
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
try {
|
|
||||||
const response = await api.get('/setup/objects')
|
|
||||||
objects.value = response.data || response || []
|
|
||||||
} catch (e) {
|
|
||||||
console.error('Failed to load objects:', e)
|
|
||||||
} finally {
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<NuxtLayout name="default">
|
|
||||||
<div class="container mx-auto p-8">
|
|
||||||
<h1 class="text-3xl font-bold mb-6">Objects</h1>
|
|
||||||
|
|
||||||
<div v-if="loading" class="text-center py-12">
|
|
||||||
<div class="animate-spin rounded-full h-12 w-12 border-b-2 border-primary mx-auto"></div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-else class="grid gap-4 md:grid-cols-2 lg:grid-cols-3">
|
|
||||||
<NuxtLink
|
|
||||||
v-for="obj in objects"
|
|
||||||
:key="obj.id"
|
|
||||||
:to="`/app/objects/${obj.apiName}/`"
|
|
||||||
class="block p-6 border rounded-lg hover:bg-accent transition-colors"
|
|
||||||
>
|
|
||||||
<h3 class="text-xl font-semibold mb-2">{{ obj.label }}</h3>
|
|
||||||
<p v-if="obj.description" class="text-sm text-muted-foreground">{{ obj.description }}</p>
|
|
||||||
</NuxtLink>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</NuxtLayout>
|
|
||||||
</template>
|
|
||||||
@@ -7,13 +7,11 @@ import EditView from '@/components/views/EditView.vue'
|
|||||||
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'
|
||||||
import {
|
import {
|
||||||
FieldType,
|
FieldType,
|
||||||
ViewMode
|
ViewMode,
|
||||||
} from '@/types/field-types'
|
type ListViewConfig,
|
||||||
import type {
|
type DetailViewConfig,
|
||||||
ListViewConfig,
|
type EditViewConfig,
|
||||||
DetailViewConfig,
|
type FieldConfig
|
||||||
EditViewConfig,
|
|
||||||
FieldConfig
|
|
||||||
} from '@/types/field-types'
|
} from '@/types/field-types'
|
||||||
|
|
||||||
// Example: Contact Object
|
// Example: Contact Object
|
||||||
|
|||||||
@@ -15,15 +15,7 @@ const authMessage = useCookie('authMessage')
|
|||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
if (authMessage.value) {
|
if (authMessage.value) {
|
||||||
console.log('Displaying auth message: ' + authMessage.value)
|
console.log('Displaying auth message: ' + authMessage.value)
|
||||||
const message = authMessage.value
|
toast.error(authMessage.value)
|
||||||
|
|
||||||
// Show success toast for logout, error for auth failures
|
|
||||||
if (message.toLowerCase().includes('logged out')) {
|
|
||||||
toast.success(message)
|
|
||||||
} else {
|
|
||||||
toast.error(message)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clear the message after displaying
|
// Clear the message after displaying
|
||||||
authMessage.value = null
|
authMessage.value = null
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,119 +13,38 @@
|
|||||||
|
|
||||||
<h1 class="text-3xl font-bold mb-6">{{ object.label }}</h1>
|
<h1 class="text-3xl font-bold mb-6">{{ object.label }}</h1>
|
||||||
|
|
||||||
<!-- Tabs -->
|
|
||||||
<div class="mb-8">
|
<div class="mb-8">
|
||||||
<Tabs v-model="activeTab" default-value="fields" class="w-full">
|
<h2 class="text-2xl font-semibold mb-4">Fields</h2>
|
||||||
<TabsList class="grid w-full grid-cols-2 max-w-md">
|
<div class="space-y-2">
|
||||||
<TabsTrigger value="fields">Fields</TabsTrigger>
|
<div
|
||||||
<TabsTrigger value="layouts">Page Layouts</TabsTrigger>
|
v-for="field in object.fields"
|
||||||
</TabsList>
|
:key="field.id"
|
||||||
|
class="p-4 border rounded-lg bg-card"
|
||||||
<!-- Fields Tab -->
|
>
|
||||||
<TabsContent value="fields" class="mt-6">
|
<div class="flex items-center justify-between">
|
||||||
<div class="space-y-2">
|
<div>
|
||||||
<div
|
<h3 class="font-semibold">{{ field.label }}</h3>
|
||||||
v-for="field in object.fields"
|
<p class="text-sm text-muted-foreground">
|
||||||
:key="field.id"
|
Type: {{ field.type }} | API Name: {{ field.apiName }}
|
||||||
class="p-4 border rounded-lg bg-card"
|
</p>
|
||||||
>
|
|
||||||
<div class="flex items-center justify-between">
|
|
||||||
<div>
|
|
||||||
<h3 class="font-semibold">{{ field.label }}</h3>
|
|
||||||
<p class="text-sm text-muted-foreground">
|
|
||||||
Type: {{ field.type }} | API Name: {{ field.apiName }}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div class="flex gap-2 text-xs">
|
|
||||||
<span
|
|
||||||
v-if="field.isRequired"
|
|
||||||
class="px-2 py-1 bg-destructive/10 text-destructive rounded"
|
|
||||||
>
|
|
||||||
Required
|
|
||||||
</span>
|
|
||||||
<span
|
|
||||||
v-if="field.isUnique"
|
|
||||||
class="px-2 py-1 bg-primary/10 text-primary rounded"
|
|
||||||
>
|
|
||||||
Unique
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
<div class="flex gap-2 text-xs">
|
||||||
</TabsContent>
|
<span
|
||||||
|
v-if="field.isRequired"
|
||||||
<!-- Page Layouts Tab -->
|
class="px-2 py-1 bg-destructive/10 text-destructive rounded"
|
||||||
<TabsContent value="layouts" class="mt-6">
|
|
||||||
<div v-if="!selectedLayout" class="space-y-4">
|
|
||||||
<div class="flex justify-between items-center mb-4">
|
|
||||||
<h2 class="text-xl font-semibold">Page Layouts</h2>
|
|
||||||
<Button @click="handleCreateLayout">
|
|
||||||
<Plus class="w-4 h-4 mr-2" />
|
|
||||||
New Layout
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-if="loadingLayouts" class="text-center py-8">
|
|
||||||
Loading layouts...
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-else-if="layouts.length === 0" class="text-center py-8 text-muted-foreground">
|
|
||||||
No page layouts yet. Create one to get started.
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-else class="space-y-2">
|
|
||||||
<div
|
|
||||||
v-for="layout in layouts"
|
|
||||||
:key="layout.id"
|
|
||||||
class="p-4 border rounded-lg bg-card hover:border-primary cursor-pointer transition-colors"
|
|
||||||
@click="handleSelectLayout(layout)"
|
|
||||||
>
|
>
|
||||||
<div class="flex items-center justify-between">
|
Required
|
||||||
<div>
|
</span>
|
||||||
<h3 class="font-semibold">{{ layout.name }}</h3>
|
<span
|
||||||
<p v-if="layout.description" class="text-sm text-muted-foreground">
|
v-if="field.isUnique"
|
||||||
{{ layout.description }}
|
class="px-2 py-1 bg-primary/10 text-primary rounded"
|
||||||
</p>
|
>
|
||||||
</div>
|
Unique
|
||||||
<div class="flex items-center gap-2">
|
</span>
|
||||||
<span
|
|
||||||
v-if="layout.isDefault"
|
|
||||||
class="px-2 py-1 bg-primary/10 text-primary rounded text-xs"
|
|
||||||
>
|
|
||||||
Default
|
|
||||||
</span>
|
|
||||||
<Button
|
|
||||||
variant="ghost"
|
|
||||||
size="sm"
|
|
||||||
@click.stop="handleDeleteLayout(layout.id)"
|
|
||||||
>
|
|
||||||
<Trash2 class="w-4 h-4" />
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
<!-- Layout Editor -->
|
</div>
|
||||||
<div v-else>
|
|
||||||
<div class="mb-4">
|
|
||||||
<Button variant="outline" @click="selectedLayout = null">
|
|
||||||
<ArrowLeft class="w-4 h-4 mr-2" />
|
|
||||||
Back to Layouts
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<PageLayoutEditor
|
|
||||||
:fields="object.fields"
|
|
||||||
:initial-layout="(selectedLayout.layoutConfig || selectedLayout.layout_config)?.fields || []"
|
|
||||||
:layout-name="selectedLayout.name"
|
|
||||||
@save="handleSaveLayout"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</TabsContent>
|
|
||||||
</Tabs>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
@@ -134,26 +53,12 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { Plus, Trash2, ArrowLeft } from 'lucide-vue-next'
|
|
||||||
import { Button } from '@/components/ui/button'
|
|
||||||
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'
|
|
||||||
import PageLayoutEditor from '@/components/PageLayoutEditor.vue'
|
|
||||||
import type { PageLayout, FieldLayoutItem } from '~/types/page-layout'
|
|
||||||
|
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const { api } = useApi()
|
const { api } = useApi()
|
||||||
const { getPageLayouts, createPageLayout, updatePageLayout, deletePageLayout } = usePageLayouts()
|
|
||||||
const { toast } = useToast()
|
|
||||||
|
|
||||||
const object = ref<any>(null)
|
const object = ref<any>(null)
|
||||||
const loading = ref(true)
|
const loading = ref(true)
|
||||||
const error = ref<string | null>(null)
|
const error = ref<string | null>(null)
|
||||||
const activeTab = ref('fields')
|
|
||||||
|
|
||||||
// Page layouts state
|
|
||||||
const layouts = ref<PageLayout[]>([])
|
|
||||||
const loadingLayouts = ref(false)
|
|
||||||
const selectedLayout = ref<PageLayout | null>(null)
|
|
||||||
|
|
||||||
const fetchObject = async () => {
|
const fetchObject = async () => {
|
||||||
try {
|
try {
|
||||||
@@ -167,92 +72,7 @@ const fetchObject = async () => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const fetchLayouts = async () => {
|
onMounted(() => {
|
||||||
if (!object.value) return
|
fetchObject()
|
||||||
|
|
||||||
try {
|
|
||||||
loadingLayouts.value = true
|
|
||||||
layouts.value = await getPageLayouts(object.value.id)
|
|
||||||
} catch (e: any) {
|
|
||||||
console.error('Error fetching layouts:', e)
|
|
||||||
toast.error('Failed to load page layouts')
|
|
||||||
} finally {
|
|
||||||
loadingLayouts.value = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleCreateLayout = async () => {
|
|
||||||
const name = prompt('Enter a name for the new layout:')
|
|
||||||
if (!name) return
|
|
||||||
|
|
||||||
try {
|
|
||||||
const newLayout = await createPageLayout({
|
|
||||||
name,
|
|
||||||
objectId: object.value.id,
|
|
||||||
isDefault: layouts.value.length === 0,
|
|
||||||
layoutConfig: { fields: [] },
|
|
||||||
})
|
|
||||||
|
|
||||||
layouts.value.push(newLayout)
|
|
||||||
selectedLayout.value = newLayout
|
|
||||||
toast.success('Layout created successfully')
|
|
||||||
} catch (e: any) {
|
|
||||||
console.error('Error creating layout:', e)
|
|
||||||
toast.error('Failed to create layout')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleSelectLayout = (layout: PageLayout) => {
|
|
||||||
selectedLayout.value = layout
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleSaveLayout = async (fields: FieldLayoutItem[]) => {
|
|
||||||
if (!selectedLayout.value) return
|
|
||||||
|
|
||||||
try {
|
|
||||||
const updated = await updatePageLayout(selectedLayout.value.id, {
|
|
||||||
layoutConfig: { fields },
|
|
||||||
})
|
|
||||||
|
|
||||||
// Update the layout in the list
|
|
||||||
const index = layouts.value.findIndex(l => l.id === selectedLayout.value!.id)
|
|
||||||
if (index !== -1) {
|
|
||||||
layouts.value[index] = updated
|
|
||||||
}
|
|
||||||
|
|
||||||
selectedLayout.value = updated
|
|
||||||
toast.success('Layout saved successfully')
|
|
||||||
} catch (e: any) {
|
|
||||||
console.error('Error saving layout:', e)
|
|
||||||
toast.error('Failed to save layout')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleDeleteLayout = async (layoutId: string) => {
|
|
||||||
if (!confirm('Are you sure you want to delete this layout?')) return
|
|
||||||
|
|
||||||
try {
|
|
||||||
await deletePageLayout(layoutId)
|
|
||||||
layouts.value = layouts.value.filter(l => l.id !== layoutId)
|
|
||||||
toast.success('Layout deleted successfully')
|
|
||||||
} catch (e: any) {
|
|
||||||
console.error('Error deleting layout:', e)
|
|
||||||
toast.error('Failed to delete layout')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Watch for tab changes to load layouts
|
|
||||||
watch(activeTab, (newTab) => {
|
|
||||||
if (newTab === 'layouts' && layouts.value.length === 0 && !loadingLayouts.value) {
|
|
||||||
fetchLayouts()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
await fetchObject()
|
|
||||||
// If we start on layouts tab, load them
|
|
||||||
if (activeTab.value === 'layouts') {
|
|
||||||
await fetchLayouts()
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -40,10 +40,6 @@ export default {
|
|||||||
DEFAULT: 'hsl(var(--card))',
|
DEFAULT: 'hsl(var(--card))',
|
||||||
foreground: 'hsl(var(--card-foreground))',
|
foreground: 'hsl(var(--card-foreground))',
|
||||||
},
|
},
|
||||||
popover: {
|
|
||||||
DEFAULT: 'hsl(var(--popover))',
|
|
||||||
foreground: 'hsl(var(--popover-foreground))',
|
|
||||||
},
|
|
||||||
sidebar: {
|
sidebar: {
|
||||||
DEFAULT: 'hsl(var(--sidebar-background))',
|
DEFAULT: 'hsl(var(--sidebar-background))',
|
||||||
foreground: 'hsl(var(--sidebar-foreground))',
|
foreground: 'hsl(var(--sidebar-foreground))',
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
export * from './field-types'
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
export interface FieldLayoutItem {
|
|
||||||
fieldId: string;
|
|
||||||
x: number;
|
|
||||||
y: number;
|
|
||||||
w: number;
|
|
||||||
h: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface PageLayoutConfig {
|
|
||||||
fields: FieldLayoutItem[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface PageLayout {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
objectId: string;
|
|
||||||
isDefault: boolean;
|
|
||||||
layoutConfig: PageLayoutConfig;
|
|
||||||
description?: string;
|
|
||||||
createdAt?: string;
|
|
||||||
updatedAt?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface CreatePageLayoutRequest {
|
|
||||||
name: string;
|
|
||||||
objectId: string;
|
|
||||||
isDefault?: boolean;
|
|
||||||
layoutConfig: PageLayoutConfig;
|
|
||||||
description?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface UpdatePageLayoutRequest {
|
|
||||||
name?: string;
|
|
||||||
isDefault?: boolean;
|
|
||||||
layoutConfig?: PageLayoutConfig;
|
|
||||||
description?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface GridStackOptions {
|
|
||||||
column: number;
|
|
||||||
cellHeight: number;
|
|
||||||
minRow: number;
|
|
||||||
float: boolean;
|
|
||||||
acceptWidgets: boolean | string;
|
|
||||||
removable?: boolean | string;
|
|
||||||
animate: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface GridStackWidget {
|
|
||||||
id: string;
|
|
||||||
x: number;
|
|
||||||
y: number;
|
|
||||||
w: number;
|
|
||||||
h: number;
|
|
||||||
minW?: number;
|
|
||||||
maxW?: number;
|
|
||||||
noResize?: boolean;
|
|
||||||
noMove?: boolean;
|
|
||||||
locked?: boolean;
|
|
||||||
content?: string;
|
|
||||||
}
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# Page Layouts Setup Script
|
|
||||||
# This script helps set up and test the page layouts feature
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
echo "🎨 Page Layouts Setup"
|
|
||||||
echo "===================="
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
# Colors for output
|
|
||||||
GREEN='\033[0;32m'
|
|
||||||
BLUE='\033[0;34m'
|
|
||||||
YELLOW='\033[1;33m'
|
|
||||||
NC='\033[0m' # No Color
|
|
||||||
|
|
||||||
# Check if we're in the right directory
|
|
||||||
if [ ! -d "backend" ] || [ ! -d "frontend" ]; then
|
|
||||||
echo "❌ Error: This script must be run from the project root directory"
|
|
||||||
echo " (The directory containing backend/ and frontend/ folders)"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo -e "${BLUE}Step 1: Database migration${NC}"
|
|
||||||
echo "Note: You'll need to run the migration for each tenant."
|
|
||||||
echo ""
|
|
||||||
echo "Run the following command for each tenant:"
|
|
||||||
echo " cd backend && npm run migrate:tenant <tenant-slug-or-id>"
|
|
||||||
echo ""
|
|
||||||
read -p "Press Enter to continue with frontend setup..."
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
echo -e "${BLUE}Step 2: Installing frontend dependencies...${NC}"
|
|
||||||
cd frontend
|
|
||||||
if ! npm list gridstack &> /dev/null; then
|
|
||||||
npm install gridstack
|
|
||||||
echo -e "${GREEN}✓ GridStack installed${NC}"
|
|
||||||
else
|
|
||||||
echo -e "${YELLOW}⚠ GridStack already installed${NC}"
|
|
||||||
fi
|
|
||||||
cd ..
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
echo -e "${BLUE}Step 3: Checking GridStack CSS...${NC}"
|
|
||||||
if [ -f "frontend/node_modules/gridstack/dist/gridstack.min.css" ]; then
|
|
||||||
echo -e "${GREEN}✓ GridStack CSS available${NC}"
|
|
||||||
else
|
|
||||||
echo -e "${YELLOW}⚠ GridStack CSS not found, may need manual installation${NC}"
|
|
||||||
fi
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
echo -e "${GREEN}✅ Setup Complete!${NC}"
|
|
||||||
echo ""
|
|
||||||
echo "📚 Next Steps:"
|
|
||||||
echo "1. Start the backend: cd backend && npm run start:dev"
|
|
||||||
echo "2. Start the frontend: cd frontend && npm run dev"
|
|
||||||
echo "3. Navigate to Setup → Objects → [Object] → Page Layouts tab"
|
|
||||||
echo "4. Create and configure your first page layout"
|
|
||||||
echo ""
|
|
||||||
echo "📖 For more information, see PAGE_LAYOUTS_GUIDE.md"
|
|
||||||
Reference in New Issue
Block a user