Claude Code Plugins

Community-maintained marketplace

Feedback

lexical-editor-image-management

@Hieubkav/wincellarCloneBackend
0
0

Implement Lexical Editor with automatic image management using Laravel Observers. Converts base64 images to file storage, deletes unused images, and handles cleanup. Use when building WYSIWYG editors with rich content, managing media uploads in editors, implementing automatic image optimization, or setting up Observer-based storage management for rich text editors.

Install Skill

1Download skill
2Enable skills in Claude

Open claude.ai/settings/capabilities and find the "Skills" section

3Upload to Claude

Click "Upload skill" and select the downloaded ZIP file

Note: Please verify skill by going through its instructions before using it.

SKILL.md

name lexical-editor-image-management
description Implement Lexical Editor with automatic image management using Laravel Observers. Converts base64 images to file storage, deletes unused images, and handles cleanup. Use when building WYSIWYG editors with rich content, managing media uploads in editors, implementing automatic image optimization, or setting up Observer-based storage management for rich text editors.

Lexical Editor Image Management

Complete guide for implementing Lexical Editor with automatic image management using Laravel Observers. This skill transforms base64 images from rich text editors into optimized storage files and automatically cleans up unused images.

When to use this Skill

Use this Skill when you need to:

  • Implement Lexical Editor or similar WYSIWYG editors with image uploads
  • Convert base64 images to actual file storage
  • Automatically delete unused images from storage
  • Manage media files in rich text content
  • Optimize storage usage in Laravel applications
  • Build Observer-based image lifecycle management
  • Handle cleanup of old images when content is updated or deleted

Quick Start

Step 1: Install Lexical Editor Package

composer require malzariey/filament-lexical-editor
npm install --save-dev @lexical/react @lexical/plain-text

Step 2: Create Model and Migration

php artisan make:model Post -m

Add content field to migration:

$table->text('content')->nullable(); // For Lexical Editor HTML
$table->string('image')->nullable();
$table->string('pdf')->nullable();

Step 3: Register Observer

php artisan make:observer PostObserver --model=Post

Register in EventServiceProvider:

use App\Models\Post;
use App\Observers\PostObserver;

public function boot(): void
{
    Post::observe(PostObserver::class);
}

Step 4: Configure Filament Resource

In your PostResource.php:

use Malzariey\FilamentLexicalEditor\FilamentLexicalEditor;
use Malzariey\FilamentLexicalEditor\Enums\ToolbarItem;

FilamentLexicalEditor::make('content')
    ->label('Nội dung')
    ->required()
    ->columnSpanFull()
    ->enabledToolbars([
        ToolbarItem::BOLD,
        ToolbarItem::ITALIC,
        ToolbarItem::IMAGE,
        ToolbarItem::H1,
        ToolbarItem::H2,
        // ... other toolbar items
    ])

Step 5: Apply Observer Logic

See observer-implementation.md for complete Observer code.

How It Works

Architecture Overview

┌─────────────────────────────────────────────────────┐
│        Lexical Editor (Frontend)                    │
│  - User uploads image in editor                     │
│  - Image converted to base64                        │
│  - HTML content stored with base64 images           │
└──────────────────┬──────────────────────────────────┘
                   │
                   ▼
┌─────────────────────────────────────────────────────┐
│      Laravel Observer (Backend)                     │
│  - Listen to Model saving events                    │
│  - Extract base64 images from content               │
│  - Convert base64 to actual files                   │
│  - Save files to storage                            │
│  - Replace base64 with file URLs                    │
└──────────────────┬──────────────────────────────────┘
                   │
                   ▼
┌─────────────────────────────────────────────────────┐
│      File Storage                                   │
│  storage/app/public/uploads/content/                │
│  - lexical-1701234567-abc123.jpg                    │
│  - lexical-1701234568-def456.png                    │
└──────────────────┬──────────────────────────────────┘
                   │
                   ▼
┌─────────────────────────────────────────────────────┐
│      Database (Model)                               │
│  - content: HTML with file URLs instead of base64   │
│  - <img src="/storage/uploads/content/image.jpg">   │
└─────────────────────────────────────────────────────┘

