Enhanced Actions
Advanced action system with AJAX calls, chaining, conditions, and callbacks
Enhanced Actions
ObjectUI's action system has been significantly enhanced to support complex workflows, including AJAX API calls, confirmation dialogs, action chaining, conditional execution, and comprehensive tracking.
Overview
The enhanced ActionSchema provides:
- New action types:
ajax,confirm,dialog - Action chaining: Execute multiple actions sequentially or in parallel
- Conditional logic: If/then/else execution based on data
- Callbacks: Success and failure handlers
- Tracking: Event logging and analytics
- Retry logic: Automatic retry with configurable backoff
Interactive Examples
Action Buttons
Action Button Variants
Different button styles for various action types
{
"type": "stack",
"spacing": 4,
"children": [
{
"type": "flex",
"className": "gap-2 flex-wrap",
"children": [
{
"type": "button",
"label": "Submit",
"variant": "default",
"icon": "send"
},
{
"type": "button",
"label": "Save Draft",
"variant": "outline",
"icon": "save"
},
{
"type": "button",
"label": "Delete",
"variant": "destructive",
"icon": "trash-2"
},
{
"type": "button",
"label": "Cancel",
"variant": "ghost"
}
]
}
]
}Confirmation Dialog Pattern
Confirmation Dialog
Safety confirmation before destructive operations
{
"type": "card",
"className": "max-w-md mx-auto",
"header": [
{
"type": "stack",
"spacing": 2,
"children": [
{
"type": "text",
"content": "Confirm Deletion",
"className": "text-lg font-semibold"
},
{
"type": "text",
"content": "Are you sure you want to delete this record? This action cannot be undone.",
"className": "text-sm text-muted-foreground"
}
]
}
],
"children": [
{
"type": "flex",
"className": "justify-end gap-2 pt-4",
"children": [
{
"type": "button",
"label": "Cancel",
"variant": "outline"
},
{
"type": "button",
"label": "Yes, Delete",
"variant": "destructive",
"icon": "trash-2"
}
]
}
]
}Action Toolbar
Action Toolbar
Grouped actions in a toolbar layout for record operations
{
"type": "card",
"children": [
{
"type": "flex",
"className": "items-center justify-between border-b pb-3 mb-4",
"children": [
{
"type": "text",
"content": "Order #12345",
"className": "font-semibold"
},
{
"type": "badge",
"label": "Pending",
"variant": "outline"
}
]
},
{
"type": "stack",
"spacing": 3,
"children": [
{
"type": "flex",
"className": "gap-2 items-center",
"children": [
{
"type": "text",
"content": "Customer:",
"className": "text-sm text-muted-foreground w-24"
},
{
"type": "text",
"content": "Acme Corp",
"className": "text-sm font-medium"
}
]
},
{
"type": "flex",
"className": "gap-2 items-center",
"children": [
{
"type": "text",
"content": "Amount:",
"className": "text-sm text-muted-foreground w-24"
},
{
"type": "text",
"content": "$1,250.00",
"className": "text-sm font-medium"
}
]
}
]
},
{
"type": "flex",
"className": "justify-end gap-2 pt-4 border-t mt-4",
"children": [
{
"type": "button",
"label": "Reject",
"variant": "outline",
"size": "sm"
},
{
"type": "button",
"label": "Request Review",
"variant": "secondary",
"size": "sm",
"icon": "eye"
},
{
"type": "button",
"label": "Approve",
"variant": "default",
"size": "sm",
"icon": "check"
}
]
}
]
}Action Types
Ajax Actions
Execute API calls with full request configuration:
const ajaxAction: ActionSchema = {
type: 'action',
label: 'Load Data',
actionType: 'ajax',
api: '/api/users',
method: 'GET',
headers: {
'Authorization': 'Bearer token'
},
data: {
filter: 'active'
},
onSuccess: {
type: 'toast',
message: 'Data loaded successfully'
},
onFailure: {
type: 'message',
message: 'Failed to load data'
}
};Ajax Properties:
api- API endpoint URLmethod- HTTP method (GET, POST, PUT, DELETE, PATCH)data- Request body/payloadheaders- Custom HTTP headerstimeout- Request timeout in millisecondsretry- Retry configuration
Confirm Actions
Show confirmation dialog before executing:
const confirmAction: ActionSchema = {
type: 'action',
label: 'Delete Record',
actionType: 'confirm',
confirm: {
title: 'Confirm Deletion',
message: 'Are you sure you want to delete this record?',
confirmText: 'Yes, Delete',
cancelText: 'Cancel',
confirmVariant: 'destructive'
},
api: '/api/records/123',
method: 'DELETE'
};Confirm Properties:
title- Dialog titlemessage- Confirmation messageconfirmText- Confirm button textcancelText- Cancel button textconfirmVariant- Button style
Dialog Actions
Open a modal or dialog:
const dialogAction: ActionSchema = {
type: 'action',
label: 'Edit Details',
actionType: 'dialog',
dialog: {
title: 'Edit Record',
size: 'lg',
content: {
type: 'form',
fields: [
{ name: 'name', type: 'input', label: 'Name' },
{ name: 'email', type: 'input', label: 'Email' }
]
},
actions: [
{
type: 'action',
label: 'Save',
actionType: 'ajax',
api: '/api/records/123',
method: 'PUT'
}
]
}
};Action Chaining
Execute multiple actions in sequence or parallel:
const chainedAction: ActionSchema = {
type: 'action',
label: 'Process Order',
actionType: 'ajax',
api: '/api/orders/process',
method: 'POST',
chain: [
{
type: 'action',
label: 'Send Email',
actionType: 'ajax',
api: '/api/emails/send',
method: 'POST'
},
{
type: 'action',
label: 'Update Inventory',
actionType: 'ajax',
api: '/api/inventory/update',
method: 'PUT'
},
{
type: 'action',
label: 'Log Event',
actionType: 'ajax',
api: '/api/events/log',
method: 'POST'
}
],
chainMode: 'sequential' // or 'parallel'
};Chain Modes:
sequential- Execute actions one after another (default)parallel- Execute all actions simultaneously
Conditional Execution
Execute different actions based on conditions:
const conditionalAction: ActionSchema = {
type: 'action',
label: 'Approve',
actionType: 'button',
condition: {
expression: '${data.amount > 1000}',
then: {
type: 'action',
label: 'Require Manager Approval',
actionType: 'confirm',
confirm: {
title: 'Manager Approval Required',
message: 'Amount exceeds $1000. Manager approval needed.'
}
},
else: {
type: 'action',
label: 'Auto Approve',
actionType: 'ajax',
api: '/api/approve',
method: 'POST'
}
}
};Condition Properties:
expression- JavaScript expression to evaluatethen- Action(s) to execute if trueelse- Action(s) to execute if false
Callbacks
Handle success and failure scenarios:
const actionWithCallbacks: ActionSchema = {
type: 'action',
label: 'Submit',
actionType: 'ajax',
api: '/api/submit',
method: 'POST',
onSuccess: {
type: 'toast',
message: 'Submitted successfully!'
},
onFailure: {
type: 'dialog',
dialog: {
title: 'Submission Failed',
content: {
type: 'text',
value: 'Please try again or contact support.'
}
}
}
};Callback Types:
toast- Show toast notificationmessage- Show message dialogredirect- Navigate to URLreload- Reload dataajax- Execute another API calldialog- Open dialogcustom- Custom handler
Action Tracking
Track actions for analytics:
const trackedAction: ActionSchema = {
type: 'action',
label: 'Download Report',
actionType: 'ajax',
api: '/api/reports/download',
tracking: {
enabled: true,
event: 'report_downloaded',
metadata: {
reportType: 'sales',
format: 'pdf',
dateRange: '2024-01'
}
}
};Retry Logic
Automatically retry failed requests:
const retryAction: ActionSchema = {
type: 'action',
label: 'Submit',
actionType: 'ajax',
api: '/api/submit',
method: 'POST',
timeout: 30000, // 30 seconds
retry: {
maxAttempts: 3,
delay: 1000 // 1 second between retries
}
};Complete Example
A comprehensive action combining multiple features:
const complexAction: ActionSchema = {
type: 'action',
label: 'Process Order',
icon: 'shopping-cart',
variant: 'default',
// Confirm before processing
actionType: 'confirm',
confirm: {
title: 'Confirm Order',
message: 'Process this order for ${data.customerName}?',
confirmText: 'Process Order',
confirmVariant: 'default'
},
// Conditional logic
condition: {
expression: '${data.totalAmount > 1000}',
then: {
type: 'action',
actionType: 'ajax',
api: '/api/orders/process-premium',
method: 'POST'
},
else: {
type: 'action',
actionType: 'ajax',
api: '/api/orders/process-standard',
method: 'POST'
}
},
// Action chain
chain: [
{
type: 'action',
label: 'Send Confirmation Email',
actionType: 'ajax',
api: '/api/emails/order-confirmation',
method: 'POST',
data: {
orderId: '${data.id}',
customerEmail: '${data.customerEmail}'
}
},
{
type: 'action',
label: 'Update Inventory',
actionType: 'ajax',
api: '/api/inventory/update',
method: 'PUT',
data: {
items: '${data.items}'
}
},
{
type: 'action',
label: 'Create Invoice',
actionType: 'ajax',
api: '/api/invoices/create',
method: 'POST'
}
],
chainMode: 'sequential',
// Callbacks
onSuccess: {
type: 'toast',
message: 'Order processed successfully!'
},
onFailure: {
type: 'dialog',
dialog: {
title: 'Order Processing Failed',
content: {
type: 'text',
value: 'Unable to process order. Please try again.'
}
}
},
// Tracking
tracking: {
enabled: true,
event: 'order_processed',
metadata: {
source: 'web_app',
amount: '${data.totalAmount}'
}
},
// Retry
timeout: 60000,
retry: {
maxAttempts: 3,
delay: 2000
},
// Post-action behavior
reload: true,
close: true,
redirect: '/orders/success'
};Runtime Validation
import { ActionSchema } from '@object-ui/types/zod';
const result = ActionSchema.safeParse(myAction);
if (result.success) {
console.log('Valid action configuration');
} else {
console.error('Validation errors:', result.error);
}Use Cases
Enhanced Actions are ideal for:
- API integration - Connect to backend services and external APIs
- Multi-step processes - Execute complex workflows with multiple stages
- Form submissions - Handle form data with validation and callbacks
- Confirmation dialogs - Add safety checks for critical operations
- Event tracking - Monitor user interactions for analytics
- Batch operations - Process multiple items in sequence or parallel
Best Practices
- Use confirm for destructive actions - Always confirm delete, archive, etc.
- Provide clear feedback - Use callbacks to inform users of success/failure
- Chain related operations - Group logically related API calls
- Track important events - Enable tracking for business-critical actions
- Set appropriate timeouts - Don't let users wait indefinitely
- Retry transient failures - Use retry for network-related errors
- Keep chains short - Long chains can be hard to debug
Related
- CRUD Components - CRUD operations with actions
- Form Schema - Form submission actions
- Data Source - API integration