@nyaruka/temba-components 0.131.1 → 0.131.3

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 (485) hide show
  1. package/.github/workflows/publish.yml +4 -1
  2. package/CHANGELOG.md +75 -1
  3. package/demo/components/floating-tabs/example.html +400 -0
  4. package/demo/components/flow/index.html +1 -1
  5. package/demo/data/flows/food-order.json +2 -2
  6. package/demo/data/flows/sample-flow.json +113 -125
  7. package/demo/data/flows/voicemail.json +613 -0
  8. package/demo/index.html +6 -0
  9. package/dist/locales/es.js +5 -5
  10. package/dist/locales/es.js.map +1 -1
  11. package/dist/locales/fr.js +5 -5
  12. package/dist/locales/fr.js.map +1 -1
  13. package/dist/locales/locale-codes.js +11 -2
  14. package/dist/locales/locale-codes.js.map +1 -1
  15. package/dist/locales/pt.js +5 -5
  16. package/dist/locales/pt.js.map +1 -1
  17. package/dist/static/svg/index.svg +1 -1
  18. package/dist/temba-components.js +1773 -662
  19. package/dist/temba-components.js.map +1 -1
  20. package/out-tsc/src/Icons.js +4 -1
  21. package/out-tsc/src/Icons.js.map +1 -1
  22. package/out-tsc/src/display/FloatingTab.js +167 -0
  23. package/out-tsc/src/display/FloatingTab.js.map +1 -0
  24. package/out-tsc/src/display/ProgressBar.js +22 -2
  25. package/out-tsc/src/display/ProgressBar.js.map +1 -1
  26. package/out-tsc/src/events.js.map +1 -1
  27. package/out-tsc/src/flow/CanvasMenu.js +200 -0
  28. package/out-tsc/src/flow/CanvasMenu.js.map +1 -0
  29. package/out-tsc/src/flow/CanvasNode.js +489 -47
  30. package/out-tsc/src/flow/CanvasNode.js.map +1 -1
  31. package/out-tsc/src/flow/Editor.js +1417 -67
  32. package/out-tsc/src/flow/Editor.js.map +1 -1
  33. package/out-tsc/src/flow/NodeEditor.js +479 -112
  34. package/out-tsc/src/flow/NodeEditor.js.map +1 -1
  35. package/out-tsc/src/flow/NodeTypeSelector.js +540 -0
  36. package/out-tsc/src/flow/NodeTypeSelector.js.map +1 -0
  37. package/out-tsc/src/flow/StickyNote.js +12 -3
  38. package/out-tsc/src/flow/StickyNote.js.map +1 -1
  39. package/out-tsc/src/flow/actions/add_contact_groups.js +4 -3
  40. package/out-tsc/src/flow/actions/add_contact_groups.js.map +1 -1
  41. package/out-tsc/src/flow/actions/add_contact_urn.js +63 -4
  42. package/out-tsc/src/flow/actions/add_contact_urn.js.map +1 -1
  43. package/out-tsc/src/flow/actions/add_input_labels.js +4 -3
  44. package/out-tsc/src/flow/actions/add_input_labels.js.map +1 -1
  45. package/out-tsc/src/flow/actions/play_audio.js +3 -2
  46. package/out-tsc/src/flow/actions/play_audio.js.map +1 -1
  47. package/out-tsc/src/flow/actions/remove_contact_groups.js +7 -5
  48. package/out-tsc/src/flow/actions/remove_contact_groups.js.map +1 -1
  49. package/out-tsc/src/flow/actions/request_optin.js +3 -2
  50. package/out-tsc/src/flow/actions/request_optin.js.map +1 -1
  51. package/out-tsc/src/flow/actions/say_msg.js +3 -2
  52. package/out-tsc/src/flow/actions/say_msg.js.map +1 -1
  53. package/out-tsc/src/flow/actions/send_broadcast.js +77 -23
  54. package/out-tsc/src/flow/actions/send_broadcast.js.map +1 -1
  55. package/out-tsc/src/flow/actions/send_email.js +5 -5
  56. package/out-tsc/src/flow/actions/send_email.js.map +1 -1
  57. package/out-tsc/src/flow/actions/send_msg.js +101 -21
  58. package/out-tsc/src/flow/actions/send_msg.js.map +1 -1
  59. package/out-tsc/src/flow/actions/set_contact_channel.js +6 -9
  60. package/out-tsc/src/flow/actions/set_contact_channel.js.map +1 -1
  61. package/out-tsc/src/flow/actions/set_contact_field.js +20 -20
  62. package/out-tsc/src/flow/actions/set_contact_field.js.map +1 -1
  63. package/out-tsc/src/flow/actions/set_contact_language.js +3 -2
  64. package/out-tsc/src/flow/actions/set_contact_language.js.map +1 -1
  65. package/out-tsc/src/flow/actions/set_contact_name.js +3 -12
  66. package/out-tsc/src/flow/actions/set_contact_name.js.map +1 -1
  67. package/out-tsc/src/flow/actions/set_contact_status.js +3 -2
  68. package/out-tsc/src/flow/actions/set_contact_status.js.map +1 -1
  69. package/out-tsc/src/flow/actions/set_run_result.js +4 -3
  70. package/out-tsc/src/flow/actions/set_run_result.js.map +1 -1
  71. package/out-tsc/src/flow/actions/start_session.js +181 -6
  72. package/out-tsc/src/flow/actions/start_session.js.map +1 -1
  73. package/out-tsc/src/flow/config.js +11 -23
  74. package/out-tsc/src/flow/config.js.map +1 -1
  75. package/out-tsc/src/flow/currencies.js +45 -0
  76. package/out-tsc/src/flow/currencies.js.map +1 -0
  77. package/out-tsc/src/flow/nodes/shared-rules.js +257 -0
  78. package/out-tsc/src/flow/nodes/shared-rules.js.map +1 -0
  79. package/out-tsc/src/flow/nodes/shared.js +71 -0
  80. package/out-tsc/src/flow/nodes/shared.js.map +1 -0
  81. package/out-tsc/src/flow/nodes/split_by_airtime.js +211 -5
  82. package/out-tsc/src/flow/nodes/split_by_airtime.js.map +1 -1
  83. package/out-tsc/src/flow/nodes/split_by_contact_field.js +152 -3
  84. package/out-tsc/src/flow/nodes/split_by_contact_field.js.map +1 -1
  85. package/out-tsc/src/flow/nodes/split_by_expression.js +73 -2
  86. package/out-tsc/src/flow/nodes/split_by_expression.js.map +1 -1
  87. package/out-tsc/src/flow/nodes/split_by_groups.js +18 -10
  88. package/out-tsc/src/flow/nodes/split_by_groups.js.map +1 -1
  89. package/out-tsc/src/flow/nodes/split_by_intent.js +8 -0
  90. package/out-tsc/src/flow/nodes/split_by_intent.js.map +1 -0
  91. package/out-tsc/src/flow/nodes/split_by_llm.js +11 -3
  92. package/out-tsc/src/flow/nodes/split_by_llm.js.map +1 -1
  93. package/out-tsc/src/flow/nodes/split_by_llm_categorize.js +10 -3
  94. package/out-tsc/src/flow/nodes/split_by_llm_categorize.js.map +1 -1
  95. package/out-tsc/src/flow/nodes/split_by_random.js +10 -4
  96. package/out-tsc/src/flow/nodes/split_by_random.js.map +1 -1
  97. package/out-tsc/src/flow/nodes/split_by_resthook.js +113 -0
  98. package/out-tsc/src/flow/nodes/split_by_resthook.js.map +1 -0
  99. package/out-tsc/src/flow/nodes/split_by_run_result.js +211 -3
  100. package/out-tsc/src/flow/nodes/split_by_run_result.js.map +1 -1
  101. package/out-tsc/src/flow/nodes/split_by_scheme.js +158 -2
  102. package/out-tsc/src/flow/nodes/split_by_scheme.js.map +1 -1
  103. package/out-tsc/src/flow/nodes/split_by_subflow.js +13 -5
  104. package/out-tsc/src/flow/nodes/split_by_subflow.js.map +1 -1
  105. package/out-tsc/src/flow/nodes/split_by_ticket.js +10 -3
  106. package/out-tsc/src/flow/nodes/split_by_ticket.js.map +1 -1
  107. package/out-tsc/src/flow/nodes/split_by_webhook.js +10 -3
  108. package/out-tsc/src/flow/nodes/split_by_webhook.js.map +1 -1
  109. package/out-tsc/src/flow/nodes/wait_for_digits.js +3 -2
  110. package/out-tsc/src/flow/nodes/wait_for_digits.js.map +1 -1
  111. package/out-tsc/src/flow/nodes/wait_for_menu.js +3 -2
  112. package/out-tsc/src/flow/nodes/wait_for_menu.js.map +1 -1
  113. package/out-tsc/src/flow/nodes/wait_for_response.js +38 -568
  114. package/out-tsc/src/flow/nodes/wait_for_response.js.map +1 -1
  115. package/out-tsc/src/flow/types.js +86 -12
  116. package/out-tsc/src/flow/types.js.map +1 -1
  117. package/out-tsc/src/flow/utils.js +101 -14
  118. package/out-tsc/src/flow/utils.js.map +1 -1
  119. package/out-tsc/src/form/FieldRenderer.js +2 -4
  120. package/out-tsc/src/form/FieldRenderer.js.map +1 -1
  121. package/out-tsc/src/interfaces.js +3 -0
  122. package/out-tsc/src/interfaces.js.map +1 -1
  123. package/out-tsc/src/layout/FloatingWindow.js +346 -0
  124. package/out-tsc/src/layout/FloatingWindow.js.map +1 -0
  125. package/out-tsc/src/list/SortableList.js +98 -33
  126. package/out-tsc/src/list/SortableList.js.map +1 -1
  127. package/out-tsc/src/live/ContactChat.js +6 -25
  128. package/out-tsc/src/live/ContactChat.js.map +1 -1
  129. package/out-tsc/src/locales/es.js +5 -5
  130. package/out-tsc/src/locales/es.js.map +1 -1
  131. package/out-tsc/src/locales/fr.js +5 -5
  132. package/out-tsc/src/locales/fr.js.map +1 -1
  133. package/out-tsc/src/locales/locale-codes.js +11 -2
  134. package/out-tsc/src/locales/locale-codes.js.map +1 -1
  135. package/out-tsc/src/locales/pt.js +5 -5
  136. package/out-tsc/src/locales/pt.js.map +1 -1
  137. package/out-tsc/src/store/AppState.js +120 -0
  138. package/out-tsc/src/store/AppState.js.map +1 -1
  139. package/out-tsc/src/utils.js +254 -13
  140. package/out-tsc/src/utils.js.map +1 -1
  141. package/out-tsc/temba-modules.js +8 -0
  142. package/out-tsc/temba-modules.js.map +1 -1
  143. package/out-tsc/test/ActionHelper.js +3 -3
  144. package/out-tsc/test/ActionHelper.js.map +1 -1
  145. package/out-tsc/test/NodeHelper.js +6 -3
  146. package/out-tsc/test/NodeHelper.js.map +1 -1
  147. package/out-tsc/test/actions/add_contact_urn.test.js +202 -0
  148. package/out-tsc/test/actions/add_contact_urn.test.js.map +1 -0
  149. package/out-tsc/test/actions/send_broadcast.test.js +148 -0
  150. package/out-tsc/test/actions/send_broadcast.test.js.map +1 -0
  151. package/out-tsc/test/actions/send_email.test.js +17 -23
  152. package/out-tsc/test/actions/send_email.test.js.map +1 -1
  153. package/out-tsc/test/actions/send_msg.test.js +33 -15
  154. package/out-tsc/test/actions/send_msg.test.js.map +1 -1
  155. package/out-tsc/test/actions/start_session.test.js +116 -0
  156. package/out-tsc/test/actions/start_session.test.js.map +1 -0
  157. package/out-tsc/test/nodes/split_by_airtime.test.js +604 -0
  158. package/out-tsc/test/nodes/split_by_airtime.test.js.map +1 -0
  159. package/out-tsc/test/nodes/split_by_contact_field.test.js +387 -0
  160. package/out-tsc/test/nodes/split_by_contact_field.test.js.map +1 -0
  161. package/out-tsc/test/nodes/split_by_expression.test.js +614 -0
  162. package/out-tsc/test/nodes/split_by_expression.test.js.map +1 -0
  163. package/out-tsc/test/nodes/split_by_random.test.js +3 -3
  164. package/out-tsc/test/nodes/split_by_random.test.js.map +1 -1
  165. package/out-tsc/test/nodes/split_by_resthook.test.js +337 -0
  166. package/out-tsc/test/nodes/split_by_resthook.test.js.map +1 -0
  167. package/out-tsc/test/nodes/split_by_run_result.test.js +920 -0
  168. package/out-tsc/test/nodes/split_by_run_result.test.js.map +1 -0
  169. package/out-tsc/test/nodes/split_by_scheme.test.js +399 -0
  170. package/out-tsc/test/nodes/split_by_scheme.test.js.map +1 -0
  171. package/out-tsc/test/nodes/split_by_subflow.test.js +333 -0
  172. package/out-tsc/test/nodes/split_by_subflow.test.js.map +1 -0
  173. package/out-tsc/test/nodes/wait_for_digits.test.js +2 -2
  174. package/out-tsc/test/nodes/wait_for_digits.test.js.map +1 -1
  175. package/out-tsc/test/nodes/wait_for_response.test.js +2 -1
  176. package/out-tsc/test/nodes/wait_for_response.test.js.map +1 -1
  177. package/out-tsc/test/temba-action-drag-between-nodes.test.js +252 -0
  178. package/out-tsc/test/temba-action-drag-between-nodes.test.js.map +1 -0
  179. package/out-tsc/test/temba-canvas-menu.test.js +122 -0
  180. package/out-tsc/test/temba-canvas-menu.test.js.map +1 -0
  181. package/out-tsc/test/temba-floating-tab.test.js +91 -0
  182. package/out-tsc/test/temba-floating-tab.test.js.map +1 -0
  183. package/out-tsc/test/temba-floating-window.test.js +301 -0
  184. package/out-tsc/test/temba-floating-window.test.js.map +1 -0
  185. package/out-tsc/test/temba-flow-editor-node.test.js +202 -2
  186. package/out-tsc/test/temba-flow-editor-node.test.js.map +1 -1
  187. package/out-tsc/test/temba-flow-editor.test.js +7 -8
  188. package/out-tsc/test/temba-flow-editor.test.js.map +1 -1
  189. package/out-tsc/test/temba-localization.test.js +471 -0
  190. package/out-tsc/test/temba-localization.test.js.map +1 -0
  191. package/out-tsc/test/temba-node-editor.test.js +3 -1
  192. package/out-tsc/test/temba-node-editor.test.js.map +1 -1
  193. package/out-tsc/test/temba-node-type-selector.test.js +265 -0
  194. package/out-tsc/test/temba-node-type-selector.test.js.map +1 -0
  195. package/out-tsc/test/temba-omnibox.test.js +2 -1
  196. package/out-tsc/test/temba-omnibox.test.js.map +1 -1
  197. package/out-tsc/test/temba-sortable-list.test.js +51 -0
  198. package/out-tsc/test/temba-sortable-list.test.js.map +1 -1
  199. package/out-tsc/test/temba-utils-index.test.js +1 -27
  200. package/out-tsc/test/temba-utils-index.test.js.map +1 -1
  201. package/out-tsc/test/utils.test.js +20 -0
  202. package/out-tsc/test/utils.test.js.map +1 -1
  203. package/package.json +2 -1
  204. package/screenshots/truth/actions/add_contact_groups/editor/descriptive-group-names.png +0 -0
  205. package/screenshots/truth/actions/add_contact_groups/editor/long-group-names.png +0 -0
  206. package/screenshots/truth/actions/add_contact_groups/editor/many-groups.png +0 -0
  207. package/screenshots/truth/actions/add_contact_groups/editor/multiple-groups.png +0 -0
  208. package/screenshots/truth/actions/add_contact_groups/editor/single-group.png +0 -0
  209. package/screenshots/truth/actions/add_contact_groups/render/descriptive-group-names.png +0 -0
  210. package/screenshots/truth/actions/add_contact_groups/render/long-group-names.png +0 -0
  211. package/screenshots/truth/actions/add_contact_groups/render/many-groups.png +0 -0
  212. package/screenshots/truth/actions/add_contact_groups/render/multiple-groups.png +0 -0
  213. package/screenshots/truth/actions/add_contact_groups/render/single-group.png +0 -0
  214. package/screenshots/truth/actions/add_contact_urn/editor/expression-facebook.png +0 -0
  215. package/screenshots/truth/actions/add_contact_urn/editor/expression-phone.png +0 -0
  216. package/screenshots/truth/actions/add_contact_urn/editor/facebook-id.png +0 -0
  217. package/screenshots/truth/actions/add_contact_urn/editor/instagram-handle.png +0 -0
  218. package/screenshots/truth/actions/add_contact_urn/editor/line-id.png +0 -0
  219. package/screenshots/truth/actions/add_contact_urn/editor/phone-number.png +0 -0
  220. package/screenshots/truth/actions/add_contact_urn/editor/telegram-id.png +0 -0
  221. package/screenshots/truth/actions/add_contact_urn/editor/viber-id.png +0 -0
  222. package/screenshots/truth/actions/add_contact_urn/editor/wechat-id.png +0 -0
  223. package/screenshots/truth/actions/add_contact_urn/editor/whatsapp.png +0 -0
  224. package/screenshots/truth/actions/add_contact_urn/render/expression-facebook.png +0 -0
  225. package/screenshots/truth/actions/add_contact_urn/render/expression-phone.png +0 -0
  226. package/screenshots/truth/actions/add_contact_urn/render/facebook-id.png +0 -0
  227. package/screenshots/truth/actions/add_contact_urn/render/instagram-handle.png +0 -0
  228. package/screenshots/truth/actions/add_contact_urn/render/line-id.png +0 -0
  229. package/screenshots/truth/actions/add_contact_urn/render/phone-number.png +0 -0
  230. package/screenshots/truth/actions/add_contact_urn/render/telegram-id.png +0 -0
  231. package/screenshots/truth/actions/add_contact_urn/render/viber-id.png +0 -0
  232. package/screenshots/truth/actions/add_contact_urn/render/wechat-id.png +0 -0
  233. package/screenshots/truth/actions/add_contact_urn/render/whatsapp.png +0 -0
  234. package/screenshots/truth/actions/remove_contact_groups/editor/cleanup-groups.png +0 -0
  235. package/screenshots/truth/actions/remove_contact_groups/editor/long-descriptive-group-names.png +0 -0
  236. package/screenshots/truth/actions/remove_contact_groups/editor/many-groups.png +0 -0
  237. package/screenshots/truth/actions/remove_contact_groups/editor/multiple-groups.png +0 -0
  238. package/screenshots/truth/actions/remove_contact_groups/editor/remove-from-all-groups.png +0 -0
  239. package/screenshots/truth/actions/remove_contact_groups/editor/single-group.png +0 -0
  240. package/screenshots/truth/actions/remove_contact_groups/render/cleanup-groups.png +0 -0
  241. package/screenshots/truth/actions/remove_contact_groups/render/long-descriptive-group-names.png +0 -0
  242. package/screenshots/truth/actions/remove_contact_groups/render/many-groups.png +0 -0
  243. package/screenshots/truth/actions/remove_contact_groups/render/multiple-groups.png +0 -0
  244. package/screenshots/truth/actions/remove_contact_groups/render/remove-from-all-groups.png +0 -0
  245. package/screenshots/truth/actions/remove_contact_groups/render/single-group.png +0 -0
  246. package/screenshots/truth/actions/send_broadcast/editor/contacts-only.png +0 -0
  247. package/screenshots/truth/actions/send_broadcast/editor/groups-and-contacts.png +0 -0
  248. package/screenshots/truth/actions/send_broadcast/editor/groups-only.png +0 -0
  249. package/screenshots/truth/actions/send_broadcast/editor/many-groups.png +0 -0
  250. package/screenshots/truth/actions/send_broadcast/editor/multiline-text.png +0 -0
  251. package/screenshots/truth/actions/send_broadcast/editor/with-attachments.png +0 -0
  252. package/screenshots/truth/actions/send_broadcast/render/contacts-only.png +0 -0
  253. package/screenshots/truth/actions/send_broadcast/render/groups-and-contacts.png +0 -0
  254. package/screenshots/truth/actions/send_broadcast/render/groups-only.png +0 -0
  255. package/screenshots/truth/actions/send_broadcast/render/many-groups.png +0 -0
  256. package/screenshots/truth/actions/send_broadcast/render/multiline-text.png +0 -0
  257. package/screenshots/truth/actions/send_broadcast/render/with-attachments.png +0 -0
  258. package/screenshots/truth/actions/send_email/editor/complex-business-email.png +0 -0
  259. package/screenshots/truth/actions/send_email/editor/empty-body.png +0 -0
  260. package/screenshots/truth/actions/send_email/editor/empty-subject.png +0 -0
  261. package/screenshots/truth/actions/send_email/editor/long-subject.png +0 -0
  262. package/screenshots/truth/actions/send_email/editor/multiline-body.png +0 -0
  263. package/screenshots/truth/actions/send_email/editor/multiple-recipients.png +0 -0
  264. package/screenshots/truth/actions/send_email/editor/simple-email.png +0 -0
  265. package/screenshots/truth/actions/send_email/editor/with-expressions.png +0 -0
  266. package/screenshots/truth/actions/send_email/render/complex-business-email.png +0 -0
  267. package/screenshots/truth/actions/send_email/render/empty-body.png +0 -0
  268. package/screenshots/truth/actions/send_email/render/empty-subject.png +0 -0
  269. package/screenshots/truth/actions/send_email/render/long-subject.png +0 -0
  270. package/screenshots/truth/actions/send_email/render/multiline-body.png +0 -0
  271. package/screenshots/truth/actions/send_email/render/multiple-recipients.png +0 -0
  272. package/screenshots/truth/actions/send_email/render/simple-email.png +0 -0
  273. package/screenshots/truth/actions/send_email/render/with-expressions.png +0 -0
  274. package/screenshots/truth/actions/send_msg/editor/long-quick-replies.png +0 -0
  275. package/screenshots/truth/actions/send_msg/editor/multiline-text-with-replies.png +0 -0
  276. package/screenshots/truth/actions/send_msg/editor/simple-text.png +0 -0
  277. package/screenshots/truth/actions/send_msg/editor/text-with-linebreaks.png +0 -0
  278. package/screenshots/truth/actions/send_msg/editor/text-with-many-quick-replies.png +0 -0
  279. package/screenshots/truth/actions/send_msg/editor/text-with-quick-replies.png +0 -0
  280. package/screenshots/truth/actions/send_msg/editor/text-without-quick-replies.png +0 -0
  281. package/screenshots/truth/actions/send_msg/render/long-quick-replies.png +0 -0
  282. package/screenshots/truth/actions/send_msg/render/multiline-text-with-replies.png +0 -0
  283. package/screenshots/truth/actions/send_msg/render/simple-text.png +0 -0
  284. package/screenshots/truth/actions/send_msg/render/text-with-linebreaks.png +0 -0
  285. package/screenshots/truth/actions/send_msg/render/text-with-many-quick-replies.png +0 -0
  286. package/screenshots/truth/actions/send_msg/render/text-with-quick-replies.png +0 -0
  287. package/screenshots/truth/actions/send_msg/render/text-without-quick-replies.png +0 -0
  288. package/screenshots/truth/actions/start_session/editor/contact-query.png +0 -0
  289. package/screenshots/truth/actions/start_session/editor/contacts-only.png +0 -0
  290. package/screenshots/truth/actions/start_session/editor/create-contact.png +0 -0
  291. package/screenshots/truth/actions/start_session/editor/groups-and-contacts.png +0 -0
  292. package/screenshots/truth/actions/start_session/editor/groups-only.png +0 -0
  293. package/screenshots/truth/actions/start_session/editor/many-recipients.png +0 -0
  294. package/screenshots/truth/actions/start_session/render/contact-query.png +0 -0
  295. package/screenshots/truth/actions/start_session/render/contacts-only.png +0 -0
  296. package/screenshots/truth/actions/start_session/render/create-contact.png +0 -0
  297. package/screenshots/truth/actions/start_session/render/groups-and-contacts.png +0 -0
  298. package/screenshots/truth/actions/start_session/render/groups-only.png +0 -0
  299. package/screenshots/truth/actions/start_session/render/many-recipients.png +0 -0
  300. package/screenshots/truth/canvas-menu/open.png +0 -0
  301. package/screenshots/truth/editor/router.png +0 -0
  302. package/screenshots/truth/editor/wait.png +0 -0
  303. package/screenshots/truth/floating-tab/default.png +0 -0
  304. package/screenshots/truth/floating-tab/gray.png +0 -0
  305. package/screenshots/truth/floating-tab/green.png +0 -0
  306. package/screenshots/truth/floating-tab/hidden.png +0 -0
  307. package/screenshots/truth/floating-tab/hover.png +0 -0
  308. package/screenshots/truth/floating-tab/purple.png +0 -0
  309. package/screenshots/truth/floating-window/chromeless.png +0 -0
  310. package/screenshots/truth/floating-window/custom-size.png +0 -0
  311. package/screenshots/truth/floating-window/default.png +0 -0
  312. package/screenshots/truth/floating-window/with-header.png +0 -0
  313. package/screenshots/truth/list/fields-dragging.png +0 -0
  314. package/screenshots/truth/list/sortable-dragging.png +0 -0
  315. package/screenshots/truth/node-type-selector/action-mode.png +0 -0
  316. package/screenshots/truth/node-type-selector/split-mode.png +0 -0
  317. package/screenshots/truth/nodes/split_by_llm/editor/information-extraction.png +0 -0
  318. package/screenshots/truth/nodes/split_by_llm/editor/sentiment-analysis.png +0 -0
  319. package/screenshots/truth/nodes/split_by_llm/editor/summarization.png +0 -0
  320. package/screenshots/truth/nodes/split_by_llm/editor/translation-task.png +0 -0
  321. package/screenshots/truth/nodes/split_by_llm/render/information-extraction.png +0 -0
  322. package/screenshots/truth/nodes/split_by_llm/render/sentiment-analysis.png +0 -0
  323. package/screenshots/truth/nodes/split_by_llm/render/summarization.png +0 -0
  324. package/screenshots/truth/nodes/split_by_llm/render/translation-task.png +0 -0
  325. package/screenshots/truth/nodes/split_by_llm_categorize/editor/basic-categorization.png +0 -0
  326. package/screenshots/truth/nodes/split_by_llm_categorize/editor/custom-input-and-result-name.png +0 -0
  327. package/screenshots/truth/nodes/split_by_llm_categorize/editor/feedback-categorization.png +0 -0
  328. package/screenshots/truth/nodes/split_by_llm_categorize/editor/many-categories.png +0 -0
  329. package/screenshots/truth/nodes/split_by_llm_categorize/editor/minimal-categories.png +0 -0
  330. package/screenshots/truth/nodes/split_by_llm_categorize/render/basic-categorization.png +0 -0
  331. package/screenshots/truth/nodes/split_by_llm_categorize/render/custom-input-and-result-name.png +0 -0
  332. package/screenshots/truth/nodes/split_by_llm_categorize/render/feedback-categorization.png +0 -0
  333. package/screenshots/truth/nodes/split_by_llm_categorize/render/many-categories.png +0 -0
  334. package/screenshots/truth/nodes/split_by_llm_categorize/render/minimal-categories.png +0 -0
  335. package/screenshots/truth/nodes/split_by_random/editor/ab-test-multiple-variants.png +0 -0
  336. package/screenshots/truth/nodes/split_by_random/editor/sampling-split.png +0 -0
  337. package/screenshots/truth/nodes/split_by_random/editor/three-way-split.png +0 -0
  338. package/screenshots/truth/nodes/split_by_random/editor/two-bucket-split.png +0 -0
  339. package/screenshots/truth/nodes/split_by_random/render/ab-test-multiple-variants.png +0 -0
  340. package/screenshots/truth/nodes/split_by_random/render/sampling-split.png +0 -0
  341. package/screenshots/truth/nodes/split_by_random/render/three-way-split.png +0 -0
  342. package/screenshots/truth/nodes/split_by_random/render/two-bucket-split.png +0 -0
  343. package/screenshots/truth/nodes/wait_for_digits/editor/basic-digits-wait.png +0 -0
  344. package/screenshots/truth/nodes/wait_for_digits/editor/phone-number-collection.png +0 -0
  345. package/screenshots/truth/nodes/wait_for_digits/editor/single-digit-with-timeout.png +0 -0
  346. package/screenshots/truth/nodes/wait_for_digits/editor/verification-code.png +0 -0
  347. package/screenshots/truth/nodes/wait_for_digits/render/basic-digits-wait.png +0 -0
  348. package/screenshots/truth/nodes/wait_for_digits/render/phone-number-collection.png +0 -0
  349. package/screenshots/truth/nodes/wait_for_digits/render/single-digit-with-timeout.png +0 -0
  350. package/screenshots/truth/nodes/wait_for_digits/render/verification-code.png +0 -0
  351. package/screenshots/truth/nodes/wait_for_response/editor/basic-wait.png +0 -0
  352. package/screenshots/truth/nodes/wait_for_response/editor/custom-result-name.png +0 -0
  353. package/screenshots/truth/nodes/wait_for_response/editor/no-timeout.png +0 -0
  354. package/screenshots/truth/nodes/wait_for_response/editor/short-timeout.png +0 -0
  355. package/screenshots/truth/nodes/wait_for_response/render/basic-wait.png +0 -0
  356. package/screenshots/truth/nodes/wait_for_response/render/custom-result-name.png +0 -0
  357. package/screenshots/truth/nodes/wait_for_response/render/no-timeout.png +0 -0
  358. package/screenshots/truth/nodes/wait_for_response/render/short-timeout.png +0 -0
  359. package/src/Icons.ts +4 -1
  360. package/src/display/FloatingTab.ts +174 -0
  361. package/src/display/ProgressBar.ts +22 -2
  362. package/src/events.ts +2 -8
  363. package/src/flow/CanvasMenu.ts +217 -0
  364. package/src/flow/CanvasNode.ts +596 -40
  365. package/src/flow/Editor.ts +1721 -45
  366. package/src/flow/NodeEditor.ts +621 -144
  367. package/src/flow/NodeTypeSelector.ts +636 -0
  368. package/src/flow/StickyNote.ts +12 -3
  369. package/src/flow/actions/add_contact_groups.ts +5 -4
  370. package/src/flow/actions/add_contact_urn.ts +78 -4
  371. package/src/flow/actions/add_input_labels.ts +5 -4
  372. package/src/flow/actions/play_audio.ts +3 -2
  373. package/src/flow/actions/remove_contact_groups.ts +16 -6
  374. package/src/flow/actions/request_optin.ts +3 -2
  375. package/src/flow/actions/say_msg.ts +3 -2
  376. package/src/flow/actions/send_broadcast.ts +86 -23
  377. package/src/flow/actions/send_email.ts +12 -6
  378. package/src/flow/actions/send_msg.ts +155 -34
  379. package/src/flow/actions/set_contact_channel.ts +6 -11
  380. package/src/flow/actions/set_contact_field.ts +21 -25
  381. package/src/flow/actions/set_contact_language.ts +11 -4
  382. package/src/flow/actions/set_contact_name.ts +4 -15
  383. package/src/flow/actions/set_contact_status.ts +4 -3
  384. package/src/flow/actions/set_run_result.ts +5 -4
  385. package/src/flow/actions/start_session.ts +210 -6
  386. package/src/flow/config.ts +11 -23
  387. package/src/flow/currencies.ts +51 -0
  388. package/src/flow/nodes/shared-rules.ts +301 -0
  389. package/src/flow/nodes/shared.ts +87 -0
  390. package/src/flow/nodes/split_by_airtime.ts +255 -5
  391. package/src/flow/nodes/split_by_contact_field.ts +195 -3
  392. package/src/flow/nodes/split_by_expression.ts +104 -2
  393. package/src/flow/nodes/split_by_groups.ts +26 -11
  394. package/src/flow/nodes/split_by_intent.ts +8 -0
  395. package/src/flow/nodes/split_by_llm.ts +22 -4
  396. package/src/flow/nodes/split_by_llm_categorize.ts +22 -5
  397. package/src/flow/nodes/split_by_random.ts +16 -6
  398. package/src/flow/nodes/split_by_resthook.ts +140 -0
  399. package/src/flow/nodes/split_by_run_result.ts +259 -3
  400. package/src/flow/nodes/split_by_scheme.ts +202 -2
  401. package/src/flow/nodes/split_by_subflow.ts +17 -5
  402. package/src/flow/nodes/split_by_ticket.ts +15 -4
  403. package/src/flow/nodes/split_by_webhook.ts +17 -6
  404. package/src/flow/nodes/wait_for_digits.ts +3 -2
  405. package/src/flow/nodes/wait_for_menu.ts +3 -2
  406. package/src/flow/nodes/wait_for_response.ts +59 -680
  407. package/src/flow/types.ts +156 -23
  408. package/src/flow/utils.ts +108 -14
  409. package/src/form/FieldRenderer.ts +2 -4
  410. package/src/interfaces.ts +3 -0
  411. package/src/layout/FloatingWindow.ts +386 -0
  412. package/src/list/SortableList.ts +109 -34
  413. package/src/live/ContactChat.ts +7 -25
  414. package/src/locales/es.ts +18 -13
  415. package/src/locales/fr.ts +18 -13
  416. package/src/locales/locale-codes.ts +11 -2
  417. package/src/locales/pt.ts +18 -13
  418. package/src/store/AppState.ts +173 -0
  419. package/src/store/flow-definition.d.ts +2 -5
  420. package/src/utils.ts +332 -12
  421. package/static/api/channels.json +46 -0
  422. package/static/api/llms.json +18 -0
  423. package/static/api/resthooks.json +31 -0
  424. package/static/svg/index.svg +1 -1
  425. package/static/svg/work/traced/lightning-02.svg +1 -0
  426. package/static/svg/work/used/lightning-02.svg +3 -0
  427. package/temba-modules.ts +8 -0
  428. package/test/ActionHelper.ts +3 -3
  429. package/test/NodeHelper.ts +6 -3
  430. package/test/actions/add_contact_urn.test.ts +287 -0
  431. package/test/actions/send_broadcast.test.ts +190 -0
  432. package/test/actions/send_email.test.ts +17 -23
  433. package/test/actions/send_msg.test.ts +39 -15
  434. package/test/actions/start_session.test.ts +151 -0
  435. package/test/nodes/split_by_airtime.test.ts +673 -0
  436. package/test/nodes/split_by_contact_field.test.ts +451 -0
  437. package/test/nodes/split_by_expression.test.ts +751 -0
  438. package/test/nodes/split_by_random.test.ts +3 -3
  439. package/test/nodes/split_by_resthook.test.ts +398 -0
  440. package/test/nodes/split_by_run_result.test.ts +1109 -0
  441. package/test/nodes/split_by_scheme.test.ts +486 -0
  442. package/test/nodes/split_by_subflow.test.ts +381 -0
  443. package/test/nodes/wait_for_digits.test.ts +2 -2
  444. package/test/nodes/wait_for_response.test.ts +2 -1
  445. package/test/temba-action-drag-between-nodes.test.ts +301 -0
  446. package/test/temba-canvas-menu.test.ts +156 -0
  447. package/test/temba-floating-tab.test.ts +110 -0
  448. package/test/temba-floating-window.test.ts +477 -0
  449. package/test/temba-flow-editor-node.test.ts +246 -2
  450. package/test/temba-flow-editor.test.ts +7 -8
  451. package/test/temba-localization.test.ts +611 -0
  452. package/test/temba-node-editor.test.ts +3 -1
  453. package/test/temba-node-type-selector.test.ts +355 -0
  454. package/test/temba-omnibox.test.ts +2 -1
  455. package/test/temba-sortable-list.test.ts +69 -0
  456. package/test/temba-utils-index.test.ts +0 -35
  457. package/test/utils.test.ts +22 -0
  458. package/test-assets/contacts/history.json +14 -21
  459. package/test-assets/select/llms.json +2 -2
  460. package/web-dev-server.config.mjs +49 -1
  461. package/web-test-runner.config.mjs +0 -1
  462. package/out-tsc/src/flow/actions/call_classifier.js +0 -11
  463. package/out-tsc/src/flow/actions/call_classifier.js.map +0 -1
  464. package/out-tsc/src/flow/actions/call_resthook.js +0 -11
  465. package/out-tsc/src/flow/actions/call_resthook.js.map +0 -1
  466. package/out-tsc/src/flow/actions/split_by_expression_example.js +0 -77
  467. package/out-tsc/src/flow/actions/split_by_expression_example.js.map +0 -1
  468. package/out-tsc/src/flow/actions/transfer_airtime.js +0 -11
  469. package/out-tsc/src/flow/actions/transfer_airtime.js.map +0 -1
  470. package/out-tsc/src/flow/nodes/wait_for_audio.js +0 -7
  471. package/out-tsc/src/flow/nodes/wait_for_audio.js.map +0 -1
  472. package/out-tsc/src/flow/nodes/wait_for_image.js +0 -7
  473. package/out-tsc/src/flow/nodes/wait_for_image.js.map +0 -1
  474. package/out-tsc/src/flow/nodes/wait_for_location.js +0 -7
  475. package/out-tsc/src/flow/nodes/wait_for_location.js.map +0 -1
  476. package/out-tsc/src/flow/nodes/wait_for_video.js +0 -7
  477. package/out-tsc/src/flow/nodes/wait_for_video.js.map +0 -1
  478. package/src/flow/actions/call_classifier.ts +0 -12
  479. package/src/flow/actions/call_resthook.ts +0 -12
  480. package/src/flow/actions/split_by_expression_example.ts +0 -88
  481. package/src/flow/actions/transfer_airtime.ts +0 -12
  482. package/src/flow/nodes/wait_for_audio.ts +0 -7
  483. package/src/flow/nodes/wait_for_image.ts +0 -7
  484. package/src/flow/nodes/wait_for_location.ts +0 -7
  485. package/src/flow/nodes/wait_for_video.ts +0 -7
