@capillarytech/creatives-library 8.0.345-alpha.14 → 8.0.345-alpha.15

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 (129) hide show
  1. package/constants/unified.js +29 -0
  2. package/package.json +1 -1
  3. package/services/tests/api.test.js +13 -0
  4. package/utils/commonUtils.js +19 -1
  5. package/utils/rcsPayloadUtils.js +92 -0
  6. package/utils/templateVarUtils.js +201 -0
  7. package/utils/tests/templateVarUtils.test.js +204 -0
  8. package/v2Components/CapActionButton/constants.js +7 -0
  9. package/v2Components/CapActionButton/index.js +167 -109
  10. package/v2Components/CapActionButton/index.scss +157 -6
  11. package/v2Components/CapActionButton/messages.js +19 -3
  12. package/v2Components/CapActionButton/tests/index.test.js +41 -17
  13. package/v2Components/CapTagList/index.js +10 -0
  14. package/v2Components/CommonTestAndPreview/CustomValuesEditor.js +70 -49
  15. package/v2Components/CommonTestAndPreview/DeliverySettings/DeliverySettings.scss +8 -2
  16. package/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.js +207 -21
  17. package/v2Components/CommonTestAndPreview/DeliverySettings/constants.js +16 -0
  18. package/v2Components/CommonTestAndPreview/DeliverySettings/index.js +85 -10
  19. package/v2Components/CommonTestAndPreview/DeliverySettings/messages.js +30 -0
  20. package/v2Components/CommonTestAndPreview/DeliverySettings/utils/parseSenderDetailsResponse.js +79 -11
  21. package/v2Components/CommonTestAndPreview/SendTestMessage.js +10 -5
  22. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js +160 -15
  23. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js.rej +18 -0
  24. package/v2Components/CommonTestAndPreview/UnifiedPreview/_unifiedPreview.scss +341 -76
  25. package/v2Components/CommonTestAndPreview/UnifiedPreview/index.js +133 -4
  26. package/v2Components/CommonTestAndPreview/_commonTestAndPreview.scss +11 -0
  27. package/v2Components/CommonTestAndPreview/constants.js +38 -2
  28. package/v2Components/CommonTestAndPreview/index.js +676 -186
  29. package/v2Components/CommonTestAndPreview/messages.js +49 -3
  30. package/v2Components/CommonTestAndPreview/previewApiUtils.js +59 -0
  31. package/v2Components/CommonTestAndPreview/sagas.js +15 -6
  32. package/v2Components/CommonTestAndPreview/tests/CustomValuesEditor.test.js +308 -284
  33. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/ModifyDeliverySettings.test.js +231 -65
  34. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/index.test.js +118 -5
  35. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/utils/parseSenderDetailsResponse.test.js +341 -0
  36. package/v2Components/CommonTestAndPreview/tests/PreviewSection.test.js +8 -1
  37. package/v2Components/CommonTestAndPreview/tests/SendTestMessage.test.js +34 -13
  38. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/RcsPreviewContent.test.js +281 -283
  39. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/index.test.js +199 -1
  40. package/v2Components/CommonTestAndPreview/tests/index.test.js +132 -4
  41. package/v2Components/CommonTestAndPreview/tests/previewApiUtils.test.js +67 -0
  42. package/v2Components/CommonTestAndPreview/tests/sagas.test.js +2 -2
  43. package/v2Components/FormBuilder/index.js +8 -10
  44. package/v2Components/SmsFallback/SmsFallbackLocalSelector.js +87 -0
  45. package/v2Components/SmsFallback/constants.js +73 -0
  46. package/v2Components/SmsFallback/index.js +955 -0
  47. package/v2Components/SmsFallback/index.scss +265 -0
  48. package/v2Components/SmsFallback/messages.js +78 -0
  49. package/v2Components/SmsFallback/smsFallbackUtils.js +118 -0
  50. package/v2Components/SmsFallback/tests/SmsFallbackLocalSelector.test.js +50 -0
  51. package/v2Components/SmsFallback/tests/rcsSmsFallback.acceptance.test.js +147 -0
  52. package/v2Components/SmsFallback/tests/smsFallbackHandlers.test.js +304 -0
  53. package/v2Components/SmsFallback/tests/smsFallbackUi.test.js +197 -0
  54. package/v2Components/SmsFallback/tests/smsFallbackUtils.test.js +277 -0
  55. package/v2Components/SmsFallback/tests/useLocalTemplateList.test.js +422 -0
  56. package/v2Components/SmsFallback/useLocalTemplateList.js +92 -0
  57. package/v2Components/TemplatePreview/_templatePreview.scss +33 -23
  58. package/v2Components/TemplatePreview/constants.js +2 -0
  59. package/v2Components/TemplatePreview/index.js +143 -28
  60. package/v2Components/TemplatePreview/tests/index.test.js +142 -0
  61. package/v2Components/TestAndPreviewSlidebox/index.js +13 -1
  62. package/v2Components/TestAndPreviewSlidebox/sagas.js +11 -4
  63. package/v2Components/TestAndPreviewSlidebox/tests/saga.test.js +3 -1
  64. package/v2Components/VarSegmentMessageEditor/constants.js +2 -0
  65. package/v2Components/VarSegmentMessageEditor/index.js +125 -0
  66. package/v2Components/VarSegmentMessageEditor/index.scss +46 -0
  67. package/v2Containers/CreativesContainer/CreativesSlideBoxWrapper.js +43 -0
  68. package/v2Containers/CreativesContainer/SlideBoxContent.js +36 -4
  69. package/v2Containers/CreativesContainer/SlideBoxFooter.js +10 -1
  70. package/v2Containers/CreativesContainer/SlideBoxHeader.js +29 -4
  71. package/v2Containers/CreativesContainer/constants.js +9 -0
  72. package/v2Containers/CreativesContainer/embeddedSlideboxUtils.js +67 -0
  73. package/v2Containers/CreativesContainer/index.js +300 -103
  74. package/v2Containers/CreativesContainer/index.scss +51 -1
  75. package/v2Containers/CreativesContainer/tests/SlideBoxContent.localTemplates.test.js +90 -0
  76. package/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +78 -34
  77. package/v2Containers/CreativesContainer/tests/SlideBoxHeader.test.js +79 -16
  78. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +8 -0
  79. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxHeader.test.js.snap +357 -98
  80. package/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +20 -15
  81. package/v2Containers/CreativesContainer/tests/embeddedSlideboxUtils.test.js +258 -0
  82. package/v2Containers/CreativesContainer/tests/index.test.js +71 -9
  83. package/v2Containers/CreativesContainer/tests/useLocalTemplatesProp.test.js +125 -0
  84. package/v2Containers/Email/reducer.js +3 -11
  85. package/v2Containers/Email/sagas.js +5 -9
  86. package/v2Containers/Email/tests/__snapshots__/reducer.test.js.snap +0 -4
  87. package/v2Containers/Email/tests/sagas.test.js +3 -21
  88. package/v2Containers/Rcs/constants.js +119 -8
  89. package/v2Containers/Rcs/index.js +2379 -807
  90. package/v2Containers/Rcs/index.js.rej +1336 -0
  91. package/v2Containers/Rcs/index.scss +276 -6
  92. package/v2Containers/Rcs/index.scss.rej +74 -0
  93. package/v2Containers/Rcs/messages.js +38 -3
  94. package/v2Containers/Rcs/rcsLibraryHydrationUtils.js +225 -0
  95. package/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +98018 -70073
  96. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap +0 -5
  97. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap.rej +128 -0
  98. package/v2Containers/Rcs/tests/index.test.js +152 -121
  99. package/v2Containers/Rcs/tests/mockData.js +38 -0
  100. package/v2Containers/Rcs/tests/rcsLibraryHydrationUtils.test.js +318 -0
  101. package/v2Containers/Rcs/tests/utils.test.js +646 -30
  102. package/v2Containers/Rcs/utils.js +478 -11
  103. package/v2Containers/Sms/Create/index.js +100 -40
  104. package/v2Containers/Sms/smsFormDataHelpers.js +67 -0
  105. package/v2Containers/Sms/tests/smsFormDataHelpers.test.js +253 -0
  106. package/v2Containers/SmsTrai/Create/index.js +9 -4
  107. package/v2Containers/SmsTrai/Edit/constants.js +2 -0
  108. package/v2Containers/SmsTrai/Edit/index.js +636 -130
  109. package/v2Containers/SmsTrai/Edit/index.scss +121 -0
  110. package/v2Containers/SmsTrai/Edit/messages.js +14 -4
  111. package/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +4328 -2375
  112. package/v2Containers/SmsWrapper/index.js +37 -8
  113. package/v2Containers/TagList/index.js +6 -0
  114. package/v2Containers/Templates/TemplatesActionBar.js +101 -0
  115. package/v2Containers/Templates/_templates.scss +163 -2
  116. package/v2Containers/Templates/actions.js +11 -0
  117. package/v2Containers/Templates/constants.js +2 -0
  118. package/v2Containers/Templates/index.js +119 -54
  119. package/v2Containers/Templates/sagas.js +57 -12
  120. package/v2Containers/Templates/tests/TemplatesActionBar.test.js +120 -0
  121. package/v2Containers/Templates/tests/__snapshots__/index.test.js.snap +1043 -1079
  122. package/v2Containers/Templates/tests/sagas.test.js +193 -123
  123. package/v2Containers/Templates/tests/smsTemplatesListApi.test.js +180 -0
  124. package/v2Containers/Templates/utils/smsTemplatesListApi.js +79 -0
  125. package/v2Containers/TemplatesV2/TemplatesV2.style.js +72 -1
  126. package/v2Containers/TemplatesV2/index.js +86 -23
  127. package/v2Containers/TemplatesV2/tests/TemplatesV2.localTemplates.test.js +131 -0
  128. package/v2Containers/Whatsapp/index.js +3 -20
  129. package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +578 -34
