Add CAD request workflows and focus actions

- Add request hydration, submission, closing, and response handling
- Wire UI events for dispatch orders, support requests, and map focus
- Expand dispatcher layout for alerts, requests, and detail views
This commit is contained in:
Jacob Schmidt 2026-03-31 20:14:45 -05:00
parent 112488f82e
commit 4ea7cf7d05
38 changed files with 5432 additions and 403 deletions

View File

@ -24,3 +24,17 @@ if (isNil QGVAR(CADUIBridge)) then { call FUNC(initUIBridge); };
GVAR(CADUIBridge) call ["handleGroupUpdateResponse", [_result]];
}] call CFUNC(addEventHandler);
[QGVAR(responseCadRequest), {
params [["_result", createHashMap, [createHashMap]]];
GVAR(CADUIBridge) call ["handleRequestResponse", [_result]];
}] call CFUNC(addEventHandler);
[QGVAR(invalidateCadState), {
if (isNil QGVAR(CADRepository)) exitWith {};
if !(GVAR(CADRepository) getOrDefault ["isOpen", false]) exitWith {};
if (isNil QGVAR(CADUIBridge)) exitWith {};
GVAR(CADUIBridge) call ["requestHydrate", []];
}] call CFUNC(addEventHandler);

View File

@ -72,6 +72,48 @@ switch (_event) do {
GVAR(CADUIBridge) call ["requestAssignTask", [_taskID, _groupID, _note]];
};
case "cad::dispatchOrder::create": {
private _assigneeGroupID = "";
private _targetGroupID = "";
private _note = "";
private _priority = "priority";
if (_data isEqualType createHashMap) then {
_assigneeGroupID = _data getOrDefault ["assigneeGroupID", ""];
_targetGroupID = _data getOrDefault ["targetGroupID", ""];
_note = _data getOrDefault ["note", ""];
_priority = _data getOrDefault ["priority", "priority"];
};
GVAR(CADUIBridge) call ["requestCreateDispatchOrder", [_assigneeGroupID, _targetGroupID, _note, _priority]];
};
case "cad::supportRequest::submit": {
private _type = "";
private _fields = createHashMap;
private _priority = "priority";
if (_data isEqualType createHashMap) then {
_type = _data getOrDefault ["type", ""];
_fields = _data getOrDefault ["fields", createHashMap];
_priority = _data getOrDefault ["priority", "priority"];
};
GVAR(CADUIBridge) call ["requestSubmitSupportRequest", [_type, _fields, _priority]];
};
case "cad::dispatchOrder::close": {
private _taskID = "";
if (_data isEqualType createHashMap) then {
_taskID = _data getOrDefault ["taskID", ""];
};
GVAR(CADUIBridge) call ["requestCloseDispatchOrder", [_taskID]];
};
case "cad::supportRequest::close": {
private _requestID = "";
if (_data isEqualType createHashMap) then {
_requestID = _data getOrDefault ["requestID", ""];
};
GVAR(CADUIBridge) call ["requestCloseSupportRequest", [_requestID]];
};
case "cad::tasks::acknowledge": {
private _taskID = "";
if (_data isEqualType createHashMap) then {
@ -108,6 +150,42 @@ switch (_event) do {
GVAR(CADUIBridge) call ["requestGroupRole", [_groupID, _role]];
};
case "cad::groups::profile": {
private _groupID = "";
private _status = "";
private _role = "";
if (_data isEqualType createHashMap) then {
_groupID = _data getOrDefault ["groupID", ""];
_status = _data getOrDefault ["status", ""];
_role = _data getOrDefault ["role", ""];
};
GVAR(CADUIBridge) call ["requestGroupProfile", [_groupID, _status, _role]];
};
case "cad::groups::focus": {
private _groupID = "";
if (_data isEqualType createHashMap) then {
_groupID = _data getOrDefault ["groupID", ""];
};
GVAR(CADUIBridge) call ["focusGroup", [_groupID]];
};
case "cad::tasks::focus": {
private _taskID = "";
if (_data isEqualType createHashMap) then {
_taskID = _data getOrDefault ["taskID", ""];
};
GVAR(CADUIBridge) call ["focusTask", [_taskID]];
};
case "cad::requests::focus": {
private _requestID = "";
if (_data isEqualType createHashMap) then {
_requestID = _data getOrDefault ["requestID", ""];
};
GVAR(CADUIBridge) call ["focusRequest", [_requestID]];
};
case "map::zoomIn": {
private _mapCtrl = uiNamespace getVariable [QGVAR(MapCtrl), controlNull];
if (isNull _mapCtrl) exitWith {};

View File

@ -27,6 +27,7 @@ GVAR(CADRepository) = createHashMapObject [[
_self set ["isOpen", false];
_self set ["groups", []];
_self set ["contracts", []];
_self set ["requests", []];
_self set ["assignments", []];
_self set ["activity", []];
_self set ["session", createHashMap];
@ -37,6 +38,7 @@ GVAR(CADRepository) = createHashMapObject [[
createHashMapFromArray [
["groups", +(_self getOrDefault ["groups", []])],
["contracts", +(_self getOrDefault ["contracts", []])],
["requests", +(_self getOrDefault ["requests", []])],
["assignments", +(_self getOrDefault ["assignments", []])],
["activity", +(_self getOrDefault ["activity", []])],
["session", +(_self getOrDefault ["session", createHashMap])],
@ -67,6 +69,7 @@ GVAR(CADRepository) = createHashMapObject [[
_self set ["groups", +(_payload getOrDefault ["groups", []])];
_self set ["contracts", +(_payload getOrDefault ["contracts", []])];
_self set ["requests", +(_payload getOrDefault ["requests", []])];
_self set ["assignments", +(_payload getOrDefault ["assignments", []])];
_self set ["activity", +(_payload getOrDefault ["activity", []])];
_self set ["session", +(_payload getOrDefault ["session", createHashMap])];

View File

@ -213,6 +213,47 @@ GVAR(CADUIBridgeBaseClass) = compileFinal createHashMapFromArray [
[SRPC(cad,requestAssignCadTask), [getPlayerUID player, _taskID, _groupID, _note]] call CFUNC(serverEvent);
true
}],
["requestCreateDispatchOrder", compileFinal {
params [
["_assigneeGroupID", "", [""]],
["_targetGroupID", "", [""]],
["_note", "", [""]],
["_priority", "priority", [""]]
];
if (_assigneeGroupID isEqualTo "" || { _targetGroupID isEqualTo "" }) exitWith { false };
[SRPC(cad,requestCreateCadDispatchOrder), [getPlayerUID player, _assigneeGroupID, _targetGroupID, _note, _priority]] call CFUNC(serverEvent);
true
}],
["requestSubmitSupportRequest", compileFinal {
params [
["_type", "", [""]],
["_fields", createHashMap, [createHashMap]],
["_priority", "priority", [""]]
];
if (_type isEqualTo "") exitWith { false };
[SRPC(cad,requestSubmitCadSupportRequest), [getPlayerUID player, _type, _fields, _priority]] call CFUNC(serverEvent);
true
}],
["requestCloseDispatchOrder", compileFinal {
params [["_taskID", "", [""]]];
if (_taskID isEqualTo "") exitWith { false };
[SRPC(cad,requestCloseCadDispatchOrder), [getPlayerUID player, _taskID]] call CFUNC(serverEvent);
true
}],
["requestCloseSupportRequest", compileFinal {
params [["_requestID", "", [""]]];
if (_requestID isEqualTo "") exitWith { false };
[SRPC(cad,requestCloseCadSupportRequest), [getPlayerUID player, _requestID]] call CFUNC(serverEvent);
true
}],
["requestAcknowledgeTask", compileFinal {
params [["_taskID", "", [""]]];
@ -245,6 +286,87 @@ GVAR(CADUIBridgeBaseClass) = compileFinal createHashMapFromArray [
[SRPC(cad,requestUpdateCadGroupRole), [getPlayerUID player, _groupID, _role]] call CFUNC(serverEvent);
true
}],
["requestGroupProfile", compileFinal {
params [["_groupID", "", [""]], ["_status", "", [""]], ["_role", "", [""]]];
if (_groupID isEqualTo "") exitWith { false };
if (_status isEqualTo "" && { _role isEqualTo "" }) exitWith { false };
[SRPC(cad,requestUpdateCadGroupProfile), [getPlayerUID player, _groupID, _status, _role]] call CFUNC(serverEvent);
true
}],
["focusGroup", compileFinal {
params [["_groupID", "", [""]]];
if (_groupID isEqualTo "") exitWith { false };
if (isNil QGVAR(CADRepository)) exitWith { false };
private _groups = GVAR(CADRepository) getOrDefault ["groups", []];
private _groupIndex = _groups findIf { (_x getOrDefault ["groupId", ""]) isEqualTo _groupID };
if (_groupIndex < 0) exitWith { false };
private _group = _groups # _groupIndex;
private _position = _group getOrDefault ["position", []];
if !(_position isEqualType []) exitWith { false };
if ((count _position) < 2) exitWith { false };
private _mapCtrl = _self call ["getMapControl", []];
if (isNull _mapCtrl) exitWith { false };
private _targetPosition = [_position # 0, _position # 1, 0];
_mapCtrl ctrlMapAnimAdd [0.35, ctrlMapScale _mapCtrl, _targetPosition];
ctrlMapAnimCommit _mapCtrl;
true
}],
["focusTask", compileFinal {
params [["_taskID", "", [""]]];
if (_taskID isEqualTo "") exitWith { false };
if (isNil QGVAR(CADRepository)) exitWith { false };
private _contracts = GVAR(CADRepository) getOrDefault ["contracts", []];
private _taskIndex = _contracts findIf {
private _entryTaskID = _x getOrDefault ["taskId", _x getOrDefault ["taskID", ""]];
_entryTaskID isEqualTo _taskID
};
if (_taskIndex < 0) exitWith { false };
private _task = _contracts # _taskIndex;
private _position = _task getOrDefault ["position", []];
if !(_position isEqualType []) exitWith { false };
if ((count _position) < 2) exitWith { false };
private _mapCtrl = _self call ["getMapControl", []];
if (isNull _mapCtrl) exitWith { false };
private _targetPosition = [_position # 0, _position # 1, 0];
_mapCtrl ctrlMapAnimAdd [0.35, ctrlMapScale _mapCtrl, _targetPosition];
ctrlMapAnimCommit _mapCtrl;
true
}],
["focusRequest", compileFinal {
params [["_requestID", "", [""]]];
if (_requestID isEqualTo "") exitWith { false };
if (isNil QGVAR(CADRepository)) exitWith { false };
private _requests = GVAR(CADRepository) getOrDefault ["requests", []];
private _requestIndex = _requests findIf { (_x getOrDefault ["requestId", ""]) isEqualTo _requestID };
if (_requestIndex < 0) exitWith { false };
private _request = _requests # _requestIndex;
private _position = _request getOrDefault ["position", []];
if !(_position isEqualType []) exitWith { false };
if ((count _position) < 2) exitWith { false };
private _mapCtrl = _self call ["getMapControl", []];
if (isNull _mapCtrl) exitWith { false };
private _targetPosition = [_position # 0, _position # 1, 0];
_mapCtrl ctrlMapAnimAdd [0.35, ctrlMapScale _mapCtrl, _targetPosition];
ctrlMapAnimCommit _mapCtrl;
true
}],
["refreshHydrate", compileFinal {
if (isNil QGVAR(CADRepository)) exitWith { false };
GVAR(CADRepository) call ["pushHydratePayload", [_self]]
@ -301,6 +423,25 @@ GVAR(CADUIBridgeBaseClass) = compileFinal createHashMapFromArray [
["message", _result getOrDefault ["message", "Group update processed."]],
["success", _result getOrDefault ["success", false]]
]]]
}],
["handleRequestResponse", compileFinal {
params [["_result", createHashMap, [createHashMap]]];
if (_self getOrDefault ["dispatcherReady", false]) then {
private _dispatcherCtrl = _self call ["getDispatcherControl", []];
if !(isNull _dispatcherCtrl) then {
_dispatcherCtrl ctrlWebBrowserAction ["ExecJS", format [
"window.cadDispatcher && window.cadDispatcher.setStatus(%1, %2);",
str (_result getOrDefault ["message", "Request processed."]),
str (["error", "success"] select (_result getOrDefault ["success", false]))
]];
};
};
_self call ["sendEvent", ["cad::request::response", createHashMapFromArray [
["message", _result getOrDefault ["message", "Request processed."]],
["success", _result getOrDefault ["success", false]]
]]]
}]
];

View File

@ -89,9 +89,9 @@ class RscMapUI {
class SidePanelBrowser: RscText {
type = 106;
idc = 1005;
x = "safeZoneX + (safeZoneW * 0.1) + (safeZoneW * 0.8) - 0.4630"; // Right edge of 80% box minus panel width
x = "safeZoneX + (safeZoneW * 0.1) + (safeZoneW * 0.8) - 0.5550"; // Right edge of 80% box minus panel width
y = "safeZoneY + (safeZoneH * 0.1) + 0.10372"; // Below visible top bar
w = "0.4630"; // ~250px width
w = "0.5550"; // Wider panel for four-tab operations layout
h = "(safeZoneH * 0.8) - 0.10372 - 0.0556"; // Full height minus visible bars
colorBackground[] = {0, 0, 0, 0};
};

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1 +1 @@
window.mapUIState={layersPanelVisible:!0,sidePanelElement:null},window.mapUI={sendEvent(e,t){A3API.SendAlert(JSON.stringify({event:e,data:t}))},updateCoordinates(e,t){const n=document.getElementById("coordsDisplay");n&&(n.textContent=`X: ${Math.round(e).toString().padStart(4,"0")} Y: ${Math.round(t).toString().padStart(4,"0")}`)},updateScale(e){const t=document.getElementById("scaleDisplay");t&&(t.textContent=`Scale: 1:${Math.round(e)}`)},updateStatus(e){const t=document.getElementById("statusText");t&&(t.textContent=e)}},window.updateCoordinates=window.mapUI.updateCoordinates,window.updateScale=window.mapUI.updateScale,window.updateStatus=window.mapUI.updateStatus,window.ForgeBridge=window.ForgeBridge||{_handlers:{},on(e,t){this._handlers[e]=this._handlers[e]||[],this._handlers[e].push(t)},ready:e=>(window.mapUI.sendEvent("cad::ready",e||{}),!0),receive(e){if(!e||"object"!=typeof e)return;(this._handlers[e.event]||[]).forEach(t=>t(e.data||{}))},send:(e,t)=>(window.mapUI.sendEvent(e,t||{}),!0),close:e=>(window.mapUI.sendEvent("map::close",e||{}),!0)};
window.mapUIState={layersPanelVisible:!0,sidePanelElement:null},window.mapUI={formatGridCoordinate:t=>Math.round(Number(t)||0).toString().padStart(4,"0"),formatPosition(t){const e=Array.isArray(t)?t:[0,0,0];return`X: ${this.formatGridCoordinate(e[0])} Y: ${this.formatGridCoordinate(e[1])}`},sendEvent(t,e){A3API.SendAlert(JSON.stringify({event:t,data:e}))},updateCoordinates(t,e){const n=document.getElementById("coordsDisplay");n&&(n.textContent=this.formatPosition([t,e,0]))},updateScale(t){const e=document.getElementById("scaleDisplay");e&&(e.textContent=`Scale: 1:${Math.round(t)}`)},updateStatus(t){const e=document.getElementById("statusText");e&&(e.textContent=t)}},window.updateCoordinates=window.mapUI.updateCoordinates,window.updateScale=window.mapUI.updateScale,window.updateStatus=window.mapUI.updateStatus,window.ForgeBridge=window.ForgeBridge||{_handlers:{},on(t,e){this._handlers[t]=this._handlers[t]||[],this._handlers[t].push(e)},ready:t=>(window.mapUI.sendEvent("cad::ready",t||{}),!0),receive(t){if(!t||"object"!=typeof t)return;(this._handlers[t.event]||[]).forEach(e=>e(t.data||{}))},send:(t,e)=>(window.mapUI.sendEvent(t,e||{}),!0),close:t=>(window.mapUI.sendEvent("map::close",t||{}),!0)};

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1 +1 @@
window.cadTopbar={mode:"operations",dispatchView:"board",currentGroup:null,session:{},init(){document.getElementById("btnClose").addEventListener("click",()=>{window.mapUI.sendEvent("map::close",null)}),document.getElementById("modeToggle").addEventListener("change",e=>{window.mapUI.sendEvent("cad::mode::set",{mode:e.target.checked?"dispatch":"operations"})}),document.getElementById("dispatchRefreshBtn").addEventListener("click",()=>{window.mapUI.sendEvent("cad::refresh",{})}),document.getElementById("dispatchBoardBtn").addEventListener("click",()=>{window.mapUI.sendEvent("cad::dispatchView::set",{dispatchView:"board"})}),document.getElementById("dispatchMapBtn").addEventListener("click",()=>{window.mapUI.sendEvent("cad::dispatchView::set",{dispatchView:"map"})}),document.getElementById("operatorRoleBtn").addEventListener("click",()=>{this.currentGroup&&window.mapUI.sendEvent("cad::groups::role",{groupID:this.currentGroup.groupId||"",role:document.getElementById("operatorRoleSelect").value})}),document.getElementById("operatorStatusBtn").addEventListener("click",()=>{this.currentGroup&&window.mapUI.sendEvent("cad::groups::status",{groupID:this.currentGroup.groupId||"",status:document.getElementById("operatorStatusSelect").value})}),window.mapUI.sendEvent("cad::topbar::ready",{})},formatLocation(e){const t=Array.isArray(e?.position)?e.position:[0,0,0];return`X: ${Math.round(t[0]||0).toString().padStart(4,"0")} Y: ${Math.round(t[1]||0).toString().padStart(4,"0")}`},receiveState(e){this.session=e&&e.session&&"object"==typeof e.session?e.session:{},this.mode=e&&"string"==typeof e.mode?e.mode:"operations",this.dispatchView=e&&"string"==typeof e.dispatchView?e.dispatchView:"board",this.currentGroup=e&&e.currentGroup&&"object"==typeof e.currentGroup?e.currentGroup:null;const t=document.getElementById("modeControls"),o=!!this.session.isDispatcher,s=!(!this.currentGroup||!this.session.isLeader&&!this.session.isDispatcher),n=document.getElementById("operatorStrip"),d=document.getElementById("operatorControls"),r=document.getElementById("dispatchViewControls"),i=document.getElementById("dispatchRefreshBtn"),a=document.getElementById("dispatchBoardBtn"),c=document.getElementById("dispatchMapBtn");t.classList.toggle("is-hidden",!o),r.classList.toggle("is-hidden",!o||"dispatch"!==this.mode),n.classList.toggle("is-hidden","operations"!==this.mode||!this.currentGroup),d.classList.toggle("is-hidden",!s),document.body.dataset.mode=this.mode,document.body.dataset.dispatcher=o?"true":"false",document.getElementById("modeToggle").checked="dispatch"===this.mode,a.classList.toggle("is-active","board"===this.dispatchView),c.classList.toggle("is-active","map"===this.dispatchView),i.title="dispatch"===this.mode?"Refresh dispatch board":"Refresh CAD",i.setAttribute("aria-label","dispatch"===this.mode?"Refresh dispatch board":"Refresh CAD"),document.getElementById("operatorGroupName").textContent=this.currentGroup?this.currentGroup.callsign||this.currentGroup.groupId||"Current Group":"No Group",document.getElementById("operatorLocation").textContent=this.currentGroup?this.formatLocation(this.currentGroup):"Unavailable",this.currentGroup&&(document.getElementById("operatorRoleSelect").value=this.currentGroup.role||"infantry",document.getElementById("operatorStatusSelect").value=this.currentGroup.status||"available")}},window.cadTopbar.init();
window.cadTopbar={mode:"operations",dispatchView:"board",currentGroup:null,session:{},init(){document.getElementById("btnClose").addEventListener("click",()=>{window.mapUI.sendEvent("map::close",null)}),document.getElementById("modeToggle").addEventListener("change",e=>{window.mapUI.sendEvent("cad::mode::set",{mode:e.target.checked?"dispatch":"operations"})}),document.getElementById("dispatchRefreshBtn").addEventListener("click",()=>{window.mapUI.sendEvent("cad::refresh",{})}),document.getElementById("dispatchBoardBtn").addEventListener("click",()=>{window.mapUI.sendEvent("cad::dispatchView::set",{dispatchView:"board"})}),document.getElementById("dispatchMapBtn").addEventListener("click",()=>{window.mapUI.sendEvent("cad::dispatchView::set",{dispatchView:"map"})}),document.getElementById("operatorRoleBtn").addEventListener("click",()=>{this.currentGroup&&window.mapUI.sendEvent("cad::groups::role",{groupID:this.currentGroup.groupId||"",role:document.getElementById("operatorRoleSelect").value})}),document.getElementById("operatorStatusBtn").addEventListener("click",()=>{this.currentGroup&&window.mapUI.sendEvent("cad::groups::status",{groupID:this.currentGroup.groupId||"",status:document.getElementById("operatorStatusSelect").value})}),window.mapUI.sendEvent("cad::topbar::ready",{})},formatLocation(e){const t=Array.isArray(e?.position)?e.position:[0,0,0];return window.mapUI.formatPosition(t)},receiveState(e){this.session=e&&e.session&&"object"==typeof e.session?e.session:{},this.mode=e&&"string"==typeof e.mode?e.mode:"operations",this.dispatchView=e&&"string"==typeof e.dispatchView?e.dispatchView:"board",this.currentGroup=e&&e.currentGroup&&"object"==typeof e.currentGroup?e.currentGroup:null;const t=document.getElementById("modeControls"),o=!!this.session.isDispatcher,s=!(!this.currentGroup||!this.session.isLeader&&!this.session.isDispatcher),n=document.getElementById("operatorStrip"),d=document.getElementById("operatorControls"),i=document.getElementById("dispatchViewControls"),r=document.getElementById("dispatchRefreshBtn"),a=document.getElementById("dispatchBoardBtn"),c=document.getElementById("dispatchMapBtn");t.classList.toggle("is-hidden",!o),i.classList.toggle("is-hidden",!o||"dispatch"!==this.mode),n.classList.toggle("is-hidden","operations"!==this.mode||!this.currentGroup),d.classList.toggle("is-hidden",!s),document.body.dataset.mode=this.mode,document.body.dataset.dispatcher=o?"true":"false",document.getElementById("modeToggle").checked="dispatch"===this.mode,a.classList.toggle("is-active","board"===this.dispatchView),c.classList.toggle("is-active","map"===this.dispatchView),r.title="dispatch"===this.mode?"Refresh dispatch board":"Refresh CAD",r.setAttribute("aria-label","dispatch"===this.mode?"Refresh dispatch board":"Refresh CAD"),document.getElementById("operatorGroupName").textContent=this.currentGroup?this.currentGroup.callsign||this.currentGroup.groupId||"Current Group":"No Group",document.getElementById("operatorLocation").textContent=this.currentGroup?this.formatLocation(this.currentGroup):"Unavailable",this.currentGroup&&(document.getElementById("operatorRoleSelect").value=this.currentGroup.role||"infantry",document.getElementById("operatorStatusSelect").value=this.currentGroup.status||"available")}},window.cadTopbar.init();

File diff suppressed because one or more lines are too long

View File

@ -1 +1 @@
<!doctype html><html><head><meta charset="UTF-8"></head><body><div class="panel-header"><h3>CAD System</h3></div><div class="panel-content"><div id="cadStatusMessage" class="task-status-message"></div><div class="cad-tabs" role="tablist" aria-label="CAD Sections"><button id="tabContractsBtn" class="cad-tab is-active" type="button" data-tab="contracts">Contracts</button> <button id="tabRosterBtn" class="cad-tab" type="button" data-tab="roster">Roster</button> <button id="tabActivityBtn" class="cad-tab" type="button" data-tab="activity">Activity</button></div><div class="cad-tab-panels"><div id="contractsPanel" class="cad-section is-active" data-panel="contracts"><div class="cad-section-header">Contracts</div><div id="taskList" class="task-list"><div class="placeholder-message"><p>Loading contracts...</p></div></div></div><div id="rosterPanel" class="cad-section" data-panel="roster"><div class="cad-section-header">Roster</div><div id="rosterList" class="task-list"><div class="placeholder-message"><p>Loading roster...</p></div></div></div><div id="activityPanel" class="cad-section" data-panel="activity"><div class="cad-section-header">Activity</div><div id="activityList" class="task-list"><div class="placeholder-message"><p>No recent activity.</p></div></div></div></div></div><script>window.MapLoader={loadCSS:e=>A3API.RequestFile(e).then(e=>{const d=document.createElement("style");d.textContent=e,document.head.appendChild(d)}),loadJS(path){return A3API.RequestFile(path).then(js=>{eval(js)})},loadAll(e){return e.reduce((e,d)=>e.then(()=>d.endsWith(".css")?this.loadCSS(d):d.endsWith(".js")?this.loadJS(d):Promise.resolve()),Promise.resolve())}},MapLoader.loadAll(["forge\\forge_client\\addons\\cad\\ui\\_site\\cad-common.css","forge\\forge_client\\addons\\cad\\ui\\_site\\cad-sidepanel.css","forge\\forge_client\\addons\\cad\\ui\\_site\\cad-shared.js","forge\\forge_client\\addons\\cad\\ui\\_site\\cad-sidepanel.js"]).catch(e=>console.error("[SIDEPANEL] Load error:",e))</script></body></html>
<!doctype html><html><head><meta charset="UTF-8"></head><body><div class="panel-header"><h3>CAD System</h3></div><div class="panel-content"><div id="cadStatusMessage" class="task-status-message"></div><div id="cadDangerAlert" class="cad-danger-alert is-hidden"></div><div id="cadRequestAlert" class="cad-warning-alert is-hidden"></div><div class="cad-tabs" role="tablist" aria-label="CAD Sections"><button id="tabContractsBtn" class="cad-tab is-active" type="button" data-tab="contracts">Contracts</button> <button id="tabRosterBtn" class="cad-tab" type="button" data-tab="roster">Roster</button> <button id="tabRequestsBtn" class="cad-tab" type="button" data-tab="requests">Requests</button> <button id="tabActivityBtn" class="cad-tab" type="button" data-tab="activity">Activity</button></div><div class="cad-tab-panels"><div id="contractsPanel" class="cad-section is-active" data-panel="contracts"><div class="cad-section-header">Contracts</div><div id="taskList" class="task-list"><div class="placeholder-message"><p>Loading contracts...</p></div></div></div><div id="rosterPanel" class="cad-section" data-panel="roster"><div class="cad-section-header">Roster</div><div id="rosterList" class="task-list"><div class="placeholder-message"><p>Loading roster...</p></div></div></div><div id="requestsPanel" class="cad-section" data-panel="requests"><div class="cad-section-header">Support Requests</div><div id="requestList" class="task-list"><div class="placeholder-message"><p>No support requests.</p></div></div></div><div id="activityPanel" class="cad-section" data-panel="activity"><div class="cad-section-header">Activity</div><div id="activityList" class="task-list"><div class="placeholder-message"><p>No recent activity.</p></div></div></div></div></div><div id="cadRequestModal" class="cad-modal is-hidden"><div class="cad-modal-backdrop"></div><div class="cad-modal-dialog" role="dialog" aria-modal="true" aria-labelledby="cadRequestModalTitle"><div class="cad-modal-header"><div><div class="cad-section-header">Support Request</div><h3 id="cadRequestModalTitle">Submit Request</h3></div><button id="cadRequestModalCloseBtn" class="cad-icon-btn" type="button" aria-label="Close support request form">x</button></div><div class="cad-modal-body"><div class="cad-modal-fields"><label class="cad-field"><span>Priority</span> <select id="cadRequestPrioritySelect" class="cad-select"><option value="routine">routine</option><option value="priority" selected>priority</option><option value="emergency">emergency</option></select></label><div id="cadRequestFields" class="cad-modal-fields"></div></div></div><div class="cad-modal-actions"><button id="cadRequestModalSaveBtn" type="button" class="task-accept-btn">Submit Request</button></div></div></div><script>window.MapLoader={loadCSS:e=>A3API.RequestFile(e).then(e=>{const d=document.createElement("style");d.textContent=e,document.head.appendChild(d)}),loadJS(path){return A3API.RequestFile(path).then(js=>{eval(js)})},loadAll(e){return e.reduce((e,d)=>e.then(()=>d.endsWith(".css")?this.loadCSS(d):d.endsWith(".js")?this.loadJS(d):Promise.resolve()),Promise.resolve())}},MapLoader.loadAll(["forge\\forge_client\\addons\\cad\\ui\\_site\\cad-common.css","forge\\forge_client\\addons\\cad\\ui\\_site\\cad-sidepanel.css","forge\\forge_client\\addons\\cad\\ui\\_site\\cad-shared.js","forge\\forge_client\\addons\\cad\\ui\\_site\\cad-sidepanel.js"]).catch(e=>console.error("[SIDEPANEL] Load error:",e))</script></body></html>

View File

@ -13,6 +13,14 @@
</header>
<div id="dispatcherStatusMessage" class="dispatch-status"></div>
<div
id="dispatcherDangerAlert"
class="dispatch-danger-alert is-hidden"
></div>
<div
id="dispatcherRequestAlert"
class="dispatch-warning-alert is-hidden"
></div>
<section class="dispatch-metrics">
<div class="metric-card">
@ -27,7 +35,11 @@
<span class="metric-label">Active Groups</span>
<strong id="metricActiveGroups">0</strong>
</div>
<div class="metric-card">
<div id="metricOpenRequestsCard" class="metric-card">
<span class="metric-label">Open Requests</span>
<strong id="metricOpenRequests">0</strong>
</div>
<div id="metricDangerGroupsCard" class="metric-card">
<span class="metric-label">Groups In Danger</span>
<strong id="metricDangerGroups">0</strong>
</div>
@ -37,6 +49,15 @@
<section class="dispatch-panel dispatch-panel-open">
<div class="dispatch-panel-header">
<h3>Available Contracts</h3>
<button
id="dispatcherCreateOrderBtn"
type="button"
class="dispatch-icon-btn"
aria-label="Create dispatch order"
title="Create dispatch order"
>
+
</button>
</div>
<div
id="dispatcherOpenContracts"
@ -63,7 +84,7 @@
<section class="dispatch-panel dispatch-panel-activity">
<div class="dispatch-panel-header">
<h3>Activity Feed</h3>
<h3>Requests & Activity</h3>
</div>
<div id="dispatcherActivity" class="dispatch-list"></div>
</section>
@ -146,6 +167,160 @@
</div>
</div>
</div>
<div id="dispatcherOrderModal" class="dispatch-modal is-hidden">
<div class="dispatch-modal-backdrop"></div>
<div
class="dispatch-modal-dialog"
role="dialog"
aria-modal="true"
aria-labelledby="dispatcherOrderModalTitle"
>
<div class="dispatch-modal-header">
<div>
<p class="dispatch-kicker">Dispatch Order</p>
<h3 id="dispatcherOrderModalTitle">
Create Support Order
</h3>
</div>
<button
id="dispatcherOrderModalCloseBtn"
class="dispatch-icon-btn"
type="button"
aria-label="Close dispatch order editor"
>
x
</button>
</div>
<div class="dispatch-modal-body">
<div class="dispatch-modal-fields">
<label class="dispatch-field">
<span>Assignee Group</span>
<select
id="dispatcherOrderAssigneeSelect"
class="dispatch-select"
></select>
</label>
<label class="dispatch-field">
<span>Target Group</span>
<select
id="dispatcherOrderTargetSelect"
class="dispatch-select"
></select>
</label>
<label class="dispatch-field">
<span>Priority</span>
<select
id="dispatcherOrderPrioritySelect"
class="dispatch-select"
>
<option value="routine">routine</option>
<option value="priority" selected>
priority
</option>
<option value="emergency">emergency</option>
</select>
</label>
<label class="dispatch-field">
<span>Order Note</span>
<textarea
id="dispatcherOrderNoteInput"
class="dispatch-textarea"
rows="4"
placeholder="Optional order note for the assigned group."
></textarea>
</label>
</div>
</div>
<div class="dispatch-modal-actions">
<button
id="dispatcherOrderModalSaveBtn"
type="button"
class="dispatch-btn"
>
Create Order
</button>
</div>
</div>
</div>
<div id="dispatcherRequestModal" class="dispatch-modal is-hidden">
<div class="dispatch-modal-backdrop"></div>
<div
class="dispatch-modal-dialog"
role="dialog"
aria-modal="true"
aria-labelledby="dispatcherRequestModalTitle"
>
<div class="dispatch-modal-header">
<div>
<p class="dispatch-kicker">Support Request</p>
<h3 id="dispatcherRequestModalTitle">
Request Details
</h3>
</div>
<button
id="dispatcherRequestModalCloseBtn"
class="dispatch-icon-btn"
type="button"
aria-label="Close support request details"
>
x
</button>
</div>
<div class="dispatch-modal-body">
<div class="dispatch-meta-grid">
<div>
<span class="metric-label">Title</span>
<strong id="dispatcherRequestTitle"
>Support Request</strong
>
</div>
<div>
<span class="metric-label">Priority</span>
<strong id="dispatcherRequestPriority"
>priority</strong
>
</div>
<div>
<span class="metric-label">Group</span>
<strong id="dispatcherRequestGroup"
>Unknown</strong
>
</div>
<div>
<span class="metric-label">Type</span>
<strong id="dispatcherRequestType"
>request</strong
>
</div>
</div>
<div class="dispatch-field">
<span>Summary</span>
<div
id="dispatcherRequestSummary"
class="dispatch-detail-block"
></div>
</div>
<div class="dispatch-field">
<span>Submitted Fields</span>
<div
id="dispatcherRequestFields"
class="dispatch-detail-list"
></div>
</div>
</div>
<div class="dispatch-modal-actions">
<button
id="dispatcherRequestModalDoneBtn"
type="button"
class="dispatch-btn"
>
Done
</button>
</div>
</div>
</div>
</div>
<script>

View File

@ -1,9 +1,11 @@
window.cadDispatcher = {
contracts: [],
requests: [],
groups: [],
activity: [],
session: {},
editingGroupId: "",
viewingRequestId: "",
statuses: [
"available",
"en_route",
@ -14,6 +16,12 @@ window.cadDispatcher = {
],
roles: ["infantry", "recon", "armor", "air", "logistics", "support"],
init() {
document
.getElementById("dispatcherCreateOrderBtn")
.addEventListener("click", () => {
this.openOrderModal();
});
document
.getElementById("dispatcherGroupModalCloseBtn")
.addEventListener("click", () => {
@ -32,12 +40,49 @@ window.cadDispatcher = {
this.closeGroupModal();
});
document
.getElementById("dispatcherOrderModalCloseBtn")
.addEventListener("click", () => {
this.closeOrderModal();
});
document
.getElementById("dispatcherOrderModalSaveBtn")
.addEventListener("click", () => {
this.createDispatchOrder();
});
document
.querySelector("#dispatcherOrderModal .dispatch-modal-backdrop")
.addEventListener("click", () => {
this.closeOrderModal();
});
document
.getElementById("dispatcherRequestModalCloseBtn")
.addEventListener("click", () => {
this.closeRequestModal();
});
document
.getElementById("dispatcherRequestModalDoneBtn")
.addEventListener("click", () => {
this.closeRequestModal();
});
document
.querySelector("#dispatcherRequestModal .dispatch-modal-backdrop")
.addEventListener("click", () => {
this.closeRequestModal();
});
window.mapUI.sendEvent("cad::dispatcher::ready", {});
},
receiveHydrate(payload) {
this.contracts = Array.isArray(payload.contracts)
? payload.contracts
: [];
this.requests = Array.isArray(payload.requests) ? payload.requests : [];
this.groups = Array.isArray(payload.groups) ? payload.groups : [];
this.activity = Array.isArray(payload.activity) ? payload.activity : [];
this.session =
@ -54,6 +99,8 @@ window.cadDispatcher = {
}
this.syncOpenModal();
this.syncOrderModal();
this.syncRequestModal();
this.render();
},
setStatus(message, type) {
@ -65,6 +112,288 @@ window.cadDispatcher = {
statusEl.textContent = message || "";
statusEl.dataset.type = type || "";
},
getDangerGroups() {
return this.groups.filter((group) => (group.status || "") === "danger");
},
getSupportAlertRequests() {
return this.requests.filter((request) =>
["medevac_9line", "fire_support", "air_support"].includes(
request.type || "",
),
);
},
buildSupportAlertMessage() {
const alertRequests = this.getSupportAlertRequests();
if (!alertRequests.length) {
return "";
}
const labels = alertRequests.map((request) => {
const groupLabel =
request.groupCallsign || request.groupId || "Unknown Group";
const typeLabel = this.getRequestTypeLabel(
request.type || "request",
);
return `${groupLabel} ${typeLabel}`;
});
return `Support request alert: ${labels.join(", ")}`;
},
getSortedGroups() {
return this.groups.slice().sort((left, right) => {
const leftDanger = (left.status || "") === "danger" ? 0 : 1;
const rightDanger = (right.status || "") === "danger" ? 0 : 1;
if (leftDanger !== rightDanger) {
return leftDanger - rightDanger;
}
const leftCallsign = left.callsign || left.groupId || "";
const rightCallsign = right.callsign || right.groupId || "";
return leftCallsign.localeCompare(rightCallsign);
});
},
isDispatchOrder(entry) {
return (
!!entry.isDispatchOrder || (entry.type || "") === "dispatch_order"
);
},
formatTypeLabel(entry) {
const typeLabel = (entry.type || "task").replaceAll("_", " ");
return this.isDispatchOrder(entry) ? "dispatch order" : typeLabel;
},
getRequestTypeLabel(typeID) {
switch (typeID) {
case "medevac_9line":
return "9-Line MEDEVAC";
case "ace_lace":
return "ACE/LACE";
case "fire_support":
return "Fire Support";
case "air_support":
return "Air Support";
case "logreq":
return "LOGREQ";
default:
return (typeID || "request").replaceAll("_", " ");
}
},
buildGroupOptions(selectedGroupID) {
return this.getSortedGroups()
.map((group) => {
const groupID = group.groupId || "";
return `<option value="${groupID}" ${groupID === selectedGroupID ? "selected" : ""}>${group.callsign || groupID}</option>`;
})
.join("");
},
updateDangerAlert() {
const alertEl = document.getElementById("dispatcherDangerAlert");
if (!alertEl) {
return;
}
const dangerGroups = this.getDangerGroups();
if (!dangerGroups.length) {
alertEl.textContent = "";
alertEl.classList.add("is-hidden");
return;
}
const callsigns = dangerGroups.map(
(group) => group.callsign || group.groupId || "Unknown Group",
);
alertEl.textContent = `Danger alert active: ${callsigns.join(", ")}`;
alertEl.classList.remove("is-hidden");
},
updateRequestAlert() {
const alertEl = document.getElementById("dispatcherRequestAlert");
if (!alertEl) {
return;
}
const alertMessage = this.buildSupportAlertMessage();
if (!alertMessage) {
alertEl.textContent = "";
alertEl.classList.add("is-hidden");
return;
}
alertEl.textContent = alertMessage;
alertEl.classList.remove("is-hidden");
},
openOrderModal() {
this.populateOrderModal();
document
.getElementById("dispatcherOrderModal")
.classList.remove("is-hidden");
},
closeOrderModal() {
document.getElementById("dispatcherOrderNoteInput").value = "";
document.getElementById("dispatcherOrderPrioritySelect").value =
"priority";
document
.getElementById("dispatcherOrderModal")
.classList.add("is-hidden");
},
openRequestModal(requestID) {
const request = this.requests.find(
(entry) => entry.requestId === requestID,
);
if (!request) {
return;
}
this.viewingRequestId = requestID;
this.populateRequestModal(request);
document
.getElementById("dispatcherRequestModal")
.classList.remove("is-hidden");
},
closeRequestModal() {
this.viewingRequestId = "";
document
.getElementById("dispatcherRequestModal")
.classList.add("is-hidden");
},
syncRequestModal() {
if (!this.viewingRequestId) {
return;
}
const request = this.requests.find(
(entry) => entry.requestId === this.viewingRequestId,
);
if (!request) {
this.closeRequestModal();
return;
}
this.populateRequestModal(request);
},
formatRequestFieldLabel(fieldID) {
return (fieldID || "field")
.replaceAll("_", " ")
.replace(/\b\w/g, (character) => character.toUpperCase());
},
formatRequestFieldValue(value) {
if (Array.isArray(value)) {
return value.join(", ");
}
if (value && typeof value === "object") {
return JSON.stringify(value);
}
const text = String(value ?? "").trim();
return text || "Not provided";
},
populateRequestModal(request) {
const fields =
request.fields && typeof request.fields === "object"
? Object.entries(request.fields)
: [];
const fieldsHTML = fields.length
? fields
.map(
([fieldID, value]) => `
<div class="dispatch-detail-row">
<span class="dispatch-detail-label">${this.formatRequestFieldLabel(fieldID)}</span>
<span class="dispatch-detail-value">${this.formatRequestFieldValue(value)}</span>
</div>
`,
)
.join("")
: '<div class="placeholder-message"><p>No submitted fields.</p></div>';
document.getElementById("dispatcherRequestTitle").textContent =
request.title || request.requestId || "Support Request";
document.getElementById("dispatcherRequestPriority").textContent = (
request.priority || "priority"
).replaceAll("_", " ");
document.getElementById("dispatcherRequestGroup").textContent =
request.groupCallsign || request.groupId || "Unknown";
document.getElementById("dispatcherRequestType").textContent =
this.getRequestTypeLabel(request.type || "request");
document.getElementById("dispatcherRequestSummary").textContent =
request.summary || "No summary provided.";
document.getElementById("dispatcherRequestFields").innerHTML =
fieldsHTML;
},
populateOrderModal(selectedAssigneeID, selectedTargetID) {
const sortedGroups = this.getSortedGroups();
const assigneeSelect = document.getElementById(
"dispatcherOrderAssigneeSelect",
);
const targetSelect = document.getElementById(
"dispatcherOrderTargetSelect",
);
if (!assigneeSelect || !targetSelect) {
return;
}
const fallbackAssignee =
selectedAssigneeID || sortedGroups[0]?.groupId || "";
const fallbackTarget =
selectedTargetID ||
sortedGroups.find(
(group) => (group.groupId || "") !== fallbackAssignee,
)?.groupId ||
sortedGroups[0]?.groupId ||
"";
assigneeSelect.innerHTML = this.buildGroupOptions(fallbackAssignee);
targetSelect.innerHTML = this.buildGroupOptions(fallbackTarget);
},
syncOrderModal() {
const modalEl = document.getElementById("dispatcherOrderModal");
if (!modalEl || modalEl.classList.contains("is-hidden")) {
return;
}
this.populateOrderModal(
document.getElementById("dispatcherOrderAssigneeSelect")?.value ||
"",
document.getElementById("dispatcherOrderTargetSelect")?.value || "",
);
},
createDispatchOrder() {
const assigneeGroupID = document.getElementById(
"dispatcherOrderAssigneeSelect",
).value;
const targetGroupID = document.getElementById(
"dispatcherOrderTargetSelect",
).value;
const priority = document.getElementById(
"dispatcherOrderPrioritySelect",
).value;
const note = document.getElementById("dispatcherOrderNoteInput").value;
if (!assigneeGroupID || !targetGroupID) {
this.setStatus(
"Select both an assignee and a target group.",
"error",
);
return;
}
if (assigneeGroupID === targetGroupID) {
this.setStatus(
"Assignee and target groups must be different.",
"error",
);
return;
}
this.setStatus("Creating dispatch order...", "info");
window.mapUI.sendEvent("cad::dispatchOrder::create", {
assigneeGroupID: assigneeGroupID,
targetGroupID: targetGroupID,
note: note.trim(),
priority: priority,
});
this.closeOrderModal();
},
assignTask(taskID) {
const selector = document.getElementById(
`dispatcher-assign-group-${taskID}`,
@ -168,32 +497,39 @@ window.cadDispatcher = {
const statusValue = document.getElementById(
"dispatcherModalStatusSelect",
).value;
let hasChanges = false;
if (roleValue && roleValue !== (group.role || "")) {
hasChanges = true;
this.setStatus("Updating group role...", "info");
window.mapUI.sendEvent("cad::groups::role", {
groupID: this.editingGroupId,
role: roleValue,
});
}
if (statusValue && statusValue !== (group.status || "")) {
hasChanges = true;
this.setStatus("Updating group status...", "info");
window.mapUI.sendEvent("cad::groups::status", {
groupID: this.editingGroupId,
status: statusValue,
});
}
const nextRole =
roleValue && roleValue !== (group.role || "") ? roleValue : "";
const nextStatus =
statusValue && statusValue !== (group.status || "")
? statusValue
: "";
const hasChanges = nextRole || nextStatus;
if (!hasChanges) {
this.setStatus("No group changes to save.", "info");
this.closeGroupModal();
return;
}
this.setStatus("Updating group profile...", "info");
window.mapUI.sendEvent("cad::groups::profile", {
groupID: this.editingGroupId,
role: nextRole,
status: nextStatus,
});
this.closeGroupModal();
},
closeDispatchOrder(taskID) {
if (!taskID) {
return;
}
this.setStatus("Closing dispatch order...", "info");
window.mapUI.sendEvent("cad::dispatchOrder::close", {
taskID: taskID,
});
},
buildGroupEditorButton(groupID) {
return `
@ -208,6 +544,38 @@ window.cadDispatcher = {
</button>
`;
},
buildCloseOrderButton(taskID) {
return `
<button
type="button"
class="dispatch-btn dispatch-btn-secondary"
onclick="window.cadDispatcher.closeDispatchOrder('${taskID}')"
>
Close
</button>
`;
},
buildCloseRequestButton(requestID) {
return `
<button
type="button"
class="dispatch-btn dispatch-btn-secondary"
onclick="event.stopPropagation(); window.cadDispatcher.closeSupportRequest('${requestID}')"
>
Close
</button>
`;
},
closeSupportRequest(requestID) {
if (!requestID) {
return;
}
this.setStatus("Closing support request...", "info");
window.mapUI.sendEvent("cad::supportRequest::close", {
requestID: requestID,
});
},
renderMetrics() {
const assignedContracts = this.contracts.filter(
(entry) => (entry.assignmentState || "unassigned") !== "unassigned",
@ -215,6 +583,8 @@ window.cadDispatcher = {
const openContracts = this.contracts.filter(
(entry) => (entry.assignmentState || "unassigned") === "unassigned",
);
const openRequests = this.requests.length;
const supportAlertRequests = this.getSupportAlertRequests();
const dangerGroups = this.groups.filter(
(group) => (group.status || "") === "danger",
);
@ -225,8 +595,30 @@ window.cadDispatcher = {
assignedContracts.length;
document.getElementById("metricActiveGroups").textContent =
this.groups.length;
document.getElementById("metricOpenRequests").textContent =
openRequests;
document.getElementById("metricDangerGroups").textContent =
dangerGroups.length;
const dangerMetricCard = document.getElementById(
"metricDangerGroupsCard",
);
if (dangerMetricCard) {
dangerMetricCard.classList.toggle(
"is-danger",
dangerGroups.length > 0,
);
}
const requestMetricCard = document.getElementById(
"metricOpenRequestsCard",
);
if (requestMetricCard) {
requestMetricCard.classList.toggle(
"is-warning",
supportAlertRequests.length > 0,
);
}
},
renderOpenContracts() {
const container = document.getElementById("dispatcherOpenContracts");
@ -240,12 +632,7 @@ window.cadDispatcher = {
return;
}
const groupOptions = this.groups
.map(
(group) =>
`<option value="${group.groupId}">${group.callsign || group.groupId}</option>`,
)
.join("");
const groupOptions = this.buildGroupOptions("");
container.innerHTML = openContracts
.map((task) => {
@ -253,17 +640,24 @@ window.cadDispatcher = {
const position = Array.isArray(task.position)
? task.position
: [0, 0, 0];
const targetGroup = this.groups.find(
(group) => group.groupId === (task.targetGroupId || ""),
);
return `
<article class="dispatch-card">
<header class="dispatch-card-header">
<strong>${task.title || taskId}</strong>
<span class="dispatch-badge">${task.type || "task"}</span>
<span class="dispatch-badge">${this.formatTypeLabel(task)}</span>
</header>
<p class="dispatch-description">${task.description || ""}</p>
<div class="dispatch-meta">
<span>Unassigned</span>
<span>X: ${Math.round(position[0] || 0)} Y: ${Math.round(position[1] || 0)}</span>
<span>${window.mapUI.formatPosition(position)}</span>
</div>
<div class="dispatch-meta">
<span>Target: ${targetGroup ? targetGroup.callsign : task.targetGroupCallsign || "None"}</span>
<span>Priority: ${(task.priority || "priority").replaceAll("_", " ")}</span>
</div>
<div class="dispatch-actions">
<select id="dispatcher-assign-group-${taskId}" class="dispatch-select">
@ -297,6 +691,10 @@ window.cadDispatcher = {
const assignedGroup = this.groups.find(
(group) => group.groupId === (task.assignedGroupId || ""),
);
const targetGroup = this.groups.find(
(group) => group.groupId === (task.targetGroupId || ""),
);
const isDispatchOrder = this.isDispatchOrder(task);
return `
<article class="dispatch-card">
@ -307,8 +705,13 @@ window.cadDispatcher = {
<p class="dispatch-description">${task.description || ""}</p>
<div class="dispatch-meta">
<span>Group: ${assignedGroup ? assignedGroup.callsign : task.assignedGroupId || "Unknown"}</span>
<span>Type: ${task.type || "task"}</span>
<span>Type: ${this.formatTypeLabel(task)}</span>
</div>
<div class="dispatch-meta">
<span>Target: ${targetGroup ? targetGroup.callsign : task.targetGroupCallsign || "None"}</span>
<span>Priority: ${(task.priority || "priority").replaceAll("_", " ")}</span>
</div>
${isDispatchOrder ? `<div class="dispatch-actions dispatch-actions-split">${this.buildCloseOrderButton(taskId)}</div>` : ""}
</article>
`;
})
@ -322,14 +725,16 @@ window.cadDispatcher = {
return;
}
container.innerHTML = this.groups
container.innerHTML = this.getSortedGroups()
.map((group) => {
const isDanger = (group.status || "") === "danger";
return `
<article class="dispatch-card dispatch-card-group">
<article class="dispatch-card dispatch-card-group ${isDanger ? "is-danger" : ""}">
<header class="dispatch-card-header">
<div class="dispatch-card-header-main">
<strong>${group.callsign || group.groupId}</strong>
<span class="dispatch-badge">${group.role || "group"}</span>
${isDanger ? '<span class="dispatch-alert-badge">Danger</span>' : ""}
</div>
<div class="dispatch-card-header-actions">
${this.buildGroupEditorButton(group.groupId)}
@ -350,30 +755,62 @@ window.cadDispatcher = {
},
renderActivity() {
const container = document.getElementById("dispatcherActivity");
if (!this.activity.length) {
container.innerHTML =
'<div class="placeholder-message"><p>No recent activity.</p></div>';
return;
}
const requestsHTML = this.requests.length
? this.requests
.map(
(request) => `
<article class="dispatch-card dispatch-card-interactive ${["medevac_9line", "fire_support", "air_support"].includes(request.type || "") ? "is-warning" : ""}" onclick="window.cadDispatcher.openRequestModal('${request.requestId || ""}')">
<header class="dispatch-card-header">
<strong>${request.title || request.requestId || "Support Request"}</strong>
<span class="dispatch-badge">${(request.priority || "priority").replaceAll("_", " ")}</span>
</header>
<p class="dispatch-description">${request.summary || ""}</p>
<div class="dispatch-meta">
<span>Group: ${request.groupCallsign || request.groupId || "Unknown"}</span>
<span>${this.getRequestTypeLabel(request.type || "request")}</span>
</div>
<div class="dispatch-actions dispatch-actions-split">
${this.buildCloseRequestButton(request.requestId || "")}
</div>
</article>
`,
)
.join("")
: '<div class="placeholder-message"><p>No active support requests.</p></div>';
container.innerHTML = this.activity
.slice()
.reverse()
.slice(0, 12)
.map(
(entry) => `
<article class="dispatch-card">
<header class="dispatch-card-header">
<strong>${entry.type || "activity"}</strong>
<span class="dispatch-badge">${Math.round(entry.timestamp || 0)}s</span>
</header>
<p class="dispatch-description">${entry.message || ""}</p>
</article>
`,
)
.join("");
const activityHTML = this.activity.length
? this.activity
.slice()
.reverse()
.slice(0, 8)
.map(
(entry) => `
<article class="dispatch-card">
<header class="dispatch-card-header">
<strong>${entry.type || "activity"}</strong>
<span class="dispatch-badge">${Math.round(entry.timestamp || 0)}s</span>
</header>
<p class="dispatch-description">${entry.message || ""}</p>
</article>
`,
)
.join("")
: '<div class="placeholder-message"><p>No recent activity.</p></div>';
container.innerHTML = `
<div class="dispatch-inline-section">
<div class="dispatch-inline-header">Support Requests</div>
${requestsHTML}
</div>
<div class="dispatch-inline-section">
<div class="dispatch-inline-header">Recent Activity</div>
${activityHTML}
</div>
`;
},
render() {
this.updateDangerAlert();
this.updateRequestAlert();
this.renderMetrics();
this.renderOpenContracts();
this.renderAssignedContracts();

View File

@ -9,17 +9,22 @@ window.mapUIState = {
};
window.mapUI = {
formatGridCoordinate(value) {
return Math.round(Number(value) || 0)
.toString()
.padStart(4, "0");
},
formatPosition(position) {
const safePosition = Array.isArray(position) ? position : [0, 0, 0];
return `X: ${this.formatGridCoordinate(safePosition[0])} Y: ${this.formatGridCoordinate(safePosition[1])}`;
},
sendEvent(event, data) {
A3API.SendAlert(JSON.stringify({ event: event, data: data }));
},
updateCoordinates(x, y) {
const coordDisplay = document.getElementById("coordsDisplay");
if (coordDisplay) {
coordDisplay.textContent = `X: ${Math.round(x)
.toString()
.padStart(4, "0")} Y: ${Math.round(y)
.toString()
.padStart(4, "0")}`;
coordDisplay.textContent = this.formatPosition([x, y, 0]);
}
},
updateScale(scale) {

View File

@ -9,6 +9,8 @@
</div>
<div class="panel-content">
<div id="cadStatusMessage" class="task-status-message"></div>
<div id="cadDangerAlert" class="cad-danger-alert is-hidden"></div>
<div id="cadRequestAlert" class="cad-warning-alert is-hidden"></div>
<div class="cad-tabs" role="tablist" aria-label="CAD Sections">
<button
id="tabContractsBtn"
@ -26,6 +28,14 @@
>
Roster
</button>
<button
id="tabRequestsBtn"
class="cad-tab"
type="button"
data-tab="requests"
>
Requests
</button>
<button
id="tabActivityBtn"
class="cad-tab"
@ -56,6 +66,18 @@
</div>
</div>
</div>
<div
id="requestsPanel"
class="cad-section"
data-panel="requests"
>
<div class="cad-section-header">Support Requests</div>
<div id="requestList" class="task-list">
<div class="placeholder-message">
<p>No support requests.</p>
</div>
</div>
</div>
<div
id="activityPanel"
class="cad-section"
@ -71,6 +93,61 @@
</div>
</div>
<div id="cadRequestModal" class="cad-modal is-hidden">
<div class="cad-modal-backdrop"></div>
<div
class="cad-modal-dialog"
role="dialog"
aria-modal="true"
aria-labelledby="cadRequestModalTitle"
>
<div class="cad-modal-header">
<div>
<div class="cad-section-header">Support Request</div>
<h3 id="cadRequestModalTitle">Submit Request</h3>
</div>
<button
id="cadRequestModalCloseBtn"
class="cad-icon-btn"
type="button"
aria-label="Close support request form"
>
x
</button>
</div>
<div class="cad-modal-body">
<div class="cad-modal-fields">
<label class="cad-field">
<span>Priority</span>
<select
id="cadRequestPrioritySelect"
class="cad-select"
>
<option value="routine">routine</option>
<option value="priority" selected>
priority
</option>
<option value="emergency">emergency</option>
</select>
</label>
<div
id="cadRequestFields"
class="cad-modal-fields"
></div>
</div>
</div>
<div class="cad-modal-actions">
<button
id="cadRequestModalSaveBtn"
type="button"
class="task-accept-btn"
>
Submit Request
</button>
</div>
</div>
</div>
<script>
window.MapLoader = {
loadCSS(path) {

File diff suppressed because it is too large Load Diff

View File

@ -81,9 +81,49 @@ body {
color: #ff8a80;
}
.dispatch-danger-alert {
padding: 10px 12px;
border: 1px solid rgba(255, 107, 107, 0.38);
background: linear-gradient(
90deg,
rgba(92, 18, 18, 0.94),
rgba(128, 29, 29, 0.82)
);
color: #ffd4cf;
font-size: 12px;
font-weight: 700;
letter-spacing: 0.06em;
text-transform: uppercase;
animation: cad-danger-pulse 1.35s ease-in-out infinite;
}
.dispatch-danger-alert.is-hidden {
display: none;
}
.dispatch-warning-alert {
padding: 10px 12px;
border: 1px solid rgba(246, 198, 84, 0.42);
background: linear-gradient(
90deg,
rgba(89, 64, 12, 0.94),
rgba(125, 92, 18, 0.84)
);
color: #ffe9b2;
font-size: 12px;
font-weight: 700;
letter-spacing: 0.06em;
text-transform: uppercase;
animation: cad-warning-pulse 1.35s ease-in-out infinite;
}
.dispatch-warning-alert.is-hidden {
display: none;
}
.dispatch-metrics {
display: grid;
grid-template-columns: repeat(4, minmax(0, 1fr));
grid-template-columns: repeat(5, minmax(0, 1fr));
gap: 12px;
}
@ -107,6 +147,28 @@ body {
font-weight: 700;
}
.metric-card.is-danger {
border-color: rgba(255, 107, 107, 0.34);
background: linear-gradient(
180deg,
rgba(74, 17, 17, 0.86),
rgba(22, 13, 16, 0.92)
);
box-shadow: inset 0 0 0 1px rgba(255, 107, 107, 0.12);
animation: cad-danger-pulse 1.35s ease-in-out infinite;
}
.metric-card.is-warning {
border-color: rgba(246, 198, 84, 0.34);
background: linear-gradient(
180deg,
rgba(92, 65, 14, 0.86),
rgba(29, 22, 11, 0.92)
);
box-shadow: inset 0 0 0 1px rgba(246, 198, 84, 0.12);
animation: cad-warning-pulse 1.35s ease-in-out infinite;
}
.dispatch-grid {
flex: 1;
display: grid;
@ -166,12 +228,35 @@ body {
padding: 12px;
}
.dispatch-inline-section {
display: flex;
flex-direction: column;
gap: 10px;
}
.dispatch-inline-header {
color: var(--accent);
font-size: 11px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.08em;
}
.dispatch-card {
padding: 12px;
border: 1px solid rgba(255, 255, 255, 0.06);
background: rgba(19, 26, 34, 0.72);
}
.dispatch-card-interactive {
cursor: pointer;
}
.dispatch-card-interactive:hover {
border-color: rgba(91, 187, 255, 0.2);
background: rgba(23, 31, 40, 0.82);
}
.dispatch-card-header,
.dispatch-meta {
display: flex;
@ -218,6 +303,17 @@ body {
text-transform: uppercase;
}
.dispatch-alert-badge {
padding: 3px 7px;
border: 1px solid rgba(255, 107, 107, 0.44);
background: rgba(95, 23, 23, 0.88);
color: #ffd8d1;
font-size: 11px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.08em;
}
.dispatch-icon-btn {
width: 32px;
height: 32px;
@ -238,6 +334,38 @@ body {
gap: 8px;
}
.dispatch-card.is-danger {
border-color: rgba(255, 107, 107, 0.34);
background: linear-gradient(
180deg,
rgba(69, 20, 22, 0.78),
rgba(28, 17, 21, 0.92)
);
box-shadow: inset 0 0 0 1px rgba(255, 107, 107, 0.1);
animation: cad-danger-pulse 1.35s ease-in-out infinite;
}
.dispatch-card.is-danger .dispatch-meta,
.dispatch-card.is-danger .dispatch-description {
color: rgba(255, 232, 228, 0.82);
}
.dispatch-card.is-warning {
border-color: rgba(246, 198, 84, 0.34);
background: linear-gradient(
180deg,
rgba(86, 64, 17, 0.78),
rgba(34, 27, 16, 0.92)
);
box-shadow: inset 0 0 0 1px rgba(246, 198, 84, 0.1);
animation: cad-warning-pulse 1.35s ease-in-out infinite;
}
.dispatch-card.is-warning .dispatch-meta,
.dispatch-card.is-warning .dispatch-description {
color: rgba(255, 243, 214, 0.84);
}
.dispatch-actions-split {
margin-top: 10px;
}
@ -247,6 +375,18 @@ body {
padding: 9px 10px;
}
.dispatch-textarea {
width: 100%;
min-height: 92px;
padding: 10px 12px;
border: 1px solid rgba(255, 255, 255, 0.12);
background: rgba(24, 31, 40, 0.92);
color: var(--text);
font: inherit;
resize: vertical;
box-sizing: border-box;
}
.placeholder-message {
padding: 18px;
text-align: center;
@ -257,6 +397,11 @@ body {
position: fixed;
inset: 0;
z-index: 30;
display: flex;
align-items: center;
justify-content: center;
padding: 32px 24px;
box-sizing: border-box;
}
.dispatch-modal.is-hidden {
@ -271,8 +416,11 @@ body {
.dispatch-modal-dialog {
position: relative;
width: min(480px, calc(100% - 48px));
margin: 72px auto 0;
display: flex;
flex-direction: column;
width: min(560px, calc(100% - 48px));
max-height: calc(100vh - 64px);
margin: 0;
border: 1px solid rgba(255, 255, 255, 0.12);
background: rgba(11, 17, 24, 0.98);
box-shadow: 0 24px 64px rgba(0, 0, 0, 0.42);
@ -298,7 +446,10 @@ body {
}
.dispatch-modal-body {
flex: 1;
min-height: 0;
padding: 16px;
overflow: auto;
}
.dispatch-meta-grid {
@ -337,3 +488,75 @@ body {
justify-content: flex-end;
border-top: 1px solid rgba(255, 255, 255, 0.08);
}
.dispatch-detail-block,
.dispatch-detail-list {
border: 1px solid rgba(255, 255, 255, 0.08);
background: rgba(19, 26, 34, 0.72);
}
.dispatch-detail-block {
padding: 12px;
color: rgba(241, 246, 251, 0.82);
line-height: 1.45;
white-space: pre-wrap;
}
.dispatch-detail-list {
display: grid;
gap: 1px;
overflow: hidden;
}
.dispatch-detail-row {
display: grid;
grid-template-columns: minmax(0, 180px) minmax(0, 1fr);
gap: 12px;
padding: 10px 12px;
background: rgba(14, 20, 28, 0.92);
}
.dispatch-detail-label {
color: rgba(233, 241, 248, 0.64);
font-size: 12px;
font-weight: 650;
text-transform: uppercase;
letter-spacing: 0.06em;
}
.dispatch-detail-value {
color: rgba(241, 246, 251, 0.84);
line-height: 1.4;
word-break: break-word;
white-space: pre-wrap;
}
@keyframes cad-danger-pulse {
0%,
100% {
box-shadow:
inset 0 0 0 1px rgba(255, 107, 107, 0.08),
0 0 0 rgba(255, 107, 107, 0);
}
50% {
box-shadow:
inset 0 0 0 1px rgba(255, 141, 141, 0.22),
0 0 18px rgba(255, 107, 107, 0.16);
}
}
@keyframes cad-warning-pulse {
0%,
100% {
box-shadow:
inset 0 0 0 1px rgba(246, 198, 84, 0.08),
0 0 0 rgba(246, 198, 84, 0);
}
50% {
box-shadow:
inset 0 0 0 1px rgba(251, 212, 118, 0.22),
0 0 18px rgba(246, 198, 84, 0.16);
}
}

View File

@ -57,19 +57,29 @@ body {
.cad-tabs {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 6px;
grid-template-columns: repeat(4, 1fr);
gap: 5px;
margin-bottom: 12px;
}
.cad-tabs.is-two-col {
grid-template-columns: repeat(2, 1fr);
}
.cad-tabs.is-three-col {
grid-template-columns: repeat(3, 1fr);
}
.cad-tab {
padding: 8px 10px;
min-width: 0;
padding: 8px 7px;
border: 1px solid rgba(255, 255, 255, 0.14);
background: rgba(20, 27, 33, 0.88);
color: rgba(243, 246, 249, 0.78);
text-transform: uppercase;
letter-spacing: 0.08em;
font-size: 11px;
font-size: 10px;
white-space: nowrap;
cursor: pointer;
}
@ -146,12 +156,162 @@ body {
color: #ff8a80;
}
.cad-modal {
position: fixed;
inset: 0;
z-index: 40;
}
.cad-modal.is-hidden {
display: none;
}
.cad-modal-backdrop {
position: absolute;
inset: 0;
background: rgba(4, 8, 12, 0.76);
}
.cad-modal-dialog {
position: relative;
width: min(480px, calc(100% - 28px));
margin: 32px auto 0;
border: 1px solid rgba(255, 255, 255, 0.12);
background: rgba(11, 17, 24, 0.98);
box-shadow: 0 24px 64px rgba(0, 0, 0, 0.42);
}
.cad-modal-header,
.cad-modal-actions {
display: flex;
align-items: center;
justify-content: space-between;
gap: 12px;
padding: 12px 14px;
}
.cad-modal-header {
border-bottom: 1px solid rgba(255, 255, 255, 0.08);
}
.cad-modal-header h3 {
margin: 4px 0 0;
font-size: 18px;
font-weight: 650;
}
.cad-modal-body {
padding: 14px;
max-height: 62vh;
overflow: auto;
}
.cad-modal-fields {
display: grid;
gap: 10px;
}
.cad-field {
display: grid;
gap: 6px;
}
.cad-field span {
font-size: 11px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.08em;
color: rgba(233, 241, 248, 0.7);
}
.cad-input,
.cad-textarea {
width: 100%;
padding: 8px 10px;
border: 1px solid rgba(255, 255, 255, 0.2);
background: rgba(30, 37, 43, 0.9);
color: #f3f6f9;
box-sizing: border-box;
font: inherit;
}
.cad-textarea {
min-height: 74px;
resize: vertical;
}
.cad-icon-btn {
width: 30px;
height: 30px;
padding: 0;
border: 1px solid rgba(255, 255, 255, 0.14);
background: rgba(24, 31, 40, 0.92);
color: var(--text);
cursor: pointer;
}
.cad-modal-actions {
justify-content: flex-end;
border-top: 1px solid rgba(255, 255, 255, 0.08);
}
.cad-danger-alert {
margin-bottom: 10px;
padding: 8px 10px;
border: 1px solid rgba(255, 107, 107, 0.36);
background: linear-gradient(
90deg,
rgba(92, 18, 18, 0.94),
rgba(128, 29, 29, 0.82)
);
color: #ffd4cf;
font-size: 11px;
font-weight: 700;
letter-spacing: 0.06em;
text-transform: uppercase;
animation: cad-danger-pulse 1.35s ease-in-out infinite;
}
.cad-danger-alert.is-hidden {
display: none;
}
.cad-warning-alert {
margin-bottom: 10px;
padding: 8px 10px;
border: 1px solid rgba(246, 198, 84, 0.4);
background: linear-gradient(
90deg,
rgba(89, 64, 12, 0.94),
rgba(125, 92, 18, 0.84)
);
color: #ffe9b2;
font-size: 11px;
font-weight: 700;
letter-spacing: 0.06em;
text-transform: uppercase;
animation: cad-warning-pulse 1.35s ease-in-out infinite;
}
.cad-warning-alert.is-hidden {
display: none;
}
.task-list {
display: flex;
flex-direction: column;
gap: 10px;
}
.cad-request-actions {
display: grid;
gap: 8px;
}
.cad-request-btn {
text-align: left;
}
.task-action-stack,
.task-action-row {
display: flex;
@ -169,6 +329,18 @@ body {
background: rgba(12, 16, 20, 0.62);
}
.task-card.is-danger,
.roster-summary-card.is-danger {
border-color: rgba(255, 107, 107, 0.34);
background: linear-gradient(
180deg,
rgba(69, 20, 22, 0.78),
rgba(28, 17, 21, 0.92)
);
box-shadow: inset 0 0 0 1px rgba(255, 107, 107, 0.1);
animation: cad-danger-pulse 1.35s ease-in-out infinite;
}
.task-card-header {
display: flex;
justify-content: space-between;
@ -207,10 +379,137 @@ body {
background: rgba(16, 23, 29, 0.82);
}
.task-alert-badge {
display: inline-flex;
align-items: center;
padding: 2px 8px;
border: 1px solid rgba(255, 107, 107, 0.44);
background: rgba(95, 23, 23, 0.88);
color: #ffd8d1;
font-size: 10px;
font-weight: 700;
letter-spacing: 0.08em;
text-transform: uppercase;
}
.roster-member-card {
background: rgba(12, 16, 20, 0.74);
}
.dispatch-map-group-card {
width: 100%;
text-align: left;
appearance: none;
-webkit-appearance: none;
border-radius: 0;
color: var(--text);
font: inherit;
cursor: pointer;
transition:
border-color 120ms ease,
background 120ms ease,
transform 120ms ease;
}
.dispatch-map-group-card strong {
color: var(--text);
}
.dispatch-map-group-card .task-type {
color: var(--accent);
opacity: 0.9;
}
.dispatch-map-group-card .task-meta {
color: var(--muted);
opacity: 1;
}
.dispatch-map-group-card:hover {
border-color: rgba(91, 187, 255, 0.26);
background: rgba(18, 29, 38, 0.9);
transform: translateX(-2px);
}
.dispatch-map-group-card.is-selected {
border-color: rgba(91, 187, 255, 0.52);
background: rgba(15, 40, 58, 0.92);
box-shadow: inset 0 0 0 1px rgba(91, 187, 255, 0.18);
}
.dispatch-map-group-card.is-danger:not(.is-selected) {
border-color: rgba(255, 107, 107, 0.34);
background: linear-gradient(
180deg,
rgba(69, 20, 22, 0.78),
rgba(28, 17, 21, 0.92)
);
}
.dispatch-map-group-card.is-danger .task-meta,
.roster-summary-card.is-danger .task-meta {
color: rgba(255, 232, 228, 0.82);
}
.dispatch-map-card {
width: 100%;
text-align: left;
appearance: none;
-webkit-appearance: none;
border-radius: 0;
color: var(--text);
font: inherit;
cursor: pointer;
transition:
border-color 120ms ease,
background 120ms ease,
transform 120ms ease;
}
.dispatch-map-card strong {
color: var(--text);
}
.dispatch-map-card .task-type {
color: var(--accent);
opacity: 0.9;
}
.dispatch-map-card .task-description {
color: var(--muted);
}
.dispatch-map-card .task-meta {
color: var(--muted);
opacity: 1;
}
.dispatch-map-card:hover {
border-color: rgba(91, 187, 255, 0.26);
background: rgba(18, 29, 38, 0.9);
transform: translateX(-2px);
}
.dispatch-map-card.is-selected {
border-color: rgba(91, 187, 255, 0.52);
background: rgba(15, 40, 58, 0.92);
box-shadow: inset 0 0 0 1px rgba(91, 187, 255, 0.18);
}
.dispatch-map-card.is-warning:not(.is-selected) {
border-color: rgba(246, 198, 84, 0.34);
background: linear-gradient(
180deg,
rgba(86, 64, 17, 0.78),
rgba(34, 27, 16, 0.92)
);
}
.dispatch-map-card.is-warning .task-meta,
.dispatch-map-card.is-warning .task-description {
color: rgba(255, 243, 214, 0.84);
}
.roster-leader-badge {
display: inline-flex;
align-items: center;
@ -223,3 +522,33 @@ body {
letter-spacing: 0.06em;
text-transform: uppercase;
}
@keyframes cad-danger-pulse {
0%,
100% {
box-shadow:
inset 0 0 0 1px rgba(255, 107, 107, 0.08),
0 0 0 rgba(255, 107, 107, 0);
}
50% {
box-shadow:
inset 0 0 0 1px rgba(255, 141, 141, 0.22),
0 0 14px rgba(255, 107, 107, 0.14);
}
}
@keyframes cad-warning-pulse {
0%,
100% {
box-shadow:
inset 0 0 0 1px rgba(246, 198, 84, 0.08),
0 0 0 rgba(246, 198, 84, 0);
}
50% {
box-shadow:
inset 0 0 0 1px rgba(251, 212, 118, 0.22),
0 0 18px rgba(246, 198, 84, 0.16);
}
}

View File

@ -71,11 +71,7 @@ window.cadTopbar = {
const position = Array.isArray(group?.position)
? group.position
: [0, 0, 0];
return `X: ${Math.round(position[0] || 0)
.toString()
.padStart(4, "0")} Y: ${Math.round(position[1] || 0)
.toString()
.padStart(4, "0")}`;
return window.mapUI.formatPosition(position);
},
receiveState(payload) {
this.session =

View File

@ -3,3 +3,5 @@ PREP(initAssignmentRepository);
PREP(initCadStore);
PREP(initGroupRepository);
PREP(initPermissionService);
PREP(initPersistenceService);
PREP(initRequestRepository);

View File

@ -9,11 +9,7 @@ call FUNC(initCadStore);
[QGVAR(requestHydrateCad), {
params [["_uid", "", [""]]];
if (_uid isEqualTo "") exitWith {
["WARNING", "CAD hydrate request received with empty UID."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
private _player = GVAR(CadStore) call ["resolveRequestPlayer", [_uid, "CAD hydrate request received with empty UID."]];
if (_player isEqualTo objNull) exitWith {};
private _payload = GVAR(CadStore) call ["buildHydratePayload", [_uid]];
@ -28,74 +24,195 @@ call FUNC(initCadStore);
["_note", "", [""]]
];
if (_uid isEqualTo "" || { _taskID isEqualTo "" } || { _groupID isEqualTo "" }) exitWith {
if (_taskID isEqualTo "" || { _groupID isEqualTo "" }) exitWith {
["WARNING", "Invalid CAD task assignment payload."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
if (_player isEqualTo objNull) exitWith {};
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD task assignment payload.",
CRPC(cad,responseCadAssignment),
"assignTaskToGroup",
[_uid, _taskID, _groupID, _note],
true,
false
]];
}] call CFUNC(addEventHandler);
private _result = GVAR(CadStore) call ["assignTaskToGroup", [_uid, _taskID, _groupID, _note]];
[CRPC(cad,responseCadAssignment), [_result], _player] call CFUNC(targetEvent);
[CRPC(cad,responseHydrateCad), [GVAR(CadStore) call ["buildHydratePayload", [_uid]]], _player] call CFUNC(targetEvent);
[QGVAR(requestCreateCadDispatchOrder), {
params [
["_uid", "", [""]],
["_assigneeGroupID", "", [""]],
["_targetGroupID", "", [""]],
["_note", "", [""]],
["_priority", "priority", [""]]
];
if (_assigneeGroupID isEqualTo "" || { _targetGroupID isEqualTo "" }) exitWith {
["WARNING", "Invalid CAD dispatch order payload."] call EFUNC(common,log);
};
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD dispatch order payload.",
CRPC(cad,responseCadAssignment),
"createDispatchOrder",
[_uid, _assigneeGroupID, _targetGroupID, _note, _priority],
true,
false
]];
}] call CFUNC(addEventHandler);
[QGVAR(requestSubmitCadSupportRequest), {
params [
["_uid", "", [""]],
["_type", "", [""]],
["_fields", createHashMap, [createHashMap]],
["_priority", "priority", [""]]
];
if (_type isEqualTo "") exitWith {
["WARNING", "Invalid CAD support request payload."] call EFUNC(common,log);
};
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD support request payload.",
CRPC(cad,responseCadRequest),
"submitSupportRequest",
[_uid, _type, _fields, _priority],
true,
false
]];
}] call CFUNC(addEventHandler);
[QGVAR(requestCloseCadSupportRequest), {
params [["_uid", "", [""]], ["_requestID", "", [""]]];
if (_requestID isEqualTo "") exitWith {
["WARNING", "Invalid CAD support request close payload."] call EFUNC(common,log);
};
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD support request close payload.",
CRPC(cad,responseCadRequest),
"closeSupportRequest",
[_uid, _requestID],
true,
false
]];
}] call CFUNC(addEventHandler);
[QGVAR(requestAcknowledgeCadTask), {
params [["_uid", "", [""]], ["_taskID", "", [""]]];
if (_uid isEqualTo "" || { _taskID isEqualTo "" }) exitWith {
if (_taskID isEqualTo "") exitWith {
["WARNING", "Invalid CAD acknowledge payload."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
if (_player isEqualTo objNull) exitWith {};
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD acknowledge payload.",
CRPC(cad,responseCadAssignment),
"acknowledgeTask",
[_uid, _taskID],
true,
false
]];
}] call CFUNC(addEventHandler);
private _result = GVAR(CadStore) call ["acknowledgeTask", [_uid, _taskID]];
[CRPC(cad,responseCadAssignment), [_result], _player] call CFUNC(targetEvent);
[CRPC(cad,responseHydrateCad), [GVAR(CadStore) call ["buildHydratePayload", [_uid]]], _player] call CFUNC(targetEvent);
[QGVAR(requestCloseCadDispatchOrder), {
params [["_uid", "", [""]], ["_taskID", "", [""]]];
if (_taskID isEqualTo "") exitWith {
["WARNING", "Invalid CAD dispatch order close payload."] call EFUNC(common,log);
};
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD dispatch order close payload.",
CRPC(cad,responseCadAssignment),
"closeDispatchOrder",
[_uid, _taskID],
true,
false
]];
}] call CFUNC(addEventHandler);
[QGVAR(requestDeclineCadTask), {
params [["_uid", "", [""]], ["_taskID", "", [""]]];
if (_uid isEqualTo "" || { _taskID isEqualTo "" }) exitWith {
if (_taskID isEqualTo "") exitWith {
["WARNING", "Invalid CAD decline payload."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
if (_player isEqualTo objNull) exitWith {};
private _result = GVAR(CadStore) call ["declineTask", [_uid, _taskID]];
[CRPC(cad,responseCadAssignment), [_result], _player] call CFUNC(targetEvent);
[CRPC(cad,responseHydrateCad), [GVAR(CadStore) call ["buildHydratePayload", [_uid]]], _player] call CFUNC(targetEvent);
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD decline payload.",
CRPC(cad,responseCadAssignment),
"declineTask",
[_uid, _taskID],
true,
false
]];
}] call CFUNC(addEventHandler);
[QGVAR(requestUpdateCadGroupStatus), {
params [["_uid", "", [""]], ["_groupID", "", [""]], ["_status", "", [""]]];
if (_uid isEqualTo "" || { _groupID isEqualTo "" } || { _status isEqualTo "" }) exitWith {
if (_groupID isEqualTo "" || { _status isEqualTo "" }) exitWith {
["WARNING", "Invalid CAD group status payload."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
if (_player isEqualTo objNull) exitWith {};
private _result = GVAR(CadStore) call ["updateGroupStatus", [_uid, _groupID, _status]];
[CRPC(cad,responseCadGroupUpdate), [_result], _player] call CFUNC(targetEvent);
[CRPC(cad,responseHydrateCad), [GVAR(CadStore) call ["buildHydratePayload", [_uid]]], _player] call CFUNC(targetEvent);
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD group status payload.",
CRPC(cad,responseCadGroupUpdate),
"updateGroupStatus",
[_uid, _groupID, _status],
true,
true
]];
}] call CFUNC(addEventHandler);
[QGVAR(requestUpdateCadGroupRole), {
params [["_uid", "", [""]], ["_groupID", "", [""]], ["_role", "", [""]]];
if (_uid isEqualTo "" || { _groupID isEqualTo "" } || { _role isEqualTo "" }) exitWith {
if (_groupID isEqualTo "" || { _role isEqualTo "" }) exitWith {
["WARNING", "Invalid CAD group role payload."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
if (_player isEqualTo objNull) exitWith {};
private _result = GVAR(CadStore) call ["updateGroupRole", [_uid, _groupID, _role]];
[CRPC(cad,responseCadGroupUpdate), [_result], _player] call CFUNC(targetEvent);
[CRPC(cad,responseHydrateCad), [GVAR(CadStore) call ["buildHydratePayload", [_uid]]], _player] call CFUNC(targetEvent);
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD group role payload.",
CRPC(cad,responseCadGroupUpdate),
"updateGroupRole",
[_uid, _groupID, _role],
true,
true
]];
}] call CFUNC(addEventHandler);
[QGVAR(requestUpdateCadGroupProfile), {
params [
["_uid", "", [""]],
["_groupID", "", [""]],
["_status", "", [""]],
["_role", "", [""]]
];
if (_groupID isEqualTo "") exitWith {
["WARNING", "Invalid CAD group profile payload."] call EFUNC(common,log);
};
GVAR(CadStore) call ["dispatchRpcMutation", [
_uid,
"Invalid CAD group profile payload.",
CRPC(cad,responseCadGroupUpdate),
"updateGroupProfile",
[_uid, _groupID, _status, _role],
true,
true
]];
}] call CFUNC(addEventHandler);

View File

@ -24,6 +24,46 @@ GVAR(ActivityRepositoryBaseClass) = compileFinal createHashMapFromArray [
["#type", "CadActivityRepositoryBaseClass"],
["#create", compileFinal {
_self set ["activityRegistry", []];
_self set ["persistenceLoaded", false];
}],
["restorePersistedActivity", compileFinal {
if (_self getOrDefault ["persistenceLoaded", false]) exitWith { true };
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith { false };
private _result = _persistenceService call ["loadActivity", []];
if !(_result getOrDefault ["success", false]) exitWith { false };
_self set ["activityRegistry", +(_result getOrDefault ["data", []])];
_self set ["persistenceLoaded", true];
true
}],
["appendEntry", compileFinal {
params [["_entry", createHashMap, [createHashMap]]];
if (_entry isEqualTo createHashMap) exitWith { false };
_self call ["restorePersistedActivity", []];
private _activityRegistry = +(_self getOrDefault ["activityRegistry", []]);
private _finalEntry = +_entry;
if ((_finalEntry getOrDefault ["timestamp", -1]) < 0) then {
_finalEntry set ["timestamp", serverTime];
};
_activityRegistry pushBack _finalEntry;
if ((count _activityRegistry) > 50) then {
_activityRegistry deleteRange [0, (count _activityRegistry) - 50];
};
_self set ["activityRegistry", _activityRegistry];
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isNotEqualTo createHashMap) then {
_persistenceService call ["appendActivity", [_finalEntry]];
};
true
}],
["appendActivity", compileFinal {
params [
@ -35,25 +75,17 @@ GVAR(ActivityRepositoryBaseClass) = compileFinal createHashMapFromArray [
];
if (_type isEqualTo "" || { _message isEqualTo "" }) exitWith { false };
private _activityRegistry = +(_self getOrDefault ["activityRegistry", []]);
_activityRegistry pushBack createHashMapFromArray [
private _entry = createHashMapFromArray [
["type", _type],
["message", _message],
["timestamp", serverTime],
["taskId", _taskID],
["groupId", _groupID],
["actorUid", _actorUid]
];
if ((count _activityRegistry) > 50) then {
_activityRegistry deleteRange [0, (count _activityRegistry) - 50];
};
_self set ["activityRegistry", _activityRegistry];
true
_self call ["appendEntry", [_entry]]
}],
["getActivity", compileFinal {
_self call ["restorePersistedActivity", []];
+(_self getOrDefault ["activityRegistry", []])
}]
];

View File

@ -25,12 +25,21 @@ GVAR(AssignmentRepositoryBaseClass) = compileFinal createHashMapFromArray [
["#type", "CadAssignmentRepositoryBaseClass"],
["#create", compileFinal {
_self set ["assignmentRegistry", createHashMap];
_self set ["dispatchOrderRegistry", createHashMap];
_self set ["persistenceLoaded", false];
}],
["pruneAssignments", compileFinal {
_self call ["restorePersistedState", []];
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
private _dispatchOrderRegistry = _self getOrDefault ["dispatchOrderRegistry", createHashMap];
private _keysToRemove = [];
{
if ((_dispatchOrderRegistry getOrDefault [_x, createHashMap]) isNotEqualTo createHashMap) then {
continue;
};
private _status = EGVAR(task,TaskStore) call ["getTaskStatus", [_x]];
if !(_status in ["active", ""]) then {
_keysToRemove pushBack _x;
@ -42,43 +51,90 @@ GVAR(AssignmentRepositoryBaseClass) = compileFinal createHashMapFromArray [
} forEach _keysToRemove;
_self set ["assignmentRegistry", _assignmentRegistry];
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isNotEqualTo createHashMap) then {
{
_persistenceService call ["deleteAssignment", [_x]];
} forEach _keysToRemove;
};
count _keysToRemove
}],
["getAssignments", compileFinal {
_self call ["restorePersistedState", []];
values (_self getOrDefault ["assignmentRegistry", createHashMap])
}],
["buildContracts", compileFinal {
params [["_uid", "", [""]]];
["isDispatchOrder", compileFinal {
params [["_taskID", "", [""]]];
_self call ["pruneAssignments", []];
if (_taskID isEqualTo "") exitWith { false };
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
private _contracts = [];
private _permissionService = _self getOrDefault ["permissionService", createHashMap];
private _groupRepository = _self getOrDefault ["groupRepository", createHashMap];
private _canDispatch = _permissionService call ["canDispatch", [_uid]];
private _playerGroupId = _groupRepository call ["getPlayerGroupId", [_uid]];
((_self getOrDefault ["dispatchOrderRegistry", createHashMap]) getOrDefault [_taskID, createHashMap]) isNotEqualTo createHashMap
}],
["restorePersistedState", compileFinal {
if (_self getOrDefault ["persistenceLoaded", false]) exitWith { true };
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith { false };
private _assignmentsResult = _persistenceService call ["loadAssignments", []];
if !(_assignmentsResult getOrDefault ["success", false]) exitWith { false };
private _ordersResult = _persistenceService call ["loadDispatchOrders", []];
if !(_ordersResult getOrDefault ["success", false]) exitWith { false };
private _assignmentRegistry = +(_assignmentsResult getOrDefault ["data", createHashMap]);
private _dispatchOrderRegistry = +(_ordersResult getOrDefault ["data", createHashMap]);
_self set ["assignmentRegistry", _assignmentRegistry];
_self set ["dispatchOrderRegistry", _dispatchOrderRegistry];
_self set ["persistenceLoaded", true];
{
private _taskID = _x getOrDefault ["taskID", ""];
if (_taskID isEqualTo "") then { continue; };
if ((_y getOrDefault ["state", ""]) isNotEqualTo "acknowledged") then { continue; };
if (((_y getOrDefault ["acknowledgedByUid", ""]) isEqualTo "")) then { continue; };
if ((_dispatchOrderRegistry getOrDefault [_x, createHashMap]) isNotEqualTo createHashMap) then { continue; };
if ((EGVAR(task,TaskStore) call ["getTaskStatus", [_x]]) isNotEqualTo "active") then { continue; };
EGVAR(task,TaskStore) call ["bindTaskOwnership", [_x, _y getOrDefault ["acknowledgedByUid", ""]]];
} forEach _assignmentRegistry;
private _assignment = _assignmentRegistry getOrDefault [_taskID, createHashMap];
private _entry = +_x;
_entry set ["taskId", _taskID];
_entry set ["assignedGroupId", _assignment getOrDefault ["groupId", ""]];
_entry set ["assignmentState", [_assignment getOrDefault ["state", ""], "unassigned"] select (_assignment isEqualTo createHashMap)];
true
}],
["buildDispatchOrderEntry", compileFinal {
params [
["_taskID", "", [""]],
["_order", createHashMap, [createHashMap]],
["_assignmentRegistry", createHashMap, [createHashMap]],
["_groupRepository", createHashMap, [createHashMap]]
];
if (!_canDispatch) then {
private _assignedGroupId = _entry getOrDefault ["assignedGroupId", ""];
if (_assignedGroupId isEqualTo "") then { continue; };
if (_assignedGroupId isNotEqualTo _playerGroupId) then { continue; };
if (_taskID isEqualTo "" || { _order isEqualTo createHashMap }) exitWith { createHashMap };
private _entry = +_order;
private _targetGroupID = _order getOrDefault ["targetGroupId", ""];
if (_targetGroupID isNotEqualTo "") then {
private _targetGroup = _groupRepository call ["getGroupRecord", [_targetGroupID]];
if (_targetGroup isNotEqualTo createHashMap) then {
private _targetCallsign = _targetGroup getOrDefault ["callsign", _targetGroupID];
_entry set ["targetGroupCallsign", _targetCallsign];
_entry set ["position", +(_targetGroup getOrDefault ["position", _entry getOrDefault ["position", []]])];
_entry set ["title", format ["Backup %1", _targetCallsign]];
if ((_order getOrDefault ["note", ""]) isEqualTo "") then {
_entry set ["description", format ["Dispatch order to back up %1 at its current position.", _targetCallsign]];
};
};
};
_contracts pushBack _entry;
} forEach (EGVAR(task,TaskStore) call ["getActiveTaskCatalog", []]);
_contracts
private _assignment = _assignmentRegistry getOrDefault [_taskID, createHashMap];
_entry set ["taskId", _taskID];
_entry set ["taskID", _taskID];
_entry set ["type", _entry getOrDefault ["type", "dispatch_order"]];
_entry set ["isDispatchOrder", true];
_entry set ["assignedGroupId", _assignment getOrDefault ["groupId", ""]];
_entry set ["assignmentState", [_assignment getOrDefault ["state", ""], "unassigned"] select (_assignment isEqualTo createHashMap)];
_entry
}],
["assignTaskToGroup", compileFinal {
params [
@ -94,24 +150,29 @@ GVAR(AssignmentRepositoryBaseClass) = compileFinal createHashMapFromArray [
["assignment", createHashMap]
];
_self call ["restorePersistedState", []];
private _permissionService = _self getOrDefault ["permissionService", createHashMap];
if !(_permissionService call ["canDispatch", [_requesterUid]]) exitWith {
_result set ["message", "You are not authorized to assign contracts."];
_result
};
if ((EGVAR(task,TaskStore) call ["getTaskStatus", [_taskID]]) isNotEqualTo "active") exitWith {
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
private _dispatchOrderRegistry = _self getOrDefault ["dispatchOrderRegistry", createHashMap];
private _isDispatchOrder = (_dispatchOrderRegistry getOrDefault [_taskID, createHashMap]) isNotEqualTo createHashMap;
if (!_isDispatchOrder && { (EGVAR(task,TaskStore) call ["getTaskStatus", [_taskID]]) isNotEqualTo "active" }) exitWith {
_result set ["message", "Task is no longer active."];
_result
};
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
private _existingAssignment = +(_assignmentRegistry getOrDefault [_taskID, createHashMap]);
if (
_existingAssignment isNotEqualTo createHashMap
&& { (_existingAssignment getOrDefault ["state", ""]) in ["assigned", "acknowledged"] }
) exitWith {
_result set ["message", "Task is already assigned and must be declined or completed before reassignment."];
_result set ["message", ["Task is already assigned and must be declined or completed before reassignment.", "Dispatch order is already assigned and must be declined or closed before reassignment."] select _isDispatchOrder];
_result set ["assignment", _existingAssignment];
_result
};
@ -133,6 +194,7 @@ GVAR(AssignmentRepositoryBaseClass) = compileFinal createHashMapFromArray [
private _assignment = createHashMapFromArray [
["taskId", _taskID],
["groupId", _groupID],
["groupCallsign", _groupRecord getOrDefault ["callsign", _groupID]],
["assignedByUid", _requesterUid],
["assignedByName", ["Dispatcher", name _requesterPlayer] select (_requesterPlayer isNotEqualTo objNull)],
["assignedAt", serverTime],
@ -140,114 +202,341 @@ GVAR(AssignmentRepositoryBaseClass) = compileFinal createHashMapFromArray [
["note", _note]
];
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension state is unavailable."];
_result
};
private _assignResult = _persistenceService call ["assignAssignment", [_taskID, _assignment]];
if !(_assignResult getOrDefault ["success", false]) exitWith {
_result set ["message", "CAD extension rejected the assignment."];
_result
};
private _assignData = +(_assignResult getOrDefault ["data", createHashMap]);
_assignment = +(_assignData getOrDefault ["assignment", createHashMap]);
if (_assignment isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension returned an invalid assignment."];
_result
};
_assignmentRegistry set [_taskID, _assignment];
_self set ["assignmentRegistry", _assignmentRegistry];
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendActivity", [
"task_assigned",
format ["%1 assigned %2 to %3.", _assignment get "assignedByName", _taskID, _groupRecord getOrDefault ["callsign", _groupID]],
_taskID,
_groupID,
_requesterUid
]];
private _activityEntry = +(_assignData getOrDefault ["activity", createHashMap]);
if (_activityEntry isNotEqualTo createHashMap) then {
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendEntry", [_activityEntry]];
};
_result set ["success", true];
_result set ["message", "Task assigned."];
_result set ["message", _assignData getOrDefault ["message", ["Task assigned.", "Dispatch order assigned."] select _isDispatchOrder]];
_result set ["assignment", _assignment];
_result set ["leaderUid", _leaderUid];
_result set ["isDispatchOrder", _isDispatchOrder];
_result
}],
["acknowledgeTask", compileFinal {
params [["_requesterUid", "", [""]], ["_taskID", "", [""]]];
["createDispatchOrder", compileFinal {
params [
["_requesterUid", "", [""]],
["_assigneeGroupID", "", [""]],
["_targetGroupID", "", [""]],
["_note", "", [""]],
["_priority", "priority", [""]]
];
private _result = createHashMapFromArray [
["success", false],
["message", "Unable to acknowledge task."],
["assignment", createHashMap]
["message", "Unable to create dispatch order."],
["assignment", createHashMap],
["order", createHashMap]
];
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
private _assignment = +(_assignmentRegistry getOrDefault [_taskID, createHashMap]);
if (_assignment isEqualTo createHashMap) exitWith {
_result set ["message", "Task is not assigned."];
_self call ["restorePersistedState", []];
private _permissionService = _self getOrDefault ["permissionService", createHashMap];
if !(_permissionService call ["canDispatch", [_requesterUid]]) exitWith {
_result set ["message", "You are not authorized to create dispatch orders."];
_result
};
if (_assigneeGroupID isEqualTo "" || { _targetGroupID isEqualTo "" }) exitWith {
_result set ["message", "Assignee and target groups are required."];
_result
};
if (_assigneeGroupID isEqualTo _targetGroupID) exitWith {
_result set ["message", "Assignee and target groups must be different."];
_result
};
private _groupID = _assignment getOrDefault ["groupId", ""];
private _groupRepository = _self getOrDefault ["groupRepository", createHashMap];
if !(_groupRepository call ["isGroupLeader", [_requesterUid, _groupID]]) exitWith {
_result set ["message", "Only the assigned group leader can acknowledge this task."];
private _assigneeGroup = _groupRepository call ["getGroupRecord", [_assigneeGroupID]];
if (_assigneeGroup isEqualTo createHashMap) exitWith {
_result set ["message", "Selected assignee group is unavailable."];
_result
};
private _bindResult = EGVAR(task,TaskStore) call ["bindTaskOwnership", [_taskID, _requesterUid]];
if !(_bindResult getOrDefault ["success", false]) exitWith {
_result set ["message", _bindResult getOrDefault ["message", "Failed to bind task ownership."]];
private _assigneeLeaderUid = _assigneeGroup getOrDefault ["leaderUid", ""];
if (_assigneeLeaderUid isEqualTo "") exitWith {
_result set ["message", "Selected assignee group has no online leader."];
_result
};
_assignment set ["state", "acknowledged"];
_assignment set ["acknowledgedAt", serverTime];
private _targetGroup = _groupRepository call ["getGroupRecord", [_targetGroupID]];
if (_targetGroup isEqualTo createHashMap) exitWith {
_result set ["message", "Selected target group is unavailable."];
_result
};
private _validPriorities = ["routine", "priority", "emergency"];
private _finalPriority = toLowerANSI _priority;
if !(_finalPriority in _validPriorities) then {
_finalPriority = "priority";
};
private _requesterPlayer = [_requesterUid] call EFUNC(common,getPlayer);
private _dispatchContext = createHashMapFromArray [
["assigneeGroupId", _assigneeGroupID],
["assigneeGroupCallsign", _assigneeGroup getOrDefault ["callsign", _assigneeGroupID]],
["targetGroupId", _targetGroupID],
["targetGroupCallsign", _targetGroup getOrDefault ["callsign", _targetGroupID]],
["targetPosition", +(_targetGroup getOrDefault ["position", []])],
["createdByUid", _requesterUid],
["createdByName", ["Dispatcher", name _requesterPlayer] select (_requesterPlayer isNotEqualTo objNull)],
["note", _note],
["priority", _finalPriority],
["createdAt", serverTime]
];
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension state is unavailable."];
_result
};
private _createResult = _persistenceService call ["createDispatchOrderFromContext", [_dispatchContext]];
if !(_createResult getOrDefault ["success", false]) exitWith {
_result set ["message", "CAD extension rejected the dispatch order."];
_result
};
private _createData = +(_createResult getOrDefault ["data", createHashMap]);
private _taskID = _createData getOrDefault ["taskId", ""];
private _order = +(_createData getOrDefault ["order", createHashMap]);
private _assignment = +(_createData getOrDefault ["assignment", createHashMap]);
if (_taskID isEqualTo "" || { _order isEqualTo createHashMap } || { _assignment isEqualTo createHashMap }) exitWith {
_result set ["message", "CAD extension returned an invalid dispatch order."];
_result
};
private _dispatchOrderRegistry = _self getOrDefault ["dispatchOrderRegistry", createHashMap];
_dispatchOrderRegistry set [_taskID, _order];
_self set ["dispatchOrderRegistry", _dispatchOrderRegistry];
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
_assignmentRegistry set [_taskID, _assignment];
_self set ["assignmentRegistry", _assignmentRegistry];
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendActivity", [
"task_acknowledged",
format ["%1 acknowledged %2.", _requesterUid, _taskID],
_taskID,
_groupID,
_requesterUid
]];
private _activityEntry = +(_createData getOrDefault ["activity", createHashMap]);
if (_activityEntry isNotEqualTo createHashMap) then {
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendEntry", [_activityEntry]];
};
_result set ["success", true];
_result set ["message", "Task acknowledged."];
_result set ["message", _createData getOrDefault ["message", "Dispatch order created."]];
_result set ["assignment", _assignment];
_result set ["order", _order];
_result set ["leaderUid", _assigneeLeaderUid];
_result set ["isDispatchOrder", true];
_result
}],
["declineTask", compileFinal {
["closeDispatchOrder", compileFinal {
params [["_requesterUid", "", [""]], ["_taskID", "", [""]]];
private _result = createHashMapFromArray [
["success", false],
["message", "Unable to decline task."],
["message", "Unable to close dispatch order."],
["assignment", createHashMap]
];
_self call ["restorePersistedState", []];
private _permissionService = _self getOrDefault ["permissionService", createHashMap];
if !(_permissionService call ["canDispatch", [_requesterUid]]) exitWith {
_result set ["message", "You are not authorized to close dispatch orders."];
_result
};
private _dispatchOrderRegistry = _self getOrDefault ["dispatchOrderRegistry", createHashMap];
private _order = +(_dispatchOrderRegistry getOrDefault [_taskID, createHashMap]);
if (_order isEqualTo createHashMap) exitWith {
_result set ["message", "Dispatch order could not be resolved."];
_result
};
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
private _assignment = +(_assignmentRegistry getOrDefault [_taskID, createHashMap]);
if (_assignment isEqualTo createHashMap) exitWith {
_result set ["message", "Task is not assigned."];
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension state is unavailable."];
_result
};
private _groupID = _assignment getOrDefault ["groupId", ""];
private _groupRepository = _self getOrDefault ["groupRepository", createHashMap];
if !(_groupRepository call ["isGroupLeader", [_requesterUid, _groupID]]) exitWith {
_result set ["message", "Only the assigned group leader can decline this task."];
private _closeResult = _persistenceService call ["closeDispatchOrder", [_taskID]];
if !(_closeResult getOrDefault ["success", false]) exitWith {
_result set ["message", "CAD extension rejected the dispatch order close."];
_result
};
_assignment set ["state", "declined"];
_assignment set ["declinedAt", serverTime];
EGVAR(task,TaskStore) call ["releaseTaskOwnership", [_taskID]];
private _closeData = +(_closeResult getOrDefault ["data", createHashMap]);
_order = +(_closeData getOrDefault ["order", _order]);
_assignment = +(_closeData getOrDefault ["assignment", _assignment]);
_dispatchOrderRegistry deleteAt _taskID;
_self set ["dispatchOrderRegistry", _dispatchOrderRegistry];
_assignmentRegistry deleteAt _taskID;
_self set ["assignmentRegistry", _assignmentRegistry];
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendActivity", [
"task_declined",
format ["%1 declined %2.", _requesterUid, _taskID],
_taskID,
_groupID,
_requesterUid
]];
private _activityEntry = +(_closeData getOrDefault ["activity", createHashMap]);
if (_activityEntry isNotEqualTo createHashMap) then {
_activityEntry set ["actorUid", _requesterUid];
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendEntry", [_activityEntry]];
};
_result set ["success", true];
_result set ["message", "Task declined and returned to the contract board."];
_result set ["message", _closeData getOrDefault ["message", "Dispatch order closed."]];
_result set ["assignment", _assignment];
_result set ["isDispatchOrder", true];
_result
}],
["applyAssignmentTransition", compileFinal {
params [["_requesterUid", "", [""]], ["_taskID", "", [""]]];
private _result = createHashMapFromArray [
["success", false],
["message", "Unable to update task assignment."],
["assignment", createHashMap]
];
private _transition = _this param [2, "acknowledge", [""]];
_self call ["restorePersistedState", []];
private _assignmentRegistry = _self getOrDefault ["assignmentRegistry", createHashMap];
private _assignment = +(_assignmentRegistry getOrDefault [_taskID, createHashMap]);
private _isDispatchOrder = _self call ["isDispatchOrder", [_taskID]];
if (_assignment isEqualTo createHashMap) exitWith {
_result set ["message", "Task is not assigned."];
_result
};
private _groupID = _assignment getOrDefault ["groupId", ""];
private _groupRepository = _self getOrDefault ["groupRepository", createHashMap];
if !(_groupRepository call ["isGroupLeader", [_requesterUid, _groupID]]) exitWith {
_result set ["message", format ["Only the assigned group leader can %1 this task.", _transition]];
_result
};
switch (_transition) do {
case "acknowledge": {
if (!_isDispatchOrder) then {
private _bindResult = EGVAR(task,TaskStore) call ["bindTaskOwnership", [_taskID, _requesterUid]];
if !(_bindResult getOrDefault ["success", false]) exitWith {
_result set ["message", _bindResult getOrDefault ["message", "Failed to bind task ownership."]];
_result
};
};
};
case "decline": {
if (!_isDispatchOrder) then {
EGVAR(task,TaskStore) call ["releaseTaskOwnership", [_taskID]];
};
};
};
if (_result getOrDefault ["success", false]) exitWith { _result };
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension state is unavailable."];
_result
};
private _patch = switch (_transition) do {
case "decline": {
createHashMapFromArray [
["state", "declined"],
["declinedAt", serverTime],
["declinedByUid", _requesterUid]
]
};
default {
createHashMapFromArray [
["state", "acknowledged"],
["acknowledgedAt", serverTime],
["acknowledgedByUid", _requesterUid]
]
};
};
private _transitionResult = switch (_transition) do {
case "decline": { _persistenceService call ["declineAssignment", [_taskID, _patch]] };
default { _persistenceService call ["acknowledgeAssignment", [_taskID, _patch]] };
};
if !(_transitionResult getOrDefault ["success", false]) exitWith {
_result set ["message", switch (_transition) do {
case "decline": { "CAD extension rejected the decline." };
default { "CAD extension rejected the acknowledgement." };
}];
_result
};
private _transitionData = +(_transitionResult getOrDefault ["data", createHashMap]);
_assignment = +(_transitionData getOrDefault ["assignment", createHashMap]);
if (_assignment isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension returned an invalid assignment."];
_result
};
switch (_transition) do {
case "decline": {
_assignmentRegistry deleteAt _taskID;
};
default {
_assignmentRegistry set [_taskID, _assignment];
};
};
_self set ["assignmentRegistry", _assignmentRegistry];
private _activityEntry = +(_transitionData getOrDefault ["activity", createHashMap]);
if (_activityEntry isNotEqualTo createHashMap) then {
if (_isDispatchOrder) then {
_activityEntry set ["type", format ["dispatch_order_%1", _transition]];
_activityEntry set ["message", format ["%1 %2d %3.", _requesterUid, _transition, _taskID]];
};
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendEntry", [_activityEntry]];
};
_result set ["success", true];
_result set ["message", switch (_transition) do {
case "decline": { [_transitionData getOrDefault ["message", "Task declined and returned to the contract board."], "Dispatch order declined and returned to the dispatch board."] select _isDispatchOrder };
default { [_transitionData getOrDefault ["message", "Task acknowledged."], "Dispatch order acknowledged."] select _isDispatchOrder };
}];
_result set ["assignment", _assignment];
_result set ["isDispatchOrder", _isDispatchOrder];
_result
}],
["acknowledgeTask", compileFinal {
_self call ["applyAssignmentTransition", [_this # 0, _this # 1, "acknowledge"]]
}],
["declineTask", compileFinal {
_self call ["applyAssignmentTransition", [_this # 0, _this # 1, "decline"]]
}]
];

View File

@ -28,55 +28,35 @@ GVAR(CadStoreBaseClass) = compileFinal createHashMapFromArray [
private _permissionService = call FUNC(initPermissionService);
private _groupRepository = call FUNC(initGroupRepository);
private _assignmentRepository = call FUNC(initAssignmentRepository);
private _persistenceService = call FUNC(initPersistenceService);
private _requestRepository = call FUNC(initRequestRepository);
_groupRepository set ["activityRepository", _activityRepository];
_groupRepository set ["assignmentRepository", _assignmentRepository];
_groupRepository set ["permissionService", _permissionService];
_groupRepository set ["persistenceService", _persistenceService];
_assignmentRepository set ["activityRepository", _activityRepository];
_assignmentRepository set ["groupRepository", _groupRepository];
_assignmentRepository set ["permissionService", _permissionService];
_assignmentRepository set ["persistenceService", _persistenceService];
_requestRepository set ["activityRepository", _activityRepository];
_requestRepository set ["groupRepository", _groupRepository];
_requestRepository set ["permissionService", _permissionService];
_requestRepository set ["persistenceService", _persistenceService];
_activityRepository set ["persistenceService", _persistenceService];
_self set ["ActivityRepository", _activityRepository];
_self set ["PermissionService", _permissionService];
_self set ["GroupRepository", _groupRepository];
_self set ["AssignmentRepository", _assignmentRepository];
_self set ["PersistenceService", _persistenceService];
_self set ["RequestRepository", _requestRepository];
["INFO", "CAD Store Initialized!"] call EFUNC(common,log);
}],
["appendActivity", compileFinal {
(_self get "ActivityRepository") call ["appendActivity", _this]
}],
["resolveGroupId", compileFinal {
(_self get "GroupRepository") call ["resolveGroupId", _this]
}],
["canDispatch", compileFinal {
(_self get "PermissionService") call ["canDispatch", _this]
}],
["getCurrentTaskIdForGroup", compileFinal {
(_self get "GroupRepository") call ["getCurrentTaskIdForGroup", _this]
}],
["syncGroups", compileFinal {
(_self get "GroupRepository") call ["syncGroups", _this]
}],
["getGroupRecord", compileFinal {
(_self get "GroupRepository") call ["getGroupRecord", _this]
}],
["getPlayerGroupId", compileFinal {
(_self get "GroupRepository") call ["getPlayerGroupId", _this]
}],
["isGroupLeader", compileFinal {
(_self get "GroupRepository") call ["isGroupLeader", _this]
}],
["pruneAssignments", compileFinal {
(_self get "AssignmentRepository") call ["pruneAssignments", _this]
}],
["buildContracts", compileFinal {
(_self get "AssignmentRepository") call ["buildContracts", _this]
}],
["buildGroups", compileFinal {
(_self get "GroupRepository") call ["buildGroups", _this]
}],
["notifyPlayer", compileFinal {
params [
["_uid", "", [""]],
@ -93,23 +73,115 @@ GVAR(CadStoreBaseClass) = compileFinal createHashMapFromArray [
[CRPC(notifications,recieveNotification), [_type, _title, _message], _player] call CFUNC(targetEvent);
true
}],
["assignTaskToGroup", compileFinal {
private _result = (_self get "AssignmentRepository") call ["assignTaskToGroup", _this];
if !(_result getOrDefault ["success", false]) exitWith { _result };
["resolveRequestPlayer", compileFinal {
params [
["_uid", "", [""]],
["_warning", "Invalid CAD payload.", [""]]
];
if (_uid isEqualTo "") exitWith {
["WARNING", _warning] call EFUNC(common,log);
objNull
};
[_uid] call EFUNC(common,getPlayer)
}],
["sendRpcResult", compileFinal {
params [
["_player", objNull, [objNull]],
["_responseRpc", "", [""]],
["_result", createHashMap, [createHashMap]],
["_invalidateOnSuccess", false, [false]],
["_requireChanged", false, [false]]
];
if (_player isEqualTo objNull || { _responseRpc isEqualTo "" }) exitWith {};
[_responseRpc, [_result], _player] call CFUNC(targetEvent);
if (
_invalidateOnSuccess
&& { _result getOrDefault ["success", false] }
&& { !_requireChanged || { _result getOrDefault ["changed", true] } }
) then {
[CRPC(cad,invalidateCadState), []] call CFUNC(globalEvent);
};
}],
["dispatchRpcMutation", compileFinal {
params [
["_uid", "", [""]],
["_warning", "Invalid CAD payload.", [""]],
["_responseRpc", "", [""]],
["_method", "", [""]],
["_arguments", [], [[]]],
["_invalidateOnSuccess", false, [false]],
["_requireChanged", false, [false]]
];
private _player = _self call ["resolveRequestPlayer", [_uid, _warning]];
if (_player isEqualTo objNull || { _method isEqualTo "" }) exitWith { createHashMap };
private _result = _self call [_method, _arguments];
_self call ["sendRpcResult", [_player, _responseRpc, _result, _invalidateOnSuccess, _requireChanged]];
_result
}],
["notifyAssignmentLeader", compileFinal {
params [["_result", createHashMap, [createHashMap]]];
if !(_result getOrDefault ["success", false]) exitWith { false };
private _assignment = _result getOrDefault ["assignment", createHashMap];
private _taskID = _assignment getOrDefault ["taskId", ""];
private _leaderUid = _result getOrDefault ["leaderUid", ""];
if (_leaderUid isEqualTo "") exitWith { false };
private _message = if (_result getOrDefault ["isDispatchOrder", false]) then {
private _order = _result getOrDefault ["order", createHashMap];
if (_order isEqualTo createHashMap) then {
private _assignment = _result getOrDefault ["assignment", createHashMap];
private _taskID = _assignment getOrDefault ["taskId", ""];
_order = (_self get "AssignmentRepository") call ["buildDispatchOrderEntry", [
_taskID,
((_self get "AssignmentRepository") getOrDefault ["dispatchOrderRegistry", createHashMap]) getOrDefault [_taskID, createHashMap],
(_self get "AssignmentRepository") getOrDefault ["assignmentRegistry", createHashMap],
_self get "GroupRepository"
]];
};
format ["Dispatch order assigned: %1. Open CAD to review and acknowledge.", _order getOrDefault ["title", "Dispatch Order"]]
} else {
private _assignment = _result getOrDefault ["assignment", createHashMap];
format ["Contract assigned: %1. Open CAD to review and acknowledge.", _assignment getOrDefault ["taskId", "Task"]]
};
_self call ["notifyPlayer", [
_leaderUid,
"info",
"Tasks",
format ["Contract assigned: %1. Open CAD to review and acknowledge.", _taskID]
]];
_message
]]
}],
["assignTaskToGroup", compileFinal {
private _result = (_self get "AssignmentRepository") call ["assignTaskToGroup", _this];
if !(_result getOrDefault ["success", false]) exitWith { _result };
_self call ["notifyAssignmentLeader", [_result]];
_result
}],
["createDispatchOrder", compileFinal {
private _result = (_self get "AssignmentRepository") call ["createDispatchOrder", _this];
if !(_result getOrDefault ["success", false]) exitWith { _result };
_self call ["notifyAssignmentLeader", [_result]];
_result
}],
["closeDispatchOrder", compileFinal {
(_self get "AssignmentRepository") call ["closeDispatchOrder", _this]
}],
["submitSupportRequest", compileFinal {
(_self get "RequestRepository") call ["submitRequest", _this]
}],
["closeSupportRequest", compileFinal {
(_self get "RequestRepository") call ["closeRequest", _this]
}],
["acknowledgeTask", compileFinal {
(_self get "AssignmentRepository") call ["acknowledgeTask", _this]
}],
@ -122,13 +194,14 @@ GVAR(CadStoreBaseClass) = compileFinal createHashMapFromArray [
["updateGroupRole", compileFinal {
(_self get "GroupRepository") call ["updateGroupRole", _this]
}],
["updateGroupProfile", compileFinal {
(_self get "GroupRepository") call ["updateGroupProfile", _this]
}],
["buildHydratePayload", compileFinal {
params [["_uid", "", [""]]];
private _activityRepository = _self get "ActivityRepository";
private _permissionService = _self get "PermissionService";
private _groupRepository = _self get "GroupRepository";
private _assignmentRepository = _self get "AssignmentRepository";
private _actor = EGVAR(actor,Registry) getOrDefault [_uid, createHashMap];
if (_actor isEqualTo createHashMap && { _uid isNotEqualTo "" }) then {
@ -136,20 +209,40 @@ GVAR(CadStoreBaseClass) = compileFinal createHashMapFromArray [
};
private _groupID = _groupRepository call ["getPlayerGroupId", [_uid]];
createHashMapFromArray [
private _session = createHashMapFromArray [
["uid", _uid],
["orgId", _actor getOrDefault ["organization", "default"]],
["isDispatcher", _permissionService call ["canDispatch", [_uid]]],
["groupId", _groupID],
["isLeader", _groupRepository call ["isGroupLeader", [_uid, _groupID]]]
];
private _seed = createHashMapFromArray [
["groups", _groupRepository call ["buildGroups", []]],
["contracts", _assignmentRepository call ["buildContracts", [_uid]]],
["assignments", _assignmentRepository call ["getAssignments", []]],
["activity", _activityRepository call ["getActivity", []]],
["session", createHashMapFromArray [
["uid", _uid],
["orgId", _actor getOrDefault ["organization", "default"]],
["isDispatcher", _permissionService call ["canDispatch", [_uid]]],
["groupId", _groupID],
["isLeader", _groupRepository call ["isGroupLeader", [_uid, _groupID]]]
]]
]
["activeTasks", EGVAR(task,TaskStore) call ["getActiveTaskCatalog", []]],
["session", _session]
];
private _emptyPayload = createHashMapFromArray [
["groups", _seed get "groups"],
["contracts", []],
["requests", []],
["assignments", []],
["activity", []],
["session", _session]
];
private _persistenceService = _self getOrDefault ["PersistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
["WARNING", "CAD hydrate extension state is unavailable; returning seed-only payload."] call EFUNC(common,log);
_emptyPayload
};
private _hydrateResult = _persistenceService call ["buildHydratePayload", [_seed]];
if (_hydrateResult getOrDefault ["success", false]) exitWith {
_hydrateResult getOrDefault ["data", createHashMap]
};
["WARNING", "CAD hydrate failed in the extension; returning seed-only payload."] call EFUNC(common,log);
_emptyPayload
}]
];

View File

@ -64,22 +64,31 @@ GVAR(GroupRepositoryBaseClass) = compileFinal createHashMapFromArray [
private _assignmentRepository = _self getOrDefault ["assignmentRepository", createHashMap];
private _assignmentRegistry = _assignmentRepository getOrDefault ["assignmentRegistry", createHashMap];
private _dispatchOrderRegistry = _assignmentRepository getOrDefault ["dispatchOrderRegistry", createHashMap];
private _taskID = "";
{
if ((_y getOrDefault ["groupId", ""]) isNotEqualTo _groupID) then { continue; };
if !((_y getOrDefault ["state", ""]) in ["assigned", "acknowledged"]) then { continue; };
if ((EGVAR(task,TaskStore) call ["getTaskStatus", [_x]]) isNotEqualTo "active") then { continue; };
private _dispatchOrder = +(_dispatchOrderRegistry getOrDefault [_x, createHashMap]);
if (_dispatchOrder isEqualTo createHashMap) then {
if ((EGVAR(task,TaskStore) call ["getTaskStatus", [_x]]) isNotEqualTo "active") then { continue; };
_taskID = _x;
} else {
_taskID = _dispatchOrder getOrDefault ["title", _x];
};
_taskID = _x;
} forEach _assignmentRegistry;
_taskID
}],
["syncGroups", compileFinal {
private _previousRegistry = _self getOrDefault ["groupRegistry", createHashMap];
private _profileRegistry = _self getOrDefault ["groupProfileRegistry", createHashMap];
private _nextRegistry = createHashMap;
private _assignmentRepository = _self getOrDefault ["assignmentRepository", createHashMap];
if (_assignmentRepository isNotEqualTo createHashMap) then {
_assignmentRepository call ["restorePersistedState", []];
};
private _liveGroups = [];
{
private _group = _x;
@ -104,9 +113,6 @@ GVAR(GroupRepositoryBaseClass) = compileFinal createHashMapFromArray [
private _orgID = _actor getOrDefault ["organization", "default"];
if (_orgID isEqualTo "") then { _orgID = "default"; };
private _existingRecord = +(_previousRegistry getOrDefault [_groupID, createHashMap]);
private _profile = +(_profileRegistry getOrDefault [_groupID, createHashMap]);
private _memberUids = [];
private _memberRoster = [];
@ -126,7 +132,7 @@ GVAR(GroupRepositoryBaseClass) = compileFinal createHashMapFromArray [
]);
} forEach _members;
private _record = createHashMapFromArray [
_liveGroups pushBack (createHashMapFromArray [
["groupId", _groupID],
["callsign", [groupId _group, _groupID] select ((groupId _group) isEqualTo "")],
["leaderUid", _leaderUid],
@ -134,20 +140,39 @@ GVAR(GroupRepositoryBaseClass) = compileFinal createHashMapFromArray [
["memberUids", _memberUids],
["members", _memberRoster],
["orgId", _orgID],
["role", [_existingRecord getOrDefault ["role", "infantry"], _profile getOrDefault ["role", _existingRecord getOrDefault ["role", "infantry"]]] select (_profile isNotEqualTo createHashMap)],
["status", [_existingRecord getOrDefault ["status", "available"], _profile getOrDefault ["status", _existingRecord getOrDefault ["status", "available"]]] select (_profile isNotEqualTo createHashMap)],
["role", "infantry"],
["status", "available"],
["position", getPosATL _leader],
["currentTaskId", _self call ["getCurrentTaskIdForGroup", [_groupID]]],
["lastUpdate", serverTime]
];
_nextRegistry set [_groupID, _record];
_profileRegistry set [_groupID, createHashMapFromArray [
["role", _record getOrDefault ["role", "infantry"]],
["status", _record getOrDefault ["status", "available"]]
]];
]);
} forEach allGroups;
private _mergedGroups = _liveGroups;
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isNotEqualTo createHashMap) then {
private _buildResult = _persistenceService call ["buildGroups", [_liveGroups]];
if (_buildResult getOrDefault ["success", false]) then {
_mergedGroups = +(_buildResult getOrDefault ["data", _liveGroups]);
};
};
private _nextRegistry = createHashMap;
private _profileRegistry = createHashMap;
{
if !(_x isEqualType createHashMap) then { continue; };
private _groupID = _x getOrDefault ["groupId", ""];
if (_groupID isEqualTo "") then { continue; };
private _groupRecord = +_x;
_nextRegistry set [_groupID, _groupRecord];
_profileRegistry set [_groupID, createHashMapFromArray [
["groupId", _groupID],
["role", _groupRecord getOrDefault ["role", "infantry"]],
["status", _groupRecord getOrDefault ["status", "available"]]
]];
} forEach _mergedGroups;
_self set ["groupProfileRegistry", _profileRegistry];
_self set ["groupRegistry", _nextRegistry];
_nextRegistry
@ -188,21 +213,52 @@ GVAR(GroupRepositoryBaseClass) = compileFinal createHashMapFromArray [
_groups
}],
["updateGroupStatus", compileFinal {
params [["_requesterUid", "", [""]], ["_groupID", "", [""]], ["_status", "", [""]]];
["applyGroupProfileUpdate", compileFinal {
params [
["_requesterUid", "", [""]],
["_groupID", "", [""]],
["_status", "", [""]],
["_role", "", [""]],
["_mode", "profile", [""]]
];
private _result = createHashMapFromArray [
["success", false],
["message", "Unable to update group status."],
["message", "Unable to update group profile."],
["changed", false],
["group", createHashMap]
];
private _finalStatus = toLowerANSI _status;
if !(_finalStatus in (_self getOrDefault ["validStatuses", []])) exitWith {
private _finalRole = toLowerANSI _role;
private _hasStatus = _finalStatus isNotEqualTo "";
private _hasRole = _finalRole isNotEqualTo "";
if (_mode isEqualTo "status" && !_hasStatus) exitWith {
_result set ["message", "Invalid group status."];
_result
};
if (_mode isEqualTo "role" && !_hasRole) exitWith {
_result set ["message", "Invalid group role."];
_result
};
if (_mode isEqualTo "profile" && !(_hasStatus || _hasRole)) exitWith {
_result set ["message", "No group changes were provided."];
_result
};
if (_hasStatus && !(_finalStatus in (_self getOrDefault ["validStatuses", []]))) exitWith {
_result set ["message", "Invalid group status."];
_result
};
if (_hasRole && !(_finalRole in (_self getOrDefault ["validRoles", []]))) exitWith {
_result set ["message", "Invalid group role."];
_result
};
private _permissionService = _self getOrDefault ["permissionService", createHashMap];
private _isAuthorized = (_self call ["isGroupLeader", [_requesterUid, _groupID]]) || { _permissionService call ["canDispatch", [_requesterUid]] };
if !_isAuthorized exitWith {
@ -217,86 +273,68 @@ GVAR(GroupRepositoryBaseClass) = compileFinal createHashMapFromArray [
_result
};
_groupRecord set ["status", _finalStatus];
private _didChangeStatus = _hasStatus && { (_groupRecord getOrDefault ["status", ""]) isNotEqualTo _finalStatus };
private _didChangeRole = _hasRole && { (_groupRecord getOrDefault ["role", ""]) isNotEqualTo _finalRole };
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension state is unavailable."];
_result
};
private _updateContext = createHashMapFromArray [
["groupId", _groupID],
["groupCallsign", _groupRecord getOrDefault ["callsign", _groupID]],
["requesterUid", _requesterUid],
["currentRole", _groupRecord getOrDefault ["role", "infantry"]],
["currentStatus", _groupRecord getOrDefault ["status", "available"]],
["role", [_finalRole, ""] select !_hasRole],
["status", [_finalStatus, ""] select !_hasStatus],
["mode", _mode]
];
private _profileResult = _persistenceService call ["updateGroupProfileFromContext", [_updateContext]];
if !(_profileResult getOrDefault ["success", false]) exitWith {
_result set ["message", "CAD extension rejected the group profile update."];
_result
};
private _profileData = +(_profileResult getOrDefault ["data", createHashMap]);
private _profile = +(_profileData getOrDefault ["profile", createHashMap]);
if (_profile isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension returned an invalid group profile."];
_result
};
_groupRecord set ["role", _profile getOrDefault ["role", _groupRecord getOrDefault ["role", "infantry"]]];
_groupRecord set ["status", _profile getOrDefault ["status", _groupRecord getOrDefault ["status", "available"]]];
_groupRecord set ["lastUpdate", serverTime];
_groupRegistry set [_groupID, _groupRecord];
_self set ["groupRegistry", _groupRegistry];
private _profileRegistry = _self getOrDefault ["groupProfileRegistry", createHashMap];
private _profile = +(_profileRegistry getOrDefault [_groupID, createHashMap]);
_profile set ["role", _groupRecord getOrDefault ["role", "infantry"]];
_profile set ["status", _finalStatus];
_groupRegistry set [_groupID, _groupRecord];
_self set ["groupRegistry", _groupRegistry];
_profileRegistry set [_groupID, _profile];
_self set ["groupProfileRegistry", _profileRegistry];
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendActivity", [
"group_status",
format ["%1 updated %2 to %3.", _requesterUid, _groupRecord getOrDefault ["callsign", _groupID], _finalStatus],
"",
_groupID,
_requesterUid
]];
private _activityEntry = +(_profileData getOrDefault ["activity", createHashMap]);
if (_activityEntry isNotEqualTo createHashMap) then {
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendEntry", [_activityEntry]];
};
_result set ["success", true];
_result set ["message", "Group status updated."];
_result set ["message", _profileData getOrDefault ["message", "Group profile updated."]];
_result set ["changed", _profileData getOrDefault ["changed", (_didChangeStatus || _didChangeRole)]];
_result set ["group", _groupRecord];
_result
}],
["updateGroupStatus", compileFinal {
_self call ["applyGroupProfileUpdate", [_this # 0, _this # 1, _this # 2, "", "status"]]
}],
["updateGroupRole", compileFinal {
params [["_requesterUid", "", [""]], ["_groupID", "", [""]], ["_role", "", [""]]];
private _result = createHashMapFromArray [
["success", false],
["message", "Unable to update group role."],
["group", createHashMap]
];
private _finalRole = toLowerANSI _role;
if !(_finalRole in (_self getOrDefault ["validRoles", []])) exitWith {
_result set ["message", "Invalid group role."];
_result
};
private _permissionService = _self getOrDefault ["permissionService", createHashMap];
private _isAuthorized = (_self call ["isGroupLeader", [_requesterUid, _groupID]]) || { _permissionService call ["canDispatch", [_requesterUid]] };
if !_isAuthorized exitWith {
_result set ["message", "You are not authorized to update that group role."];
_result
};
private _groupRegistry = _self call ["syncGroups", []];
private _groupRecord = +(_groupRegistry getOrDefault [_groupID, createHashMap]);
if (_groupRecord isEqualTo createHashMap) exitWith {
_result set ["message", "Group could not be resolved."];
_result
};
_groupRecord set ["role", _finalRole];
_groupRecord set ["lastUpdate", serverTime];
_groupRegistry set [_groupID, _groupRecord];
_self set ["groupRegistry", _groupRegistry];
private _profileRegistry = _self getOrDefault ["groupProfileRegistry", createHashMap];
private _profile = +(_profileRegistry getOrDefault [_groupID, createHashMap]);
_profile set ["role", _finalRole];
_profile set ["status", _groupRecord getOrDefault ["status", "available"]];
_profileRegistry set [_groupID, _profile];
_self set ["groupProfileRegistry", _profileRegistry];
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendActivity", [
"group_role",
format ["%1 updated %2 role to %3.", _requesterUid, _groupRecord getOrDefault ["callsign", _groupID], _finalRole],
"",
_groupID,
_requesterUid
]];
_result set ["success", true];
_result set ["message", "Group role updated."];
_result set ["group", _groupRecord];
_result
_self call ["applyGroupProfileUpdate", [_this # 0, _this # 1, "", _this # 2, "role"]]
}],
["updateGroupProfile", compileFinal {
_self call ["applyGroupProfileUpdate", [_this # 0, _this # 1, _this # 2, _this # 3, "profile"]]
}]
];

View File

@ -0,0 +1,209 @@
#include "..\script_component.hpp"
/*
* File: fnc_initPersistenceService.sqf
* Author: IDSolutions
* Date: 2026-03-31
* Public: No
*
* Description:
* Initializes the CAD extension-state service that bridges live SQF
* state to the Rust extension for hot CAD storage and recent history.
*
* Arguments:
* None
*
* Return Value:
* CAD persistence service object [HASHMAP OBJECT]
*
* Example:
* call forge_server_cad_fnc_initPersistenceService
*/
#pragma hemtt ignore_variables ["_self"]
GVAR(PersistenceServiceBaseClass) = compileFinal createHashMapFromArray [
["#type", "CadPersistenceServiceBaseClass"],
["makeResult", compileFinal {
params [
["_success", false, [false]],
["_data", nil, [createHashMap, []]]
];
createHashMapFromArray [
["success", _success],
["data", _data]
]
}],
["loadObject", compileFinal {
params [["_function", "", [""]], ["_arguments", [], [[]]]];
private _result = _self call ["makeResult", [false, createHashMap]];
if (_function isEqualTo "") exitWith { _result };
[_function, _arguments] call EFUNC(extension,extCall) params ["_payload", "_isSuccess"];
if (!_isSuccess || { !(_payload isEqualType "") } || { (_payload find "Error:") == 0 }) exitWith {
_result
};
private _data = fromJSON _payload;
if !(_data isEqualType createHashMap) exitWith { _result };
_result set ["success", true];
_result set ["data", _data];
_result
}],
["loadCollection", compileFinal {
params [["_function", "", [""]], ["_arguments", [], [[]]]];
private _result = _self call ["makeResult", [false, []]];
if (_function isEqualTo "") exitWith { _result };
[_function, _arguments] call EFUNC(extension,extCall) params ["_payload", "_isSuccess"];
if (!_isSuccess || { !(_payload isEqualType "") } || { (_payload find "Error:") == 0 }) exitWith {
_result
};
private _data = fromJSON _payload;
if !(_data isEqualType []) exitWith { _result };
_result set ["success", true];
_result set ["data", _data];
_result
}],
["loadRegistry", compileFinal {
params [["_function", "", [""]], ["_idField", "", [""]]];
private _result = _self call ["makeResult", [false, createHashMap]];
if (_function isEqualTo "" || { _idField isEqualTo "" }) exitWith { _result };
private _collectionResult = _self call ["loadCollection", [_function, []]];
if !(_collectionResult getOrDefault ["success", false]) exitWith { _result };
private _registry = createHashMap;
{
if !(_x isEqualType createHashMap) then { continue; };
private _entryId = _x getOrDefault [_idField, ""];
if (_entryId isEqualTo "") then { continue; };
_registry set [_entryId, +_x];
} forEach (_collectionResult getOrDefault ["data", []]);
_result set ["success", true];
_result set ["data", _registry];
_result
}],
["saveEntry", compileFinal {
params [
["_function", "", [""]],
["_entryID", "", [""]],
["_entry", createHashMap, [createHashMap]]
];
if (_function isEqualTo "" || { _entryID isEqualTo "" } || { _entry isEqualTo createHashMap }) exitWith { false };
[_function, [_entryID, toJSON _entry]] call EFUNC(extension,extCall) params ["_payload", "_isSuccess"];
_isSuccess && { !(_payload isEqualType "") || { (_payload find "Error:") != 0 } }
}],
["deleteEntry", compileFinal {
params [["_function", "", [""]], ["_entryID", "", [""]]];
if (_function isEqualTo "" || { _entryID isEqualTo "" }) exitWith { false };
[_function, [_entryID]] call EFUNC(extension,extCall) params ["_payload", "_isSuccess"];
_isSuccess && { !(_payload isEqualType "") || { (_payload find "Error:") != 0 } }
}],
["appendActivity", compileFinal {
params [["_entry", createHashMap, [createHashMap]]];
if (_entry isEqualTo createHashMap) exitWith { false };
["cad:activity:append", [toJSON _entry]] call EFUNC(extension,extCall) params ["_payload", "_isSuccess"];
_isSuccess && { !(_payload isEqualType "") || { (_payload find "Error:") != 0 } }
}],
["loadActivity", compileFinal {
_self call ["loadCollection", ["cad:activity:recent", [str 50]]]
}],
["buildHydratePayload", compileFinal {
_self call ["loadObject", ["cad:view:hydrate", [toJSON (_this # 0)]]]
}],
["loadAssignments", compileFinal {
_self call ["loadRegistry", ["cad:assignments:list", "taskId"]]
}],
["assignAssignment", compileFinal {
_self call ["loadObject", ["cad:assignments:assign", [_this # 0, toJSON (_this # 1)]]]
}],
["acknowledgeAssignment", compileFinal {
_self call ["loadObject", ["cad:assignments:acknowledge", [_this # 0, toJSON (_this # 1)]]]
}],
["declineAssignment", compileFinal {
_self call ["loadObject", ["cad:assignments:decline", [_this # 0, toJSON (_this # 1)]]]
}],
["saveAssignment", compileFinal {
_self call ["saveEntry", ["cad:assignments:upsert", _this # 0, _this # 1]]
}],
["deleteAssignment", compileFinal {
_self call ["deleteEntry", ["cad:assignments:delete", _this # 0]]
}],
["loadDispatchOrders", compileFinal {
_self call ["loadRegistry", ["cad:orders:list", "taskID"]]
}],
["createDispatchOrder", compileFinal {
params [
["_orderSeed", createHashMap, [createHashMap]],
["_assignmentSeed", createHashMap, [createHashMap]]
];
_self call ["loadObject", ["cad:orders:create", [toJSON (createHashMapFromArray [
["order", _orderSeed],
["assignment", _assignmentSeed]
])]]]
}],
["createDispatchOrderFromContext", compileFinal {
_self call ["loadObject", ["cad:orders:create_from_context", [toJSON (_this # 0)]]]
}],
["closeDispatchOrder", compileFinal {
_self call ["loadObject", ["cad:orders:close", [_this # 0]]]
}],
["saveDispatchOrder", compileFinal {
_self call ["saveEntry", ["cad:orders:upsert", _this # 0, _this # 1]]
}],
["deleteDispatchOrder", compileFinal {
_self call ["deleteEntry", ["cad:orders:delete", _this # 0]]
}],
["loadRequests", compileFinal {
_self call ["loadRegistry", ["cad:requests:list", "requestId"]]
}],
["submitSupportRequest", compileFinal {
_self call ["loadObject", ["cad:requests:submit", [toJSON (_this # 0)]]]
}],
["submitSupportRequestFromContext", compileFinal {
_self call ["loadObject", ["cad:requests:submit_from_context", [toJSON (_this # 0)]]]
}],
["closeSupportRequest", compileFinal {
_self call ["loadObject", ["cad:requests:close", [_this # 0]]]
}],
["saveRequest", compileFinal {
_self call ["saveEntry", ["cad:requests:upsert", _this # 0, _this # 1]]
}],
["deleteRequest", compileFinal {
_self call ["deleteEntry", ["cad:requests:delete", _this # 0]]
}],
["loadGroupProfiles", compileFinal {
_self call ["loadRegistry", ["cad:profiles:list", "groupId"]]
}],
["buildGroups", compileFinal {
_self call ["loadCollection", ["cad:groups:build", [toJSON (createHashMapFromArray [
["liveGroups", _this # 0]
])]]]
}],
["updateGroupProfileFromContext", compileFinal {
_self call ["loadObject", ["cad:profiles:update_from_context", [toJSON (_this # 0)]]]
}],
["saveGroupProfile", compileFinal {
_self call ["saveEntry", ["cad:profiles:upsert", _this # 0, _this # 1]]
}],
["deleteGroupProfile", compileFinal {
_self call ["deleteEntry", ["cad:profiles:delete", _this # 0]]
}]
];
createHashMapObject [GVAR(PersistenceServiceBaseClass)]

View File

@ -0,0 +1,208 @@
#include "..\script_component.hpp"
/*
* File: fnc_initRequestRepository.sqf
* Author: IDSolutions
* Date: 2026-03-31
* Public: No
*
* Description:
* Initializes the CAD request repository for structured support
* requests submitted by groups and triaged by dispatch.
*
* Arguments:
* None
*
* Return Value:
* CAD request repository object [HASHMAP OBJECT]
*
* Example:
* call forge_server_cad_fnc_initRequestRepository
*/
#pragma hemtt ignore_variables ["_self"]
GVAR(RequestRepositoryBaseClass) = compileFinal createHashMapFromArray [
["#type", "CadRequestRepositoryBaseClass"],
["#create", compileFinal {
_self set ["requestRegistry", createHashMap];
_self set ["persistenceLoaded", false];
_self set ["validTypes", [
"medevac_9line",
"ace_lace",
"fire_support",
"air_support",
"logreq"
]];
_self set ["validPriorities", [
"routine",
"priority",
"emergency"
]];
}],
["restorePersistedState", compileFinal {
if (_self getOrDefault ["persistenceLoaded", false]) exitWith { true };
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith { false };
private _result = _persistenceService call ["loadRequests", []];
if !(_result getOrDefault ["success", false]) exitWith { false };
private _requestRegistry = +(_result getOrDefault ["data", createHashMap]);
_self set ["requestRegistry", _requestRegistry];
_self set ["persistenceLoaded", true];
true
}],
["submitRequest", compileFinal {
params [
["_requesterUid", "", [""]],
["_type", "", [""]],
["_fields", createHashMap, [createHashMap]],
["_priority", "priority", [""]]
];
private _result = createHashMapFromArray [
["success", false],
["message", "Unable to submit support request."],
["request", createHashMap]
];
_self call ["restorePersistedState", []];
private _finalType = toLowerANSI _type;
if !(_finalType in (_self getOrDefault ["validTypes", []])) exitWith {
_result set ["message", "Invalid support request type."];
_result
};
private _groupRepository = _self getOrDefault ["groupRepository", createHashMap];
private _groupID = _groupRepository call ["getPlayerGroupId", [_requesterUid]];
if (_groupID isEqualTo "") exitWith {
_result set ["message", "You are not currently assigned to a group."];
_result
};
if !(_groupRepository call ["isGroupLeader", [_requesterUid, _groupID]]) exitWith {
_result set ["message", "Only the current group leader can submit support requests."];
_result
};
private _groupRecord = _groupRepository call ["getGroupRecord", [_groupID]];
if (_groupRecord isEqualTo createHashMap) exitWith {
_result set ["message", "Your group could not be resolved."];
_result
};
private _validPriorities = _self getOrDefault ["validPriorities", []];
private _finalPriority = toLowerANSI _priority;
if !(_finalPriority in _validPriorities) then {
_finalPriority = "priority";
};
private _requestContext = createHashMapFromArray [
["type", _finalType],
["fields", +_fields],
["groupId", _groupID],
["groupCallsign", _groupRecord getOrDefault ["callsign", _groupID]],
["submittedByUid", _requesterUid],
["submittedByName", _groupRecord getOrDefault ["leaderName", _requesterUid]],
["priority", _finalPriority],
["position", +(_groupRecord getOrDefault ["position", []])],
["createdAt", serverTime]
];
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension state is unavailable."];
_result
};
private _submitResult = _persistenceService call ["submitSupportRequestFromContext", [_requestContext]];
if !(_submitResult getOrDefault ["success", false]) exitWith {
_result set ["message", "CAD extension rejected the support request."];
_result
};
private _submitData = +(_submitResult getOrDefault ["data", createHashMap]);
private _request = +(_submitData getOrDefault ["request", createHashMap]);
private _requestID = _request getOrDefault ["requestId", ""];
if (_requestID isEqualTo "") exitWith {
_result set ["message", "CAD extension returned an invalid support request."];
_result
};
private _requestRegistry = _self getOrDefault ["requestRegistry", createHashMap];
_requestRegistry set [_requestID, _request];
_self set ["requestRegistry", _requestRegistry];
private _activityEntry = +(_submitData getOrDefault ["activity", createHashMap]);
if (_activityEntry isNotEqualTo createHashMap) then {
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendEntry", [_activityEntry]];
};
_result set ["success", true];
_result set ["message", _submitData getOrDefault ["message", "Support request submitted."]];
_result set ["request", _request];
_result
}],
["closeRequest", compileFinal {
params [["_requesterUid", "", [""]], ["_requestID", "", [""]]];
private _result = createHashMapFromArray [
["success", false],
["message", "Unable to close support request."],
["request", createHashMap]
];
_self call ["restorePersistedState", []];
private _requestRegistry = _self getOrDefault ["requestRegistry", createHashMap];
private _request = +(_requestRegistry getOrDefault [_requestID, createHashMap]);
if (_request isEqualTo createHashMap) exitWith {
_result set ["message", "Support request could not be resolved."];
_result
};
private _permissionService = _self getOrDefault ["permissionService", createHashMap];
private _groupRepository = _self getOrDefault ["groupRepository", createHashMap];
private _groupID = _request getOrDefault ["groupId", ""];
private _isAuthorized = (_permissionService call ["canDispatch", [_requesterUid]]) || { _groupRepository call ["isGroupLeader", [_requesterUid, _groupID]] };
if !_isAuthorized exitWith {
_result set ["message", "You are not authorized to close that support request."];
_result
};
private _persistenceService = _self getOrDefault ["persistenceService", createHashMap];
if (_persistenceService isEqualTo createHashMap) exitWith {
_result set ["message", "CAD extension state is unavailable."];
_result
};
private _closeResult = _persistenceService call ["closeSupportRequest", [_requestID]];
if !(_closeResult getOrDefault ["success", false]) exitWith {
_result set ["message", "CAD extension rejected the support request close."];
_result
};
private _closeData = +(_closeResult getOrDefault ["data", createHashMap]);
_request = +(_closeData getOrDefault ["request", _request]);
_requestRegistry deleteAt _requestID;
_self set ["requestRegistry", _requestRegistry];
private _activityEntry = +(_closeData getOrDefault ["activity", createHashMap]);
if (_activityEntry isNotEqualTo createHashMap) then {
_activityEntry set ["actorUid", _requesterUid];
private _activityRepository = _self getOrDefault ["activityRepository", createHashMap];
_activityRepository call ["appendEntry", [_activityEntry]];
};
_result set ["success", true];
_result set ["message", _closeData getOrDefault ["message", "Support request closed."]];
_result set ["request", _request];
_result
}]
];
createHashMapObject [GVAR(RequestRepositoryBaseClass)]

View File

@ -5,31 +5,3 @@ PREP_RECOMPILE_START;
PREP_RECOMPILE_END;
call FUNC(initTaskStore);
[QGVAR(requestTaskCatalog), {
params [["_uid", "", [""]]];
if (_uid isEqualTo "") exitWith {
["WARNING", "Task catalog request received with empty UID."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
if (_player isEqualTo objNull) exitWith {};
[CRPC(cad,responseTaskCatalog), [GVAR(TaskStore) call ["getActiveTaskCatalog", []]], _player] call CFUNC(targetEvent);
}] call CFUNC(addEventHandler);
[QGVAR(requestAcceptTask), {
params [["_uid", "", [""]], ["_taskID", "", [""]]];
if (_uid isEqualTo "" || { _taskID isEqualTo "" }) exitWith {
["WARNING", "Invalid task accept request payload."] call EFUNC(common,log);
};
private _player = [_uid] call EFUNC(common,getPlayer);
if (_player isEqualTo objNull) exitWith {};
private _result = GVAR(TaskStore) call ["acceptTask", [_taskID, _uid]];
[CRPC(cad,responseTaskAccept), [_result], _player] call CFUNC(targetEvent);
[CRPC(cad,responseTaskCatalog), [GVAR(TaskStore) call ["getActiveTaskCatalog", []]], _player] call CFUNC(targetEvent);
}] call CFUNC(addEventHandler);

View File

@ -0,0 +1,183 @@
//! CAD hot-state operations for the Arma 3 server extension.
//!
//! The extension owns the in-memory CAD state store, while the shared service
//! layer handles mutation rules and hydrate shaping. This keeps the extension
//! surface thin and aligned with the workspace architecture.
use arma_rs::Group;
use forge_repositories::InMemoryCadRepository;
use forge_services::CadStateService;
use serde::Serialize;
use std::sync::LazyLock;
static CAD_SERVICE: LazyLock<CadStateService<InMemoryCadRepository>> =
LazyLock::new(|| CadStateService::new(InMemoryCadRepository::new()));
pub fn group() -> Group {
Group::new()
.group(
"activity",
Group::new()
.command("append", append_activity)
.command("recent", recent_activity),
)
.group(
"assignments",
Group::new()
.command("list", list_assignments)
.command("assign", assign_assignment)
.command("acknowledge", acknowledge_assignment)
.command("decline", decline_assignment)
.command("upsert", upsert_assignment)
.command("delete", delete_assignment),
)
.group(
"orders",
Group::new()
.command("list", list_orders)
.command("create", create_order)
.command("create_from_context", create_order_from_context)
.command("close", close_order)
.command("upsert", upsert_order)
.command("delete", delete_order),
)
.group(
"requests",
Group::new()
.command("list", list_requests)
.command("submit", submit_request)
.command("submit_from_context", submit_request_from_context)
.command("close", close_request)
.command("upsert", upsert_request)
.command("delete", delete_request),
)
.group(
"profiles",
Group::new()
.command("list", list_profiles)
.command("update_from_context", update_profile_from_context)
.command("upsert", upsert_profile)
.command("delete", delete_profile),
)
.group("groups", Group::new().command("build", build_groups))
.group("view", Group::new().command("hydrate", hydrate_view))
}
fn append_activity(json_data: String) -> String {
serialize_ok(CAD_SERVICE.append_activity(json_data))
}
fn recent_activity(limit: String) -> String {
serialize_json(CAD_SERVICE.recent_activity(limit))
}
fn list_assignments() -> String {
serialize_json(CAD_SERVICE.list_assignments())
}
fn assign_assignment(entry_id: String, json_data: String) -> String {
serialize_json(CAD_SERVICE.assign_assignment(entry_id, json_data))
}
fn acknowledge_assignment(entry_id: String, json_data: String) -> String {
serialize_json(CAD_SERVICE.acknowledge_assignment(entry_id, json_data))
}
fn decline_assignment(entry_id: String, json_data: String) -> String {
serialize_json(CAD_SERVICE.decline_assignment(entry_id, json_data))
}
fn upsert_assignment(entry_id: String, json_data: String) -> String {
serialize_ok(CAD_SERVICE.upsert_assignment(entry_id, json_data))
}
fn delete_assignment(entry_id: String) -> String {
serialize_ok(CAD_SERVICE.delete_assignment(entry_id))
}
fn list_orders() -> String {
serialize_json(CAD_SERVICE.list_orders())
}
fn create_order(json_data: String) -> String {
serialize_json(CAD_SERVICE.create_order(json_data))
}
fn create_order_from_context(json_data: String) -> String {
serialize_json(CAD_SERVICE.create_order_from_context(json_data))
}
fn close_order(entry_id: String) -> String {
serialize_json(CAD_SERVICE.close_order(entry_id))
}
fn upsert_order(entry_id: String, json_data: String) -> String {
serialize_ok(CAD_SERVICE.upsert_order(entry_id, json_data))
}
fn delete_order(entry_id: String) -> String {
serialize_ok(CAD_SERVICE.delete_order(entry_id))
}
fn list_requests() -> String {
serialize_json(CAD_SERVICE.list_requests())
}
fn submit_request(json_data: String) -> String {
serialize_json(CAD_SERVICE.submit_request(json_data))
}
fn submit_request_from_context(json_data: String) -> String {
serialize_json(CAD_SERVICE.submit_request_from_context(json_data))
}
fn close_request(entry_id: String) -> String {
serialize_json(CAD_SERVICE.close_request(entry_id))
}
fn upsert_request(entry_id: String, json_data: String) -> String {
serialize_ok(CAD_SERVICE.upsert_request(entry_id, json_data))
}
fn delete_request(entry_id: String) -> String {
serialize_ok(CAD_SERVICE.delete_request(entry_id))
}
fn list_profiles() -> String {
serialize_json(CAD_SERVICE.list_profiles())
}
fn update_profile_from_context(json_data: String) -> String {
serialize_json(CAD_SERVICE.update_profile_from_context(json_data))
}
fn upsert_profile(entry_id: String, json_data: String) -> String {
serialize_ok(CAD_SERVICE.upsert_profile(entry_id, json_data))
}
fn delete_profile(entry_id: String) -> String {
serialize_ok(CAD_SERVICE.delete_profile(entry_id))
}
fn build_groups(json_data: String) -> String {
serialize_json(CAD_SERVICE.build_groups(json_data))
}
fn hydrate_view(json_data: String) -> String {
serialize_json(CAD_SERVICE.build_hydrate_payload(json_data))
}
fn serialize_ok(result: Result<(), String>) -> String {
match result {
Ok(()) => "OK".to_string(),
Err(error) => format!("Error: {error}"),
}
}
fn serialize_json<T: Serialize>(result: Result<T, String>) -> String {
match result {
Ok(value) => serde_json::to_string(&value)
.unwrap_or_else(|error| format!("Error: Failed to serialize CAD state: {error}")),
Err(error) => format!("Error: {error}"),
}
}

View File

@ -14,6 +14,7 @@ use tokio::sync::RwLock as TokioRwLock;
pub mod actor;
pub mod adapters;
pub mod bank;
pub mod cad;
pub mod garage;
pub mod helpers;
pub mod icom;
@ -63,6 +64,7 @@ fn init() -> Extension {
.group("redis", redis::group())
.group("actor", actor::group())
.group("bank", bank::group())
.group("cad", cad::group())
.group("garage", garage::group())
.group("icom", icom::group())
.group("locker", locker::group())

227
lib/models/src/cad.rs Normal file
View File

@ -0,0 +1,227 @@
use serde::{Deserialize, Serialize};
use serde_json::{Map, Value};
pub type CadJsonMap = Map<String, Value>;
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(transparent)]
pub struct CadRecord {
pub fields: CadJsonMap,
}
impl CadRecord {
pub fn into_value(self) -> Value {
Value::Object(self.fields)
}
pub fn to_value(&self) -> Value {
Value::Object(self.fields.clone())
}
pub fn is_empty(&self) -> bool {
self.fields.is_empty()
}
pub fn merge(mut self, patch: CadRecord) -> Self {
for (key, value) in patch.fields {
self.fields.insert(key, value);
}
self
}
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadDispatchOrderCreateSeed {
#[serde(default)]
pub order: CadRecord,
#[serde(default)]
pub assignment: CadRecord,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadDispatchOrderContextSeed {
#[serde(default)]
pub assignee_group_id: String,
#[serde(default)]
pub assignee_group_callsign: String,
#[serde(default)]
pub target_group_id: String,
#[serde(default)]
pub target_group_callsign: String,
#[serde(default)]
pub target_position: Value,
#[serde(default)]
pub created_by_uid: String,
#[serde(default)]
pub created_by_name: String,
#[serde(default)]
pub note: String,
#[serde(default)]
pub priority: String,
#[serde(default)]
pub created_at: f64,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadDispatchOrderMutationResult {
#[serde(default)]
pub task_id: String,
#[serde(default)]
pub order: Value,
#[serde(default)]
pub assignment: Value,
#[serde(default)]
pub message: String,
#[serde(default)]
pub activity: CadActivityEntry,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadActivityEntry {
#[serde(default)]
#[serde(rename = "type")]
pub entry_type: String,
#[serde(default)]
pub message: String,
#[serde(default)]
pub task_id: String,
#[serde(default)]
pub group_id: String,
#[serde(default)]
pub actor_uid: String,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadAssignmentMutationResult {
#[serde(default)]
pub assignment: Value,
#[serde(default)]
pub message: String,
#[serde(default)]
pub activity: CadActivityEntry,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadRequestMutationResult {
#[serde(default)]
pub request: Value,
#[serde(default)]
pub message: String,
#[serde(default)]
pub activity: CadActivityEntry,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadGroupProfileUpdateSeed {
#[serde(default)]
pub group_id: String,
#[serde(default)]
pub group_callsign: String,
#[serde(default)]
pub requester_uid: String,
#[serde(default)]
pub current_role: String,
#[serde(default)]
pub current_status: String,
#[serde(default)]
pub role: String,
#[serde(default)]
pub status: String,
#[serde(default)]
pub mode: String,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadGroupProfileMutationResult {
#[serde(default)]
pub profile: Value,
#[serde(default)]
pub message: String,
#[serde(default)]
pub activity: CadActivityEntry,
#[serde(default)]
pub changed: bool,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadSupportRequestSubmitSeed {
#[serde(rename = "type")]
#[serde(default)]
pub request_type: String,
#[serde(default)]
pub fields: CadRecord,
#[serde(default)]
pub group_id: String,
#[serde(default)]
pub group_callsign: String,
#[serde(default)]
pub submitted_by_uid: String,
#[serde(default)]
pub submitted_by_name: String,
#[serde(default)]
pub priority: String,
#[serde(default)]
pub position: Value,
#[serde(default)]
pub created_at: f64,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadGroupBuildSeed {
#[serde(default)]
pub live_groups: Vec<Value>,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadSession {
#[serde(default)]
pub uid: String,
#[serde(default)]
pub org_id: String,
#[serde(default)]
pub is_dispatcher: bool,
#[serde(default)]
pub group_id: String,
#[serde(default)]
pub is_leader: bool,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadHydrateSeed {
#[serde(default)]
pub groups: Vec<Value>,
#[serde(default)]
pub active_tasks: Vec<Value>,
#[serde(default)]
pub session: CadSession,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct CadHydratePayload {
#[serde(default)]
pub groups: Vec<Value>,
#[serde(default)]
pub contracts: Vec<Value>,
#[serde(default)]
pub requests: Vec<Value>,
#[serde(default)]
pub assignments: Vec<Value>,
#[serde(default)]
pub activity: Vec<Value>,
#[serde(default)]
pub session: CadSession,
}

View File

@ -1,5 +1,6 @@
pub mod actor;
pub mod bank;
pub mod cad;
pub mod garage;
pub mod locker;
pub mod org;
@ -8,6 +9,12 @@ pub mod v_locker;
pub use actor::Actor;
pub use bank::Bank;
pub use cad::{
CadActivityEntry, CadAssignmentMutationResult, CadDispatchOrderContextSeed,
CadDispatchOrderCreateSeed, CadDispatchOrderMutationResult, CadGroupBuildSeed,
CadGroupProfileMutationResult, CadGroupProfileUpdateSeed, CadHydratePayload, CadHydrateSeed,
CadJsonMap, CadRecord, CadRequestMutationResult, CadSession, CadSupportRequestSubmitSeed,
};
pub use garage::{Garage, HitPoints, Vehicle};
pub use locker::{Item, Locker};
pub use org::{CreditLineSummary, MemberSummary, Org, OrgAssetEntry, OrgFleetEntry};

236
lib/repositories/src/cad.rs Normal file
View File

@ -0,0 +1,236 @@
use forge_models::CadRecord;
use serde_json::Value;
use std::collections::HashMap;
use std::sync::{Arc, RwLock};
const CAD_ACTIVITY_LIMIT: usize = 200;
pub trait CadRepository: Send + Sync {
fn append_activity(&self, entry: Value) -> Result<(), String>;
fn recent_activity(&self, limit: usize) -> Result<Vec<Value>, String>;
fn snapshot_activity(&self) -> Result<Vec<Value>, String>;
fn list_assignments(&self) -> Result<HashMap<String, CadRecord>, String>;
fn get_assignment(&self, id: &str) -> Result<Option<CadRecord>, String>;
fn save_assignment(&self, id: String, entry: CadRecord) -> Result<(), String>;
fn delete_assignment(&self, id: &str) -> Result<(), String>;
fn list_orders(&self) -> Result<HashMap<String, CadRecord>, String>;
fn get_order(&self, id: &str) -> Result<Option<CadRecord>, String>;
fn save_order(&self, id: String, entry: CadRecord) -> Result<(), String>;
fn delete_order(&self, id: &str) -> Result<(), String>;
fn list_requests(&self) -> Result<HashMap<String, CadRecord>, String>;
fn get_request(&self, id: &str) -> Result<Option<CadRecord>, String>;
fn save_request(&self, id: String, entry: CadRecord) -> Result<(), String>;
fn delete_request(&self, id: &str) -> Result<(), String>;
fn list_profiles(&self) -> Result<HashMap<String, CadRecord>, String>;
fn get_profile(&self, id: &str) -> Result<Option<CadRecord>, String>;
fn save_profile(&self, id: String, entry: CadRecord) -> Result<(), String>;
fn delete_profile(&self, id: &str) -> Result<(), String>;
fn next_order_id(&self) -> Result<String, String>;
fn next_request_id(&self) -> Result<String, String>;
}
#[derive(Debug, Default)]
struct CadState {
activity: Vec<Value>,
assignments: HashMap<String, CadRecord>,
orders: HashMap<String, CadRecord>,
requests: HashMap<String, CadRecord>,
profiles: HashMap<String, CadRecord>,
order_sequence: u64,
request_sequence: u64,
}
#[derive(Clone, Debug, Default)]
pub struct InMemoryCadRepository {
state: Arc<RwLock<CadState>>,
}
impl InMemoryCadRepository {
pub fn new() -> Self {
Self::default()
}
}
impl CadRepository for InMemoryCadRepository {
fn append_activity(&self, entry: Value) -> Result<(), String> {
let mut state = self
.state
.write()
.map_err(|_| "CAD activity state lock poisoned.".to_string())?;
state.activity.push(entry);
if state.activity.len() > CAD_ACTIVITY_LIMIT {
let overflow = state.activity.len() - CAD_ACTIVITY_LIMIT;
state.activity.drain(0..overflow);
}
Ok(())
}
fn recent_activity(&self, limit: usize) -> Result<Vec<Value>, String> {
let state = self
.state
.read()
.map_err(|_| "CAD activity state lock poisoned.".to_string())?;
let start = state.activity.len().saturating_sub(limit);
Ok(state.activity[start..].to_vec())
}
fn snapshot_activity(&self) -> Result<Vec<Value>, String> {
self.state
.read()
.map(|state| state.activity.clone())
.map_err(|_| "CAD activity state lock poisoned.".to_string())
}
fn list_assignments(&self) -> Result<HashMap<String, CadRecord>, String> {
self.state
.read()
.map(|state| state.assignments.clone())
.map_err(|_| "CAD assignments state lock poisoned.".to_string())
}
fn get_assignment(&self, id: &str) -> Result<Option<CadRecord>, String> {
self.state
.read()
.map(|state| state.assignments.get(id).cloned())
.map_err(|_| "CAD assignments state lock poisoned.".to_string())
}
fn save_assignment(&self, id: String, entry: CadRecord) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD assignments state lock poisoned.".to_string())?
.assignments
.insert(id, entry);
Ok(())
}
fn delete_assignment(&self, id: &str) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD assignments state lock poisoned.".to_string())?
.assignments
.remove(id);
Ok(())
}
fn list_orders(&self) -> Result<HashMap<String, CadRecord>, String> {
self.state
.read()
.map(|state| state.orders.clone())
.map_err(|_| "CAD orders state lock poisoned.".to_string())
}
fn get_order(&self, id: &str) -> Result<Option<CadRecord>, String> {
self.state
.read()
.map(|state| state.orders.get(id).cloned())
.map_err(|_| "CAD orders state lock poisoned.".to_string())
}
fn save_order(&self, id: String, entry: CadRecord) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD orders state lock poisoned.".to_string())?
.orders
.insert(id, entry);
Ok(())
}
fn delete_order(&self, id: &str) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD orders state lock poisoned.".to_string())?
.orders
.remove(id);
Ok(())
}
fn list_requests(&self) -> Result<HashMap<String, CadRecord>, String> {
self.state
.read()
.map(|state| state.requests.clone())
.map_err(|_| "CAD requests state lock poisoned.".to_string())
}
fn get_request(&self, id: &str) -> Result<Option<CadRecord>, String> {
self.state
.read()
.map(|state| state.requests.get(id).cloned())
.map_err(|_| "CAD requests state lock poisoned.".to_string())
}
fn save_request(&self, id: String, entry: CadRecord) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD requests state lock poisoned.".to_string())?
.requests
.insert(id, entry);
Ok(())
}
fn delete_request(&self, id: &str) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD requests state lock poisoned.".to_string())?
.requests
.remove(id);
Ok(())
}
fn list_profiles(&self) -> Result<HashMap<String, CadRecord>, String> {
self.state
.read()
.map(|state| state.profiles.clone())
.map_err(|_| "CAD profiles state lock poisoned.".to_string())
}
fn get_profile(&self, id: &str) -> Result<Option<CadRecord>, String> {
self.state
.read()
.map(|state| state.profiles.get(id).cloned())
.map_err(|_| "CAD profiles state lock poisoned.".to_string())
}
fn save_profile(&self, id: String, entry: CadRecord) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD profiles state lock poisoned.".to_string())?
.profiles
.insert(id, entry);
Ok(())
}
fn delete_profile(&self, id: &str) -> Result<(), String> {
self.state
.write()
.map_err(|_| "CAD profiles state lock poisoned.".to_string())?
.profiles
.remove(id);
Ok(())
}
fn next_order_id(&self) -> Result<String, String> {
let mut state = self
.state
.write()
.map_err(|_| "CAD order sequence lock poisoned.".to_string())?;
state.order_sequence += 1;
Ok(format!("cad-order:{}", state.order_sequence))
}
fn next_request_id(&self) -> Result<String, String> {
let mut state = self
.state
.write()
.map_err(|_| "CAD request sequence lock poisoned.".to_string())?;
state.request_sequence += 1;
Ok(format!("cad-request:{}", state.request_sequence))
}
}

View File

@ -1,5 +1,6 @@
pub mod actor;
pub mod bank;
pub mod cad;
pub mod garage;
pub mod locker;
pub mod org;
@ -8,6 +9,7 @@ pub mod v_locker;
pub use actor::{ActorRepository, RedisActorRepository};
pub use bank::{BankRepository, RedisBankRepository};
pub use cad::{CadRepository, InMemoryCadRepository};
pub use garage::{GarageRepository, RedisGarageRepository};
pub use locker::{LockerRepository, RedisLockerRepository};
pub use org::{OrgRepository, RedisOrgRepository};

1092
lib/services/src/cad.rs Normal file

File diff suppressed because it is too large Load Diff

View File

@ -1,5 +1,6 @@
pub mod actor;
pub mod bank;
pub mod cad;
pub mod garage;
pub mod locker;
pub mod org;
@ -8,6 +9,7 @@ pub mod v_locker;
pub use actor::ActorService;
pub use bank::BankService;
pub use cad::{CadStateService, CadViewService};
pub use garage::GarageService;
pub use locker::LockerService;
pub use org::OrgService;