@@ -1,10 +1,11 @@
1
1
  import { html } from 'lit-html';
2
- import { ActionConfig, COLORS, ValidationResult } from '../types';
2
+ import { ActionConfig, ACTION_GROUPS, FlowTypes } from '../types';
3
3
  import { Node, SetContactChannel } from '../../store/flow-definition';
4
4
 
5
5
  export const set_contact_channel: ActionConfig = {
6
6
  name: 'Update Channel',
7
- color: COLORS.update,
7
+ group: ACTION_GROUPS.contacts,
8
+ flowTypes: [FlowTypes.VOICE, FlowTypes.MESSAGE, FlowTypes.BACKGROUND],
8
9
  render: (_node: Node, action: SetContactChannel) => {
9
10
  return html`<div>Set to <strong>${action.channel.name}</strong></div>`;
10
11
  },
@@ -21,16 +22,10 @@ export const set_contact_channel: ActionConfig = {
21
22
  helpText: 'Select the channel to set for the contact'
22
23
  }
23
24
  },
24
- validate: (formData: SetContactChannel): ValidationResult => {
25
- const errors: { [key: string]: string } = {};
26
-
27
- if (!formData.channel) {
28
- errors.channel = 'Channel is required';
29
- }
30
-
25
+ toFormData: (action: SetContactChannel) => {
31
26
  return {
32
- valid: Object.keys(errors).length === 0,
33
- errors
27
+ uuid: action.uuid,
28
+ channel: action.channel ? [action.channel] : null
34
29
  };
35
30
  }
36
31
  };