@@ -1,10 +1,11 @@
1
1
  /* eslint-disable no-restricted-syntax */
2
2
  /* eslint-disable no-undef */
3
- import React, { useState, useEffect } from 'react';
3
+ import React, { useState, useEffect, useMemo, useRef } from 'react';
4
+ import PropTypes from 'prop-types';
4
5
  import { createStructuredSelector } from 'reselect';
5
6
  import { bindActionCreators } from 'redux';
6
7
  import { FormattedMessage } from 'react-intl';
7
- import { get, cloneDeep, isEmpty, isObject } from 'lodash';
8
+ import { get, cloneDeep, isEmpty } from 'lodash';
8
9
  import styled from 'styled-components';
9
10
  import CapRow from '@capillarytech/cap-ui-library/CapRow';
10
11
  import CapColumn from '@capillarytech/cap-ui-library/CapColumn';
@@ -27,13 +28,13 @@ import {
27
28
  CAP_SPACE_32,
28
29
  CAP_SPACE_04,
29
30
  CAP_WHITE,
30
- CAP_G10,
31
31
  } from '@capillarytech/cap-ui-library/styled/variables';
32
32
  import { makeSelectTemplateDetailsResponse } from '../../Sms/Edit/selectors';
33
33
  import { makeSelectMetaEntities, selectLiquidStateDetails, setInjectedTags } from '../../Cap/selectors';
34
34
  import * as smsEditActions from '../../Sms/Edit/actions';
35
35
  import * as globalActions from '../../Cap/actions';
36
36
  import messages from './messages';
37
+ import globalMessages from '../../Cap/messages';
37
38
  import TagList from '../../TagList';
38
39
  import formBuilderMessages from '../../../v2Components/FormBuilder/messages';
39
40
  import UnifiedPreview from '../../../v2Components/CommonTestAndPreview/UnifiedPreview';
@@ -41,6 +42,8 @@ import TestAndPreviewSlidebox from '../../../v2Components/TestAndPreviewSlidebox
41
42
  import withCreatives from '../../../hoc/withCreatives';