Workflow Events

  1. Creating: Auto-generate slug from title
  2. Saving: Convert all base64 images to storage files
  3. Updating:
    • Delete old main image if replaced
    • Delete old PDF if replaced
    • Compare old/new content and delete unused images
  4. Deleted: Clean up all associated files

Implementation Details

Base64 Conversion Process

The Observer scans content for base64 images using regex pattern:

data:image/{type};base64,{data}

For each match:

  1. Extract MIME type (png, jpg, gif, etc.)
  2. Decode base64 data to binary
  3. Generate unique filename: lexical-{timestamp}-{uniqid}.{ext}
  4. Save to storage/app/public/uploads/content/
  5. Replace base64 string with file URL

Example:

<!-- Before (base64) -->
<img src="...">

<!-- After (file URL) -->
<img src="/storage/uploads/content/lexical-1701234567-abc123.png">

Image Cleanup Strategy

The Observer intelligently tracks images:

  • Old images: Extracted from previous content version
  • New images: Extracted from current content version
  • Unused images: Images in old but not in new = deleted

This prevents accidental deletion of images still in use while cleaning up removed images.

Storage Structure

storage/app/public/
├── uploads/
│   ├── main-photo.jpg          # Main post image
│   ├── document.pdf            # PDF file
│   └── content/
│       ├── lexical-17012345-1a2b.jpg
│       ├── lexical-17012346-3c4d.png
│       └── lexical-17012347-5e6f.gif
└── service-content/            # For ServicePost model
    ├── service-lexical-1701234-7g8h.jpg
    └── service-lexical-1701235-9i0j.png

Keep different content types in separate directories to:

  • Track which images belong to which model
  • Prevent accidental deletion of shared images
  • Organize cleanup commands by type

Best Practices

1. Image Naming Convention

Use descriptive prefixes for different contexts:

// Post model
$filename = 'lexical-' . time() . '-' . uniqid() . '.' . $extension;

// ServicePost model
$filename = 'service-lexical-' . time() . '-' . uniqid() . '.' . $extension;

// Custom models
$filename = 'custom-lexical-' . time() . '-' . uniqid() . '.' . $extension;

2. Directory Organization

Separate storage directories by model:

// In Observer
$path = 'uploads/content/' . $filename;              // For Post
$path = 'uploads/service-content/' . $filename;     // For ServicePost
$path = 'uploads/custom-content/' . $filename;      // For custom models

3. Logging and Monitoring

Always log important operations:

Log::info("Converted base64 image to storage: {$filePath}");
Log::info("Deleted unused content image: {$imagePath}");
Log::error("Failed to convert base64 image: " . $e->getMessage());

4. Error Handling

Implement graceful error handling:

try {
    // Convert base64 to file
    $filePath = $this->saveBase64AsFile($base64Data, $extension);
} catch (\Exception $e) {
    Log::error("Conversion failed: " . $e->getMessage());
    // Continue processing other images
    continue;
}

5. Soft Deletes

Consider using soft deletes for safety:

// In Model
use SoftDeletes;

// Allow recovery of deleted models and their images
public function restore()
{
    // Restore images from backup if available
}

Advanced Usage

Multiple Image Formats

Support different image types:

$supported = ['png', 'jpg', 'jpeg', 'gif', 'webp', 'svg+xml'];

// In regex pattern
preg_match_all('/data:image\/(png|jpg|jpeg|gif|webp|svg\+xml);base64,/i', $content);

Image Validation

Validate before saving:

private function validateImage(string $base64Data, string $extension): bool
{
    $imageData = base64_decode($base64Data);
    
    // Check file size (max 5MB)
    if (strlen($imageData) > 5 * 1024 * 1024) {
        throw new \Exception("Image exceeds 5MB limit");
    }
    
    // Verify MIME type
    $finfo = finfo_open(FILEINFO_MIME_TYPE);
    $mime = finfo_buffer($finfo, $imageData);
    finfo_close($finfo);
    
    if (!str_starts_with($mime, 'image/')) {
        throw new \Exception("Invalid image file");
    }
    
    return true;
}

