@capillarytech/creatives-library 8.0.353-alpha.3 → 8.0.353-alpha.4

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.
@@ -1,4 +1,48 @@
1
1
  import React from 'react';
2
+
3
+ // Isolated input for the email template name field.
4
+ // Manages its own value in local state so keystrokes only re-render this
5
+ // small component, not the entire CreativesContainer → Email → FormBuilder tree.
6
+ class TemplateNameInputField extends React.Component {
7
+ constructor(props) {
8
+ super(props);
9
+ this.state = { localValue: props.initialValue || '' };
10
+ }
11
+
12
+ componentDidUpdate(prevProps) {
13
+ // Sync from props only when the external value changed AND the user hasn't
14
+ // diverged from the previous prop value. This handles async data-load in edit
15
+ // mode without overwriting what the user is actively typing.
16
+ if (
17
+ prevProps.initialValue !== this.props.initialValue &&
18
+ this.state.localValue === (prevProps.initialValue || '')
19
+ ) {
20
+ this.setState({ localValue: this.props.initialValue || '' });
21
+ }
22
+ }
23
+
24
+ handleChange = (ev) => {
25
+ const { value } = ev.currentTarget;
26
+ this.setState({ localValue: value });
27
+ if (this.props.onChange) this.props.onChange(value);
28
+ };
29
+
30
+ handleBlur = () => {
31
+ if (this.props.onBlur) this.props.onBlur(this.state.localValue);
32
+ };
33
+
34
+ render() {
35
+ const { onChange: _onChange, initialValue: _initialValue, onBlur: _ob, ...rest } = this.props;
36
+ return (
37
+ <CapInput
38
+ {...rest}
39
+ value={this.state.localValue}
40
+ onChange={this.handleChange}
41
+ onBlur={this.handleBlur}
42
+ />
43
+ );
44
+ }
45
+ }
2
46
  import PropTypes from 'prop-types';
3
47
  import {
4
48
  CAP_SPACE_16, CAP_SPACE_32, CAP_SPACE_56, CAP_SPACE_64,
@@ -1753,30 +1797,22 @@ export class Creatives extends React.Component {
1753
1797
  } />
1754
1798
  )
1755
1799
 
1756
- templateNameComponentInput = ({ formData, onFormDataChange, name }) => {
1757
- // Use local state for immediate UI feedback, fallback to prop value
1758
- const displayValue = this.state.localTemplateName !== '' ? this.state.localTemplateName : name;
1759
-
1760
- return (
1761
- <CapInput
1762
- value={displayValue}
1763
- suffix={<span />}
1764
- onBlur={() => {
1765
- this.setState({
1766
- isEditName: false,
1767
- localTemplateName: '', // Clear local state on blur
1768
- }, () => {
1769
- this.showTemplateName({ formData, onFormDataChange });
1770
- });
1771
- }}
1772
- onChange={(ev) => {
1773
- const { value } = ev.currentTarget;
1774
- // Use optimized update for better performance
1775
- this.updateTemplateNameImmediately(value, formData, onFormDataChange);
1776
- }}
1777
- />
1778
- );
1779
- }
1800
+ templateNameComponentInput = ({ formData, onFormDataChange, name }) => (
1801
+ <TemplateNameInputField
1802
+ initialValue={name}
1803
+ suffix={<span />}
1804
+ onBlur={(committedValue) => {
1805
+ this.performTemplateNameUpdate(committedValue, formData, onFormDataChange);
1806
+ this.setState({ isEditName: false });
1807
+ }}
1808
+ onChange={(value) => {
1809
+ const isEmptyTemplateName = !value.trim();
1810
+ if (this.state.isTemplateNameEmpty !== isEmptyTemplateName) {
1811
+ this.setState({ isTemplateNameEmpty: isEmptyTemplateName });
1812
+ }
1813
+ }}
1814
+ />
1815
+ )
1780
1816
 
