@webex/contact-center 3.12.0-next.8 → 3.12.0-task-refactor.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (200) hide show
  1. package/AGENTS.md +438 -0
  2. package/ai-docs/README.md +131 -0
  3. package/ai-docs/RULES.md +455 -0
  4. package/ai-docs/patterns/event-driven-patterns.md +485 -0
  5. package/ai-docs/patterns/testing-patterns.md +480 -0
  6. package/ai-docs/patterns/typescript-patterns.md +365 -0
  7. package/ai-docs/templates/README.md +102 -0
  8. package/ai-docs/templates/documentation/create-agents-md.md +240 -0
  9. package/ai-docs/templates/documentation/create-architecture-md.md +295 -0
  10. package/ai-docs/templates/existing-service/bug-fix.md +254 -0
  11. package/ai-docs/templates/existing-service/feature-enhancement.md +450 -0
  12. package/ai-docs/templates/new-method/00-master.md +80 -0
  13. package/ai-docs/templates/new-method/01-requirements.md +232 -0
  14. package/ai-docs/templates/new-method/02-implementation.md +295 -0
  15. package/ai-docs/templates/new-method/03-tests.md +201 -0
  16. package/ai-docs/templates/new-method/04-validation.md +141 -0
  17. package/ai-docs/templates/new-service/00-master.md +109 -0
  18. package/ai-docs/templates/new-service/01-pre-questions.md +159 -0
  19. package/ai-docs/templates/new-service/02-code-generation.md +346 -0
  20. package/ai-docs/templates/new-service/03-integration.md +178 -0
  21. package/ai-docs/templates/new-service/04-test-generation.md +205 -0
  22. package/ai-docs/templates/new-service/05-validation.md +145 -0
  23. package/dist/cc.js +65 -123
  24. package/dist/cc.js.map +1 -1
  25. package/dist/constants.js +13 -2
  26. package/dist/constants.js.map +1 -1
  27. package/dist/index.js +13 -5
  28. package/dist/index.js.map +1 -1
  29. package/dist/metrics/behavioral-events.js +26 -13
  30. package/dist/metrics/behavioral-events.js.map +1 -1
  31. package/dist/metrics/constants.js +7 -6
  32. package/dist/metrics/constants.js.map +1 -1
  33. package/dist/services/ApiAiAssistant.js +0 -3
  34. package/dist/services/ApiAiAssistant.js.map +1 -1
  35. package/dist/services/config/Util.js +2 -3
  36. package/dist/services/config/Util.js.map +1 -1
  37. package/dist/services/config/types.js +16 -14
  38. package/dist/services/config/types.js.map +1 -1
  39. package/dist/services/constants.js +0 -1
  40. package/dist/services/constants.js.map +1 -1
  41. package/dist/services/core/Err.js.map +1 -1
  42. package/dist/services/core/Utils.js +79 -55
  43. package/dist/services/core/Utils.js.map +1 -1
  44. package/dist/services/core/aqm-reqs.js +17 -92
  45. package/dist/services/core/aqm-reqs.js.map +1 -1
  46. package/dist/services/core/websocket/WebSocketManager.js +5 -25
  47. package/dist/services/core/websocket/WebSocketManager.js.map +1 -1
  48. package/dist/services/core/websocket/types.js.map +1 -1
  49. package/dist/services/index.js +1 -2
  50. package/dist/services/index.js.map +1 -1
  51. package/dist/services/task/Task.js +644 -0
  52. package/dist/services/task/Task.js.map +1 -0
  53. package/dist/services/task/TaskFactory.js +45 -0
  54. package/dist/services/task/TaskFactory.js.map +1 -0
  55. package/dist/services/task/TaskManager.js +556 -532
  56. package/dist/services/task/TaskManager.js.map +1 -1
  57. package/dist/services/task/TaskUtils.js +132 -28
  58. package/dist/services/task/TaskUtils.js.map +1 -1
  59. package/dist/services/task/constants.js +7 -6
  60. package/dist/services/task/constants.js.map +1 -1
  61. package/dist/services/task/dialer.js +0 -51
  62. package/dist/services/task/dialer.js.map +1 -1
  63. package/dist/services/task/digital/Digital.js +77 -0
  64. package/dist/services/task/digital/Digital.js.map +1 -0
  65. package/dist/services/task/state-machine/TaskStateMachine.js +634 -0
  66. package/dist/services/task/state-machine/TaskStateMachine.js.map +1 -0
  67. package/dist/services/task/state-machine/actions.js +366 -0
  68. package/dist/services/task/state-machine/actions.js.map +1 -0
  69. package/dist/services/task/state-machine/constants.js +139 -0
  70. package/dist/services/task/state-machine/constants.js.map +1 -0
  71. package/dist/services/task/state-machine/guards.js +256 -0
  72. package/dist/services/task/state-machine/guards.js.map +1 -0
  73. package/dist/services/task/state-machine/index.js +53 -0
  74. package/dist/services/task/state-machine/index.js.map +1 -0
  75. package/dist/services/task/state-machine/types.js +54 -0
  76. package/dist/services/task/state-machine/types.js.map +1 -0
  77. package/dist/services/task/state-machine/uiControlsComputer.js +369 -0
  78. package/dist/services/task/state-machine/uiControlsComputer.js.map +1 -0
  79. package/dist/services/task/taskDataNormalizer.js +99 -0
  80. package/dist/services/task/taskDataNormalizer.js.map +1 -0
  81. package/dist/services/task/types.js +157 -18
  82. package/dist/services/task/types.js.map +1 -1
  83. package/dist/services/task/voice/Voice.js +1031 -0
  84. package/dist/services/task/voice/Voice.js.map +1 -0
  85. package/dist/services/task/voice/WebRTC.js +149 -0
  86. package/dist/services/task/voice/WebRTC.js.map +1 -0
  87. package/dist/types/cc.d.ts +4 -33
  88. package/dist/types/constants.d.ts +13 -2
  89. package/dist/types/index.d.ts +11 -5
  90. package/dist/types/metrics/constants.d.ts +5 -3
  91. package/dist/types/services/ApiAiAssistant.d.ts +1 -1
  92. package/dist/types/services/config/types.d.ts +97 -25
  93. package/dist/types/services/core/Err.d.ts +0 -2
  94. package/dist/types/services/core/Utils.d.ts +25 -23
  95. package/dist/types/services/core/aqm-reqs.d.ts +0 -49
  96. package/dist/types/services/core/websocket/WebSocketManager.d.ts +1 -1
  97. package/dist/types/services/core/websocket/connection-service.d.ts +0 -1
  98. package/dist/types/services/core/websocket/types.d.ts +1 -1
  99. package/dist/types/services/index.d.ts +1 -1
  100. package/dist/types/services/task/Task.d.ts +146 -0
  101. package/dist/types/services/task/TaskFactory.d.ts +12 -0
  102. package/dist/types/services/task/TaskUtils.d.ts +39 -8
  103. package/dist/types/services/task/constants.d.ts +5 -4
  104. package/dist/types/services/task/dialer.d.ts +0 -15
  105. package/dist/types/services/task/digital/Digital.d.ts +22 -0
  106. package/dist/types/services/task/state-machine/TaskStateMachine.d.ts +906 -0
  107. package/dist/types/services/task/state-machine/actions.d.ts +8 -0
  108. package/dist/types/services/task/state-machine/constants.d.ts +91 -0
  109. package/dist/types/services/task/state-machine/guards.d.ts +78 -0
  110. package/dist/types/services/task/state-machine/index.d.ts +13 -0
  111. package/dist/types/services/task/state-machine/types.d.ts +256 -0
  112. package/dist/types/services/task/state-machine/uiControlsComputer.d.ts +9 -0
  113. package/dist/types/services/task/taskDataNormalizer.d.ts +10 -0
  114. package/dist/types/services/task/types.d.ts +539 -88
  115. package/dist/types/services/task/voice/Voice.d.ts +183 -0
  116. package/dist/types/services/task/voice/WebRTC.d.ts +53 -0
  117. package/dist/types/types.d.ts +68 -0
  118. package/dist/types/webex.d.ts +1 -0
  119. package/dist/types.js +70 -0
  120. package/dist/types.js.map +1 -1
  121. package/dist/webex.js +14 -2
  122. package/dist/webex.js.map +1 -1
  123. package/package.json +14 -11
  124. package/src/cc.ts +91 -177
  125. package/src/constants.ts +13 -2
  126. package/src/index.ts +14 -5
  127. package/src/metrics/ai-docs/AGENTS.md +348 -0
  128. package/src/metrics/ai-docs/ARCHITECTURE.md +336 -0
  129. package/src/metrics/behavioral-events.ts +28 -14
  130. package/src/metrics/constants.ts +7 -8
  131. package/src/services/ApiAiAssistant.ts +2 -4
  132. package/src/services/agent/ai-docs/AGENTS.md +238 -0
  133. package/src/services/agent/ai-docs/ARCHITECTURE.md +302 -0
  134. package/src/services/ai-docs/AGENTS.md +384 -0
  135. package/src/services/config/Util.ts +2 -3
  136. package/src/services/config/ai-docs/AGENTS.md +253 -0
  137. package/src/services/config/ai-docs/ARCHITECTURE.md +424 -0
  138. package/src/services/config/types.ts +108 -20
  139. package/src/services/constants.ts +0 -1
  140. package/src/services/core/Err.ts +0 -1
  141. package/src/services/core/Utils.ts +90 -67
  142. package/src/services/core/ai-docs/AGENTS.md +379 -0
  143. package/src/services/core/ai-docs/ARCHITECTURE.md +696 -0
  144. package/src/services/core/aqm-reqs.ts +22 -100
  145. package/src/services/core/websocket/WebSocketManager.ts +4 -23
  146. package/src/services/core/websocket/types.ts +1 -1
  147. package/src/services/index.ts +1 -2
  148. package/src/services/task/Task.ts +785 -0
  149. package/src/services/task/TaskFactory.ts +55 -0
  150. package/src/services/task/TaskManager.ts +567 -633
  151. package/src/services/task/TaskUtils.ts +175 -31
  152. package/src/services/task/ai-docs/AGENTS.md +448 -0
  153. package/src/services/task/ai-docs/ARCHITECTURE.md +573 -0
  154. package/src/services/task/constants.ts +5 -4
  155. package/src/services/task/dialer.ts +1 -56
  156. package/src/services/task/digital/Digital.ts +95 -0
  157. package/src/services/task/state-machine/TaskStateMachine.ts +793 -0
  158. package/src/services/task/state-machine/actions.ts +409 -0
  159. package/src/services/task/state-machine/ai-docs/AGENTS.md +495 -0
  160. package/src/services/task/state-machine/ai-docs/ARCHITECTURE.md +1135 -0
  161. package/src/services/task/state-machine/constants.ts +150 -0
  162. package/src/services/task/state-machine/guards.ts +295 -0
  163. package/src/services/task/state-machine/index.ts +28 -0
  164. package/src/services/task/state-machine/types.ts +228 -0
  165. package/src/services/task/state-machine/uiControlsComputer.ts +529 -0
  166. package/src/services/task/taskDataNormalizer.ts +137 -0
  167. package/src/services/task/types.ts +641 -95
  168. package/src/services/task/voice/Voice.ts +1255 -0
  169. package/src/services/task/voice/WebRTC.ts +187 -0
  170. package/src/types.ts +88 -5
  171. package/src/utils/AGENTS.md +276 -0
  172. package/src/webex.js +2 -0
  173. package/test/unit/spec/cc.ts +59 -142
  174. package/test/unit/spec/logger-proxy.ts +70 -0
  175. package/test/unit/spec/services/ApiAiAssistant.ts +17 -0
  176. package/test/unit/spec/services/config/index.ts +26 -55
  177. package/test/unit/spec/services/core/Utils.ts +103 -52
  178. package/test/unit/spec/services/core/websocket/WebSocketManager.ts +48 -112
  179. package/test/unit/spec/services/core/websocket/connection-service.ts +5 -4
  180. package/test/unit/spec/services/task/AutoWrapup.ts +63 -0
  181. package/test/unit/spec/services/task/Task.ts +416 -0
  182. package/test/unit/spec/services/task/TaskFactory.ts +62 -0
  183. package/test/unit/spec/services/task/TaskManager.ts +781 -1735
  184. package/test/unit/spec/services/task/TaskUtils.ts +125 -0
  185. package/test/unit/spec/services/task/dialer.ts +112 -198
  186. package/test/unit/spec/services/task/digital/Digital.ts +105 -0
  187. package/test/unit/spec/services/task/state-machine/TaskStateMachine.ts +473 -0
  188. package/test/unit/spec/services/task/state-machine/guards.ts +288 -0
  189. package/test/unit/spec/services/task/state-machine/types.ts +18 -0
  190. package/test/unit/spec/services/task/state-machine/uiControlsComputer.ts +147 -0
  191. package/test/unit/spec/services/task/taskTestUtils.ts +87 -0
  192. package/test/unit/spec/services/task/voice/Voice.ts +587 -0
  193. package/test/unit/spec/services/task/voice/WebRTC.ts +242 -0
  194. package/umd/contact-center.min.js +2 -2
  195. package/umd/contact-center.min.js.map +1 -1
  196. package/dist/services/task/index.js +0 -1525
  197. package/dist/services/task/index.js.map +0 -1
  198. package/dist/types/services/task/index.d.ts +0 -650
  199. package/src/services/task/index.ts +0 -1801
  200. package/test/unit/spec/services/task/index.ts +0 -2184