Automatic Resizing

Resize large images:

use Intervention\Image\Facades\Image;

private function optimizeImage(string $path): void
{
    $image = Image::make(storage_path('app/public/' . $path));
    
    // Resize if larger than 1920px
    if ($image->width() > 1920) {
        $image->resize(1920, null, ['aspect_ratio' => true]);
    }
    
    // Compress and save as WebP
    $image->save(str_replace($path, '.webp', $path), 75);
}

Cleanup Command

Create a command to clean unused images:

php artisan make:command ImagesCleanUnused

See cleanup-command.md for complete implementation.

Common Issues and Solutions

Issue 1: Images Not Converting

Symptom: Base64 images remain in content instead of being saved

Solutions:

  • Verify Observer is registered in EventServiceProvider
  • Check storage permissions: chmod -R 775 storage/app/public/
  • Verify public disk is configured in config/filesystems.php
  • Check logs for conversion errors: tail -f storage/logs/laravel.log

Issue 2: Images Not Displaying

Symptom: Images saved but not visible in frontend

Solutions:

  • Run storage symlink: php artisan storage:link
  • Verify storage/app/public directory exists
  • Check file permissions: chmod -R 755 storage/app/public/uploads/
  • Verify public URL in config/app.php

Issue 3: Orphaned Files

Symptom: Old images remain in storage after deletion

Solutions:

  • Verify deleted() event is firing: Add logging
  • Check handleContentImages() logic for content extraction
  • Run cleanup command: php artisan images:clean-unused --dry-run
  • Manually delete orphaned files: Find files modified before X days

Issue 4: Observer Not Running

Symptom: Model events not triggering

Solutions:

# Clear configuration cache
php artisan config:clear

# Clear application cache
php artisan cache:clear

# Verify Observer is registered
php artisan tinker
>>> Post::getObservableEvents()

Testing

Unit Testing

public function test_base64_converts_to_file()
{
    $post = Post::create([
        'name' => 'Test Post',
        'content' => '<img src="...">',
    ]);
    
    $this->assertStringContainsString('/storage/uploads/content/', $post->content);
    $this->assertStringNotContainsString('data:image/png;base64', $post->content);
}

public function test_old_image_deleted_on_update()
{
    $post = Post::create(['name' => 'Test', 'image' => 'old.jpg']);
    
    Storage::disk('public')->put('old.jpg', 'dummy content');
    
    $post->update(['image' => 'new.jpg']);
    
    Storage::disk('public')->assertMissing('old.jpg');
}

Feature Testing

public function test_image_cleanup_on_content_update()
{
    $post = Post::create([
        'content' => '<img src="/storage/uploads/content/image1.jpg">
                      <img src="/storage/uploads/content/image2.jpg">',
    ]);
    
    $post->update([
        'content' => '<img src="/storage/uploads/content/image1.jpg">',
    ]);
    
    // image2.jpg should be deleted
    Storage::disk('public')->assertMissing('uploads/content/image2.jpg');
}

Related Resources

Examples

See EXAMPLES.md for:

  • Complete working Observer class
  • Filament Resource configuration
  • Cleanup command implementation
  • Real-world use cases

References

Validation Checklist

Before deploying to production:

  • Observers registered in EventServiceProvider
  • Storage permissions set correctly (775)
  • Storage symlink created: php artisan storage:link
  • Base64 regex pattern tested with sample content
  • Image cleanup logic verified with test data
  • Logging configured and monitored
  • Cleanup command scheduled in Kernel.php
  • Backup system in place before cleanup
  • File size limits enforced
  • Image format validation in place

Output Format

When implementing this Skill, I will:

  1. Analyze your current model structure
  2. Generate complete Observer class with all lifecycle hooks
  3. Provide Filament Resource configuration
  4. Create cleanup command if needed
  5. Set up logging and monitoring
  6. Include comprehensive error handling
  7. Provide testing examples
  8. Document storage structure
  9. Create troubleshooting guide

This ensures a production-ready image management system with automatic cleanup and error recovery.