1781
1817
  showTemplateName = ({ formData, onFormDataChange }) => { //gets called from email/index after template data is fetched
1782
1818
  const {
@@ -798,6 +798,7 @@ export class Email extends React.Component { // eslint-disable-line react/prefer
798
798
  onFormDataChange = (updatedFormData, tabCount, currentTab) => {
799
799
  // this.transformFormData(formData);
800
800
  const formData = {...updatedFormData};
801
+
801
802
  const {defaultData = {}, isFullMode, showTemplateName} = this.props;
802
803
  const templateName = formData['template-name'];
803
804
  const defaultTemplateName = _.get(defaultData, 'template-name', "");
@@ -809,7 +810,6 @@ export class Email extends React.Component { // eslint-disable-line react/prefer
809
810
  formData['template-name'] = templateName;
810
811
  }
811
812
 
812
-
813
813
  this.setState({formData, tabCount, isSchemaChanged: false}, () => {
814
814
  if (this.props.isFullMode && showTemplateName) {
815
815
  showTemplateName({formData, onFormDataChange: this.onFormDataChange});
@@ -153,83 +153,6 @@
153
153
  line-clamp: 3;
154
154
  }
155
155
  }
156
- .viber-carousel-static-content {
157
- width: 100%;
158
- overflow: hidden;
159
- .viber-carousel-static-message-box {
160
- width: 100%;
161
- height: 2.25rem;
162
- border-radius: $CAP_SPACE_04;
163
- background: $CAP_WHITE;
164
- margin-bottom: $CAP_SPACE_08;
165
- padding: 0 $CAP_SPACE_08;
166
- display: flex;
167
- align-items: center;
168
- }
169
- .viber-carousel-static-message {
170
- display: block;
171
- width: 100%;
172
- color: $CAP_G01;
173
- overflow: hidden;
174
- text-overflow: ellipsis;
175
- white-space: nowrap;
176
- }
177
- .viber-carousel-static-cards {
178
- display: flex;
179
- gap: $CAP_SPACE_08;
180
- overflow: hidden;
181
- }
182
- .viber-carousel-static-card {
183
- flex: 1;
184
- min-width: 0;
185
- background: $CAP_WHITE;
186
- border-radius: $CAP_SPACE_04;
187
- padding: $CAP_SPACE_06;
188
- overflow: hidden;
189
- }
190
- .viber-carousel-static-image,
191
- .viber-carousel-static-image-placeholder {
192
- width: 100%;
193
- height: 5.357rem;
194
- border-radius: $CAP_SPACE_04;
195
- }
196
- .viber-carousel-static-image {
197
- object-fit: cover;
198
- }
199
- .viber-carousel-static-image-placeholder {
200
- background: $CAP_G07;
201
- }
202
- .viber-carousel-static-text {
203
- display: block;
204
- color: $CAP_G01;
205
- margin: $CAP_SPACE_04 0;
206
- line-height: 1.3;
207
- overflow: hidden;
208
- text-overflow: ellipsis;
209
- white-space: normal;
210
- }
211
- .viber-carousel-static-buttons {
212
- display: flex;
213
- flex-direction: column;
214
- gap: $CAP_SPACE_04;
215
- }
216
- .viber-carousel-static-button {
217
- display: block;
218
- min-height: 1.5rem;
219
- border-radius: $CAP_SPACE_12;
220
- background: $CAP_PURPLE01;
221
- color: $CAP_WHITE;
222
- text-align: center;
223
- padding: 0.179rem $CAP_SPACE_08;
224
- white-space: normal;
225
- }
226
- .viber-carousel-static-button-secondary {
227
- color: $CAP_PURPLE01;
228
- background: transparent;
229
- border: none;
230
- box-shadow: none;
231
- }
232
- }
233
156
 
234
157
  }
235
158
 
@@ -72,7 +72,6 @@ import * as ebillActions from '../Ebill/actions';
72
72
  import * as emailActions from '../Email/actions';
73
73
  import * as lineActions from '../Line/Container/actions';
74
74
  import * as viberActions from '../Viber/actions';
75
- import { VIBER_MEDIA_TYPES } from '../Viber/constants';
76
75
  import * as facebookActions from '../Facebook/actions';
77
76
  import * as whatsappActions from '../Whatsapp/actions';
78
77
  import * as rcsActions from '../Rcs/actions';
@@ -1372,11 +1371,6 @@ export class Templates extends React.Component { // eslint-disable-line react/pr
1372
1371
  const image = viberContent.image || {};
1373
1372
  const video = viberContent.video || {};
1374
1373
  const button = viberContent.button || {};
1375
- const messageType = viberContent.type || '';
1376
- const cardsRaw = viberContent.cards;
1377
- const cards = Array.isArray(cardsRaw) ? cardsRaw : [];
1378
- const isCarousel =
1379
- messageType === VIBER_MEDIA_TYPES.CAROUSEL || cards.length > 0;
1380
1374
 
1381
1375
  const viberPreview = {
1382
1376
  messageContent: text,
@@ -1394,39 +1388,14 @@ export class Templates extends React.Component { // eslint-disable-line react/pr
1394
1388
  };
1395
1389
  }
1396
1390
 
1397
- if (isCarousel) {
1398
- viberPreview.type = VIBER_MEDIA_TYPES.CAROUSEL;
1399
- viberPreview.cards = cards.map((card) => ({
1400
- text: card?.text || '',
1401
- mediaUrl: card?.mediaUrl || '',
1402
- buttons: (card?.buttons || []).map((btn) => ({
1403
- title: btn?.title || '',
1404
- action: btn?.action || '',
1405
- })),
1406
- }));
1407
- viberPreview.showCarouselEditorPreview = true;
1408
- }
1409
-
1410
1391
  // Extract account name
1411
1392
  const accountName = get(template, 'definition.accountName', '');
1412
1393
 
1413
1394
  // Return Viber content object (same as Viber/index.js getTemplateContent)
1414
1395
  return {
1415
1396
  viberPreviewContent: viberPreview,
1416
- accountName: accountName || '',
1417
- brandName: accountName || '',
1418
- messageContent: text,
1419
- ...(isCarousel && {
1420
- type: VIBER_MEDIA_TYPES.CAROUSEL,
1421
- cards: viberPreview.cards,
1422
- }),
1423
- ...(!isCarousel && !isEmpty(button) && button.text && (button.url || viberContent.buttonURL) && {
1424
- button: {
1425
- text: button.text,
1426
- url: button.url || viberContent.buttonURL || '',
1427
- },
1428
- }),
1429
- buttonURL: button.url || viberContent.buttonURL || '',
1397
+ accountName: accountName ? [accountName] : [],
1398
+ brandName: accountName ? [accountName] : [],
1430
1399
  };
1431
1400
  }
1432
1401
 
@@ -2372,52 +2341,10 @@ export class Templates extends React.Component { // eslint-disable-line react/pr
2372
2341
  image = {},
2373
2342
  button = {},
2374
2343
  video = {},
2375
- type = '',
2376
- cards = [],
2377
2344
  } = {},
2378
2345
  } = {},
2379
2346
  } = template.versions.base;
2380
- const isViberCarousel = type === VIBER_MEDIA_TYPES.CAROUSEL;
2381
2347
  templateData.content = text;
2382
- if (isViberCarousel) {
2383
- const previewCards = Array.isArray(cards) ? cards.slice(0, 1) : [];
2384
- templateData.className = 'viber-carousel-static';
2385
- templateData.content = (
2386
- <CapRow className="viber-carousel-static-content">
2387
- <CapRow className="viber-carousel-static-message-box">
2388
- <CapLabel type="label1" className="viber-carousel-static-message">
2389
- {text}
2390
- </CapLabel>
2391
- </CapRow>
2392
- <CapRow className="viber-carousel-static-cards">
2393
- {previewCards.map((card, cardIdx) => (
2394
- <CapRow className="viber-carousel-static-card" key={`viber-static-card-${cardIdx}`}>
2395
- {card?.mediaUrl ? (
2396
- <CapImage src={card.mediaUrl} className="viber-carousel-static-image" />
2397
- ) : (
2398
- <CapRow className="viber-carousel-static-image-placeholder" />
2399
- )}
2400
- <CapLabel type="label1" className="viber-carousel-static-text">
2401
- {card?.text}
2402
- </CapLabel>
2403
- <CapRow className="viber-carousel-static-buttons">
2404
- {(Array.isArray(card?.buttons) && card.buttons.length ? card.buttons : [{}, {}]).slice(0, 2).map((carouselButton, btnIdx) => (
2405
- <CapLabel
2406
- key={`viber-static-btn-${cardIdx}-${btnIdx}`}
2407
- type="label1"
2408
- className={`viber-carousel-static-button ${btnIdx === 1 ? 'viber-carousel-static-button-secondary' : ''}`}
2409
- >
2410
- {(carouselButton?.title || '').trim()}
2411
- </CapLabel>
2412
- ))}
2413
- </CapRow>
2414
- </CapRow>
2415
- ))}
2416
- </CapRow>
2417
- </CapRow>
2418
- );
2419
- break;
2420
- }
2421
2348
  if (!isEmpty(image)) {
2422
2349
  const { url = '' } = image;
2423
2350
  templateData.url = url;
@@ -47,22 +47,7 @@ export const VIBER_MEDIA_TYPES = {
47
47
  TEXT: 'TEXT',
48
48
  IMAGE: 'IMAGE',
49
49
  VIDEO: 'VIDEO',
50
- CAROUSEL: 'CAROUSEL',
51
50
  };
52
- export const VIBER_CAROUSEL_MIN_CARDS = 2;
53
- export const VIBER_CAROUSEL_MAX_CARDS = 5;
54
- export const VIBER_CAROUSEL_MAX_BUTTONS = 2;
55
- export const VIBER_CAROUSEL_CARD_TITLE_MIN_LENGTH = 2;
56
- export const VIBER_CAROUSEL_CARD_TITLE_MAX_LENGTH = 38;
57
- export const VIBER_CAROUSEL_FIRST_BUTTON_TITLE_MAX_LENGTH = 10;
58
- export const VIBER_CAROUSEL_SECOND_BUTTON_TITLE_MAX_LENGTH = 12;
59
- export const VIBER_CAROUSEL_BUTTON_URL_MAX_LENGTH = 1000;
60
- /** Recommended / validated carousel image height in pixels (passed to image upload). */
61
- export const VIBER_CAROUSEL_IMG_HEIGHT = 600;
62
- /** Recommended / validated carousel image width in pixels (passed to image upload). */
63
- export const VIBER_CAROUSEL_IMG_WIDTH = 696;
64
- /** Maximum carousel image file size (bytes). 10_000_000 ≈ 10 MB (decimal). */
65
- export const VIBER_CAROUSEL_IMG_SIZE = 10000000;
66
51
  export const ALLOWED_IMAGE_EXTENSIONS_REGEX_VIBER = /\.(jpe?g|png)$/i;
67
52
  export const ALLOWED_EXTENSIONS_VIDEO_REGEX_VIBER = /\.(mp4|3gp|m4v|mov)$/i;
68
53
  export const NONE = 'NONE';
@@ -84,10 +69,6 @@ export const mediaRadioOptions = [
84
69
  value: VIBER_MEDIA_TYPES.VIDEO,
85
70
  label: <FormattedMessage {...messages.mediaVideo} />,
86
71
  },
87
- {
88
- value: VIBER_MEDIA_TYPES.CAROUSEL,
89
- label: <FormattedMessage {...messages.mediaCarousel} />,
90
- },
91
72
  ];
92
73
 
93
74
  export const buttonRadioOptions = [