feat(patvisit): enrich AttDoc fields in patient visit response

Add AttDocFirstName, AttDocLastName, and AttDocContactCode to showByPatient via joins to contact and site-scoped contactdetail.

Keep existing AttDoc ID field unchanged for backward compatibility.

Update PatientVisit OpenAPI schema, regenerate bundled docs, and extend PatVisitByPatientTest assertions for new fields.
This commit is contained in:
mahdahar 2026-04-21 09:40:29 +07:00
parent 4aad7d331a
commit 2053ab4596
4 changed files with 65 additions and 32 deletions

View File

@ -30,7 +30,7 @@ class PatVisitModel extends BaseModel {
} }
public function showByPatient($InternalPID) { public function showByPatient($InternalPID) {
$rows = $this->select("*, patvisit.InternalPID, patvisit.CreateDate as PVCreateDate, patdiag.CreateDate as PDCreateDate, patvisitadt.CreateDate as PVACreateDate") $rows = $this->select("*, patvisit.InternalPID, patvisit.CreateDate as PVCreateDate, patdiag.CreateDate as PDCreateDate, patvisitadt.CreateDate as PVACreateDate, attDoc.NameFirst as AttDocFirstName, attDoc.NameLast as AttDocLastName, attDocDetail.ContactCode as AttDocContactCode")
->join('patdiag', 'patdiag.InternalPVID=patvisit.InternalPVID and patdiag.DelDate is null', 'left') ->join('patdiag', 'patdiag.InternalPVID=patvisit.InternalPVID and patdiag.DelDate is null', 'left')
->join('(SELECT a1.* ->join('(SELECT a1.*
FROM patvisitadt a1 FROM patvisitadt a1
@ -43,6 +43,8 @@ class PatVisitModel extends BaseModel {
'patvisitadt.InternalPVID = patvisit.InternalPVID', 'patvisitadt.InternalPVID = patvisit.InternalPVID',
'left') 'left')
->join('location', 'location.LocationID=patvisitadt.LocationID', 'left') ->join('location', 'location.LocationID=patvisitadt.LocationID', 'left')
->join('contact attDoc', 'attDoc.ContactID = patvisitadt.AttDoc', 'left')
->join('contactdetail attDocDetail', 'attDocDetail.ContactID = attDoc.ContactID AND attDocDetail.SiteID = patvisit.SiteID', 'left')
->where('patvisit.InternalPID',$InternalPID) ->where('patvisit.InternalPID',$InternalPID)
->where('patvisit.EndDate IS NULL') // Exclude soft deleted ->where('patvisit.EndDate IS NULL') // Exclude soft deleted
->findAll(); ->findAll();
@ -107,27 +109,27 @@ class PatVisitModel extends BaseModel {
throw new \Exception("Visit not found or has been deleted."); throw new \Exception("Visit not found or has been deleted.");
} }
$visitData = array_intersect_key($input, array_flip($this->allowedFields)); $visitData = array_intersect_key($input, array_flip($this->allowedFields));
if (!empty($visitData)) { if (!empty($visitData)) {
$this->where('InternalPVID', $InternalPVID)->set($visitData)->update(); $this->where('InternalPVID', $InternalPVID)->set($visitData)->update();
} }
// patdiag // patdiag
if (array_key_exists('PatDiag', $input)) { if (array_key_exists('PatDiag', $input)) {
$exist = $modelPD->where('InternalPVID', $InternalPVID)->find(); $exist = $modelPD->where('InternalPVID', $InternalPVID)->find();
if ($exist) { if ($exist) {
if (!empty($input['PatDiag']) && (!empty($input['PatDiag']['DiagCode']) || !empty($input['PatDiag']['Diagnosis']))) { if (!empty($input['PatDiag']) && (!empty($input['PatDiag']['DiagCode']) || !empty($input['PatDiag']['Diagnosis']))) {
$tmp = $modelPD->where('InternalPVID', $InternalPVID)->set($input['PatDiag'])->update(); $tmp = $modelPD->where('InternalPVID', $InternalPVID)->set($input['PatDiag'])->update();
} else { } else {
$tmp = $modelPD->delete($InternalPVID); $tmp = $modelPD->delete($InternalPVID);
} }
} else { } else {
if (!empty($input['PatDiag']) && (!empty($input['PatDiag']['DiagCode']) || !empty($input['PatDiag']['Diagnosis']))) { if (!empty($input['PatDiag']) && (!empty($input['PatDiag']['DiagCode']) || !empty($input['PatDiag']['Diagnosis']))) {
$input['PatDiag']['InternalPVID'] = $InternalPVID; $input['PatDiag']['InternalPVID'] = $InternalPVID;
$tmp = $modelPD->insert($input['PatDiag']); $tmp = $modelPD->insert($input['PatDiag']);
} }
} }
} }
if (isset($tmp) && $tmp === false) { if (isset($tmp) && $tmp === false) {
$error = $db->error(); $error = $db->error();
throw new \Exception("Failed to update PatDiag record. ". $error['message']); throw new \Exception("Failed to update PatDiag record. ". $error['message']);
@ -148,9 +150,9 @@ class PatVisitModel extends BaseModel {
return false; return false;
} else { } else {
$db->transCommit(); $db->transCommit();
$data = [ "PVID" => $input['PVID'] ?? $visit['PVID'], "InternalPVID" => $InternalPVID ]; $data = [ "PVID" => $input['PVID'] ?? $visit['PVID'], "InternalPVID" => $InternalPVID ];
return $data; return $data;
} }
} catch (\Exception $e) { } catch (\Exception $e) {
$this->db->transRollback(); $this->db->transRollback();

View File

@ -6734,6 +6734,18 @@ components:
LastLocation: LastLocation:
type: string type: string
description: Full name of the last/current location from patvisitadt description: Full name of the last/current location from patvisitadt
AttDocFirstName:
type: string
nullable: true
description: Attending doctor first name (from contact)
AttDocLastName:
type: string
nullable: true
description: Attending doctor last name (from contact)
AttDocContactCode:
type: string
nullable: true
description: Attending doctor contact code for visit SiteID (from contactdetail)
CreateDate: CreateDate:
type: string type: string
format: date-time format: date-time

View File

@ -19,6 +19,18 @@ PatientVisit:
LastLocation: LastLocation:
type: string type: string
description: Full name of the last/current location from patvisitadt description: Full name of the last/current location from patvisitadt
AttDocFirstName:
type: string
nullable: true
description: Attending doctor first name (from contact)
AttDocLastName:
type: string
nullable: true
description: Attending doctor last name (from contact)
AttDocContactCode:
type: string
nullable: true
description: Attending doctor contact code for visit SiteID (from contactdetail)
CreateDate: CreateDate:
type: string type: string
format: date-time format: date-time

View File

@ -5,16 +5,16 @@ namespace Tests\Feature\PatVisit;
use CodeIgniter\Test\FeatureTestTrait; use CodeIgniter\Test\FeatureTestTrait;
use CodeIgniter\Test\CIUnitTestCase; use CodeIgniter\Test\CIUnitTestCase;
class PatVisitByPatientTest extends CIUnitTestCase class PatVisitByPatientTest extends CIUnitTestCase
{ {
use FeatureTestTrait; use FeatureTestTrait;
protected $endpoint = 'api/patvisit/patient'; protected $endpoint = 'api/patvisit/patient';
protected function setUp(): void protected function setUp(): void
{ {
parent::setUp(); parent::setUp();
} }
/** /**
* Test: Show all visits by valid InternalPID * Test: Show all visits by valid InternalPID
@ -37,6 +37,13 @@ class PatVisitByPatientTest extends CIUnitTestCase
// Pastikan 'data' ada // Pastikan 'data' ada
$this->assertArrayHasKey('data', $json); $this->assertArrayHasKey('data', $json);
$this->assertIsArray($json['data']); $this->assertIsArray($json['data']);
if (!empty($json['data'])) {
$this->assertArrayHasKey('AttDoc', $json['data'][0]);
$this->assertArrayHasKey('AttDocFirstName', $json['data'][0]);
$this->assertArrayHasKey('AttDocLastName', $json['data'][0]);
$this->assertArrayHasKey('AttDocContactCode', $json['data'][0]);
}
} }
/** /**