42
43
  import {
43
44
  CHARLIMIT,
45
+ SMS,
46
+ SMS_TRAI_CONTENT_MAX_LENGTH,
44
47
  SMS_TRAI_VAR,
45
48
  TAG,
46
49
  EMBEDDED,
@@ -49,24 +52,39 @@ import {
49
52
  ALL,
50
53
  LIBRARY,
51
54
  } from './constants';
52
- import { SMS } from '../../CreativesContainer/constants';
53
55
  import v2EditSmsReducer from '../../Sms/Edit/reducer';
54
56
  import { v2SmsEditSagas } from '../../Sms/Edit/sagas';
55
57
  import ErrorInfoNote from '../../../v2Components/ErrorInfoNote';
58
+ import { isTraiDLTEnable, hasTraiDltFeature } from '../../../utils/common';
56
59
  import { validateLiquidTemplateContent } from '../../../utils/commonUtils';
57
60
  import { validateTags } from '../../../utils/tagValidations';
58
- import globalMessages from '../../Cap/messages';
59
61
  import { ANDROID } from '../../../v2Components/CommonTestAndPreview/constants';
62
+ import {
63
+ getFallbackResolvedContent,
64
+ splitTemplateVarString,
65
+ COMBINED_SMS_TEMPLATE_VAR_REGEX,
66
+ isAnyTemplateVarToken,
67
+ isDltHashVarToken,
68
+ } from '../../../utils/templateVarUtils';
69
+ import VarSegmentMessageEditor from '../../../v2Components/VarSegmentMessageEditor';
70
+ import rcsMessages from '../../Rcs/messages';
71
+
72
+ import './index.scss';
60
73
 
61
- let varMap = {};
62
- let traiData = {};
63
74
  const { TextArea } = CapInput;
64
75
  const { CapLabelInline } = CapLabel;
65
76
 
77
+ /** Redux `metaEntities` may be an Immutable.Map; TagList needs plain `tags.standard`. */
78
+ const getStandardTagsFromMeta = (metaEntities) => {
79
+ if (!metaEntities) return [];
80
+ const plain = typeof metaEntities.toJS === 'function' ? metaEntities.toJS() : metaEntities;
81
+ const standard = get(plain, 'tags.standard');
82
+ return Array.isArray(standard) ? standard : [];
83
+ };
84
+
66
85
  export const SmsTraiEdit = (props) => {
67
86
  const {
68
87
  intl,
69
- handleClose,
70
88
  params,
71
89
  actions,
72
90
  templateDetails,
@@ -82,10 +100,20 @@ export const SmsTraiEdit = (props) => {
82
100
  templateData = {},
83
101
  selectedOfferDetails,
84
102
  eventContextTags,
85
- waitEventContextTags,
86
103
  fetchingLiquidTags,
87
104
  getLiquidTags,
88
105
  showLiquidErrorInFooter,
106
+ smsRegister,
107
+ // RCS -> SMS fallback edit mode
108
+ isRcsSmsFallback = false,
109
+ /** When editing an existing RCS template, lock Unicode (matches product: no mid-edit toggle). */
110
+ isRcsEditFlow = false,
111
+ showPreviewInRcsFallback = false,
112
+ hidePreview = false,
113
+ isOverview = false,
114
+ forceFullTagContext = false,
115
+ /** RCS parent: merge `rcsSmsFallbackVarMapped` into `smsFallbackData` (same idea as `cardVarMapped`). */
116
+ onRcsFallbackEditorStateChange,
89
117
  } = props || {};
90
118
 
91
119
  const { formatMessage } = intl;
@@ -95,14 +123,149 @@ export const SmsTraiEdit = (props) => {
95
123
  const [tags, updateTags] = useState([]);
96
124
  const [textAreaId, updateTextAreaId] = useState();
97
125
  const [isValidationError, updateIsValidationError] = useState(false);
126
+ const [isTagValidationError, updateIsTagValidationError] = useState(false);
98
127
  const [totalMessageLength, setTotalMessageLength] = useState(0);
99
128
  const [isUnicodeAllowed, updateIsUnicodeAllowed] = useState(true);
129
+ const [fallbackText, setFallbackText] = useState('');
130
+ const [fallbackVarMappedData, setFallbackVarMappedData] = useState({});
131
+ const [fallbackFocusedId, setFallbackFocusedId] = useState('');
100
132
  const [showMsgLengthNote, updateshowMsgLengthNote] = useState(false);
101
133
  const [liquidErrorMessages, setLiquidErrorMessages] = useState({});
102
134
  const [isLiquidValidationError, setIsLiquidValidationError] = useState(false);
103
135
  /** After user closes the validation panel, keep it hidden until Save is clicked again (even if ErrorInfoNote remounts). */
104
136
  const [liquidErrorPanelDismissed, setLiquidErrorPanelDismissed] = useState(false);
105
137
  const [showTestAndPreviewSlidebox, setShowTestAndPreviewSlidebox] = useState(false);
138
+
139
+ /** Per-instance only — was module-level (leaked across mounts / multiple editors). */
140
+ const varMapRef = useRef({});
141
+ const traiDataRef = useRef({});
142
+ const tagValidationResponseRef = useRef({});
143
+ /** De-dupe repeated /meta/TAG fetches from popover/context callbacks. */
144
+ const lastTagSchemaQueryKeyRef = useRef(null);
145
+ /** Latest template props for RCS fallback init — avoids effect deps on unstable `templateData` references. */
146
+ const rcsFallbackTemplateSourceRef = useRef({ isFullMode, templateDetails, templateData });
147
+ rcsFallbackTemplateSourceRef.current = { isFullMode, templateDetails, templateData };
148
+
149
+ const fetchTagSchemaIfNewQuery = (query) => {
150
+ const key = JSON.stringify(query);
151
+ if (lastTagSchemaQueryKeyRef.current === key) return;
152
+ lastTagSchemaQueryKeyRef.current = key;
153
+ globalActions.fetchSchemaForEntity(query);
154
+ };
155
+
156
+ const traiDltEnabled = useMemo(
157
+ () => isTraiDLTEnable(isFullMode, smsRegister),
158
+ [isFullMode, smsRegister],
159
+ );
160
+ /**
161
+ * VarSegment for RCS SMS fallback when:
162
+ * - `isTraiDLTEnable` passes (library + `smsRegister === 'DLT'`, or full-mode library), or
163
+ * - org has TRAI DLT (`hasTraiDltFeature`) — needed when slidebox used to force `isFullMode={false}`
164
+ * or `smsRegister` is missing so `isTraiDLTEnable` alone is false, or
165
+ * - RCS template inline edit (`isRcsEditFlow`).
166
+ */
167
+ const useRcsFallbackVarSegment = useMemo(() => {
168
+ if (!isRcsSmsFallback) return false;
169
+ return traiDltEnabled || hasTraiDltFeature() || isRcsEditFlow;
170
+ }, [isRcsSmsFallback, traiDltEnabled, isRcsEditFlow]);
171
+
172
+ /**
173
+ * RCS SMS fallback: always show character count vs TRAI max (`SMS_TRAI_CONTENT_MAX_LENGTH`).
174
+ * Do not use `totalCharacters` ({smsCount} SMS via length/160) here: resolved template + variable
175
+ * values can exceed one GSM segment while DLT still shows a single registered template — length/160
176
+ * is only a rough segment hint and reads as “wrong SMS count” in campaigns.
177
+ */
178
+ const renderDescriptionCharacterCount = (className = "rcs-character-count") => (
179
+ <CapLabel type="label1" className={className}>
180
+ {formatMessage(messages.charactersCountLabel, {
181
+ current: totalMessageLength,
182
+ max: SMS_TRAI_CONTENT_MAX_LENGTH,
183
+ })}
184
+ </CapLabel>
185
+ );
186
+
187
+ const dltConsecutiveRunLength = (segments, index) => {
188
+ if (!Array.isArray(segments) || typeof segments[index] !== 'string' || !isDltHashVarToken(segments[index])) {
189
+ return 0;
190
+ }
191
+ let lo = index;
192
+ while (lo > 0 && isDltHashVarToken(segments[lo - 1])) lo -= 1;
193
+ let hi = index;
194
+ while (hi < segments.length - 1 && isDltHashVarToken(segments[hi + 1])) hi += 1;
195
+ return hi - lo + 1;
196
+ };
197
+
198
+ const renderRcsFallbackMessage = (str = '') => {
199
+ if (!useRcsFallbackVarSegment) {
200
+ return (
201
+ <CapRow className="rcs-create-template-message-input">
202
+ <div className="rcs_text_area_wrapper">
203
+ <TextArea
204
+ id="rcs_fallback_message_textarea"
205
+ autosize={{ minRows: 4, maxRows: 12 }}
206
+ value={fallbackText}
207
+ onChange={(e) => setFallbackText(e.target.value)}
208
+ placeholder={formatMessage(rcsMessages.fallbackMsgPlaceholder)}
209
+ data-testid="rcs_fallback_plain_text_area"
210
+ />
211
+ {renderDescriptionCharacterCount()}
212
+ </div>
213
+ </CapRow>
214
+ );
215
+ }
216
+ return (
217
+ <CapRow className="rcs-create-template-message-input">
218
+ <div className="rcs_text_area_wrapper">
219
+ <VarSegmentMessageEditor
220
+ templateString={str}
221
+ valueMap={fallbackVarMappedData || {}}
222
+ onChange={(varSegmentFieldId, nextSlotValue) => {
223
+ setFallbackVarMappedData((previousSlotMap) => ({
224
+ ...(previousSlotMap || {}),
225
+ [varSegmentFieldId]: nextSlotValue,
226
+ }));
227
+ }}
228
+ onFocus={setFallbackFocusedId}
229
+ varRegex={COMBINED_SMS_TEMPLATE_VAR_REGEX}
230
+ placeholderPrefix=""
231
+ getPlaceholder={() => formatMessage(rcsMessages.fallbackMsgPlaceholder)}
232
+ renderVarFooter={(dltSegmentToken, dltSegmentIndex) => {
233
+ if (!isDltHashVarToken(dltSegmentToken)) return null;
234
+ const segments = splitTemplateVarString(str, COMBINED_SMS_TEMPLATE_VAR_REGEX);
235
+ const varCounts = dltConsecutiveRunLength(segments, dltSegmentIndex);
236
+ if (varCounts < 1) return null;
237
+ /* Inline layout/color so hint is visible even if CapSpin / load order blocks SCSS */
238
+ const dltVarSlotHintStyle = {
239
+ display: 'block',
240
+ marginTop: CAP_SPACE_04,
241
+ marginLeft: 'auto',
242
+ marginRight: 0,
243
+ width: 'fit-content',
244
+ maxWidth: '100%',
245
+ textAlign: 'right',
246
+ color: 'rgba(0, 0, 0, 0.45)',
247
+ fontSize: 12,
248
+ fontWeight: 400,
249
+ lineHeight: '16px',
250
+ };
251
+ return (
252
+ <span className="sms-trai-rcs-fallback-var-hint" style={dltVarSlotHintStyle}>
253
+ {formatMessage(messages.textAreaCounts, {
254
+ varCounts,
255
+ var: SMS_TRAI_VAR,
256
+ charCounts: varCounts * CHARLIMIT,
257
+ })}
258
+ </span>
259
+ );
260
+ }}
261
+ />
262
+ {(!isFullMode)
263
+ ? renderDescriptionCharacterCount('rcs-character-count rcs-character-count--compact')
264
+ : (isFullMode && renderDescriptionCharacterCount())}
265
+ </div>
266
+ </CapRow>
267
+ );
268
+ };
106
269
  const SMSTraiFooter = styled.div`
107
270
  background-color: ${CAP_WHITE};
108
271
  padding: ${CAP_SPACE_32} ${CAP_SPACE_24};
@@ -118,7 +281,6 @@ export const SmsTraiEdit = (props) => {
118
281
  .ant-btn {
119
282
  margin-right: ${CAP_SPACE_16};
120
283
  }
121
- }
122
284
  `;
123
285
  const TraiEditTemplateDetails = styled.div`
124
286
  margin-bottom: ${CAP_SPACE_16};
@@ -127,41 +289,127 @@ export const SmsTraiEdit = (props) => {
127
289
  }
128
290
  `;
129
291
 
292
+ /**
293
+ * RCS embedded SMS fallback receives a new `templateData` object from `mapFallbackValueToEditTemplateData`
294
+ * on every parent render. Depending on `[templateDetails || templateData]` re-ran init and wiped
295
+ * VarSegmentMessageEditor state whenever Test & Preview (or any) re-rendered Rcs. Key only changes
296
+ * when template id / sms-editor / header actually change.
297
+ *
298
+ * Do NOT include `unicode-validity` in the key: parent merges `onRcsFallbackEditorStateChange`
299
+ * unicode patches into `templateData`, which would change the key every toggle and re-run init —
300
+ * resetting local checkbox state (visible as the unicode control flipping).
301
+ *
302
+ * Do NOT include `rcs-sms-fallback-var-mapped` in the key: the parent merges slot edits into
303
+ * `templateData` on every change, so the key would change each keystroke and this effect would
304
+ * re-run and reset slots — visible as toggling between old and new values.
305
+ */
306
+ const rcsFallbackTemplateInitKey = useMemo(() => {
307
+ if (!isRcsSmsFallback) return null;
308
+ // RCS fallback slidebox always passes `templateData` from `mapFallbackValueToEditTemplateData`.
309
+ // `templateDetails` stays empty there (no route id fetch) — do not branch on `isFullMode` or init
310
+ // key is null, the effect never runs, and `loading` stays true forever.
311
+ const activeSmsTemplateSource = templateData;
312
+ if (!activeSmsTemplateSource || isEmpty(activeSmsTemplateSource)) return null;
313
+ const templateBase = get(activeSmsTemplateSource, 'versions.base', {}) || {};
314
+ const smsEditorTemplateString = templateBase?.['sms-editor'] ?? '';
315
+ const registeredSenderHeaderList = templateBase?.header;
316
+ const headerListFingerprint = Array.isArray(registeredSenderHeaderList)
317
+ ? registeredSenderHeaderList.join('\u001f')
318
+ : '';
319
+ const templateRecordId = activeSmsTemplateSource?._id ?? '';
320
+ return [
321
+ templateRecordId,
322
+ smsEditorTemplateString,
323
+ headerListFingerprint,
324
+ ].join('\u0000');
325
+ }, [isRcsSmsFallback, templateData]);
326
+
130
327
  useEffect(() => {
131
- //fetching tags
132
328
  const { type, module } = location.query || {};
133
329
  const isEmbedded = type === EMBEDDED;
134
330
  const query = {
135
331
  layout: SMS,
136
332
  type: TAG,
137
333
  context: isEmbedded ? module : DEFAULT,
138
- embedded: isEmbedded ? type : FULL,
334
+ embedded: forceFullTagContext ? FULL : (isEmbedded ? type : FULL),
139
335
  };
140
336
  if (getDefaultTags) {
141
337
  query.context = getDefaultTags;
142
338
  }
143
- globalActions.fetchSchemaForEntity(query);
144
- //fetching template data in fullmode
339
+ fetchTagSchemaIfNewQuery(query);
145
340
  const { id } = params || {};
146
341
  if (id) {
147
342
  actions.getTemplateDetails(id);
148
343
  }
149
- //cleanup code
150
344
  return () => {
151
345
  actions.resetEditTemplate();
152
- varMap = {};
346
+ varMapRef.current = {};
153
347
  };
154
348
  }, []);
155
349
 
156
- //computing placeholder array for mapping values and rendering dynamic form
157
350
  useEffect(() => {
158
- traiData = isFullMode ? templateDetails : templateData;
159
- if (traiData && !isEmpty(traiData)) {
160
- let msg = get(traiData, `versions.base.sms-editor`, '');
351
+ if (!isRcsSmsFallback || rcsFallbackTemplateInitKey == null) return;
352
+ const {
353
+ isFullMode: isFullModeFromRef,
354
+ templateDetails: templateDetailsFromRef,
355
+ templateData: templateDataFromRef,
356
+ } = rcsFallbackTemplateSourceRef.current;
357
+ const activeTemplateSourceForInit = isRcsSmsFallback
358
+ ? templateDataFromRef
359
+ : (isFullModeFromRef ? templateDetailsFromRef : templateDataFromRef);
360
+ if (!activeTemplateSourceForInit || isEmpty(activeTemplateSourceForInit)) return;
361
+ traiDataRef.current = activeTemplateSourceForInit;
362
+ const templateBase = get(activeTemplateSourceForInit, 'versions.base', {});
363
+ const unicodeValidity = get(templateBase, 'unicode-validity', true);
364
+ const templateMsg = get(activeTemplateSourceForInit, 'versions.base.sms-editor', '') || '';
365
+ if (!useRcsFallbackVarSegment) {
366
+ setFallbackText(templateMsg);
367
+ setFallbackVarMappedData({});
368
+ setUpdatedSmsEditor(String(templateMsg).split(''));
369
+ setTotalMessageLength(String(templateMsg).length);
370
+ updateIsUnicodeAllowed(unicodeValidity);
371
+ updateLoading(false);
372
+ return;
373
+ }
374
+ const savedVarMap = get(templateBase, 'rcs-sms-fallback-var-mapped', {}) || {};
375
+ const initialVarMap = {};
376
+ const templateSegments = splitTemplateVarString(templateMsg, COMBINED_SMS_TEMPLATE_VAR_REGEX);
377
+ let varOrdinal = 0;
378
+ templateSegments.forEach((segmentToken, segmentIndexInTemplate) => {
379
+ const isVar =
380
+ typeof segmentToken === 'string' && isAnyTemplateVarToken(segmentToken);
381
+ if (!isVar) return;
382
+ varOrdinal += 1;
383
+ const varSegmentSlotId = `${segmentToken}_${segmentIndexInTemplate}`;
384
+ const persistedSlotValue =
385
+ savedVarMap?.[varSegmentSlotId]
386
+ ?? savedVarMap?.[`${segmentToken}_${varOrdinal}`];
387
+ // Persisted '' means the user cleared the slot — must not fall back to `segmentToken` for mustache
388
+ // (that would resurrect {{…}} in the input and look like the tag "came back").
389
+ if (typeof persistedSlotValue === 'string') {
390
+ initialVarMap[varSegmentSlotId] = persistedSlotValue;
391
+ } else if (isDltHashVarToken(segmentToken)) {
392
+ initialVarMap[varSegmentSlotId] = '';
393
+ } else {
394
+ initialVarMap[varSegmentSlotId] = segmentToken;
395
+ }
396
+ });
397
+ setFallbackText(templateMsg);
398
+ setFallbackVarMappedData(initialVarMap);
399
+ const initialResolvedFallbackDisplay = getFallbackResolvedContent(templateMsg, initialVarMap);
400
+ setUpdatedSmsEditor(initialResolvedFallbackDisplay.split(''));
401
+ setTotalMessageLength(initialResolvedFallbackDisplay.length);
402
+ updateIsUnicodeAllowed(unicodeValidity);
403
+ updateLoading(false);
404
+ // eslint-disable-next-line react-hooks/exhaustive-deps -- init only when semantic key changes; template props are read from ref
405
+ }, [isRcsSmsFallback, rcsFallbackTemplateInitKey, useRcsFallbackVarSegment]);
406
+
407
+ useEffect(() => {
408
+ if (isRcsSmsFallback) return;
409
+ traiDataRef.current = isFullMode ? templateDetails : templateData;
410
+ if (traiDataRef.current && !isEmpty(traiDataRef.current)) {
411
+ let msg = get(traiDataRef.current, `versions.base.sms-editor`, '');
161
412
  const templateMessageArray = [];
162
- //converting sms-editor string to an array split at '{#var#}'
163
- //split and push string before '{#var#}[0 to index]', push '{#var#}',
164
- //split and repeat for remaining string[index+7 to length]
165
413
  while (msg.length !== 0) {
166
414
  const index = msg.search(SMS_TRAI_VAR);
167
415
  if (index !== -1) {
@@ -175,30 +423,59 @@ export const SmsTraiEdit = (props) => {
175
423
  }
176
424
  const filteredTemplateMessageArray = templateMessageArray.filter((i) => i === 0 || i);
177
425
  updateTempMsgArray(filteredTemplateMessageArray);
178
- //stop spinner
179
426
  updateLoading(false);
180
427
  }
181
- }, [templateDetails || templateData]);
428
+ }, [templateDetails || templateData, isRcsSmsFallback, isFullMode]);
182
429
 
183
- //compute/get varMapped and updated-sms-editor
184
430
  useEffect(() => {
431
+ if (!isRcsSmsFallback) return;
432
+ if (!useRcsFallbackVarSegment) {
433
+ const plainFallbackSmsText = fallbackText || '';
434
+ setUpdatedSmsEditor(plainFallbackSmsText.split(''));
435
+ setTotalMessageLength(plainFallbackSmsText.length);
436
+ return;
437
+ }
438
+ const resolvedFallbackDisplay = getFallbackResolvedContent(
439
+ fallbackText || '',
440
+ fallbackVarMappedData || {},
441
+ );
442
+ setUpdatedSmsEditor(resolvedFallbackDisplay.split(''));
443
+ setTotalMessageLength(resolvedFallbackDisplay.length);
444
+ }, [isRcsSmsFallback, useRcsFallbackVarSegment, fallbackText, fallbackVarMappedData]);
445
+
446
+ useEffect(() => {
447
+ if (!isRcsSmsFallback) return;
448
+ if (typeof onRcsFallbackEditorStateChange !== 'function') return;
449
+ onRcsFallbackEditorStateChange({
450
+ rcsSmsFallbackVarMapped: fallbackVarMappedData || {},
451
+ });
452
+ }, [isRcsSmsFallback, fallbackVarMappedData, onRcsFallbackEditorStateChange]);
453
+
454
+ useEffect(() => {
455
+ if (!isRcsSmsFallback) return;
456
+ if (typeof onRcsFallbackEditorStateChange !== 'function') return;
457
+ onRcsFallbackEditorStateChange({
458
+ unicodeValidity: isUnicodeAllowed,
459
+ });
460
+ }, [isRcsSmsFallback, isUnicodeAllowed, onRcsFallbackEditorStateChange]);
461
+
462
+ useEffect(() => {
463
+ if (isRcsSmsFallback) return;
185
464
  if (tempMsgArray.length !== 0) {
186
- const traiBase = traiData?.versions?.base || {};
465
+ const traiBase = traiDataRef.current?.versions?.base || {};
187
466
  const {
188
467
  'var-mapped': varMapped = {},
189
468
  'updated-sms-editor': traiSmsEditor = '',
190
469
  'unicode-validity': unicodeValidity = true,
191
470
  } = traiBase;
192
- //if varMap and updated-sms-editor is already present on non first edits,use those values
193
471
  if (!isEmpty(varMapped)) {
194
- varMap = cloneDeep(varMapped);
472
+ varMapRef.current = cloneDeep(varMapped);
195
473
  if (isFullMode) {
196
474
  setUpdatedSmsEditor(traiSmsEditor);
197
475
  } else {
198
476
  computeUpdatedSmsEditor();
199
477
  }
200
478
  } else {
201
- //computing and setting varMap for first edit
202
479
  let varCount = 1;
203
480
  let horizontalSpaceCount = 0;
204
481
  for (let i = 0; i < tempMsgArray.length; i += 1) {
@@ -209,7 +486,7 @@ export const SmsTraiEdit = (props) => {
209
486
  horizontalSpaceCount += 1;
210
487
  }
211
488
  if (tempMsgArray[i] !== nextElem && nextElem?.replace(/[^\S\r\n]/gm, '') !== '') {
212
- varMap[`${tempMsgArray[i]}_${(i - varCount - horizontalSpaceCount) + 1}`] = {
489
+ varMapRef.current[`${tempMsgArray[i]}_${(i - varCount - horizontalSpaceCount) + 1}`] = {
213
490
  data: '',
214
491
  count: varCount,
215
492
  };
@@ -223,7 +500,6 @@ export const SmsTraiEdit = (props) => {
223
500
  setUpdatedSmsEditor(tempMsgArray);
224
501
  }
225
502
  updateIsUnicodeAllowed(unicodeValidity);
226
- // calcaulate message length here
227
503
  calculateTotalMessageLength();
228
504
  }
229
505
  }, [tempMsgArray]);
@@ -237,17 +513,84 @@ export const SmsTraiEdit = (props) => {
237
513
  }
238
514
  }, []);
239
515
 
516
+ useEffect(() => {
517
+ const runValidateTags = (content) =>
518
+ validateTags({
519
+ content,
520
+ tagsParam: tags,
521
+ location,
522
+ tagModule: getDefaultTags,
523
+ isFullMode,
524
+ }) || {};
525
+
526
+ if (isRcsSmsFallback) {
527
+ if (isFullMode) {
528
+ tagValidationResponseRef.current = {};
529
+ updateIsTagValidationError(false);
530
+ return;
531
+ }
532
+ // TRAI/DLT VarSegment: `validateIfTagClosed` only understands paired `{{…}}` and breaks on
533
+ // legitimate `{#…#}` / mixed TRAI shapes (extra `{`/`}` counts). Do not tie Done to it;
534
+ // slot completeness is enforced by `areAllRcsSmsFallbackVarSlotsFilled` on the RCS screen.
535
+ if (useRcsFallbackVarSegment) {
536
+ tagValidationResponseRef.current = {};
537
+ updateIsTagValidationError(false);
538
+ return;
539
+ }
540
+ const validationContent = fallbackText || '';
541
+ if (!validationContent.trim()) {
542
+ tagValidationResponseRef.current = {};
543
+ updateIsTagValidationError(false);
544
+ return;
545
+ }
546
+ tagValidationResponseRef.current = validateTags({
547
+ content: validationContent,
548
+ tagsParam: tags,
549
+ location,
550
+ tagModule: getDefaultTags,
551
+ isFullMode: true,
552
+ }) || {};
553
+ const braceErr = !!tagValidationResponseRef.current.isBraceError;
554
+ // Plain (non-VarSegment) fallback textarea: only unbalanced `{{…}}` disables Done.
555
+ updateIsTagValidationError(braceErr);
556
+ return;
557
+ }
558
+ if (
559
+ !isFullMode &&
560
+ updatedSmsEditor?.length > 0 &&
561
+ !updatedSmsEditor.includes(SMS_TRAI_VAR)
562
+ ) {
563
+ tagValidationResponseRef.current = runValidateTags(updatedSmsEditor.join(''));
564
+ const missing = (tagValidationResponseRef.current.missingTags || []).length > 0;
565
+ const braceErr = !!tagValidationResponseRef.current.isBraceError;
566
+ updateIsTagValidationError(missing || braceErr);
567
+ } else if (!isRcsSmsFallback) {
568
+ tagValidationResponseRef.current = {};
569
+ updateIsTagValidationError(false);
570
+ }
571
+ }, [
572
+ updatedSmsEditor,
573
+ tags,
574
+ isRcsSmsFallback,
575
+ isFullMode,
576
+ useRcsFallbackVarSegment,
577
+ fallbackText,
578
+ fallbackVarMappedData,
579
+ getDefaultTags,
580
+ location,
581
+ ]);
582
+
240
583
  const computeUpdatedSmsEditor = () => {
241
584
  const arr = [...tempMsgArray];
242
- const varMapKeys = Object.keys(varMap)?.map((key) => Number(key.slice(8)))?.sort((a, b) => a - b) || [];
243
- for (const key in varMap) {
244
- if (varMap[key].data !== '') {
585
+ const varMapKeys = Object.keys(varMapRef.current)?.map((key) => Number(key.slice(8)))?.sort((a, b) => a - b) || [];
586
+ for (const key in varMapRef.current) {
587
+ if (varMapRef.current[key].data !== '') {
245
588
  const _id = Number(key.slice(8)); //Eg: -> extracting index 1 from keys like {#var# } _1
246
589
  const loopIndex =
247
590
  varMapKeys[varMapKeys?.indexOf(_id) + 1] || arr.length;
248
591
  for (let i = _id; i < loopIndex; i += 1) {
249
592
  if (i === _id) {
250
- arr[i] = varMap[key].data; //data for first #var# of the textbox
593
+ arr[i] = varMapRef.current[key].data; //data for first #var# of the textbox
251
594
  } else if (arr[i] === SMS_TRAI_VAR) {
252
595
  arr[i] = ''; //'' for remaining #var# of a textbox
253
596
  }
@@ -257,8 +600,8 @@ export const SmsTraiEdit = (props) => {
257
600
  setUpdatedSmsEditor(arr);
258
601
  };
259
602
 
260
- //Saving on done start
261
603
  const onUpdateTemplateComplete = (editResponse, errorMsg) => {
604
+ updateLoading(false);
262
605
  if (editResponse?.templateId) {
263
606
  CapNotification.success({
264
607
  message: formatMessage(messages.smsEditNotification),
@@ -337,66 +680,121 @@ export const SmsTraiEdit = (props) => {
337
680
  };
338
681
 
339
682
  const onDoneCallback = () => {
340
- if (updatedSmsEditor.includes(SMS_TRAI_VAR) && !isFullMode) {
341
- //during save textbox should not be empty
683
+ // RCS fallback: allow Save when only "missing tags" ({{name}} may not be in TagList); still block unbalanced braces.
684
+ if (isTagValidationError) {
685
+ if (!isRcsSmsFallback) return;
686
+ const tagValidationSnapshot = tagValidationResponseRef.current ?? {};
687
+ if (tagValidationSnapshot.isBraceError) return;
688
+ }
689
+ const editorJoined = Array.isArray(updatedSmsEditor)
690
+ ? updatedSmsEditor.join('')
691
+ : String(updatedSmsEditor || '');
692
+ if (!isRcsSmsFallback && editorJoined.includes(SMS_TRAI_VAR) && !isFullMode) {
342
693
  updateIsValidationError(true);
343
694
  } else {
344
- //start spinner
345
- updateLoading(true);
346
- const traiVersions = traiData.versions || {};
347
- traiVersions.base = {
348
- ...traiVersions.base,
349
- 'var-mapped': varMap,
350
- 'updated-sms-editor': updatedSmsEditor,
351
- 'unicode-validity': isUnicodeAllowed,
352
- };
695
+ // Only show full-screen spinner when waiting on full-mode SMS edit API — not for RCS SMS fallback
696
+ // handoff (getFormSubscriptionData → SmsFallback closeSlidebox) and not for embedded sync handoff.
697
+ if (isFullMode && !isRcsSmsFallback) {
698
+ updateLoading(true);
699
+ }
700
+ if (!traiDataRef.current.versions) {
701
+ traiDataRef.current.versions = { base: {}, history: [] };
702
+ }
703
+ const traiVersions = traiDataRef.current.versions;
704
+ if (isRcsSmsFallback) {
705
+ const {
706
+ 'var-mapped': _ignoredVarMapped,
707
+ 'updated-sms-editor': _ignoredUpdatedSmsEditor,
708
+ 'rcs-sms-fallback-var-mapped': _ignoredFallbackVarMapped,
709
+ ...baseWithoutDerivedFields
710
+ } = traiVersions.base || {};
711
+ traiVersions.base = {
712
+ ...baseWithoutDerivedFields,
713
+ 'sms-editor': fallbackText || '',
714
+ 'unicode-validity': isUnicodeAllowed,
715
+ ...(useRcsFallbackVarSegment && {
716
+ 'rcs-sms-fallback-var-mapped': fallbackVarMappedData || {},
717
+ }),
718
+ };
719
+ } else {
720
+ traiVersions.base = {
721
+ ...traiVersions.base,
722
+ 'var-mapped': varMapRef.current,
723
+ 'updated-sms-editor': updatedSmsEditor,
724
+ 'unicode-validity': isUnicodeAllowed,
725
+ };
726
+ }
353
727
  traiVersions.history = [traiVersions.base];
354
- if (isFullMode) {
355
- actions.editTemplate(traiData, onUpdateTemplateComplete);
728
+ // RCS → SMS fallback (slidebox / inline): always hand off via getFormSubscriptionData so parent
729
+ // can persist fallback state and close the slide — never the standalone TRAI editTemplate API here.
730
+ if (isFullMode && !isRcsSmsFallback) {
731
+ actions.editTemplate(traiDataRef.current, onUpdateTemplateComplete);
356
732
  } else {
357
733
  getFormSubscriptionData({
358
- value: traiData.versions,
734
+ value: traiVersions,
735
+ // Consumers/tests read `versions.base` first (see getBaseFromSmsTraiFormData, rcsDltEditCompletionHandler).
736
+ versions: traiVersions,
359
737
  _id: params && params.id,
360
738
  validity: true,
361
739
  type: SMS,
362
740
  });
741
+ updateLoading(false);
363
742
  }
364
743
  }
365
744
  };
366
- //Saving on done end
367
745
 
368
- // tag code start
746
+ const locationQueryType = location?.query?.type;
747
+ const locationQueryModule = location?.query?.module;
748
+
369
749
  useEffect(() => {
370
- let tag =
371
- metaEntities && metaEntities.tags ? metaEntities.tags.standard : [];
372
- const { type, module } = location.query || {};
373
- if (type === EMBEDDED && module === LIBRARY && !getDefaultTags) {
750
+ let tag = getStandardTagsFromMeta(metaEntities);
751
+ if ((!Array.isArray(tag) || tag.length === 0) && Array.isArray(supportedTags) && supportedTags.length > 0) {
374
752
  tag = supportedTags;
375
753
  }
376
- updateTags(tag);
377
- }, [metaEntities]);
754
+ if (
755
+ locationQueryType === EMBEDDED &&
756
+ locationQueryModule === LIBRARY &&
757
+ !getDefaultTags
758
+ ) {
759
+ tag = supportedTags || [];
760
+ }
761
+ updateTags(Array.isArray(tag) ? tag : []);
762
+ }, [metaEntities, getDefaultTags, supportedTags, locationQueryType, locationQueryModule]);
378
763
 
379
764
  const handleOnTagsContextChange = (data) => {
765
+ // CapTagList passes "Outbound" | "Loyalty" from the module filter; TagList may pass "Outbound" on init.
766
+ if (data == null || data === '') return;
767
+ const normalizedContext = String(data).toLowerCase();
380
768
  const { type } = location.query || {};
381
769
  const isEmbedded = type === EMBEDDED;
382
770
  const query = {
383
771
  layout: SMS,
384
772
  type: TAG,
385
- context:
386
- (data || '').toLowerCase() === ALL
387
- ? DEFAULT
388
- : (data || '').toLowerCase(),
389
- embedded: isEmbedded ? type : FULL,
773
+ context: normalizedContext === ALL ? DEFAULT : normalizedContext,
774
+ embedded: forceFullTagContext ? FULL : (isEmbedded ? type : FULL),
390
775
  };
391
- globalActions.fetchSchemaForEntity(query);
776
+ fetchTagSchemaIfNewQuery(query);
392
777
  };
393
778
 
394
779
  const onTagSelect = (data) => {
395
- if (textAreaId >= 0 && varMap && updatedSmsEditor) {
780
+ if (isRcsSmsFallback) {
781
+ if (!useRcsFallbackVarSegment) {
782
+ setFallbackText((prev) => `${prev || ''}{{${data}}}`);
783
+ return;
784
+ }
785
+ if (!fallbackFocusedId) return;
786
+ const prevVal = fallbackVarMappedData?.[fallbackFocusedId] ?? '';
787
+ const nextVal = `${prevVal}{{${data}}}`;
788
+ setFallbackVarMappedData((prev) => ({
789
+ ...(prev || {}),
790
+ [fallbackFocusedId]: nextVal,
791
+ }));
792
+ return;
793
+ }
794
+ if (textAreaId >= 0 && varMapRef.current && updatedSmsEditor) {
396
795
  const arr = [...updatedSmsEditor];
397
- const varMapKeys = Object.keys(varMap)?.map((key) => Number(key.slice(8)))?.sort((a, b) => a - b) || [];
796
+ const varMapKeys = Object.keys(varMapRef.current)?.map((key) => Number(key.slice(8)))?.sort((a, b) => a - b) || [];
398
797
  const loopIndex = varMapKeys[varMapKeys?.indexOf(textAreaId) + 1] || arr.length;
399
- //when trying to insert tag in empty textarea,{#var#} is replaced with "" and then tag is added
400
798
  for (let i = textAreaId; i < loopIndex; i += 1) {
401
799
  if (arr[i] === SMS_TRAI_VAR) {
402
800
  arr[i] = '';
@@ -404,26 +802,23 @@ export const SmsTraiEdit = (props) => {
404
802
  }
405
803
  const messageData = `${arr[textAreaId]}{{${data}}}`;
406
804
  arr[textAreaId] = messageData;
407
- varMap[`${SMS_TRAI_VAR}_${textAreaId}`].data = messageData;
805
+ varMapRef.current[`${SMS_TRAI_VAR}_${textAreaId}`].data = messageData;
408
806
  setUpdatedSmsEditor(arr);
409
807
  }
410
808
  };
411
- //setting the id of currently selected text area, is used onTagSelect
809
+
412
810
  const setTextAreaId = (event) => {
413
811
  updateTextAreaId(Number(event.target.id));
414
812
  };
415
- // tag code end
416
813
 
417
- // on change event of Text Area
418
814
  const textAreaValueChange = ({ target: { value, id } }) => {
815
+ if (isRcsSmsFallback) return;
419
816
  const _id = Number(id);
420
817
  const arr = [...updatedSmsEditor];
421
- const varMapKeys = Object.keys(varMap)?.map((key) => Number(key.slice(8)))?.sort((a, b) => a - b) || [];
818
+ const varMapKeys = Object.keys(varMapRef.current)?.map((key) => Number(key.slice(8)))?.sort((a, b) => a - b) || [];
422
819
  const loopIndex = varMapKeys[varMapKeys?.indexOf(_id) + 1] || arr.length;
423
820
 
424
- //assign entered value to varMap
425
- varMap[`${SMS_TRAI_VAR}_${_id}`].data = value;
426
- //based on entered value update updatedSmsEditor
821
+ varMapRef.current[`${SMS_TRAI_VAR}_${_id}`].data = value;
427
822
  if (value === '') {
428
823
  for (let i = _id; i < loopIndex; i += 1) {
429
824
  if (i === _id || arr[i] === '') {
@@ -433,7 +828,7 @@ export const SmsTraiEdit = (props) => {
433
828
  } else {
434
829
  for (let i = _id; i < loopIndex; i += 1) {
435
830
  if (i === _id) {
436
- arr[i] = varMap[`${SMS_TRAI_VAR}_${_id}`].data;
831
+ arr[i] = varMapRef.current[`${SMS_TRAI_VAR}_${_id}`].data;
437
832
  } else if (arr[i] === SMS_TRAI_VAR) {
438
833
  arr[i] = '';
439
834
  }
@@ -536,18 +931,33 @@ export const SmsTraiEdit = (props) => {
536
931
  };
537
932
 
538
933
  const smsLengthForVar = () => (
539
- <CapHeading type="h5" style={{ marginTop: CAP_SPACE_04, float: 'right' }}>
540
- {formatMessage(messages.totalCharacters, {
541
- smsCount: Math.ceil(totalMessageLength / 160),
542
- number: totalMessageLength,
543
- })}
934
+ <CapHeading
935
+ type="h5"
936
+ className={isRcsSmsFallback ? 'rcs-character-count rcs-character-count--compact' : ''}
937
+ style={isRcsSmsFallback ? {} : { marginTop: CAP_SPACE_04, marginBottom: 0 }}
938
+ >
939
+ {isRcsSmsFallback
940
+ ? formatMessage(messages.charactersCountLabel, {
941
+ current: totalMessageLength,
942
+ max: SMS_TRAI_CONTENT_MAX_LENGTH,
943
+ })
944
+ : formatMessage(messages.totalCharacters, {
945
+ smsCount: Math.ceil(totalMessageLength / 160),
946
+ number: totalMessageLength,
947
+ })}
544
948
  </CapHeading>
545
949
  );
546
950
 
547
- // to compute the length of the message
548
- //40 characters is blocked per'{#var#}' if textbox is empty otherwise it will be textbox length
549
- // and the remaining string length is added to it
550
951
  const calculateTotalMessageLength = () => {
952
+ if (isRcsSmsFallback) {
953
+ if (useRcsFallbackVarSegment) {
954
+ const resolved = getFallbackResolvedContent(fallbackText || '', fallbackVarMappedData || {});
955
+ setTotalMessageLength(resolved.length);
956
+ } else {
957
+ setTotalMessageLength((fallbackText || '').length);
958
+ }
959
+ return;
960
+ }
551
961
  const msgLenWithoutVar = tempMsgArray
552
962
  ?.filter((i) => i !== SMS_TRAI_VAR)
553
963
  .join('');
@@ -558,22 +968,43 @@ export const SmsTraiEdit = (props) => {
558
968
 
559
969
  const calculateLenForTextBox = () => {
560
970
  let countVarChar = 0;
561
- Object.keys(varMap).forEach((i) => {
562
- if (varMap[i].data) {
563
- countVarChar += varMap[i].data?.length;
564
- if (!showMsgLengthNote && varMap[i].data?.length > varMap[i].count * CHARLIMIT) {
971
+ Object.keys(varMapRef.current).forEach((i) => {
972
+ if (varMapRef.current[i].data) {
973
+ countVarChar += varMapRef.current[i].data?.length;
974
+ if (!showMsgLengthNote && varMapRef.current[i].data?.length > varMapRef.current[i].count * CHARLIMIT) {
565
975
  updateshowMsgLengthNote(true);
566
976
  }
567
977
  } else {
568
- countVarChar += varMap[i].count * CHARLIMIT;
978
+ countVarChar += varMapRef.current[i].count * CHARLIMIT;
569
979
  }
570
980
  });
571
981
  return countVarChar;
572
982
  };
573
983
 
984
+ const tagValidationErrorMessage = () => {
985
+ const { missingTags = [], unsupportedTags = [], isBraceError } = tagValidationResponseRef.current || {};
986
+ const listForMessage = (unsupportedTags && unsupportedTags.length > 0)
987
+ ? unsupportedTags
988
+ : missingTags;
989
+ if (isBraceError) {
990
+ return <CapError>{formatMessage(globalMessages.unbalanacedCurlyBraces)}</CapError>;
991
+ }
992
+ if (listForMessage.length > 0) {
993
+ return (
994
+ <CapError>
995
+ {formatMessage(messages.unsupportedTagsValidationError, {
996
+ unsupportedTags: listForMessage,
997
+ })}
998
+ </CapError>
999
+ );
1000
+ }
1001
+ return null;
1002
+ };
1003
+
574
1004
  const disablehandler = () => {
575
- if (traiData && !isEmpty(traiData)) {
576
- const msg = get(traiData, `versions.base.sms-editor`, '');
1005
+ if (isRcsSmsFallback) return false;
1006
+ if (traiDataRef.current && !isEmpty(traiDataRef.current)) {
1007
+ const msg = get(traiDataRef.current, `versions.base.sms-editor`, '');
577
1008
  const index = msg.search(SMS_TRAI_VAR);
578
1009
  if (index === -1) {
579
1010
  return true;
@@ -587,7 +1018,6 @@ export const SmsTraiEdit = (props) => {
587
1018
  updateIsUnicodeAllowed(checked);
588
1019
  };
589
1020
 
590
- // Get template content for test and preview
591
1021
  const getTemplateContent = () => {
592
1022
  if (!updatedSmsEditor || updatedSmsEditor.length === 0) {
593
1023
  return '';
@@ -595,42 +1025,69 @@ export const SmsTraiEdit = (props) => {
595
1025
  return updatedSmsEditor.join('');
596
1026
  };
597
1027
 
598
- // Build formData for TestAndPreviewSlidebox - templateConfigs with templateId and template for DLT test
599
1028
  const getFormDataForTestAndPreview = () => {
600
- const smsBase = get(traiData, 'versions.base') || get(templateDetails, 'versions.base') || get(templateData, 'versions.base');
1029
+ const smsBase = get(traiDataRef.current, 'versions.base') || get(templateDetails, 'versions.base') || get(templateData, 'versions.base');
601
1030
  if (!smsBase || !smsBase.template_id) {
602
1031
  return {};
603
1032
  }
604
1033
  const templateRaw = smsBase['updated-sms-editor'] || smsBase['sms-editor'] || '';
605
1034
  const template = Array.isArray(templateRaw) ? templateRaw.join('') : templateRaw;
1035
+ const traiDlt = isTraiDLTEnable(isFullMode, smsRegister);
1036
+ const headerIds = get(traiDataRef.current, 'versions.base.header', []) || [];
606
1037
  return {
607
1038
  templateConfigs: {
608
- templateId: smsBase.template_id, template, traiDltEnabled: true, registeredSenderIds: get(traiData, `versions.base.header`, []),
1039
+ templateId: smsBase.template_id,
1040
+ template,
1041
+ traiDltEnabled: traiDlt,
1042
+ registeredSenderIds: traiDlt ? headerIds : [],
609
1043
  },
610
1044
  };
611
1045
  };
612
1046
 
613
- // Handle test and preview button click
614
1047
  const handleTestAndPreview = () => {
615
1048
  setShowTestAndPreviewSlidebox(true);
616
1049
  };
617
1050
 
618
- // Handle close test and preview slidebox
619
1051
  const handleCloseTestAndPreview = () => {
620
1052
  setShowTestAndPreviewSlidebox(false);
621
1053
  };
622
1054
 
1055
+ const shouldShowPreview =
1056
+ !isRcsSmsFallback || (showPreviewInRcsFallback && !hidePreview);
1057
+ const smsSidePreviewColumn = (
1058
+ <CapColumn span={8} offset={1}>
1059
+ <UnifiedPreview
1060
+ channel={SMS}
1061
+ content={updatedSmsEditor.join('')}
1062
+ device={ANDROID}
1063
+ showDeviceToggle={false}
1064
+ showHeader={false}
1065
+ formatMessage={formatMessage}
1066
+ senderId={isUnicodeAllowed ? 'Unicode' : 'ASCII'}
1067
+ />
1068
+ </CapColumn>
1069
+ );
1070
+
623
1071
  return (
624
1072
  <>
625
- <CapSpin spinning={loading || fetchingLiquidTags} tip={fetchingLiquidTags && formatMessage(formBuilderMessages.liquidSpinText)}>
1073
+ <CapSpin
1074
+ spinning={loading || fetchingLiquidTags}
1075
+ tip={fetchingLiquidTags && formatMessage(formBuilderMessages.liquidSpinText)}
1076
+ className={[
1077
+ isRcsSmsFallback && 'sms-trai-edit-rcs-fallback',
1078
+ isOverview && 'sms-trai-edit--overview',
1079
+ ]
1080
+ .filter(Boolean)
1081
+ .join(' ') || undefined}
1082
+ >
626
1083
  <CapRow>
627
- {traiData && !isEmpty(traiData) && (
1084
+ {traiDataRef.current && !isEmpty(traiDataRef.current) && !isRcsSmsFallback && (
628
1085
  <TraiEditTemplateDetails>
629
1086
  <CapLabelInline type="label1">
630
1087
  {formatMessage(messages.templateLabel)}
631
1088
  </CapLabelInline>
632
1089
  <CapLabelInline type="label2">
633
- {get(traiData, `versions.base.template_name`, '')}
1090
+ {get(traiDataRef.current, `versions.base.template_name`, '')}
634
1091
  </CapLabelInline>
635
1092
 
636
1093
  <CapLabelInline type="label1">
@@ -641,15 +1098,42 @@ export const SmsTraiEdit = (props) => {
641
1098
  {formatMessage(messages.senderIdlabel)}
642
1099
  </CapLabelInline>
643
1100
  <CapLabelInline type="label2">
644
- {[...get(traiData, `versions.base.header`, [])].join(', ')}
1101
+ {[...get(traiDataRef.current, `versions.base.header`, [])].join(', ')}
1102
+ </CapLabelInline>
1103
+ </TraiEditTemplateDetails>
1104
+ )}
1105
+ {isRcsSmsFallback &&
1106
+ traiDataRef.current &&
1107
+ !isEmpty(traiDataRef.current) &&
1108
+ !loading && (
1109
+ <TraiEditTemplateDetails className="sms-trai-edit-rcs-fallback__template-meta">
1110
+ <CapLabelInline type="label1">
1111
+ {formatMessage(messages.templateNameLabel)}
1112
+ </CapLabelInline>
1113
+ <CapLabelInline type="label2">
1114
+ {get(traiDataRef.current, 'versions.base.template_name', '')
1115
+ || get(traiDataRef.current, 'name', '')}
645
1116
  </CapLabelInline>
1117
+ {traiDltEnabled ? (
1118
+ <>
1119
+ <CapLabelInline type="label1">
1120
+ {formatMessage(messages.traiEditSeperator)}
1121
+ </CapLabelInline>
1122
+ <CapLabelInline type="label1">
1123
+ {formatMessage(messages.senderIdlabel)}
1124
+ </CapLabelInline>
1125
+ <CapLabelInline type="label2">
1126
+ {[...get(traiDataRef.current, 'versions.base.header', [])].join(', ')}
1127
+ </CapLabelInline>
1128
+ </>
1129
+ ) : null}
646
1130
  </TraiEditTemplateDetails>
647
1131
  )}
648
- <CapColumn span={14}>
1132
+ <CapColumn span={shouldShowPreview ? 14 : 24}>
649
1133
  <CapRow>
650
1134
  <CapHeader
651
1135
  title={formatMessage(messages.traiEditTitle)}
652
- size="regular"
1136
+ size={isRcsSmsFallback ? 'label1' : 'regular'}
653
1137
  suffix={(
654
1138
  <TagList
655
1139
  label={formatMessage(messages.addLabels)}
@@ -658,43 +1142,55 @@ export const SmsTraiEdit = (props) => {
658
1142
  tags={tags || []}
659
1143
  onContextChange={handleOnTagsContextChange}
660
1144
  injectedTags={injectedTags || {}}
661
- hidePopover={disablehandler()}
1145
+ channel={SMS}
1146
+ hidePopover={false}
1147
+ disabled={!isRcsSmsFallback && disablehandler()}
662
1148
  selectedOfferDetails={selectedOfferDetails}
663
1149
  eventContextTags={eventContextTags}
664
- waitEventContextTags={waitEventContextTags}
1150
+ popoverOverlayStyle={isRcsSmsFallback ? { zIndex: 10020 } : undefined}
1151
+ popoverOverlayClassName={isRcsSmsFallback ? 'sms-fallback-taglist-popover rcs-sms-fallback-taglist-popover' : undefined}
665
1152
  />
666
1153
  )}
667
1154
  />
668
1155
  </CapRow>
669
1156
 
670
- <CapRow
671
- style={{
672
- backgroundColor: CAP_G10,
673
- padding: CAP_SPACE_16,
674
- }}
1157
+ {isRcsSmsFallback ? (
1158
+ <>
1159
+ {renderRcsFallbackMessage(fallbackText)}
1160
+ </>
1161
+ ) : (
1162
+ <>
1163
+ <CapRow className="sms-trai-editor-segment-row">
1164
+ <div className="sms-trai-segmented-editor">
1165
+ {renderedContent()}
1166
+ </div>
1167
+ </CapRow>
1168
+ <CapRow className="sms-trai-length-row">
1169
+ {smsLengthForVar()}
1170
+ </CapRow>
1171
+ </>
1172
+ )}
1173
+ {isRcsSmsFallback && isTagValidationError && (
1174
+ <CapRow>
1175
+ {tagValidationErrorMessage()}
1176
+ </CapRow>
1177
+ )}
1178
+ {!isRcsSmsFallback && isTagValidationError && tagValidationErrorMessage()}
1179
+ <CapCheckbox
1180
+ onChange={unicodeHandler}
1181
+ checked={isUnicodeAllowed}
1182
+ disabled={
1183
+ isRcsSmsFallback
1184
+ ? isRcsEditFlow
1185
+ : disablehandler()
1186
+ }
675
1187
  >
676
- {renderedContent()}
677
- </CapRow>
678
- <CapRow>
679
- {smsLengthForVar()}
680
- </CapRow>
681
- <CapCheckbox onChange={unicodeHandler} checked={isUnicodeAllowed} disabled={disablehandler()}>
682
1188
  {formatMessage(messages.unicodeLabel)}
683
1189
  </CapCheckbox>
684
1190
  {showMsgLengthNote && <CapInfoNote message={<FormattedMessage {...messages.msgLengthNote} values={{ var: '{#var#}' }} />} />}
685
- <div style={{ marginBottom: '100px' }} />
686
- </CapColumn>
687
- <CapColumn span={8} offset={1}>
688
- <UnifiedPreview
689
- channel={SMS}
690
- content={updatedSmsEditor.join('')}
691
- device={ANDROID}
692
- showDeviceToggle={false}
693
- showHeader={false}
694
- formatMessage={formatMessage}
695
- senderId={isUnicodeAllowed ? 'Unicode' : 'ASCII'}
696
- />
1191
+ <div className="sms-trai-edit-bottom-spacer" />
697
1192
  </CapColumn>
1193
+ {shouldShowPreview && smsSidePreviewColumn}
698
1194
  </CapRow>
699
1195
  <SMSTraiFooter>
700
1196
  {isLiquidValidationError && !liquidErrorPanelDismissed && (
@@ -720,8 +1216,9 @@ export const SmsTraiEdit = (props) => {
720
1216
  <FormattedMessage {...messages.testAndPreviewButtonLabel} />
721
1217
  </CapButton>
722
1218
  <CapButton
723
- onClick={onSubmitWrapper}
1219
+ onClick={onDoneCallback}
724
1220
  className="create-msg"
1221
+ disabled={isTagValidationError}
725
1222
  >
726
1223
  <FormattedMessage {...messages.saveButtonLabel} />
727
1224
  </CapButton>
@@ -738,6 +1235,15 @@ export const SmsTraiEdit = (props) => {
738
1235
  );
739
1236
  };
740
1237
 
1238
+ SmsTraiEdit.propTypes = {
1239
+ isRcsSmsFallback: PropTypes.bool,
1240
+ isRcsEditFlow: PropTypes.bool,
1241
+ showPreviewInRcsFallback: PropTypes.bool,
1242
+ hidePreview: PropTypes.bool,
1243
+ isOverview: PropTypes.bool,
1244
+ onRcsFallbackEditorStateChange: PropTypes.func,
1245
+ };
1246
+
741
1247
  const mapStateToProps = createStructuredSelector({
742
1248
  templateDetails: makeSelectTemplateDetailsResponse(),
743
1249
  metaEntities: makeSelectMetaEntities(),