@@ -0,0 +1,529 @@
1
+ /**
2
+ * UI Controls Computer - Centralized logic for computing UI control states
3
+ */
4
+
5
+ import {
6
+ InteractionUIControls,
7
+ TASK_CHANNEL_TYPE,
8
+ TaskData,
9
+ TaskUILeg,
10
+ TaskUIControls,
11
+ VOICE_VARIANT,
12
+ } from '../types';
13
+ import {TaskContext, UIControlConfig} from './types';
14
+ import {TaskState, MAX_PARTICIPANTS_IN_MULTIPARTY_CONFERENCE} from './constants';
15
+ import {
16
+ getIsCustomerInCall,
17
+ getConferenceParticipantsCount,
18
+ getIsConferenceInProgress,
19
+ getIsConsultInProgressForConferenceControls,
20
+ getIsConsultedAgentForControls,
21
+ getServerHoldStateForControls,
22
+ } from '../TaskUtils';
23
+
24
+ const DISABLED = {isVisible: false, isEnabled: false} as const;
25
+ const VISIBLE_ENABLED = {isVisible: true, isEnabled: true} as const;
26
+ const VISIBLE_DISABLED = {isVisible: true, isEnabled: false} as const;
27
+
28
+ function getDefaultInteractionUIControls(): InteractionUIControls {
29
+ return {
30
+ accept: DISABLED,
31
+ decline: DISABLED,
32
+ hold: DISABLED,
33
+ mute: DISABLED,
34
+ end: DISABLED,
35
+ transfer: DISABLED,
36
+ consult: DISABLED,
37
+ consultTransfer: DISABLED,
38
+ endConsult: DISABLED,
39
+ recording: DISABLED,
40
+ conference: DISABLED,
41
+ wrapup: DISABLED,
42
+ exitConference: DISABLED,
43
+ transferConference: DISABLED,
44
+ mergeToConference: DISABLED,
45
+ switch: DISABLED,
46
+ };
47
+ }
48
+
49
+ function createTaskUIControls(
50
+ main: InteractionUIControls,
51
+ consult: InteractionUIControls,
52
+ activeLeg: TaskUILeg
53
+ ): TaskUIControls {
54
+ return {
55
+ main,
56
+ consult,
57
+ activeLeg,
58
+ };
59
+ }
60
+
61
+ export function getDefaultUIControls(): TaskUIControls {
62
+ return createTaskUIControls(
63
+ getDefaultInteractionUIControls(),
64
+ getDefaultInteractionUIControls(),
65
+ 'main'
66
+ );
67
+ }
68
+
69
+ function computeVoiceInteractionUIControls(
70
+ state: TaskState,
71
+ context: TaskContext,
72
+ config: UIControlConfig,
73
+ fallbackTaskData?: TaskData,
74
+ currentLeg: TaskUILeg = 'main'
75
+ ): InteractionUIControls {
76
+ // Early exit for idle
77
+ if (state === TaskState.IDLE) {
78
+ return getDefaultInteractionUIControls();
79
+ }
80
+
81
+ // Essential data
82
+ const taskData = context.taskData ?? fallbackTaskData ?? null;
83
+ const interaction = taskData?.interaction;
84
+ const mainCallId = interaction?.mainInteractionId || taskData?.interactionId;
85
+ const isWebrtc = config.voiceVariant === VOICE_VARIANT.WEBRTC;
86
+ const isOutdial = interaction?.outboundType === 'OUTDIAL';
87
+ const serverHold = getServerHoldStateForControls(context, mainCallId, fallbackTaskData);
88
+
89
+ // Backend-derived checks
90
+ const customerInCall =
91
+ interaction && mainCallId ? getIsCustomerInCall(interaction, mainCallId) : false;
92
+ // EP-DN/secondary legs can have incomplete media participant lists; fall back to participants map.
93
+ const customerPresent =
94
+ customerInCall ||
95
+ Boolean(
96
+ interaction &&
97
+ interaction.participants &&
98
+ Object.values(interaction.participants).some(
99
+ (p: any) => p?.pType === 'Customer' && !p?.hasLeft
100
+ )
101
+ );
102
+ const participantCount =
103
+ interaction && mainCallId ? getConferenceParticipantsCount(interaction, mainCallId) : 0;
104
+ const maxParticipants = participantCount >= MAX_PARTICIPANTS_IN_MULTIPARTY_CONFERENCE;
105
+ const selfAgentId = config.agentId ?? taskData?.agentId;
106
+ const consultInProgress = getIsConsultInProgressForConferenceControls(
107
+ interaction,
108
+ mainCallId,
109
+ selfAgentId
110
+ );
111
+ const conferenceFromBackend = taskData ? getIsConferenceInProgress(taskData) : false;
112
+ // Note: ownership is used by some controls; keep computations local to those controls
113
+
114
+ // Context flags (set by state machine actions)
115
+ const {consultInitiator, consultDestinationAgentJoined, consultCallHeld, consultFromConference} =
116
+ context;
117
+ const {recordingControlsAvailable} = context;
118
+
119
+ const stateImpliesHeld = state === TaskState.HELD || state === TaskState.RESUME_INITIATING;
120
+ const stateImpliesConnected =
121
+ state === TaskState.CONNECTED || state === TaskState.HOLD_INITIATING;
122
+ const isHeld = stateImpliesHeld || serverHold === true;
123
+ const isConnected = stateImpliesConnected || (!stateImpliesHeld && serverHold === false);
124
+
125
+ // State categories for cleaner logic
126
+ const isConsulting =
127
+ state === TaskState.CONSULTING ||
128
+ state === TaskState.CONSULT_INITIATING ||
129
+ state === TaskState.CONF_INITIATING;
130
+ const isConferencing = state === TaskState.CONFERENCING;
131
+ const isWrappingUp = state === TaskState.WRAPPING_UP;
132
+ const selfInMainCall =
133
+ Boolean(selfAgentId) &&
134
+ Boolean(mainCallId) &&
135
+ Boolean(interaction?.media?.[mainCallId]?.participants?.includes(selfAgentId as string));
136
+ const conferenceActive = isConferencing || conferenceFromBackend || consultFromConference;
137
+ // Treat consult initiator as "in conference" even if mainCall participant list lags while consulting.
138
+ const inConference = conferenceActive && (isConferencing || selfInMainCall || consultInitiator);
139
+
140
+ // Check if this is a consulted agent (must be after isConsulting is computed).
141
+ const isSoleAgentOnCall = participantCount <= 1 && !isConsulting && !inConference;
142
+ const isConsulted =
143
+ inConference || isSoleAgentOnCall
144
+ ? false
145
+ : getIsConsultedAgentForControls(taskData, context, isConsulting);
146
+
147
+ // Active call = can perform call operations
148
+ const isActive =
149
+ state === TaskState.CONNECTED ||
150
+ state === TaskState.HELD ||
151
+ state === TaskState.HOLD_INITIATING ||
152
+ state === TaskState.RESUME_INITIATING ||
153
+ isConsulting ||
154
+ isConferencing;
155
+
156
+ // Consulted agents have limited controls until they're in conference or wrapup
157
+ // Use inConference (not isConferencing) so controls remain enabled after state downgrade
158
+ const hasFullControls = !isConsulted || consultInitiator || inConference || isWrappingUp;
159
+ const consultOwnedBySelf =
160
+ consultInitiator || (Boolean(selfAgentId) && taskData?.consultingAgentId === selfAgentId);
161
+ const hasConsultMedia = Boolean(
162
+ taskData?.consultMediaResourceId ||
163
+ Object.values(interaction?.media ?? {}).some((media: any) => media?.mType === 'consult')
164
+ );
165
+ const hasParallelConsultLeg =
166
+ consultOwnedBySelf &&
167
+ !isConsulting &&
168
+ !isConsulted &&
169
+ (consultInProgress || consultCallHeld || hasConsultMedia);
170
+ const consultLegOnHold = isConsulting && consultCallHeld;
171
+
172
+ return {
173
+ // Accept/Decline: Voice tasks in offered state
174
+ // For outdial, accept is disabled (auto-answer handles it), decline remains enabled
175
+ // For Extension mode (non-WebRTC), accept shows as disabled "Ringing" button
176
+ accept:
177
+ state === TaskState.OFFERED && !interaction?.isTerminated
178
+ ? {isVisible: true, isEnabled: isWebrtc && !isOutdial}
179
+ : DISABLED,
180
+ decline:
181
+ isWebrtc && state === TaskState.OFFERED && !interaction?.isTerminated
182
+ ? VISIBLE_ENABLED
183
+ : DISABLED,
184
+
185
+ // Hold: visible in connected/held/conference, disabled in conference/consulting
186
+ hold: (() => {
187
+ if (!hasFullControls) return DISABLED;
188
+ if (consultOwnedBySelf && (isConsulting || hasParallelConsultLeg || consultCallHeld)) {
189
+ return DISABLED;
190
+ }
191
+ if (hasParallelConsultLeg) return DISABLED;
192
+ if (state === TaskState.OFFERED) return DISABLED;
193
+ if (isWrappingUp) return DISABLED;
194
+ // Visibility: connected || held || inConference
195
+ if (!(isConnected || isHeld || inConference)) return DISABLED;
196
+ // Enabled: (connected || held) && !inConference && !isConsulting
197
+ const canHold = (isConnected || isHeld) && !inConference && !isConsulting;
198
+
199
+ return canHold ? VISIBLE_ENABLED : VISIBLE_DISABLED;
200
+ })(),
201
+
202
+ // Mute: WebRTC only, active calls; hidden entirely during wrapup
203
+ mute: (() => {
204
+ if (!isWebrtc) return DISABLED;
205
+ if (isWrappingUp) return DISABLED;
206
+ if (isConsulting) return VISIBLE_ENABLED;
207
+
208
+ if (isConnected || isHeld || isConferencing) {
209
+ if (inConference) return VISIBLE_ENABLED;
210
+
211
+ return isHeld ? VISIBLE_DISABLED : VISIBLE_ENABLED;
212
+ }
213
+
214
+ return DISABLED;
215
+ })(),
216
+
217
+ // End: varies by state; during consulting only on main leg (consult held)
218
+ end: (() => {
219
+ if (!config.isEndTaskEnabled) return DISABLED;
220
+ if (hasParallelConsultLeg) return VISIBLE_DISABLED;
221
+
222
+ if (isConsulting) {
223
+ if (currentLeg === 'consult' && consultCallHeld) return DISABLED;
224
+
225
+ return consultInitiator && consultCallHeld ? VISIBLE_ENABLED : DISABLED;
226
+ }
227
+
228
+ if (inConference) {
229
+ if (isConsulted) return DISABLED;
230
+
231
+ if (consultInProgress) return VISIBLE_DISABLED;
232
+
233
+ return isWrappingUp ? VISIBLE_DISABLED : VISIBLE_ENABLED;
234
+ }
235
+ if (!hasFullControls) return DISABLED;
236
+ if (isActive) return isHeld || isWrappingUp ? VISIBLE_DISABLED : VISIBLE_ENABLED;
237
+
238
+ return DISABLED;
239
+ })(),
240
+
241
+ // Transfer: connected/held, not in conference
242
+ transfer: (() => {
243
+ if (hasParallelConsultLeg) {
244
+ if (state === TaskState.CONNECTED) return VISIBLE_ENABLED;
245
+ if (state === TaskState.HELD) return VISIBLE_DISABLED;
246
+ }
247
+ if (isConsulting) {
248
+ if (!consultInitiator) return DISABLED;
249
+ if (consultLegOnHold) return VISIBLE_DISABLED;
250
+
251
+ return consultDestinationAgentJoined ? VISIBLE_ENABLED : VISIBLE_DISABLED;
252
+ }
253
+ if (!hasFullControls || inConference) return DISABLED;
254
+ if (state === TaskState.CONNECTED || state === TaskState.HELD) return VISIBLE_ENABLED;
255
+
256
+ return DISABLED;
257
+ })(),
258
+
259
+ // Consult: connected/held/conference when conditions met
260
+ consult: (() => {
261
+ const isConnectedOrHeld = state === TaskState.CONNECTED || state === TaskState.HELD;
262
+
263
+ if (hasParallelConsultLeg) return DISABLED;
264
+ if (!hasFullControls || !(isConnectedOrHeld || inConference)) {
265
+ return DISABLED;
266
+ }
267
+
268
+ // Enabled conditions differ by state
269
+ const canFromConnected =
270
+ !maxParticipants && customerPresent && !consultInProgress && !isConsulted;
271
+ const canFromConference =
272
+ !maxParticipants && customerPresent && !consultInProgress && !isConsulting;
273
+
274
+ const isEnabled = inConference ? canFromConference : canFromConnected;
275
+
276
+ return {isVisible: true, isEnabled};
277
+ })(),
278
+
279
+ // ConsultTransfer: always hidden (use transfer button)
280
+ consultTransfer: DISABLED,
281
+
282
+ // EndConsult: during consulting
283
+ endConsult: (() => {
284
+ if (!isConsulting) return DISABLED;
285
+ if (isConsulted && isConferencing) return DISABLED;
286
+ if (!isConsulted && isConferencing && !(consultInitiator && conferenceFromBackend)) {
287
+ return DISABLED;
288
+ }
289
+
290
+ return {isVisible: true, isEnabled: consultInitiator || config.isEndConsultEnabled};
291
+ })(),
292
+
293
+ // Recording: connected/held only, not in consult/conference
294
+ recording: (() => {
295
+ if (!recordingControlsAvailable || !config.isRecordingEnabled) return DISABLED;
296
+ if (!hasFullControls || isConsulting || inConference) return DISABLED;
297
+ if (state === TaskState.CONNECTED || state === TaskState.HELD) {
298
+ return VISIBLE_ENABLED;
299
+ }
300
+
301
+ return DISABLED;
302
+ })(),
303
+
304
+ // Conference: during consulting, enabled on both legs when agent joined
305
+ // Label changes based on leg: "Conference" on main leg, "Merge" on consult leg
306
+ conference: (() => {
307
+ if (hasParallelConsultLeg) {
308
+ if (state === TaskState.CONNECTED) {
309
+ return maxParticipants ? VISIBLE_DISABLED : VISIBLE_ENABLED;
310
+ }
311
+ if (state === TaskState.HELD) return VISIBLE_DISABLED;
312
+
313
+ return DISABLED;
314
+ }
315
+ if (!hasFullControls || !isConsulting) return DISABLED;
316
+ if (!consultInitiator) return DISABLED;
317
+ if (consultLegOnHold) return VISIBLE_DISABLED;
318
+
319
+ return consultDestinationAgentJoined && !maxParticipants ? VISIBLE_ENABLED : VISIBLE_DISABLED;
320
+ })(),
321
+
322
+ // Wrapup: wrapping up state
323
+ wrapup: isWrappingUp ? VISIBLE_ENABLED : DISABLED,
324
+
325
+ // ExitConference: in conference, not consulting from conference
326
+ exitConference: (() => {
327
+ if (isConsulted && !isConferencing) return DISABLED;
328
+ if (!inConference) return DISABLED;
329
+ const consultingFromConference = consultInitiator && isConsulting && conferenceFromBackend;
330
+
331
+ return consultingFromConference ? VISIBLE_DISABLED : VISIBLE_ENABLED;
332
+ })(),
333
+
334
+ // TransferConference: in conference with active consult, owner consulting from conference
335
+ transferConference: (() => {
336
+ if (hasParallelConsultLeg || consultLegOnHold) return DISABLED;
337
+ if (!inConference || !isConsulting) return DISABLED;
338
+ if (!consultInitiator || isConsulted) return DISABLED;
339
+
340
+ return consultDestinationAgentJoined ? VISIBLE_ENABLED : VISIBLE_DISABLED;
341
+ })(),
342
+
343
+ // MergeToConference: mirrors conference control, enabled on both legs
344
+ mergeToConference: (() => {
345
+ if (!isConsulting || !consultInitiator) return DISABLED;
346
+ if (consultLegOnHold) return VISIBLE_DISABLED;
347
+
348
+ return consultDestinationAgentJoined && !maxParticipants ? VISIBLE_ENABLED : VISIBLE_DISABLED;
349
+ })(),
350
+
351
+ // Switch: visible only on the currently active leg
352
+ switch: (() => {
353
+ if (currentLeg === 'consult') {
354
+ if (!isConsulting || !consultInitiator || consultCallHeld) return DISABLED;
355
+
356
+ return consultDestinationAgentJoined ? VISIBLE_ENABLED : VISIBLE_DISABLED;
357
+ }
358
+
359
+ if (hasParallelConsultLeg && state === TaskState.CONNECTED) {
360
+ return consultDestinationAgentJoined ? VISIBLE_ENABLED : VISIBLE_DISABLED;
361
+ }
362
+
363
+ return DISABLED;
364
+ })(),
365
+ };
366
+ }
367
+
368
+ function computeDigitalInteractionUIControls(
369
+ state: TaskState,
370
+ context: TaskContext,
371
+ fallbackTaskData?: TaskData
372
+ ): InteractionUIControls {
373
+ const taskData = context.taskData ?? fallbackTaskData ?? null;
374
+ const isTerminated = taskData?.interaction?.isTerminated ?? false;
375
+
376
+ const isConnected = state === TaskState.CONNECTED;
377
+ const isWrappingUp = state === TaskState.WRAPPING_UP;
378
+
379
+ return {
380
+ accept: state === TaskState.OFFERED ? VISIBLE_ENABLED : DISABLED,
381
+ decline: DISABLED,
382
+ hold: DISABLED,
383
+ mute: DISABLED,
384
+ end: isConnected && !isWrappingUp ? VISIBLE_ENABLED : DISABLED,
385
+ transfer: isConnected && !isWrappingUp ? VISIBLE_ENABLED : DISABLED,
386
+ consult: DISABLED,
387
+ consultTransfer: DISABLED,
388
+ endConsult: DISABLED,
389
+ recording: DISABLED,
390
+ conference: DISABLED,
391
+ wrapup: isTerminated || isWrappingUp ? VISIBLE_ENABLED : DISABLED,
392
+ exitConference: DISABLED,
393
+ transferConference: DISABLED,
394
+ mergeToConference: DISABLED,
395
+ switch: DISABLED,
396
+ };
397
+ }
398
+
399
+ function getVoiceLegState(
400
+ currentState: TaskState,
401
+ context: TaskContext,
402
+ config: UIControlConfig,
403
+ fallbackTaskData?: TaskData
404
+ ): {hasConsultLeg: boolean; activeLeg: TaskUILeg; mainState: TaskState; consultState: TaskState} {
405
+ if (currentState === TaskState.WRAPPING_UP) {
406
+ return {
407
+ hasConsultLeg: false,
408
+ activeLeg: 'main',
409
+ mainState: currentState,
410
+ consultState: TaskState.CONSULTING,
411
+ };
412
+ }
413
+
414
+ const taskData = context.taskData ?? fallbackTaskData ?? null;
415
+ const interaction = taskData?.interaction;
416
+ const mainCallId = interaction?.mainInteractionId || taskData?.interactionId;
417
+ const selfAgentId = config.agentId ?? taskData?.agentId;
418
+ const consultInProgress = getIsConsultInProgressForConferenceControls(
419
+ interaction,
420
+ mainCallId,
421
+ selfAgentId
422
+ );
423
+ const isConsultingState =
424
+ currentState === TaskState.CONSULTING ||
425
+ currentState === TaskState.CONSULT_INITIATING ||
426
+ currentState === TaskState.CONF_INITIATING;
427
+ const consultOwnedBySelf =
428
+ context.consultInitiator ||
429
+ (Boolean(selfAgentId) && taskData?.consultingAgentId === selfAgentId);
430
+ const hasConsultMedia = Boolean(
431
+ taskData?.consultMediaResourceId ||
432
+ Object.values(interaction?.media ?? {}).some((media: any) => media?.mType === 'consult')
433
+ );
434
+ const hasConsultLeg = Boolean(
435
+ consultOwnedBySelf &&
436
+ !taskData?.isConsulted &&
437
+ !interaction?.isTerminated &&
438
+ (consultInProgress || isConsultingState || context.consultCallHeld || hasConsultMedia)
439
+ );
440
+
441
+ if (!hasConsultLeg) {
442
+ return {
443
+ hasConsultLeg: false,
444
+ activeLeg: 'main',
445
+ mainState: currentState,
446
+ consultState: TaskState.CONSULTING,
447
+ };
448
+ }
449
+
450
+ return {
451
+ hasConsultLeg: true,
452
+ activeLeg: context.consultCallHeld ? 'main' : 'consult',
453
+ mainState: context.consultCallHeld ? TaskState.CONNECTED : TaskState.HELD,
454
+ consultState: isConsultingState ? currentState : TaskState.CONSULTING,
455
+ };
456
+ }
457
+
458
+ export function computeUIControls(
459
+ currentState: TaskState,
460
+ context: TaskContext,
461
+ fallbackTaskData?: TaskData
462
+ ): TaskUIControls {
463
+ if (currentState === TaskState.TERMINATED || currentState === TaskState.COMPLETED) {
464
+ return getDefaultUIControls();
465
+ }
466
+
467
+ switch (context.uiControlConfig.channelType) {
468
+ case TASK_CHANNEL_TYPE.VOICE: {
469
+ const {hasConsultLeg, activeLeg, mainState, consultState} = getVoiceLegState(
470
+ currentState,
471
+ context,
472
+ context.uiControlConfig,
473
+ fallbackTaskData
474
+ );
475
+
476
+ const mainControls = computeVoiceInteractionUIControls(
477
+ mainState,
478
+ context,
479
+ context.uiControlConfig,
480
+ fallbackTaskData,
481
+ 'main'
482
+ );
483
+ const consultControls = hasConsultLeg
484
+ ? computeVoiceInteractionUIControls(
485
+ consultState,
486
+ context,
487
+ context.uiControlConfig,
488
+ fallbackTaskData,
489
+ 'consult'
490
+ )
491
+ : getDefaultInteractionUIControls();
492
+
493
+ return createTaskUIControls(mainControls, consultControls, activeLeg);
494
+ }
495
+ case TASK_CHANNEL_TYPE.DIGITAL:
496
+ return createTaskUIControls(
497
+ computeDigitalInteractionUIControls(currentState, context, fallbackTaskData),
498
+ getDefaultInteractionUIControls(),
499
+ 'main'
500
+ );
501
+ default:
502
+ return getDefaultUIControls();
503
+ }
504
+ }
505
+
506
+ function haveInteractionUIControlsChanged(
507
+ previous: InteractionUIControls,
508
+ next: InteractionUIControls
509
+ ): boolean {
510
+ return (Object.keys(next) as (keyof InteractionUIControls)[]).some((key) => {
511
+ const prev = previous[key];
512
+ const curr = next[key];
513
+
514
+ return prev.isVisible !== curr.isVisible || prev.isEnabled !== curr.isEnabled;
515
+ });
516
+ }
517
+
518
+ export function haveUIControlsChanged(
519
+ previous: TaskUIControls | undefined,
520
+ next: TaskUIControls
521
+ ): boolean {
522
+ if (!previous) return true;
523
+
524
+ return (
525
+ previous.activeLeg !== next.activeLeg ||
526
+ haveInteractionUIControlsChanged(previous.main, next.main) ||
527
+ haveInteractionUIControlsChanged(previous.consult, next.consult)
528
+ );
529
+ }
@@ -0,0 +1,137 @@
1
+ import {
2
+ CallProcessingBooleanKey,
3
+ InteractionBooleanKey,
4
+ ParticipantBooleanKey,
5
+ TaskData,
6
+ } from './types';
7
+
8
+ const booleanKeys: CallProcessingBooleanKey[] = [
9
+ 'recordingStarted',
10
+ 'recordInProgress',
11
+ 'isPaused',
12
+ 'pauseResumeEnabled',
13
+ 'ctqInProgress',
14
+ 'outdialTransferToQueueEnabled',
15
+ 'taskToBeSelfServiced',
16
+ 'CONTINUE_RECORDING_ON_TRANSFER',
17
+ 'isParked',
18
+ 'participantInviteTimeout',
19
+ 'checkAgentAvailability',
20
+ ];
21
+
22
+ const interactionBooleanKeys: InteractionBooleanKey[] = [
23
+ 'isFcManaged',
24
+ 'isMediaForked',
25
+ 'isTerminated',
26
+ ];
27
+
28
+ const participantBooleanKeys: ParticipantBooleanKey[] = [
29
+ 'autoAnswerEnabled',
30
+ 'hasJoined',
31
+ 'hasLeft',
32
+ 'isConsulted',
33
+ 'isInPredial',
34
+ 'isOffered',
35
+ 'isWrapUp',
36
+ 'isWrappedUp',
37
+ ];
38
+
39
+ const toBoolean = (value: unknown): boolean | undefined => {
40
+ if (typeof value === 'boolean') {
41
+ return value;
42
+ }
43
+
44
+ if (typeof value === 'string') {
45
+ const normalized = value.toLowerCase();
46
+
47
+ if (normalized === 'true') {
48
+ return true;
49
+ }
50
+ if (normalized === 'false') {
51
+ return false;
52
+ }
53
+ }
54
+
55
+ return undefined;
56
+ };
57
+
58
+ const normalizeFields = <T extends Record<string, any>>(obj: T, keys: string[]): T | undefined => {
59
+ let updated: T | undefined;
60
+
61
+ keys.forEach((key) => {
62
+ const normalized = toBoolean(obj[key]);
63
+
64
+ if (typeof normalized !== 'undefined') {
65
+ if (!updated) {
66
+ updated = {...obj};
67
+ }
68
+ (updated as any)[key] = normalized;
69
+ }
70
+ });
71
+
72
+ return updated;
73
+ };
74
+
75
+ /**
76
+ * Normalize backend task payload quirks so downstream code can rely on actual booleans.
77
+ *
78
+ * Applies to every Agent Contact websocket event before it reaches the state machine:
79
+ * - Converts string booleans in callProcessingDetails to actual booleans.
80
+ * - Also normalizes known boolean fields on interaction and participants.
81
+ * - Keeps payload shape intact; only coerces known boolean fields.
82
+ */
83
+ export function normalizeTaskData(data: TaskData): TaskData {
84
+ const interaction = data?.interaction;
85
+
86
+ if (!interaction) {
87
+ return data;
88
+ }
89
+
90
+ const details = interaction.callProcessingDetails;
91
+ const updatedDetails = details ? normalizeFields(details, booleanKeys) : undefined;
92
+ const updatedInteractionBooleans = normalizeFields(
93
+ interaction,
94
+ interactionBooleanKeys as string[]
95
+ );
96
+
97
+ let updatedParticipants: typeof interaction.participants | undefined;
98
+ const participants = interaction.participants || {};
99
+ Object.keys(participants).forEach((id) => {
100
+ const participant = participants[id];
101
+ const normalized = normalizeFields(participant, participantBooleanKeys);
102
+ if (normalized) {
103
+ if (!updatedParticipants) {
104
+ updatedParticipants = {...participants};
105
+ }
106
+ updatedParticipants[id] = normalized;
107
+ }
108
+ });
109
+
110
+ let updatedMedia: typeof interaction.media | undefined;
111
+ const mediaEntries = interaction.media || {};
112
+ Object.keys(mediaEntries).forEach((id) => {
113
+ const media = mediaEntries[id];
114
+ const normalized = normalizeFields(media, ['isHold']);
115
+ if (normalized) {
116
+ if (!updatedMedia) {
117
+ updatedMedia = {...mediaEntries};
118
+ }
119
+ updatedMedia[id] = normalized;
120
+ }
121
+ });
122
+
123
+ if (!updatedDetails && !updatedInteractionBooleans && !updatedParticipants && !updatedMedia) {
124
+ return data;
125
+ }
126
+
127
+ return {
128
+ ...data,
129
+ interaction: {
130
+ ...interaction,
131
+ ...(updatedInteractionBooleans || {}),
132
+ callProcessingDetails: updatedDetails || details,
133
+ participants: updatedParticipants || interaction.participants,
134
+ media: updatedMedia || interaction.media,
135
+ },
136
+ };
137
+ }