375 lines
18 KiB
PHP
375 lines
18 KiB
PHP
<?php
|
|
// RMLWorkorderAdminController.php
|
|
|
|
class RMLWorkorderAdminController extends TTCrud
|
|
{
|
|
protected string $headerTitle = 'RML-Arbeitsaufträge (Admin)';
|
|
protected bool $createText = false;
|
|
protected array $permissionCheck = ['RMLAdmin'];
|
|
|
|
protected array $columns = [
|
|
['key' => 'id', 'text' => 'Auftrags-Nr.', 'table' => ['sortable' => true]],
|
|
['key' => 'preordercampaign_id', 'text' => 'Kampagne', 'modal' => false, 'table' => ['filter' => 'select']],
|
|
['key' => 'preorderInfo', 'text' => 'Kunde', 'modal' => false, 'table' => ['sortable' => false]],
|
|
['key' => 'rimo_fcp_name', 'text' => 'FCP', 'modal' => false, 'table' => ['sortable' => true]],
|
|
['key' => 'companyName', 'text' => 'Zugewiesene Firma', 'modal' => false, 'table' => ['sortable' => true]],
|
|
['key' => 'status', 'text' => 'Status', 'modal' => false, 'table' => ['filter' => 'iconSelect', 'filterOptions' => [
|
|
['value' => 'new', 'text' => 'Neu', 'icon' => 'fas fa-star text-primary'],
|
|
['value' => 'assigned', 'text' => 'Zugewiesen', 'icon' => 'fas fa-user-check text-info'],
|
|
['value' => 'scheduled', 'text' => 'Geplant', 'icon' => 'fas fa-calendar-check text-warning'],
|
|
['value' => 'correction_requested', 'text' => 'Korrektur angefordert', 'icon' => 'fas fa-exclamation-triangle text-danger'],
|
|
['value' => 'intervention_required', 'text' => 'Eingriff erforderlich', 'icon' => 'fas fa-times-circle text-danger'],
|
|
['value' => 'problem_solved', 'text' => 'Problem gelöst', 'icon' => 'fas fa-check-circle text-success'],
|
|
['value' => 'documented', 'text' => 'Dokumentiert', 'icon' => 'fas fa-file-alt text-success'],
|
|
['value' => 'completed', 'text' => 'Abgeschlossen', 'icon' => 'fas fa-check-double text-secondary'],
|
|
['value' => 'cancelled', 'text' => 'Abgebrochen', 'icon' => 'fas fa-ban text-danger'],
|
|
]]],
|
|
['key' => 'additionalInfo', 'text' => 'Notiz', 'modal' => false, 'table' => ['sortable' => true]],
|
|
['key' => 'deadlineDate', 'text' => 'Deadline', 'modal' => false, 'table' => ['filter' => 'date', 'sortable' => true]],
|
|
];
|
|
|
|
private function getStatusText(string $statusKey): string
|
|
{
|
|
foreach ($this->columns as $column) {
|
|
if ($column['key'] === 'status') {
|
|
foreach ($column['table']['filterOptions'] as $option) {
|
|
if ($option['value'] === $statusKey) {
|
|
return $option['text'];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ucfirst(str_replace('_', ' ', $statusKey)); // Fallback
|
|
}
|
|
|
|
protected function indexAction()
|
|
{
|
|
$campaigns = Helper::getPreorderCampaignFromUser($this->user, true);
|
|
$this->columns[array_search('preordercampaign_id', array_column($this->columns, 'key'))]['table']['filterOptions'] = array_map(
|
|
fn($c) => ['value' => $c->id, 'text' => $c->name],
|
|
$campaigns
|
|
);
|
|
|
|
$this->createWorkordersFromPreorders();
|
|
Helper::renderVue($this, 'RMLWorkorderAdmin', $this->headerTitle, [
|
|
"CRUD_CONFIG" => $this->getCrudConfig(),
|
|
"TABLE_URL" => $this::getUrl("RMLWorkorderAdmin/get"),
|
|
]);
|
|
}
|
|
|
|
protected function getAction()
|
|
{
|
|
$json = json_decode(file_get_contents('php://input'), true);
|
|
$pagination = $json['pagination'] ?? ['page' => 1, 'per_page' => 10];
|
|
$filters = $json['filters'] ?? [];
|
|
$order = $json['order'] ?? [];
|
|
|
|
$allowedCampaignIds = Helper::getPreorderCampaignFromUser($this->user);
|
|
|
|
if (empty($allowedCampaignIds)) {
|
|
self::returnJson([
|
|
'rows' => [],
|
|
'pagination' => array_merge($pagination, ['total_rows' => 0, 'total_pages' => 0, 'filtered_available' => 0])
|
|
]);
|
|
return;
|
|
}
|
|
|
|
$limit = $pagination['per_page'];
|
|
$offset = ($pagination['page'] - 1) * $limit;
|
|
|
|
$workorders = RMLWorkorderModel::getAdminWorkorders($filters, $limit, $offset, $order, $allowedCampaignIds);
|
|
$totalCount = RMLWorkorderModel::countAdminWorkorders($filters, $allowedCampaignIds);
|
|
|
|
$rows = array_map(function ($workorder) {
|
|
$row = (array)$workorder;
|
|
$row['companyName'] ??= 'Nicht zugewiesen';
|
|
return $row;
|
|
}, $workorders);
|
|
|
|
self::returnJson([
|
|
'rows' => $rows,
|
|
'pagination' => [
|
|
'page' => $pagination['page'],
|
|
'per_page' => $pagination['per_page'],
|
|
'total_rows' => $totalCount,
|
|
'total_pages' => ceil($totalCount / $limit),
|
|
'filtered_available' => $totalCount
|
|
]
|
|
]);
|
|
}
|
|
|
|
private function createWorkordersFromPreorders()
|
|
{
|
|
$configs = RMLWorkorderTenantConfigModel::getAll();
|
|
foreach ($configs as $config) {
|
|
$filters = json_decode($config->workorderCreationFilters, true);
|
|
if (empty($filters)) continue;
|
|
|
|
$networks = NetworkModel::getAll(['owner_id' => $config->addressId]);
|
|
if (empty($networks)) continue;
|
|
|
|
$tenantCampaigns = array_map(fn($n) => $n->id, PreordercampaignModel::getAll(['network_id' => array_map(fn($n) => $n->id, $networks)]));
|
|
if (empty($tenantCampaigns)) continue;
|
|
|
|
$filters['preordercampaign_id'] = $tenantCampaigns;
|
|
|
|
$newPreorders = PreorderModel::searchActive($filters);
|
|
if (empty($newPreorders)) continue;
|
|
|
|
foreach ($newPreorders as $preorder) {
|
|
if (!RMLWorkorderModel::getFirst(['preorderId' => $preorder->id])) {
|
|
RMLWorkorderModel::create([
|
|
'preorderId' => $preorder->id,
|
|
'clusterId' => $preorder->preordercampaign_id,
|
|
'status' => 'new',
|
|
'create' => time(),
|
|
'createBy' => 0 // System User
|
|
]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
protected function getDocumentationAction()
|
|
{
|
|
if (empty($this->request->workorderId)) self::sendError("Arbeitsauftrags-ID fehlt.");
|
|
|
|
$docs = RMLWorkorderDocumentationModel::getAll(['workorderId' => $this->request->workorderId], null, 0, ['key' => 'create', 'order' => 'ASC']);
|
|
$journals = RMLWorkorderJournalModel::getAll(['workorderId' => $this->request->workorderId], null, 0, ['key' => 'create', 'order' => 'DESC']);
|
|
|
|
$translationMap = [
|
|
'photo_hup_mounted' => 'Foto_montierter_HÜP', 'photo_hup_open' => 'Foto_offener_HÜP',
|
|
'photo_splice_cassette_hup' => 'Foto_Spleißkassette_HÜP', 'photo_splice_cassette_fcp' => 'Foto_Spleißkassette_FCP',
|
|
'photo_hup_closed_stickers' => 'Foto_geschlossener_HÜP_mit_Aufklebern', 'photo_fcp_labeled' => 'Foto_FCP_beschriftet',
|
|
'photo_patch_position_osp' => 'Foto_Patch-Position_OSP-Seite', 'photo_patch_position_anb' => 'Foto_Patch-Position_ANB-Seite',
|
|
'measurement_protocol_otdr' => 'ODTR_Messung', 'other' => 'Sonstiges_Dokument'
|
|
];
|
|
|
|
$responseDocs = [];
|
|
$typeCounts = [];
|
|
|
|
foreach ($docs as $doc) {
|
|
$file = new File($doc->fileId);
|
|
$documentTypeKey = $doc->documentType;
|
|
$typeCounts[$documentTypeKey] = ($typeCounts[$documentTypeKey] ?? 0) + 1;
|
|
$originalFilename = $file->orig_filename ?? $file->filename;
|
|
$extension = pathinfo($originalFilename, PATHINFO_EXTENSION);
|
|
$translatedType = $translationMap[$documentTypeKey] ?? $documentTypeKey;
|
|
$newFilename = "{$translatedType}_{$typeCounts[$documentTypeKey]}." . strtolower($extension);
|
|
|
|
$responseDocs[] = [
|
|
'id' => $doc->id, 'fileId' => $doc->fileId, 'fileName' => $newFilename, 'description' => $doc->description,
|
|
'documentType' => $documentTypeKey, 'userName' => UserModel::getOne($doc->createBy)->name ?? 'Unbekannt',
|
|
'mimetype' => $file->mimetype ?? 'application/octet-stream',
|
|
];
|
|
}
|
|
|
|
foreach ($journals as $journal) $journal->createByName = UserModel::getOne($journal->createBy)->name ?? 'Unbekannt';
|
|
|
|
self::returnJson(['docs' => $responseDocs, 'journals' => $journals]);
|
|
}
|
|
|
|
private function assignSingleWorkorder($workorderId, $companyId, $deadline, $userId)
|
|
{
|
|
$workorder = RMLWorkorderModel::get($workorderId);
|
|
if (!$workorder) return false;
|
|
$company = RMLWorkorderCompanyModel::get($companyId);
|
|
if (!$company) return false;
|
|
|
|
$workorder->companyId = $companyId;
|
|
$workorder->status = 'assigned';
|
|
$workorder->assignmentDate = time();
|
|
$workorder->deadlineDate = $deadline;
|
|
RMLWorkorderModel::update((array)$workorder);
|
|
|
|
RMLWorkorderJournalModel::create([
|
|
'workorderId' => $workorder->id, 'text' => "Firma '{$company->name}' wurde zugewiesen.",
|
|
'create' => time(), 'createBy' => $userId,
|
|
]);
|
|
|
|
$preorder = new Preorder($workorder->preorderId);
|
|
if ($preorder->id) {
|
|
$preorder->status_id = 10;
|
|
$preorder->edit_by = $this->user->id;
|
|
$preorder->save();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
protected function assignWorkorderAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId']) || empty($post['companyId'])) self::sendError("Erforderliche Felder fehlen.");
|
|
|
|
$deadline = !empty($post['deadlineDate']) ? $post['deadlineDate'] : strtotime('+6 weeks');
|
|
if ($this->assignSingleWorkorder($post['workorderId'], $post['companyId'], $deadline, $this->user->id)) {
|
|
self::returnJson(['success' => true, 'message' => 'Arbeitsauftrag erfolgreich zugewiesen.']);
|
|
} else self::sendError("Arbeitsauftrag konnte nicht zugewiesen werden. Er wurde möglicherweise bereits bearbeitet oder existiert nicht.");
|
|
}
|
|
|
|
protected function massAssignWorkordersAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderIds']) || empty($post['companyId'])) self::sendError("Erforderliche Felder fehlen.");
|
|
|
|
$deadline = strtotime($post['deadlineDate'] ?? '+6 weeks');
|
|
$count = 0;
|
|
foreach ($post['workorderIds'] as $workorderId) if ($this->assignSingleWorkorder($workorderId, $post['companyId'], $deadline, $this->user->id)) $count++;
|
|
self::returnJson(['success' => true, 'message' => "$count Arbeitsaufträge erfolgreich zugewiesen."]);
|
|
}
|
|
|
|
protected function requestCorrectionAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId']) || empty($post['text'])) self::sendError("Erforderliche Felder fehlen.");
|
|
$workorder = RMLWorkorderModel::get($post['workorderId']);
|
|
if (!$workorder) self::sendError("Arbeitsauftrag nicht gefunden.");
|
|
|
|
$oldStatus = $workorder->status;
|
|
$workorder->status = 'correction_requested';
|
|
RMLWorkorderModel::update((array)$workorder);
|
|
|
|
RMLWorkorderJournalModel::create([
|
|
'workorderId' => $workorder->id, 'text' => "Korrektur angefordert. Grund: " . $post['text'],
|
|
'fileIds' => !empty($post['fileIds']) ? json_encode($post['fileIds']) : null,
|
|
'statusChange' => $this->getStatusText($oldStatus) . " -> " . $this->getStatusText('correction_requested'),
|
|
'create' => time(), 'createBy' => $this->user->id,
|
|
]);
|
|
self::returnJson(['success' => true, 'message' => 'Korrektur wurde angefordert.']);
|
|
}
|
|
|
|
protected function getCompaniesAction()
|
|
{
|
|
$tenantId = $this->request->tenantId ?? null;
|
|
$companies = RMLWorkorderCompanyModel::getAll([], null, 0, ['key' => 'name', 'order' => 'ASC']);
|
|
|
|
if ($tenantId) {
|
|
$companies = array_filter($companies, function ($company) use ($tenantId) {
|
|
if ($company->addressId == 4807 && empty($company->visibleForAddressId)) return true;
|
|
$visibleFor = !empty($company->visibleForAddressId) ? json_decode($company->visibleForAddressId, true) : [];
|
|
return in_array($tenantId, $visibleFor);
|
|
});
|
|
}
|
|
self::returnJson(array_values(array_map(fn($c) => ['value' => $c->id, 'text' => $c->name], $companies)));
|
|
}
|
|
|
|
protected function addJournalAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId']) || empty(trim($post['text']))) self::sendError("Arbeitsauftrags-ID und Text sind erforderlich.");
|
|
|
|
RMLWorkorderJournalModel::create(['workorderId' => $post['workorderId'], 'text' => $post['text'], 'createBy' => $this->user->id, 'create' => time()]);
|
|
$journals = array_map(function ($j) {
|
|
$j->createByName = UserModel::getOne($j->createBy)->name ?? 'Unbekannt';
|
|
return (array)$j;
|
|
},
|
|
RMLWorkorderJournalModel::getAll(['workorderId' => $post['workorderId']], null, 0, ['key' => 'create', 'order' => 'DESC'])
|
|
);
|
|
self::returnJson(['success' => true, 'message' => 'Journaleintrag hinzugefügt.', 'journals' => $journals]);
|
|
}
|
|
|
|
protected function updateDeadlineAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId']) || empty($post['deadlineDate'])) self::sendError("Erforderliche Felder fehlen.");
|
|
|
|
$workorder = RMLWorkorderModel::get($post['workorderId']);
|
|
if (!$workorder) self::sendError("Arbeitsauftrag nicht gefunden.");
|
|
|
|
$workorder->deadlineDate = $post['deadlineDate'];
|
|
RMLWorkorderModel::update((array)$workorder);
|
|
RMLWorkorderJournalModel::create(['workorderId' => $workorder->id, 'text' => 'Deadline geändert auf ' . date('d.m.Y', $post['deadlineDate']) . '.', 'create' => time(), 'createBy' => $this->user->id]);
|
|
self::returnJson(['success' => true, 'message' => 'Deadline erfolgreich aktualisiert.']);
|
|
}
|
|
|
|
protected function acceptDocumentationAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId'])) self::sendError("Arbeitsauftrags-ID fehlt.");
|
|
$workorder = RMLWorkorderModel::get($post['workorderId']);
|
|
if (!$workorder) self::sendError("Arbeitsauftrag nicht gefunden.");
|
|
if ($workorder->status !== 'documented') self::sendError("Die Dokumentation muss zuerst von der Firma als fertig markiert werden.");
|
|
|
|
$preorder = new Preorder($workorder->preorderId);
|
|
if ($preorder->id) {
|
|
$preorder->status_id = 15;
|
|
$preorder->edit_by = $this->user->id;
|
|
$preorder->save();
|
|
}
|
|
|
|
$oldStatus = $workorder->status;
|
|
$workorder->status = 'completed';
|
|
RMLWorkorderModel::update((array)$workorder);
|
|
RMLWorkorderJournalModel::create([
|
|
'workorderId' => $workorder->id, 'text' => 'Dokumentation akzeptiert und Arbeitsauftrag abgeschlossen.',
|
|
'statusChange' => $this->getStatusText($oldStatus) . " -> " . $this->getStatusText('completed'),
|
|
'create' => time(), 'createBy' => $this->user->id,
|
|
]);
|
|
self::returnJson(['success' => true, 'message' => 'Dokumentation akzeptiert und Arbeitsauftrag abgeschlossen.']);
|
|
}
|
|
|
|
protected function setToProblemSolvedAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId']) || empty($post['text'])) self::sendError("Arbeitsauftrags-ID und Text sind erforderlich.");
|
|
|
|
$workorder = RMLWorkorderModel::get($post['workorderId']);
|
|
if (!$workorder) self::sendError("Arbeitsauftrag nicht gefunden.");
|
|
if ($workorder->status !== 'intervention_required') self::sendError("Der Arbeitsauftrag muss den Status 'Eingriff erforderlich' haben, um als gelöst markiert zu werden.");
|
|
|
|
$oldStatus = $workorder->status;
|
|
$workorder->status = 'problem_solved';
|
|
RMLWorkorderModel::update((array)$workorder);
|
|
RMLWorkorderJournalModel::create([
|
|
'workorderId' => $workorder->id, 'text' => "Problem gelöst: " . $post['text'],
|
|
'statusChange' => $this->getStatusText($oldStatus) . " -> " . $this->getStatusText('problem_solved'),
|
|
'create' => time(), 'createBy' => $this->user->id,
|
|
]);
|
|
self::returnJson(['success' => true, 'message' => 'Arbeitsauftrag als "Problem gelöst" markiert.']);
|
|
}
|
|
|
|
protected function updateAdditionalInfoAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId'])) self::sendError("Arbeitsauftrags-ID fehlt.");
|
|
$workorder = RMLWorkorderModel::get($post['workorderId']);
|
|
if (!$workorder) self::sendError("Arbeitsauftrag nicht gefunden.");
|
|
|
|
$oldInfo = $workorder->additionalInfo;
|
|
$workorder->additionalInfo = $post['additionalInfo'] ?? null;
|
|
RMLWorkorderModel::update((array)$workorder);
|
|
|
|
RMLWorkorderJournalModel::create([
|
|
'workorderId' => $workorder->id, 'text' => "Zusatzinfo geändert.\nAlt: '{$oldInfo}'\nNeu: '{$workorder->additionalInfo}'",
|
|
'create' => time(), 'createBy' => $this->user->id,
|
|
]);
|
|
self::returnJson(['success' => true, 'message' => 'Zusatzinfo aktualisiert.']);
|
|
}
|
|
|
|
protected function cancelWorkorderAction()
|
|
{
|
|
$post = json_decode(file_get_contents('php://input'), true);
|
|
if (empty($post['workorderId'])) self::sendError("Arbeitsauftrags-ID fehlt.");
|
|
$workorder = RMLWorkorderModel::get($post['workorderId']);
|
|
if (!$workorder) self::sendError("Arbeitsauftrag nicht gefunden.");
|
|
|
|
$oldStatus = $workorder->status;
|
|
$workorder->status = 'cancelled';
|
|
RMLWorkorderModel::update((array)$workorder);
|
|
|
|
RMLWorkorderJournalModel::create([
|
|
'workorderId' => $workorder->id, 'text' => 'Arbeitsauftrag wurde storniert.' . (!empty($post['reason']) ? ' Grund: ' . $post['reason'] : ''),
|
|
'statusChange' => $this->getStatusText($oldStatus) . " -> " . $this->getStatusText('cancelled'),
|
|
'create' => time(), 'createBy' => $this->user->id,
|
|
]);
|
|
|
|
$preorder = new Preorder($workorder->preorderId);
|
|
if ($preorder->id) {
|
|
$preorder->status_id = 99;
|
|
$preorder->edit_by = $this->user->id;
|
|
$preorder->save();
|
|
}
|
|
|
|
self::returnJson(['success' => true, 'message' => 'Arbeitsauftrag wurde storniert.']);
|
|
}
|
|
} |