eg-mohamed / notable
This is my package notable
Fund package maintenance!
Mohamed Said
Requires
- php: ^8.3
- illuminate/contracts: ^10.0||^11.0||^12.0
- spatie/laravel-package-tools: ^1.92.7
Requires (Dev)
- laravel/pint: ^1.24
- nunomaduro/collision: ^8.8.2||^7.10.0
- orchestra/testbench: ^10.6.0||^9.0.0||^8.22.0
- pestphp/pest: ^2.0||^3.8.4
- pestphp/pest-plugin-arch: ^2.5||^3.1.1
- pestphp/pest-plugin-laravel: ^2.0||^3.2
README
Notable is a powerful Laravel package that enables you to add notes to any Eloquent model through polymorphic relationships. Perfect for adding internal comments, audit logs, user feedback, or any textual annotations to your models.
✨ Features
- 🔗 Polymorphic Relationships - Attach notes to any Eloquent model
- 👤 Creator Tracking - Track who created each note (also polymorphic!)
- ⏰ Timestamps - Automatic created_at and updated_at tracking
- 🔍 Query Scopes - Powerful query methods for filtering notes
- 🎯 Configurable - Customize table names through config
- 🚀 Easy Integration - Simple trait-based implementation
- 📦 Laravel 10+ Ready - Built for modern Laravel applications
🚀 Installation
Install the package via Composer:
composer require mohamedsaid/notable
Publish and run the migrations:
php artisan vendor:publish --tag="notable-migrations"
php artisan migrate
Optionally, publish the config file:
php artisan vendor:publish --tag="notable-config"
🎯 Quick Start
1. Add the Trait to Your Model
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use MohamedSaid\Notable\Traits\HasNotables; class User extends Model { use HasNotables; }
2. Start Adding Notes
$user = User::find(1); // Simple note $user->addNote('User completed onboarding process'); // Note with creator $admin = User::find(2); $user->addNote('Account verified by admin', $admin); // Get all notes $notes = $user->getNotes(); // Check if model has notes if ($user->hasNotes()) { echo "This user has {$user->notesCount()} notes"; }
📚 Complete Usage Guide
Adding Notes
// Basic note $model->addNote('Something noteworthy happened'); // Note with creator (any model) $model->addNote('Action performed', $currentUser); $model->addNote('System generated note', $systemBot);
Retrieving Notes
// Get all notes (newest first) $notes = $model->getNotes(); // Get notes with creator information $notesWithCreators = $model->getNotesWithCreator(); // Get latest note $latestNote = $model->getLatestNote(); // Get notes by specific creator $adminNotes = $model->getNotesByCreator($admin); // Check if model has notes $hasNotes = $model->hasNotes(); // Count notes $count = $model->notesCount();
Managing Notes
// Update a note $model->updateNote($noteId, 'Updated note content'); // Delete a note $model->deleteNote($noteId);
Query Scopes
The Notable
model includes powerful query scopes:
use MohamedSaid\Notable\Notable; // Notes by specific creator $notes = Notable::byCreator($user)->get(); // Notes without creator (system notes) $systemNotes = Notable::withoutCreator()->get(); // Recent notes (last 7 days by default) $recentNotes = Notable::recent()->get(); $lastMonth = Notable::recent(30)->get(); // Older notes (30+ days by default) $oldNotes = Notable::olderThan(30)->get(); // Combine scopes $recentAdminNotes = Notable::byCreator($admin) ->recent(14) ->orderBy('created_at', 'desc') ->get();
🗂️ Database Schema
The package creates a notables
table with the following structure:
Column | Type | Description |
---|---|---|
id |
bigint | Primary key |
note |
text | The note content |
notable_type |
varchar | Polymorphic type (model class) |
notable_id |
bigint | Polymorphic ID (model ID) |
creator_type |
varchar | Creator polymorphic type (nullable) |
creator_id |
bigint | Creator polymorphic ID (nullable) |
created_at |
timestamp | When the note was created |
updated_at |
timestamp | When the note was last updated |
⚙️ Configuration
Publish the config file to customize the package:
php artisan vendor:publish --tag="notable-config"
<?php // config/notable.php return [ // Customize the table name 'table_name' => 'notables', ];
🎨 Advanced Examples
User Activity Log
class User extends Model { use HasNotables; public function logActivity(string $activity, ?Model $performer = null) { return $this->addNote("User activity: {$activity}", $performer); } } // Usage $user->logActivity('Logged in', $user); $user->logActivity('Password changed', $user); $user->logActivity('Account suspended', $admin);
Order Tracking
class Order extends Model { use HasNotables; public function addStatusNote(string $status, ?User $updatedBy = null) { return $this->addNote("Order status changed to: {$status}", $updatedBy); } } // Usage $order->addStatusNote('Processing', $staff); $order->addStatusNote('Shipped', $system); $order->addStatusNote('Delivered');
Support Tickets
class SupportTicket extends Model { use HasNotables; } // Customer adds note $ticket->addNote('Still experiencing the issue', $customer); // Support agent responds $ticket->addNote('Investigating the problem', $agent); // Get conversation history $conversation = $ticket->getNotesWithCreator();
🔍 Model Relationships
Access notes directly through Eloquent relationships:
// Get the polymorphic relationship $model->notables(); // Returns MorphMany relationship // With eager loading $users = User::with('notables.creator')->get(); // Load notes later $user->load('notables');
📝 Note Model Properties
Each note instance provides:
$note = $model->getLatestNote(); $note->note; // The note content $note->notable; // The parent model $note->creator; // The creator model (nullable) $note->created_at; // When it was created $note->updated_at; // When it was updated
🤝 Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
🐛 Bug Reports
If you discover any bugs, please create an issue on GitHub.
📄 License
The MIT License (MIT). Please see License File for more information.
👨💻 Credits
Made with ❤️ by Mohamed Said