@@ -1,15 +1,20 @@
1
1
  import { html } from 'lit-html';
2
- import { ActionConfig, COLORS, ValidationResult } from '../types';
2
+ import { ActionConfig, ACTION_GROUPS, FormData, FlowTypes } from '../types';
3
3
  import { Node, SetContactField } from '../../store/flow-definition';
4
4
 
5
5
  export const set_contact_field: ActionConfig = {
6
6
  name: 'Update Field',
7
- color: COLORS.update,
7
+ group: ACTION_GROUPS.contacts,
8
+ flowTypes: [FlowTypes.VOICE, FlowTypes.MESSAGE, FlowTypes.BACKGROUND],
8
9
  render: (_node: Node, action: SetContactField) => {
9
- return html`<div>
10
- Set <strong>${action.field.name}</strong> to
11
- <strong>${action.value}</strong>
12
- </div>`;
10
+ if (action.value) {
11
+ return html`<div>
12
+ Set <strong>${action.field.name}</strong> to
13
+ <strong>${action.value}</strong>
14
+ </div>`;
15
+ } else {
16
+ return html`<div>Clear <strong>${action.field.name}</strong></div>`;
17
+ }
13
18
  },
14
19
  form: {
15
20
  field: {
@@ -18,6 +23,7 @@ export const set_contact_field: ActionConfig = {
18
23
  required: true,
19
24
  searchable: true,
20
25
  clearable: false,
26
+ placeholder: 'Search for contact fields...',
21
27
  nameKey: 'name',
22
28
  valueKey: 'key',
23
29
  endpoint: '/api/v2/fields.json',
@@ -29,13 +35,19 @@ export const set_contact_field: ActionConfig = {
29
35
  type: 'text',
30
36
  label: 'Value',
31
37
  placeholder: 'Enter field value...',
32
- required: true,
33
38
  evaluated: true,
34
39
  helpText:
35
40
  'The new value for the contact field. You can use expressions like @contact.name'
36
41
  }
37
42
  },
38
- fromFormData: (formData: SetContactField): SetContactField => {
43
+ toFormData: (action: SetContactField) => {
44
+ return {
45
+ uuid: action.uuid,
46
+ field: action.field ? [action.field] : null,
47
+ value: action.value
48
+ };
49
+ },
50
+ fromFormData: (formData: FormData): SetContactField => {
39
51
  const field = formData.field[0];
40
52
  return {
41
53
  uuid: formData.uuid,
@@ -44,23 +56,7 @@ export const set_contact_field: ActionConfig = {
44
56
  value: formData.value
45
57
  };
46
58
  },
47
- validate: (formData: SetContactField): ValidationResult => {
48
- const errors: { [key: string]: string } = {};
49
-
50
- if (!formData.field) {
51
- errors.field = 'Field is required';
52
- }
53
-
54
- if (!formData.value || formData.value.trim() === '') {
55
- errors.value = 'Field value is required';
56
- }
57
-
58
- return {
59
- valid: Object.keys(errors).length === 0,
60
- errors
61
- };
62
- },
63
- sanitize: (formData: SetContactField): void => {
59
+ sanitize: (formData: FormData): void => {
64
60
  if (formData.value && typeof formData.value === 'string') {
65
61
  formData.value = formData.value.trim();
66
62
  }
@@ -1,11 +1,18 @@
1
1
  import { html } from 'lit-html';
2
- import { ActionConfig, COLORS, ValidationResult } from '../types';
2
+ import {
3
+ ActionConfig,
4
+ ACTION_GROUPS,
5
+ FormData,
6
+ ValidationResult,
7
+ FlowTypes
8
+ } from '../types';
3
9
  import { Node, SetContactLanguage } from '../../store/flow-definition';
4
10
  import { getStore } from '../../store/Store';
5
11
 
6
12
  export const set_contact_language: ActionConfig = {
7
13
  name: 'Update Language',
8
- color: COLORS.update,
14
+ group: ACTION_GROUPS.contacts,
15
+ flowTypes: [FlowTypes.VOICE, FlowTypes.MESSAGE, FlowTypes.BACKGROUND],
9
16
  render: (_node: Node, action: SetContactLanguage) => {
10
17
  const languageNames = new Intl.DisplayNames(['en'], {
11
18
  type: 'language'
@@ -61,7 +68,7 @@ export const set_contact_language: ActionConfig = {
61
68
  uuid: action.uuid
62
69
  };
63
70
  },
64
- fromFormData: (formData: any): SetContactLanguage => {
71
+ fromFormData: (formData: FormData): SetContactLanguage => {
65
72
  return {
66
73
  uuid: formData.uuid,
67
74
  type: 'set_contact_language',
@@ -69,7 +76,7 @@ export const set_contact_language: ActionConfig = {
69
76
  };
70
77
  },
71
78
 
72
- validate: (formData: any): ValidationResult => {
79
+ validate: (formData: FormData): ValidationResult => {
73
80
  const errors: { [key: string]: string } = {};
74
81
 
75
82
  if (!formData.language) {
@@ -1,10 +1,11 @@
1
1
  import { html } from 'lit-html';
2
- import { ActionConfig, COLORS, ValidationResult } from '../types';
2
+ import { ActionConfig, ACTION_GROUPS, FormData, FlowTypes } from '../types';
3
3
  import { Node, SetContactName } from '../../store/flow-definition';
4
4
 
5
5
  export const set_contact_name: ActionConfig = {
6
6
  name: 'Update Name',
7
- color: COLORS.update,
7
+ group: ACTION_GROUPS.contacts,
8
+ flowTypes: [FlowTypes.VOICE, FlowTypes.MESSAGE, FlowTypes.BACKGROUND],
8
9
  render: (_node: Node, action: SetContactName) => {
9
10
  return html`<div>Set to <strong>${action.name}</strong></div>`;
10
11
  },
@@ -19,19 +20,7 @@ export const set_contact_name: ActionConfig = {
19
20
  'The new name for the contact. You can use expressions like @contact.name'
20
21
  }
21
22
  },
22
- validate: (formData: SetContactName): ValidationResult => {
23
- const errors: { [key: string]: string } = {};
24
-
25
- if (!formData.name || formData.name.trim() === '') {
26
- errors.name = 'Name is required';
27
- }
28
-
29
- return {
30
- valid: Object.keys(errors).length === 0,
31
- errors
32
- };
33
- },
34
- sanitize: (formData: SetContactName): void => {
23
+ sanitize: (formData: FormData): void => {
35
24
  if (formData.name && typeof formData.name === 'string') {
36
25
  formData.name = formData.name.trim();
37
26
  }
@@ -1,11 +1,12 @@
1
1
  import { html } from 'lit-html';
2
- import { ActionConfig, COLORS } from '../types';
2
+ import { ActionConfig, ACTION_GROUPS, FormData, FlowTypes } from '../types';
3
3
  import { Node, SetContactStatus } from '../../store/flow-definition';
4
4
  import { titleCase } from '../../utils';
5
5
 
6
6
  export const set_contact_status: ActionConfig = {
7
7
  name: 'Update Status',
8
- color: COLORS.update,
8
+ group: ACTION_GROUPS.contacts,
9
+ flowTypes: [FlowTypes.VOICE, FlowTypes.MESSAGE, FlowTypes.BACKGROUND],
9
10
  render: (_node: Node, action: SetContactStatus) => {
10
11
  return html`<div>Set to <strong>${titleCase(action.status)}</strong></div>`;
11
12
  },
@@ -18,7 +19,7 @@ export const set_contact_status: ActionConfig = {
18
19
  }
19
20
  };
20
21
  },
21
- fromFormData: (formData: any): SetContactStatus => {
22
+ fromFormData: (formData: FormData): SetContactStatus => {
22
23
  return {
23
24
  status: formData.status[0].value,
24
25
  type: 'set_contact_status',
@@ -1,11 +1,12 @@
1
1
  import { html } from 'lit-html';
2
- import { ActionConfig, COLORS } from '../types';
2
+ import { ActionConfig, ACTION_GROUPS, FormData, FlowTypes } from '../types';
3
3
  import { Node, SetRunResult } from '../../store/flow-definition';
4
4
  import { getStore } from '../../store/Store';
5
5
 
6
6
  export const set_run_result: ActionConfig = {
7
7
  name: 'Save Flow Result',
8
- color: COLORS.save,
8
+ group: ACTION_GROUPS.save,
9
+ flowTypes: [FlowTypes.VOICE, FlowTypes.MESSAGE, FlowTypes.BACKGROUND],
9
10
  render: (_node: Node, action: SetRunResult) => {
10
11
  return html`<div>
11
12
  Save <strong>${action.value}</strong> as <strong>${action.name}</strong>
@@ -60,12 +61,12 @@ export const set_run_result: ActionConfig = {
60
61
  toFormData: (action: SetRunResult) => {
61
62
  return {
62
63
  uuid: action.uuid,
63
- name: action.name ? [{ name: action.name, value: action.name }] : [],
64
+ name: action.name ? [{ name: action.name, value: action.name }] : null,
64
65
  value: action.value || '',
65
66
  category: action.category || ''
66
67
  };
67
68
  },
68
- fromFormData: (formData: any): SetRunResult => {
69
+ fromFormData: (formData: FormData): SetRunResult => {
69
70
  // Ensure name is a simple string, handling both direct values and select option objects
70
71
  let name = formData.name || '';
71
72
  if (Array.isArray(name) && name.length > 0) {
@@ -1,12 +1,216 @@
1
1
  import { html } from 'lit-html';
2
- import { ActionConfig, COLORS } from '../types';
2
+ import {
3
+ ActionConfig,
4
+ ACTION_GROUPS,
5
+ FormData,
6
+ ValidationResult,
7
+ FlowTypes
8
+ } from '../types';
3
9
  import { Node, StartSession } from '../../store/flow-definition';
10
+ import { renderNamedObjects } from '../utils';
4
11
 
5
12
  export const start_session: ActionConfig = {
6
- name: 'Start Session',
7
- color: COLORS.execute,
8
- render: (_node: Node, _action: StartSession) => {
9
- // This will need to be implemented based on the actual render logic
10
- return html`<div>Start Session</div>`;
13
+ name: 'Start Flow',
14
+ group: ACTION_GROUPS.broadcast,
15
+ flowTypes: [FlowTypes.VOICE, FlowTypes.MESSAGE, FlowTypes.BACKGROUND],
16
+ render: (_node: Node, action: StartSession) => {
17
+ const hasGroups = action.groups && action.groups.length > 0;
18
+ const hasContacts = action.contacts && action.contacts.length > 0;
19
+ const hasRecipients = hasGroups || hasContacts;
20
+
21
+ // Build the recipients display
22
+ let recipientsDisplay = html``;
23
+ if (action.create_contact) {
24
+ recipientsDisplay = html`Create a new contact`;
25
+ } else if ((action as any).contact_query) {
26
+ recipientsDisplay = html`${(action as any).contact_query}`;
27
+ } else if (hasRecipients) {
28
+ const allRecipients = [
29
+ ...(action.contacts || []),
30
+ ...(action.groups || [])
31
+ ];
32
+ recipientsDisplay = html`${renderNamedObjects(
33
+ allRecipients,
34
+ hasGroups ? 'group' : 'contact'
35
+ )}`;
36
+ }
37
+
38
+ return html`
39
+ <div>
40
+ <div
41
+ style="padding: 3px 10px; background: #f5f5f5; padding-bottom: 0px; font-size: 11px; border-radius: var(--curvature); margin-bottom: 10px;"
42
+ >
43
+ ${recipientsDisplay}
44
+ </div>
45
+ <div style="padding: 0px 10px;">
46
+ ${renderNamedObjects([action.flow], 'flow')}
47
+ </div>
48
+ </div>
49
+ `;
50
+ },
51
+
52
+ toFormData: (action: StartSession) => {
53
+ const extendedAction = action as StartSession & {
54
+ contact_query?: string;
55
+ exclusions?: { in_a_flow?: boolean };
56
+ };
57
+
58
+ // Determine start type based on action properties
59
+ let startTypeValue = 'manual';
60
+ if (action.create_contact) {
61
+ startTypeValue = 'create';
62
+ } else if (extendedAction.contact_query) {
63
+ startTypeValue = 'query';
64
+ }
65
+
66
+ // Map value to full option object for proper display
67
+ const startTypeOptions = [
68
+ { value: 'manual', name: 'Select recipients manually' },
69
+ { value: 'query', name: 'Select a contact with a query' },
70
+ { value: 'create', name: 'Create a new contact' }
71
+ ];
72
+ const startType =
73
+ startTypeOptions.find((opt) => opt.value === startTypeValue) ||
74
+ startTypeOptions[0];
75
+
76
+ return {
77
+ flow: action.flow ? [action.flow] : null,
78
+ recipients: [...(action.contacts || []), ...(action.groups || [])],
79
+ startType: [startType],
80
+ contactQuery: extendedAction.contact_query || '',
81
+ skipContactsInFlow: extendedAction.exclusions?.in_a_flow || false,
82
+ uuid: action.uuid
83
+ };
84
+ },
85
+
86
+ form: {
87
+ flow: {
88
+ type: 'select',
89
+ label: 'Flow',
90
+ helpText: 'Select the flow to start',
91
+ required: true,
92
+ searchable: true,
93
+ endpoint: '/api/v2/flows.json',
94
+ valueKey: 'uuid',
95
+ nameKey: 'name',
96
+ placeholder: 'Select a flow...'
97
+ },
98
+ startType: {
99
+ type: 'select',
100
+ label: 'Start Type',
101
+ helpText: 'How should contacts be selected?',
102
+ required: true,
103
+ options: [
104
+ { value: 'manual', name: 'Select recipients manually' },
105
+ { value: 'query', name: 'Select a contact with a query' },
106
+ { value: 'create', name: 'Create a new contact' }
107
+ ]
108
+ },
109
+ recipients: {
110
+ type: 'select',
111
+ label: 'Recipients',
112
+ helpText: 'Select who should be started in the flow',
113
+ options: [],
114
+ multi: true,
115
+ searchable: true,
116
+ endpoint: '/api/v2/contacts.json',
117
+ valueKey: 'uuid',
118
+ nameKey: 'name',
119
+ placeholder: 'Search for contacts or groups...',
120
+ conditions: {
121
+ visible: (formData: FormData) => {
122
+ const startType = formData.startType?.[0]?.value;
123
+ return startType === 'manual';
124
+ }
125
+ }
126
+ },
127
+ contactQuery: {
128
+ type: 'text',
129
+ evaluated: true,
130
+ label: 'Contact Query',
131
+ helpText: 'Only one matching contact will be started',
132
+ placeholder: 'household_id = @fields.household_id',
133
+ conditions: {
134
+ visible: (formData: FormData) => {
135
+ const startType = formData.startType?.[0]?.value;
136
+ return startType === 'query';
137
+ }
138
+ }
139
+ },
140
+ skipContactsInFlow: {
141
+ type: 'checkbox',
142
+ label: 'Skip contacts currently in a flow',
143
+ helpText: 'Avoid interrupting a contact who is already in a flow'
144
+ }
145
+ },
146
+
147
+ layout: [
148
+ 'flow',
149
+ 'startType',
150
+ 'recipients',
151
+ 'contactQuery',
152
+ 'skipContactsInFlow'
153
+ ],
154
+
155
+ validate: (formData: FormData): ValidationResult => {
156
+ const errors: { [key: string]: string } = {};
157
+
158
+ const startType = formData.startType?.[0]?.value;
159
+
160
+ // Check if manual selection has recipients
161
+ if (
162
+ startType === 'manual' &&
163
+ (!formData.recipients || formData.recipients.length === 0)
164
+ ) {
165
+ errors.recipients = 'At least one contact or group must be selected';
166
+ }
167
+
168
+ // Check if query has a query string
169
+ if (
170
+ startType === 'query' &&
171
+ (!formData.contactQuery || !formData.contactQuery.trim())
172
+ ) {
173
+ errors.contactQuery = 'Contact query is required';
174
+ }
175
+
176
+ return {
177
+ valid: Object.keys(errors).length === 0,
178
+ errors
179
+ };
180
+ },
181
+
182
+ fromFormData: (formData: FormData): StartSession => {
183
+ const action: StartSession & {
184
+ contact_query?: string;
185
+ exclusions?: { in_a_flow: boolean };
186
+ } = {
187
+ uuid: formData.uuid,
188
+ type: 'start_session',
189
+ flow: formData.flow[0],
190
+ groups: [],
191
+ contacts: []
192
+ };
193
+
194
+ // Get the start type value from array
195
+ const startTypeValue = formData.startType[0].value;
196
+
197
+ // Handle different start types
198
+ if (startTypeValue === 'create') {
199
+ action.create_contact = true;
200
+ } else if (startTypeValue === 'query') {
201
+ action.contact_query = formData.contactQuery || '';
202
+ } else {
203
+ // Manual selection - separate contacts and groups
204
+ const recipients = formData.recipients || [];
205
+ action.contacts = recipients.filter((r: any) => !r.group);
206
+ action.groups = recipients.filter((r: any) => r.group);
207
+ }
208
+
209
+ // Add exclusions if set
210
+ if (formData.skipContactsInFlow) {
211
+ action.exclusions = { in_a_flow: true };
212
+ }
213
+
214
+ return action;
11
215
  }
12
216
  };
@@ -14,9 +14,6 @@ import { set_run_result } from './actions/set_run_result';
14
14
  import { send_msg } from './actions/send_msg';
15
15
  import { send_email } from './actions/send_email';
16
16
  import { start_session } from './actions/start_session';
17
- import { call_classifier } from './actions/call_classifier';
18
- import { call_resthook } from './actions/call_resthook';
19
- import { transfer_airtime } from './actions/transfer_airtime';
20
17
  import { set_contact_name } from './actions/set_contact_name';
21
18
  import { add_contact_groups } from './actions/add_contact_groups';
22
19
  import { remove_contact_groups } from './actions/remove_contact_groups';
@@ -36,46 +33,39 @@ import { split_by_scheme } from './nodes/split_by_scheme';
36
33
  import { split_by_subflow } from './nodes/split_by_subflow';
37
34
  import { split_by_ticket } from './nodes/split_by_ticket';
38
35
  import { split_by_webhook } from './nodes/split_by_webhook';
36
+ import { split_by_resthook } from './nodes/split_by_resthook';
39
37
  import { split_by_llm } from './nodes/split_by_llm';
40
38
  import { split_by_llm_categorize } from './nodes/split_by_llm_categorize';
41
- import { wait_for_audio } from './nodes/wait_for_audio';
42
39
  import { wait_for_digits } from './nodes/wait_for_digits';
43
- import { wait_for_image } from './nodes/wait_for_image';
44
- import { wait_for_location } from './nodes/wait_for_location';
45
40
  import { wait_for_menu } from './nodes/wait_for_menu';
46
41
  import { wait_for_response } from './nodes/wait_for_response';
47
- import { wait_for_video } from './nodes/wait_for_video';
48
42
 
49
43
  export const ACTION_CONFIG: {
50
44
  [key: string]: ActionConfig;
51
45
  } = {
52
- add_input_labels,
53
- add_contact_urn,
46
+ say_msg,
47
+ play_audio,
54
48
  set_contact_field,
55
- set_contact_channel,
56
- set_contact_language,
57
- set_contact_status,
58
49
  send_broadcast,
59
50
  set_run_result,
60
51
  send_msg,
61
52
  send_email,
62
53
  start_session,
63
- call_classifier,
64
- call_resthook,
65
- transfer_airtime,
66
54
  set_contact_name,
67
55
  add_contact_groups,
68
56
  remove_contact_groups,
69
- request_optin,
70
- say_msg,
71
- play_audio
57
+ set_contact_channel,
58
+ set_contact_language,
59
+ set_contact_status,
60
+ add_contact_urn,
61
+ add_input_labels,
62
+ request_optin
72
63
  };
73
64
 
74
65
  export const NODE_CONFIG: {
75
66
  [key: string]: NodeConfig;
76
67
  } = {
77
68
  execute_actions,
78
- split_by_airtime,
79
69
  split_by_contact_field,
80
70
  split_by_expression,
81
71
  split_by_groups,
@@ -87,11 +77,9 @@ export const NODE_CONFIG: {
87
77
  split_by_subflow,
88
78
  split_by_ticket,
89
79
  split_by_webhook,
90
- wait_for_audio,
80
+ split_by_resthook,
91
81
  wait_for_digits,
92
- wait_for_image,
93
- wait_for_location,
94
82
  wait_for_menu,
95
83
  wait_for_response,
96
- wait_for_video
84
+ split_by_airtime
97
85
  };
@@ -0,0 +1,51 @@
1
+ // Currency definitions for airtime transfers
2
+ // Based on the original React implementation
3
+
4
+ export interface Currency {
5
+ code: string;
6
+ name: string;
7
+ }
8
+
9
+ export const CURRENCIES: Record<string, Currency> = {
10
+ ARS: { code: 'ARS', name: 'Argentine Peso' },
11
+ AUD: { code: 'AUD', name: 'Australian Dollar' },
12
+ BIF: { code: 'BIF', name: 'Burundi Franc' },
13
+ BRL: { code: 'BRL', name: 'Brazilian Real' },
14
+ CAD: { code: 'CAD', name: 'Canadian Dollar' },
15
+ CDF: { code: 'CDF', name: 'Congolese Franc' },
16
+ CLP: { code: 'CLP', name: 'Chilean Peso' },
17
+ COP: { code: 'COP', name: 'Colombian Peso' },
18
+ DJF: { code: 'DJF', name: 'Djibouti Franc' },
19
+ DOP: { code: 'DOP', name: 'Dominican Peso' },
20
+ DZD: { code: 'DZD', name: 'Algerian Dinar' },
21
+ EUR: { code: 'EUR', name: 'Euro' },
22
+ GBP: { code: 'GBP', name: 'Pound Sterling' },
23
+ GHS: { code: 'GHS', name: 'Ghana Cedi' },
24
+ GNF: { code: 'GNF', name: 'Guinean Franc' },
25
+ KES: { code: 'KES', name: 'Kenyan Shilling' },
26
+ LBP: { code: 'LBP', name: 'Lebanese Pound' },
27
+ LKR: { code: 'LKR', name: 'Sri Lanka Rupee' },
28
+ LRD: { code: 'LRD', name: 'Liberian Dollar' },
29
+ MWK: { code: 'MWK', name: 'Malawian Kwacha' },
30
+ MXN: { code: 'MXN', name: 'Mexican Peso' },
31
+ MZN: { code: 'MZN', name: 'Mozambican Metical' },
32
+ NAD: { code: 'NAD', name: 'Namibian Dollar' },
33
+ NGN: { code: 'NGN', name: 'Nigerian Naira' },
34
+ PEN: { code: 'PEN', name: 'Peruvian Sol' },
35
+ PHP: { code: 'PHP', name: 'Philippine Peso' },
36
+ RWF: { code: 'RWF', name: 'Rwandan Franc' },
37
+ SZL: { code: 'SZL', name: 'Swazi Lilangeni' },
38
+ TZS: { code: 'TZS', name: 'Tanzanian Shilling' },
39
+ UGX: { code: 'UGX', name: 'Ugandan Shilling' },
40
+ USD: { code: 'USD', name: 'US Dollar' },
41
+ XAF: { code: 'XAF', name: 'Central African CFA Franc' },
42
+ XOF: { code: 'XOF', name: 'West African CFA Franc' },
43
+ ZAR: { code: 'ZAR', name: 'South African Rand' },
44
+ ZMW: { code: 'ZMW', name: 'Zambian Kwacha' }
45
+ };
46
+
47
+ // Convert currencies to select options format
48
+ export const CURRENCY_OPTIONS = Object.values(CURRENCIES).map((currency) => ({
49
+ value: currency.code,
50
+ name: `${currency.name} (${currency.code})`
51
+ }));