wip - custom worklist
This commit is contained in:
parent
280a795350
commit
f4de216946
@ -4,8 +4,13 @@
|
|||||||
|
|
||||||
use App\DAL\Studies\WorklistFactory;
|
use App\DAL\Studies\WorklistFactory;
|
||||||
use App\Models\Study;
|
use App\Models\Study;
|
||||||
|
use App\Models\User;
|
||||||
|
use App\Services\ACL\AccessControl;
|
||||||
|
use App\Services\ACL\WorklistButton;
|
||||||
|
use App\Services\ACL\WorklistColumn;
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use Carbon\CarbonImmutable;
|
use Carbon\CarbonImmutable;
|
||||||
|
use Closure;
|
||||||
use Illuminate\Database\Eloquent\Builder as QueryBuilder;
|
use Illuminate\Database\Eloquent\Builder as QueryBuilder;
|
||||||
use Illuminate\Support\Facades\Blade;
|
use Illuminate\Support\Facades\Blade;
|
||||||
use Illuminate\Support\Str;
|
use Illuminate\Support\Str;
|
||||||
@ -26,6 +31,22 @@ private static function dtFormat(Carbon|CarbonImmutable|null $dt): ?string
|
|||||||
|
|
||||||
public function dataTable(QueryBuilder $query): EloquentDataTable
|
public function dataTable(QueryBuilder $query): EloquentDataTable
|
||||||
{
|
{
|
||||||
|
$table = new EloquentDataTable($query);
|
||||||
|
$rawColumns = [
|
||||||
|
'priority_icon',
|
||||||
|
'report_status_led',
|
||||||
|
];
|
||||||
|
foreach ($this->customColumns() as $column => $content) {
|
||||||
|
$table->editColumn($column, $content);
|
||||||
|
$rawColumns[] = $column;
|
||||||
|
}
|
||||||
|
$table
|
||||||
|
->orderColumn('patient_name', 'patient_name $1')
|
||||||
|
->rawColumns($rawColumns)
|
||||||
|
->setRowId('id');
|
||||||
|
|
||||||
|
return $table;
|
||||||
|
|
||||||
return (new EloquentDataTable($query))
|
return (new EloquentDataTable($query))
|
||||||
->addColumn('action', 'worklist.action')
|
->addColumn('action', 'worklist.action')
|
||||||
->editColumn('patient_name', fn (Study $study) => $study->sanitizedPatientName())
|
->editColumn('patient_name', fn (Study $study) => $study->sanitizedPatientName())
|
||||||
@ -136,6 +157,72 @@ public function html(): HtmlBuilder
|
|||||||
*/
|
*/
|
||||||
public function getColumns(): array
|
public function getColumns(): array
|
||||||
{
|
{
|
||||||
|
$cols_allowed = AccessControl::worklistColumns();
|
||||||
|
$columns = [];
|
||||||
|
foreach ($cols_allowed as $allowed) {
|
||||||
|
switch ($allowed) {
|
||||||
|
case WorklistColumn::Priority:
|
||||||
|
$columns[] = Column::make($allowed->value)
|
||||||
|
->searchable(false)
|
||||||
|
->hidden();
|
||||||
|
$columns[] = Column::make('priority_icon')
|
||||||
|
->searchable(false)
|
||||||
|
->orderable(false)
|
||||||
|
->addClass('text-center')
|
||||||
|
->width('20px')
|
||||||
|
->title('');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::ReportStatus:
|
||||||
|
case WorklistColumn::History:
|
||||||
|
$columns[] = Column::make($allowed->value)
|
||||||
|
->searchable(false)
|
||||||
|
->orderable(false)
|
||||||
|
->addClass('text-center')
|
||||||
|
->width('20px')
|
||||||
|
->title('');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::PatientSexAge:
|
||||||
|
$columns[] = Column::make($allowed->value)
|
||||||
|
->searchable(false)
|
||||||
|
->orderable(false)
|
||||||
|
->addClass('text-center')
|
||||||
|
->title('Age');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::StudyDate:
|
||||||
|
$columns[] = Column::make($allowed->value)->searchable(false)->title('Scan Dt');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::ReceiveDate:
|
||||||
|
$columns[] = Column::make($allowed->value)->searchable(false)->title('Received');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::ReportDate:
|
||||||
|
$columns[] = Column::make($allowed->value)->searchable(false)->title('Read At');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::AssignedPhysician:
|
||||||
|
$columns[] = Column::make($allowed->value)
|
||||||
|
->searchable(false)
|
||||||
|
->title('Assigned');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::ReadingPhysician:
|
||||||
|
$columns[] = Column::make($allowed->value)
|
||||||
|
->searchable(false)
|
||||||
|
->title('Read By');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::ActionButtons:
|
||||||
|
$columns[] = Column::computed('action')
|
||||||
|
->exportable(false)
|
||||||
|
->printable(false)
|
||||||
|
->width(60)
|
||||||
|
->addClass('text-center')
|
||||||
|
->title('');
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$columns[] = Column::make($allowed->value)->title(Str::title($allowed->value));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $columns;
|
||||||
|
|
||||||
return [
|
return [
|
||||||
Column::make('priority')->hidden(),
|
Column::make('priority')->hidden(),
|
||||||
Column::make('priority_icon')
|
Column::make('priority_icon')
|
||||||
@ -171,7 +258,8 @@ public function getColumns(): array
|
|||||||
Column::make('study_description')
|
Column::make('study_description')
|
||||||
->title('Study'),
|
->title('Study'),
|
||||||
|
|
||||||
Column::make('show_study')->searchable(false)
|
Column::make('show_study')
|
||||||
|
->searchable(false)
|
||||||
->orderable(false)
|
->orderable(false)
|
||||||
->addClass('text-center')
|
->addClass('text-center')
|
||||||
->width('20px')
|
->width('20px')
|
||||||
@ -218,4 +306,110 @@ protected function filename(): string
|
|||||||
|
|
||||||
return Str::slug(implode(' ', $parts), '_');
|
return Str::slug(implode(' ', $parts), '_');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function physicianColumn(?User $user, Carbon|CarbonImmutable|null $dt): ?string
|
||||||
|
{
|
||||||
|
if ($user === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Blade::render('staff.worklist.partials._radiologist-listing',
|
||||||
|
[
|
||||||
|
'avatar_url' => $user->avatar(),
|
||||||
|
'name' => $user->display_name,
|
||||||
|
'time' => $dt?->diffForHumans() ?? '~',
|
||||||
|
]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, Closure>
|
||||||
|
*/
|
||||||
|
private function customColumns(): array
|
||||||
|
{
|
||||||
|
$columns = [];
|
||||||
|
|
||||||
|
foreach (AccessControl::worklistColumns() as $col) {
|
||||||
|
switch ($col) {
|
||||||
|
case WorklistColumn::PatientName:
|
||||||
|
$columns[$col->value] = fn (Study $study) => $study->sanitizedPatientName();
|
||||||
|
break;
|
||||||
|
case WorklistColumn::StudyDescription:
|
||||||
|
$columns[$col->value] = fn (Study $study) => $study->sanitizedStudyDescription();
|
||||||
|
break;
|
||||||
|
case WorklistColumn::AssignedPhysician:
|
||||||
|
$columns[$col->value] = fn (Study $study) => $this->physicianColumn($study->assignedPhysician, $study->assigned_at);
|
||||||
|
break;
|
||||||
|
case WorklistColumn::ReadingPhysician:
|
||||||
|
$columns[$col->value] = fn (Study $study) => $this->physicianColumn($study->readingPhysician, $study->reported_at);
|
||||||
|
break;
|
||||||
|
case WorklistColumn::Series:
|
||||||
|
$columns[$col->value] = fn (Study $study) => $study->numInstances() . '<small class="text-muted ms-2 fw-lighter fs-xsmall">' . human_filesize($study->disk_size) . '</small>';
|
||||||
|
break;
|
||||||
|
case WorklistColumn::StudyDate:
|
||||||
|
case WorklistColumn::ReceiveDate:
|
||||||
|
case WorklistColumn::ReportDate:
|
||||||
|
case WorklistColumn::AssignDate:
|
||||||
|
case WorklistColumn::AuthorizeDate:
|
||||||
|
case WorklistColumn::ArchiveDate:
|
||||||
|
$columns[$col->value] = fn (Study $study) => $study->{$col->value}?->format(self::DATE_FORMAT);
|
||||||
|
break;
|
||||||
|
case WorklistColumn::History:
|
||||||
|
$columns[$col->value] = fn (Study $study) => sprintf('
|
||||||
|
<a href="#" data-id="' . _h($study->id) . '" class="btn btn-sm btn-outline-light show-attach">
|
||||||
|
<i class="fa-light fa-file-prescription %s"></i>
|
||||||
|
</a>
|
||||||
|
', blank($study->body_part_examined) ? 'text-muted' : 'text-primary');
|
||||||
|
break;
|
||||||
|
case WorklistColumn::ActionButtons:
|
||||||
|
$columns[$col->value] = fn (Study $study) => $this->generateButtons($study);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $columns;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function renderButton(string $data_id, string $fa_icon, string $data_class, string $text, string $url = '#'): string
|
||||||
|
{
|
||||||
|
return Blade::render('staff.worklist.partials._column-button',
|
||||||
|
[
|
||||||
|
'data_id' => $data_id,
|
||||||
|
'url' => $url,
|
||||||
|
'fa_icon' => $fa_icon,
|
||||||
|
'data_class' => $data_class,
|
||||||
|
'text' => $text,
|
||||||
|
]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function generateButtons(Study $study): string
|
||||||
|
{
|
||||||
|
$btns = [];
|
||||||
|
foreach (AccessControl::worklistButtons() as $button) {
|
||||||
|
switch ($button) {
|
||||||
|
case WorklistButton::StudyMetadata:
|
||||||
|
$btns[] = $this->renderButton($study->hash, 'fa-circle-info', 'showStudy btn-outline-facebook', 'Info');
|
||||||
|
break;
|
||||||
|
case WorklistButton::History:
|
||||||
|
$btns[] = $this->renderButton($study->hash, 'fa-pen-to-square', 'btn-outline-primary', 'Edit', route('staff.history.edit', $study->hash));
|
||||||
|
break;
|
||||||
|
case WorklistButton::Notes:
|
||||||
|
$btns[] = $this->renderButton($study->hash, 'fa-user-doctor', 'btn-outline-youtube show-assign', 'Assign');
|
||||||
|
break;
|
||||||
|
case WorklistButton::Attachment:
|
||||||
|
$btns[] = $this->renderButton($study->hash, 'fa-trash', 'btn-danger archive-study', 'Archive');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return implode("\r", $btns);
|
||||||
|
|
||||||
|
$btn = '<a href="#" data-id="' . _h($study->id) . '" class="btn btn-outline-facebook btn-xs showStudy"><i class="fa-light fa-circle-info me-1"></i>Show</a>';
|
||||||
|
$btn .= '<a href="' . route('staff.history.edit', $study->hash) . '" class="edit btn btn-outline-primary btn-xs"><i class="fa-light fa-pen-to-square me-1"></i>Edit</a>';
|
||||||
|
$btn .= ' <a href="#" data-id="' . _h($study->id) . '" class="btn btn-outline-youtube fw-light btn-xs show-assign"><i class="fa-light fa-user-doctor me-1"></i>Assign</a>';
|
||||||
|
$btn .= ' <a href="#" data-id="' . _h($study->id) . '" class="btn btn-danger btn-xs deleteStudy">Delete</a>';
|
||||||
|
|
||||||
|
return $btn;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,43 +1,48 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
namespace App\Services;
|
namespace App\Services\ACL;
|
||||||
|
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
|
use Illuminate\Support\Collection;
|
||||||
enum WorklistColumn: string
|
|
||||||
{
|
|
||||||
case PatientName = 'patient_name';
|
|
||||||
case PatientId = 'patient_id';
|
|
||||||
case StudyDate = 'study_date';
|
|
||||||
case StudyTime = 'study_time';
|
|
||||||
case Modality = 'modality';
|
|
||||||
case StudyDescription = 'study_description';
|
|
||||||
case AccessionNumber = 'accession_number';
|
|
||||||
case ReferringPhysician = 'referring_physician';
|
|
||||||
case Status = 'status';
|
|
||||||
case Priority = 'priority';
|
|
||||||
case CreatedAt = 'created_at';
|
|
||||||
case UpdatedAt = 'updated_at';
|
|
||||||
}
|
|
||||||
|
|
||||||
final readonly class AccessControl
|
final readonly class AccessControl
|
||||||
{
|
{
|
||||||
public static function worklistColumns(?int $user_id): array
|
public static function worklistColumns(?int $user_id = null): Collection
|
||||||
{
|
{
|
||||||
$user = $user_id !== null ? User::findOrFail($user_id) : auth()->user();
|
$user = $user_id !== null ? User::findOrFail($user_id) : auth()->user();
|
||||||
$columns = [
|
$columns = collect([
|
||||||
'patient_name',
|
WorklistColumn::Priority,
|
||||||
'patient_id',
|
WorklistColumn::ReportStatus,
|
||||||
'study_date',
|
WorklistColumn::History,
|
||||||
'study_time',
|
WorklistColumn::Modality,
|
||||||
'modality',
|
WorklistColumn::PatientId,
|
||||||
'study_description',
|
WorklistColumn::PatientName,
|
||||||
'accession_number',
|
WorklistColumn::PatientSexAge,
|
||||||
'referring_physician',
|
WorklistColumn::StudyDescription,
|
||||||
'status',
|
WorklistColumn::StudyDate,
|
||||||
'priority',
|
WorklistColumn::AssignedPhysician,
|
||||||
'created_at',
|
WorklistColumn::ReadingPhysician,
|
||||||
'updated_at',
|
WorklistColumn::ReportDate,
|
||||||
];
|
WorklistColumn::Series,
|
||||||
|
WorklistColumn::ReceiveDate,
|
||||||
|
WorklistColumn::ActionButtons,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return $columns;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function worklistButtons(?int $user_id = null): Collection
|
||||||
|
{
|
||||||
|
$user = $user_id !== null ? User::findOrFail($user_id) : auth()->user();
|
||||||
|
$buttons = collect([
|
||||||
|
WorklistButton::StudyMetadata,
|
||||||
|
WorklistButton::History,
|
||||||
|
WorklistButton::Notes,
|
||||||
|
WorklistButton::Attachment,
|
||||||
|
WorklistButton::Assign,
|
||||||
|
WorklistButton::Report,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return $buttons;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
13
app/Services/ACL/WorklistButton.php
Normal file
13
app/Services/ACL/WorklistButton.php
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Services\ACL;
|
||||||
|
|
||||||
|
enum WorklistButton: string
|
||||||
|
{
|
||||||
|
case StudyMetadata = 'metadata';
|
||||||
|
case History = 'history';
|
||||||
|
case Notes = 'notes';
|
||||||
|
case Attachment = 'attachment';
|
||||||
|
case Assign = 'assign';
|
||||||
|
case Report = 'report';
|
||||||
|
}
|
35
app/Services/ACL/WorklistColumn.php
Normal file
35
app/Services/ACL/WorklistColumn.php
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Services\ACL;
|
||||||
|
|
||||||
|
enum WorklistColumn: string
|
||||||
|
{
|
||||||
|
case StudyStatus = 'study_status';
|
||||||
|
case ReportStatus = 'report_status';
|
||||||
|
case StudyHash = 'hash';
|
||||||
|
case PatientName = 'patient_name';
|
||||||
|
case PatientId = 'patient_id';
|
||||||
|
case PatientSexAge = 'sex_age';
|
||||||
|
case History = 'history';
|
||||||
|
case StudyDate = 'study_date';
|
||||||
|
case ReceiveDate = 'received_at';
|
||||||
|
case ReportDate = 'reported_at';
|
||||||
|
case AssignDate = 'assigned_at';
|
||||||
|
case AuthorizeDate = 'authorized_at';
|
||||||
|
case ArchiveDate = 'archived_at';
|
||||||
|
case Modality = 'modality';
|
||||||
|
case StudyDescription = 'study_description';
|
||||||
|
case AccessionNumber = 'accession_number';
|
||||||
|
case ReferringPhysician = 'referring_physician';
|
||||||
|
case AssignedPhysician = 'assigned_physician';
|
||||||
|
case ReadingPhysician = 'reader';
|
||||||
|
case LockStatus = 'lock_status';
|
||||||
|
case LinkedInstitute = 'institute';
|
||||||
|
case Facility = 'facility';
|
||||||
|
case InstitutionName = 'institution_name';
|
||||||
|
case Priority = 'priority';
|
||||||
|
case Series = 'series';
|
||||||
|
case CreatedAt = 'created_at';
|
||||||
|
case UpdatedAt = 'updated_at';
|
||||||
|
case ActionButtons = 'action';
|
||||||
|
}
|
@ -7,9 +7,6 @@
|
|||||||
|
|
||||||
return new class extends Migration
|
return new class extends Migration
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* Run the migrations
|
|
||||||
*/
|
|
||||||
public function up(): void
|
public function up(): void
|
||||||
{
|
{
|
||||||
Schema::create('radiologists', static function (Blueprint $table) {
|
Schema::create('radiologists', static function (Blueprint $table) {
|
||||||
|
@ -0,0 +1,4 @@
|
|||||||
|
<a href="{{ $url ?? '#' }}" data-id="{{ $data_id }}" class="btn btn-xs {{ $data_class }}">
|
||||||
|
<i class="fa-light {{ $fa_icon }} me-1"></i>
|
||||||
|
{{ $text }}
|
||||||
|
</a>
|
Loading…
Reference in New Issue
Block a user