@nyaruka/temba-components 0.41.7 → 0.42.0

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 (119) hide show
  1. package/CHANGELOG.md +9 -0
  2. package/demo/index.html +22 -0
  3. package/dist/{973a6cc2.js → f3b7c2f1.js} +397 -157
  4. package/dist/index.js +397 -157
  5. package/dist/sw.js +1 -1
  6. package/dist/sw.js.map +1 -1
  7. package/dist/templates/components-body.html +1 -1
  8. package/dist/templates/components-head.html +1 -1
  9. package/out-tsc/src/button/Button.js +3 -3
  10. package/out-tsc/src/button/Button.js.map +1 -1
  11. package/out-tsc/src/charcount/CharCount.js +5 -4
  12. package/out-tsc/src/charcount/CharCount.js.map +1 -1
  13. package/out-tsc/src/completion/Completion.js +5 -0
  14. package/out-tsc/src/completion/Completion.js.map +1 -1
  15. package/out-tsc/src/compose/Compose.js +531 -0
  16. package/out-tsc/src/compose/Compose.js.map +1 -0
  17. package/out-tsc/src/contacts/ContactChat.js +73 -69
  18. package/out-tsc/src/contacts/ContactChat.js.map +1 -1
  19. package/out-tsc/src/contacts/ContactHistory.js +1 -1
  20. package/out-tsc/src/contacts/ContactHistory.js.map +1 -1
  21. package/out-tsc/src/contacts/events.js +1 -0
  22. package/out-tsc/src/contacts/events.js.map +1 -1
  23. package/out-tsc/src/interfaces.js +2 -0
  24. package/out-tsc/src/interfaces.js.map +1 -1
  25. package/out-tsc/src/textinput/TextInput.js +3 -1
  26. package/out-tsc/src/textinput/TextInput.js.map +1 -1
  27. package/out-tsc/src/utils/index.js +10 -0
  28. package/out-tsc/src/utils/index.js.map +1 -1
  29. package/out-tsc/src/vectoricon/index.js +1 -0
  30. package/out-tsc/src/vectoricon/index.js.map +1 -1
  31. package/out-tsc/temba-modules.js +2 -0
  32. package/out-tsc/temba-modules.js.map +1 -1
  33. package/out-tsc/test/temba-compose.test.js +432 -0
  34. package/out-tsc/test/temba-compose.test.js.map +1 -0
  35. package/out-tsc/test/temba-contact-chat.test.js +219 -47
  36. package/out-tsc/test/temba-contact-chat.test.js.map +1 -1
  37. package/out-tsc/test/temba-contact-history.test.js +7 -3
  38. package/out-tsc/test/temba-contact-history.test.js.map +1 -1
  39. package/out-tsc/test/utils.test.js +12 -8
  40. package/out-tsc/test/utils.test.js.map +1 -1
  41. package/package.json +1 -1
  42. package/screenshots/truth/compose/attachments-and-send-button.png +0 -0
  43. package/screenshots/truth/compose/attachments-no-send-button.png +0 -0
  44. package/screenshots/truth/compose/attachments-with-all-files-and-click-send.png +0 -0
  45. package/screenshots/truth/compose/attachments-with-all-files.png +0 -0
  46. package/screenshots/truth/compose/attachments-with-failure-files.png +0 -0
  47. package/screenshots/truth/compose/attachments-with-success-files-and-click-send.png +0 -0
  48. package/screenshots/truth/compose/attachments-with-success-files.png +0 -0
  49. package/screenshots/truth/compose/chatbox-attachments-counter-and-send-button.png +0 -0
  50. package/screenshots/truth/compose/chatbox-attachments-counter-no-send-button.png +0 -0
  51. package/screenshots/truth/compose/chatbox-attachments-no-counter-and-send-button.png +0 -0
  52. package/screenshots/truth/compose/chatbox-attachments-no-counter-no-send-button.png +0 -0
  53. package/screenshots/truth/compose/chatbox-counter-and-send-button.png +0 -0
  54. package/screenshots/truth/compose/chatbox-counter-no-send-button.png +0 -0
  55. package/screenshots/truth/compose/chatbox-no-counter-and-send-button.png +0 -0
  56. package/screenshots/truth/compose/chatbox-no-counter-no-send-button.png +0 -0
  57. package/screenshots/truth/compose/chatbox-no-text-attachments-with-all-files-and-click-send.png +0 -0
  58. package/screenshots/truth/compose/chatbox-no-text-attachments-with-all-files.png +0 -0
  59. package/screenshots/truth/compose/chatbox-no-text-attachments-with-failure-files.png +0 -0
  60. package/screenshots/truth/compose/chatbox-no-text-attachments-with-success-files-and-click-send.png +0 -0
  61. package/screenshots/truth/compose/chatbox-no-text-attachments-with-success-files.png +0 -0
  62. package/screenshots/truth/compose/chatbox-with-text-and-click-send.png +0 -0
  63. package/screenshots/truth/compose/chatbox-with-text-and-hit-enter.png +0 -0
  64. package/screenshots/truth/compose/chatbox-with-text-and-spaces.png +0 -0
  65. package/screenshots/truth/compose/chatbox-with-text-and-url.png +0 -0
  66. package/screenshots/truth/compose/chatbox-with-text-attachments-no-files-and-click-send.png +0 -0
  67. package/screenshots/truth/compose/chatbox-with-text-attachments-no-files-and-hit-enter.png +0 -0
  68. package/screenshots/truth/compose/chatbox-with-text-attachments-no-files.png +0 -0
  69. package/screenshots/truth/compose/chatbox-with-text-attachments-with-all-files-and-click-send.png +0 -0
  70. package/screenshots/truth/compose/chatbox-with-text-attachments-with-all-files-and-hit-enter.png +0 -0
  71. package/screenshots/truth/compose/chatbox-with-text-attachments-with-all-files.png +0 -0
  72. package/screenshots/truth/compose/chatbox-with-text-attachments-with-failure-files.png +0 -0
  73. package/screenshots/truth/compose/chatbox-with-text-attachments-with-success-files-and-click-send.png +0 -0
  74. package/screenshots/truth/compose/chatbox-with-text-attachments-with-success-files-and-hit-enter.png +0 -0
  75. package/screenshots/truth/compose/chatbox-with-text-attachments-with-success-files.png +0 -0
  76. package/screenshots/truth/compose/chatbox-with-text-no-spaces.png +0 -0
  77. package/screenshots/truth/compose/chatbox-with-text.png +0 -0
  78. package/screenshots/truth/contacts/badges.png +0 -0
  79. package/screenshots/truth/contacts/compose-attachments-no-text-failure.png +0 -0
  80. package/screenshots/truth/contacts/compose-attachments-no-text-success.png +0 -0
  81. package/screenshots/truth/contacts/compose-text-and-attachments-failure-attachments.png +0 -0
  82. package/screenshots/truth/contacts/compose-text-and-attachments-failure-generic.png +0 -0
  83. package/screenshots/truth/contacts/compose-text-and-attachments-failure-text-and-attachments.png +0 -0
  84. package/screenshots/truth/contacts/compose-text-and-attachments-failure-text.png +0 -0
  85. package/screenshots/truth/contacts/compose-text-and-attachments-success.png +0 -0
  86. package/screenshots/truth/contacts/compose-text-no-attachments-failure.png +0 -0
  87. package/screenshots/truth/contacts/compose-text-no-attachments-success.png +0 -0
  88. package/screenshots/truth/contacts/contact-active-default.png +0 -0
  89. package/screenshots/truth/contacts/contact-active-show-chatbox.png +0 -0
  90. package/screenshots/truth/contacts/contact-active-ticket-closed-show-reopen-button.png +0 -0
  91. package/screenshots/truth/contacts/contact-active-ticket-open-show-chatbox.png +0 -0
  92. package/screenshots/truth/contacts/contact-archived-hide-chatbox.png +0 -0
  93. package/screenshots/truth/contacts/contact-archived-ticket-closed-hide-chatbox.png +0 -0
  94. package/screenshots/truth/contacts/contact-blocked-hide-chatbox.png +0 -0
  95. package/screenshots/truth/contacts/contact-stopped-hide-chatbox.png +0 -0
  96. package/screenshots/truth/contacts/history.png +0 -0
  97. package/screenshots/truth/counter/summary.png +0 -0
  98. package/screenshots/truth/counter/text.png +0 -0
  99. package/screenshots/truth/counter/unicode-variables.png +0 -0
  100. package/screenshots/truth/counter/unicode.png +0 -0
  101. package/screenshots/truth/counter/variable.png +0 -0
  102. package/src/button/Button.ts +3 -3
  103. package/src/charcount/CharCount.ts +5 -4
  104. package/src/completion/Completion.ts +4 -0
  105. package/src/compose/Compose.ts +584 -0
  106. package/src/contacts/ContactChat.ts +75 -78
  107. package/src/contacts/ContactHistory.ts +1 -1
  108. package/src/contacts/events.ts +1 -0
  109. package/src/interfaces.ts +2 -0
  110. package/src/textinput/TextInput.ts +3 -1
  111. package/src/utils/index.ts +12 -0
  112. package/src/vectoricon/index.ts +1 -0
  113. package/static/css/temba-components.css +6 -0
  114. package/temba-modules.ts +2 -0
  115. package/test/temba-compose.test.ts +633 -0
  116. package/test/temba-contact-chat.test.ts +354 -110
  117. package/test/temba-contact-history.test.ts +10 -5
  118. package/test/utils.test.ts +24 -9
  119. package/screenshots/truth/contacts/history-expanded.png +0 -0
@@ -0,0 +1,531 @@
1
+ import { __decorate } from "tslib";
2
+ import { html, css } from 'lit';
3
+ import { FormElement } from '../FormElement';
4
+ import { property } from 'lit/decorators.js';
5
+ import { Icon } from '../vectoricon';
6
+ import { CustomEventType } from '../interfaces';
7
+ import { formatFileSize, formatFileType, getClasses, postFormData, truncate, } from '../utils';
8
+ export class Compose extends FormElement {
9
+ static get styles() {
10
+ return css `
11
+ .container {
12
+ display: flex;
13
+ flex-direction: column;
14
+ justify-content: space-between;
15
+ position: relative;
16
+
17
+ border-radius: var(--curvature-widget);
18
+ background: var(--color-widget-bg);
19
+ border: 1px solid var(--color-widget-border);
20
+ transition: all ease-in-out var(--transition-speed);
21
+ box-shadow: var(--widget-box-shadow);
22
+ caret-color: var(--input-caret);
23
+ padding: var(--temba-textinput-padding);
24
+ }
25
+ .container:focus-within {
26
+ border-color: var(--color-focus);
27
+ background: var(--color-widget-bg-focused);
28
+ box-shadow: var(--widget-box-shadow-focused);
29
+ }
30
+
31
+ .drop-mask {
32
+ opacity: 0;
33
+ pointer-events: none;
34
+ position: absolute;
35
+ z-index: 1;
36
+ height: 100%;
37
+ width: 100%;
38
+ bottom: 0;
39
+ right: 0;
40
+ background: rgba(210, 243, 184, 0.8);
41
+ border-radius: var(--curvature-widget);
42
+ margin: -0.5em;
43
+ padding: 0.5em;
44
+ transition: opacity ease-in-out var(--transition-speed);
45
+ display: flex;
46
+ align-items: center;
47
+ text-align: center;
48
+ }
49
+
50
+ .highlight .drop-mask {
51
+ opacity: 1;
52
+ }
53
+
54
+ .drop-mask > div {
55
+ margin: auto;
56
+ border-radius: var(--curvature-widget);
57
+ font-weight: 400;
58
+ color: rgba(0, 0, 0, 0.5);
59
+ }
60
+
61
+ .items {
62
+ }
63
+
64
+ temba-completion {
65
+ margin-left: 0.3em;
66
+ margin-top: 0.3em;
67
+ --color-widget-border: none;
68
+ --curvature-widget: none;
69
+ --widget-box-shadow: none;
70
+ --widget-box-shadow-focused: none;
71
+ --temba-textinput-padding: 0;
72
+ }
73
+
74
+ .attachments {
75
+ display: flex;
76
+ flex-direction: column;
77
+ }
78
+ .attachments-list {
79
+ display: flex;
80
+ flex-direction: row;
81
+ flex-wrap: wrap;
82
+ }
83
+ .attachment-item {
84
+ background: rgba(100, 100, 100, 0.1);
85
+ border-radius: 2px;
86
+ margin: 0.3em;
87
+ display: flex;
88
+ color: var(--color-widget-text);
89
+ }
90
+ .attachment-item.error {
91
+ background: rgba(250, 0, 0, 0.1);
92
+ color: rgba(250, 0, 0, 0.75);
93
+ }
94
+ .remove-item {
95
+ cursor: pointer !important;
96
+ padding: 3px 6px;
97
+ border-right: 1px solid rgba(100, 100, 100, 0.2);
98
+ margin-top: 1px;
99
+ background: rgba(100, 100, 100, 0.05);
100
+ }
101
+
102
+ .remove-item:hover {
103
+ background: rgba(100, 100, 100, 0.1);
104
+ }
105
+
106
+ .remove-item.error:hover {
107
+ background: rgba(250, 0, 0, 0.1);
108
+ }
109
+
110
+ .remove-item.error {
111
+ background: rgba(250, 0, 0, 0.05);
112
+ color: rgba(250, 0, 0, 0.75);
113
+ }
114
+ .attachment-name {
115
+ align-self: center;
116
+ font-size: 12px;
117
+ padding: 2px 8px;
118
+ }
119
+
120
+ .actions {
121
+ display: flex;
122
+ justify-content: space-between;
123
+ align-items: center;
124
+ margin-left: 0.25em;
125
+ padding: 0.2em;
126
+ }
127
+
128
+ #upload-files {
129
+ display: none;
130
+ }
131
+ .upload-label {
132
+ display: flex;
133
+ align-items: center;
134
+ }
135
+ .upload-icon {
136
+ color: rgb(102, 102, 102);
137
+ }
138
+ .actions-right {
139
+ display: flex;
140
+ align-items: center;
141
+ }
142
+ temba-charcount {
143
+ margin-right: 5px;
144
+ overflow: hidden;
145
+ --temba-charcount-counts-margin-top: 0px;
146
+ --temba-charcount-summary-margin-top: 0px;
147
+ --temba-charcount-summary-position: fixed;
148
+ --temba-charcount-summary-right: 105px;
149
+ --temba-charcount-summary-bottom: 105px;
150
+ }
151
+ temba-button {
152
+ --button-y: 1px;
153
+ --button-x: 12px;
154
+ }
155
+ .send-error {
156
+ color: rgba(250, 0, 0, 0.75);
157
+ font-size: var(--help-text-size);
158
+ }
159
+ `;
160
+ }
161
+ constructor() {
162
+ super();
163
+ this.currentChat = '';
164
+ this.accept = ''; //e.g. ".xls,.xlsx"
165
+ this.endpoint = '/msgmedia/upload/';
166
+ // values = valid and uploaded attachments
167
+ // errorValues = invalid and not-uploaded attachments
168
+ this.errorValues = [];
169
+ this.buttonName = 'Send';
170
+ this.buttonDisabled = true;
171
+ this.buttonError = '';
172
+ }
173
+ updated(changes) {
174
+ super.updated(changes);
175
+ if (changes.has('currentChat') ||
176
+ changes.has('values') ||
177
+ changes.has('buttonError')) {
178
+ this.toggleButton();
179
+ }
180
+ }
181
+ firstUpdated() {
182
+ const completion = this.shadowRoot.querySelector('temba-completion');
183
+ if (completion) {
184
+ window.setTimeout(() => {
185
+ completion.click();
186
+ }, 0);
187
+ }
188
+ }
189
+ reset() {
190
+ this.currentChat = '';
191
+ this.values = [];
192
+ this.errorValues = [];
193
+ this.buttonError = '';
194
+ }
195
+ handleChatboxChange(evt) {
196
+ const completion = evt.target;
197
+ const textInput = completion.textInputElement;
198
+ this.currentChat = textInput.value;
199
+ this.preventDefaults(evt);
200
+ }
201
+ handleDragEnter(evt) {
202
+ this.highlight(evt);
203
+ }
204
+ handleDragOver(evt) {
205
+ this.highlight(evt);
206
+ }
207
+ handleDragLeave(evt) {
208
+ this.unhighlight(evt);
209
+ }
210
+ handleDrop(evt) {
211
+ this.unhighlight(evt);
212
+ const dt = evt.dataTransfer;
213
+ if (dt) {
214
+ const files = dt.files;
215
+ this.uploadFiles(files);
216
+ }
217
+ }
218
+ preventDefaults(evt) {
219
+ evt.preventDefault();
220
+ evt.stopPropagation();
221
+ }
222
+ highlight(evt) {
223
+ this.pendingDrop = true;
224
+ this.preventDefaults(evt);
225
+ }
226
+ unhighlight(evt) {
227
+ this.pendingDrop = false;
228
+ this.preventDefaults(evt);
229
+ }
230
+ handleAddAttachments() {
231
+ this.dispatchEvent(new Event('change'));
232
+ }
233
+ handleUploadFileChanged(evt) {
234
+ const target = evt.target;
235
+ const files = target.files;
236
+ this.uploadFiles(files);
237
+ }
238
+ uploadFiles(files) {
239
+ let filesToUpload = [];
240
+ if (this.values && this.values.length > 0) {
241
+ //remove duplicate files that have already been uploaded
242
+ filesToUpload = [...files].filter(file => {
243
+ const index = this.values.findIndex(value => value.name === file.name && value.size === file.size);
244
+ if (index === -1) {
245
+ return file;
246
+ }
247
+ });
248
+ }
249
+ else {
250
+ filesToUpload = [...files];
251
+ }
252
+ filesToUpload.map(fileToUpload => {
253
+ this.uploadFile(fileToUpload);
254
+ });
255
+ }
256
+ uploadFile(file) {
257
+ this.uploading = true;
258
+ const url = this.endpoint;
259
+ const payload = new FormData();
260
+ payload.append('file', file);
261
+ postFormData(url, payload)
262
+ .then((response) => {
263
+ if (response.json.error) {
264
+ this.addErrorValue(file, response.json.error);
265
+ }
266
+ else {
267
+ const attachment = response.json;
268
+ if (attachment) {
269
+ this.addValue(attachment);
270
+ this.fireCustomEvent(CustomEventType.AttachmentAdded, attachment);
271
+ }
272
+ }
273
+ })
274
+ .catch((error) => {
275
+ console.log(error);
276
+ this.addErrorValue(file, error);
277
+ })
278
+ .finally(() => {
279
+ this.uploading = false;
280
+ });
281
+ }
282
+ addErrorValue(file, error) {
283
+ const errorValue = {
284
+ uuid: Math.random().toString(36).slice(2, 6),
285
+ content_type: file.type,
286
+ type: file.type,
287
+ name: file.name,
288
+ url: file.name,
289
+ size: file.size,
290
+ error: error,
291
+ };
292
+ this.errorValues.push(errorValue);
293
+ this.requestUpdate('errorValues');
294
+ }
295
+ removeErrorValue(valueToRemove) {
296
+ this.errorValues = this.errorValues.filter((value) => value !== valueToRemove);
297
+ this.requestUpdate('errorValues');
298
+ }
299
+ handleRemoveAttachment(evt) {
300
+ const target = evt.target;
301
+ const attachment = this.values.find(({ uuid }) => uuid === target.id);
302
+ if (attachment) {
303
+ this.removeValue(attachment);
304
+ this.fireCustomEvent(CustomEventType.AttachmentRemoved, attachment);
305
+ }
306
+ const errorAttachment = this.errorValues.find(({ uuid }) => uuid === target.id);
307
+ if (errorAttachment) {
308
+ this.removeErrorValue(errorAttachment);
309
+ this.fireCustomEvent(CustomEventType.AttachmentRemoved, attachment);
310
+ }
311
+ }
312
+ toggleButton() {
313
+ if (this.button) {
314
+ if (this.buttonError && this.buttonError.length > 0) {
315
+ this.buttonDisabled = true;
316
+ }
317
+ else {
318
+ const chatboxEmpty = this.currentChat.trim().length === 0;
319
+ const attachmentsEmpty = this.values.length === 0;
320
+ if (this.chatbox && this.attachments) {
321
+ this.buttonDisabled = chatboxEmpty && attachmentsEmpty;
322
+ }
323
+ else if (this.chatbox) {
324
+ this.buttonDisabled = chatboxEmpty;
325
+ }
326
+ else if (this.attachments) {
327
+ this.buttonDisabled = attachmentsEmpty;
328
+ }
329
+ else {
330
+ this.buttonDisabled = true;
331
+ }
332
+ }
333
+ }
334
+ }
335
+ handleSendClick() {
336
+ this.handleSend();
337
+ }
338
+ handleSendEnter(evt) {
339
+ if (evt.key === 'Enter' && !evt.shiftKey) {
340
+ const chat = evt.target;
341
+ if (!chat.hasVisibleOptions()) {
342
+ this.handleSend();
343
+ }
344
+ }
345
+ }
346
+ handleSend() {
347
+ if (!this.buttonDisabled) {
348
+ this.buttonDisabled = true;
349
+ const name = this.buttonName;
350
+ this.fireCustomEvent(CustomEventType.ButtonClicked, { name });
351
+ }
352
+ }
353
+ handleSendBlur() {
354
+ if (this.buttonError.length > 0) {
355
+ this.buttonError = '';
356
+ }
357
+ }
358
+ render() {
359
+ return html `
360
+ <div
361
+ class=${getClasses({ container: true, highlight: this.pendingDrop })}
362
+ @dragenter="${this.handleDragEnter}"
363
+ @dragover="${this.handleDragOver}"
364
+ @dragleave="${this.handleDragLeave}"
365
+ @drop="${this.handleDrop}"
366
+ >
367
+ <div class="drop-mask"><div>Upload Attachment</div></div>
368
+
369
+ ${this.chatbox
370
+ ? html `<div class="items chatbox">${this.getChatbox()}</div>`
371
+ : null}
372
+ ${this.attachments
373
+ ? html `<div class="items attachments">${this.getAttachments()}</div>`
374
+ : null}
375
+ <div class="items actions">${this.getActions()}</div>
376
+ </div>
377
+ `;
378
+ }
379
+ getChatbox() {
380
+ return html ` <temba-completion
381
+ value=${this.currentChat}
382
+ gsm
383
+ textarea
384
+ autogrow
385
+ @change=${this.handleChatboxChange}
386
+ @keydown=${this.handleSendEnter}
387
+ placeholder="Write something here"
388
+ @blur=${this.handleSendBlur}
389
+ >
390
+ </temba-completion>`;
391
+ }
392
+ getAttachments() {
393
+ return html `
394
+ ${(this.values && this.values.length > 0) ||
395
+ (this.errorValues && this.errorValues.length > 0)
396
+ ? html ` <div class="attachments-list">
397
+ ${this.values.map(attachment => {
398
+ return html ` <div class="attachment-item">
399
+ <div
400
+ class="remove-item"
401
+ @click="${this.handleRemoveAttachment}"
402
+ >
403
+ <temba-icon
404
+ id="${attachment.uuid}"
405
+ name="${Icon.delete_small}"
406
+ ></temba-icon>
407
+ </div>
408
+ <div class="attachment-name">
409
+ <span
410
+ title="${attachment.name} (${formatFileSize(attachment.size, 2)}) ${attachment.type}"
411
+ >${truncate(attachment.name, 25)}
412
+ (${formatFileSize(attachment.size, 0)})
413
+ ${formatFileType(attachment.type)}</span
414
+ >
415
+ </div>
416
+ </div>`;
417
+ })}
418
+ ${this.errorValues.map(errorAttachment => {
419
+ return html ` <div class="attachment-item error">
420
+ <div
421
+ class="remove-item error"
422
+ @click="${this.handleRemoveAttachment}"
423
+ >
424
+ <temba-icon
425
+ id="${errorAttachment.uuid}"
426
+ name="${Icon.delete_small}"
427
+ ></temba-icon>
428
+ </div>
429
+ <div class="attachment-name">
430
+ <span
431
+ title="${errorAttachment.name} (${formatFileSize(0, 0)}) - Attachment failed - ${errorAttachment.error}"
432
+ >${truncate(errorAttachment.name, 25)}
433
+ (${formatFileSize(0, 0)}) - Attachment failed</span
434
+ >
435
+ </div>
436
+ </div>`;
437
+ })}
438
+ </div>`
439
+ : null}
440
+ `;
441
+ }
442
+ getActions() {
443
+ return html `
444
+ <div class="actions-left">
445
+ ${this.attachments ? this.getUploader() : null}
446
+ </div>
447
+ <div class="actions-center"></div>
448
+ <div class="actions-right">
449
+ ${this.buttonError
450
+ ? html `<div class="send-error">${this.buttonError}</div>`
451
+ : null}
452
+ ${this.counter ? this.getCounter() : null}
453
+ ${this.button ? this.getButton() : null}
454
+ </div>
455
+ `;
456
+ }
457
+ getUploader() {
458
+ if (this.uploading) {
459
+ return html `<temba-loading units="3" size="12"></temba-loading>`;
460
+ }
461
+ else {
462
+ return html ` <input
463
+ type="file"
464
+ id="upload-files"
465
+ multiple
466
+ accept="${this.accept}"
467
+ @change="${this.handleUploadFileChanged}"
468
+ />
469
+ <label class="actions-left upload-label" for="upload-files">
470
+ <temba-icon
471
+ class="upload-icon"
472
+ name="${Icon.attachment}"
473
+ @click="${this.handleAddAttachments}"
474
+ clickable
475
+ ></temba-icon>
476
+ </label>`;
477
+ }
478
+ }
479
+ getCounter() {
480
+ return html `<temba-charcount text="${this.currentChat}"></temba-charcount>`;
481
+ }
482
+ getButton() {
483
+ return html ` <temba-button
484
+ id="send-button"
485
+ name=${this.buttonName}
486
+ @click=${this.handleSendClick}
487
+ ?disabled=${this.buttonDisabled}
488
+ @blur=${this.handleSendBlur}
489
+ ></temba-button>`;
490
+ }
491
+ }
492
+ __decorate([
493
+ property({ type: Boolean })
494
+ ], Compose.prototype, "chatbox", void 0);
495
+ __decorate([
496
+ property({ type: Boolean })
497
+ ], Compose.prototype, "attachments", void 0);
498
+ __decorate([
499
+ property({ type: Boolean })
500
+ ], Compose.prototype, "counter", void 0);
501
+ __decorate([
502
+ property({ type: Boolean })
503
+ ], Compose.prototype, "pendingDrop", void 0);
504
+ __decorate([
505
+ property({ type: Boolean })
506
+ ], Compose.prototype, "button", void 0);
507
+ __decorate([
508
+ property({ type: String, attribute: false })
509
+ ], Compose.prototype, "currentChat", void 0);
510
+ __decorate([
511
+ property({ type: String })
512
+ ], Compose.prototype, "accept", void 0);
513
+ __decorate([
514
+ property({ type: String, attribute: false })
515
+ ], Compose.prototype, "endpoint", void 0);
516
+ __decorate([
517
+ property({ type: Boolean, attribute: false })
518
+ ], Compose.prototype, "uploading", void 0);
519
+ __decorate([
520
+ property({ type: Array, attribute: false })
521
+ ], Compose.prototype, "errorValues", void 0);
522
+ __decorate([
523
+ property({ type: String })
524
+ ], Compose.prototype, "buttonName", void 0);
525
+ __decorate([
526
+ property({ type: Boolean, attribute: false })
527
+ ], Compose.prototype, "buttonDisabled", void 0);
528
+ __decorate([
529
+ property({ type: String, attribute: false })
530
+ ], Compose.prototype, "buttonError", void 0);
531
+ //# sourceMappingURL=Compose.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Compose.js","sourceRoot":"","sources":["../../../src/compose/Compose.ts"],"names":[],"mappings":"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","sourcesContent":["import { TemplateResult, html, css } from 'lit';\nimport { FormElement } from '../FormElement';\nimport { property } from 'lit/decorators.js';\nimport { Icon } from '../vectoricon';\nimport { CustomEventType } from '../interfaces';\nimport {\n formatFileSize,\n formatFileType,\n getClasses,\n postFormData,\n truncate,\n WebResponse,\n} from '../utils';\nimport { Completion } from '../completion/Completion';\n\nexport interface Attachment {\n uuid: string;\n content_type: string;\n type: string; //deprecated\n url: string;\n name: string;\n size: number;\n error: string;\n}\n\nexport class Compose extends FormElement {\n static get styles() {\n return css`\n .container {\n display: flex;\n flex-direction: column;\n justify-content: space-between;\n position: relative;\n\n border-radius: var(--curvature-widget);\n background: var(--color-widget-bg);\n border: 1px solid var(--color-widget-border);\n transition: all ease-in-out var(--transition-speed);\n box-shadow: var(--widget-box-shadow);\n caret-color: var(--input-caret);\n padding: var(--temba-textinput-padding);\n }\n .container:focus-within {\n border-color: var(--color-focus);\n background: var(--color-widget-bg-focused);\n box-shadow: var(--widget-box-shadow-focused);\n }\n\n .drop-mask {\n opacity: 0;\n pointer-events: none;\n position: absolute;\n z-index: 1;\n height: 100%;\n width: 100%;\n bottom: 0;\n right: 0;\n background: rgba(210, 243, 184, 0.8);\n border-radius: var(--curvature-widget);\n margin: -0.5em;\n padding: 0.5em;\n transition: opacity ease-in-out var(--transition-speed);\n display: flex;\n align-items: center;\n text-align: center;\n }\n\n .highlight .drop-mask {\n opacity: 1;\n }\n\n .drop-mask > div {\n margin: auto;\n border-radius: var(--curvature-widget);\n font-weight: 400;\n color: rgba(0, 0, 0, 0.5);\n }\n\n .items {\n }\n\n temba-completion {\n margin-left: 0.3em;\n margin-top: 0.3em;\n --color-widget-border: none;\n --curvature-widget: none;\n --widget-box-shadow: none;\n --widget-box-shadow-focused: none;\n --temba-textinput-padding: 0;\n }\n\n .attachments {\n display: flex;\n flex-direction: column;\n }\n .attachments-list {\n display: flex;\n flex-direction: row;\n flex-wrap: wrap;\n }\n .attachment-item {\n background: rgba(100, 100, 100, 0.1);\n border-radius: 2px;\n margin: 0.3em;\n display: flex;\n color: var(--color-widget-text);\n }\n .attachment-item.error {\n background: rgba(250, 0, 0, 0.1);\n color: rgba(250, 0, 0, 0.75);\n }\n .remove-item {\n cursor: pointer !important;\n padding: 3px 6px;\n border-right: 1px solid rgba(100, 100, 100, 0.2);\n margin-top: 1px;\n background: rgba(100, 100, 100, 0.05);\n }\n\n .remove-item:hover {\n background: rgba(100, 100, 100, 0.1);\n }\n\n .remove-item.error:hover {\n background: rgba(250, 0, 0, 0.1);\n }\n\n .remove-item.error {\n background: rgba(250, 0, 0, 0.05);\n color: rgba(250, 0, 0, 0.75);\n }\n .attachment-name {\n align-self: center;\n font-size: 12px;\n padding: 2px 8px;\n }\n\n .actions {\n display: flex;\n justify-content: space-between;\n align-items: center;\n margin-left: 0.25em;\n padding: 0.2em;\n }\n\n #upload-files {\n display: none;\n }\n .upload-label {\n display: flex;\n align-items: center;\n }\n .upload-icon {\n color: rgb(102, 102, 102);\n }\n .actions-right {\n display: flex;\n align-items: center;\n }\n temba-charcount {\n margin-right: 5px;\n overflow: hidden;\n --temba-charcount-counts-margin-top: 0px;\n --temba-charcount-summary-margin-top: 0px;\n --temba-charcount-summary-position: fixed;\n --temba-charcount-summary-right: 105px;\n --temba-charcount-summary-bottom: 105px;\n }\n temba-button {\n --button-y: 1px;\n --button-x: 12px;\n }\n .send-error {\n color: rgba(250, 0, 0, 0.75);\n font-size: var(--help-text-size);\n }\n `;\n }\n\n @property({ type: Boolean })\n chatbox: boolean;\n\n @property({ type: Boolean })\n attachments: boolean;\n\n @property({ type: Boolean })\n counter: boolean;\n\n @property({ type: Boolean })\n pendingDrop: boolean;\n\n @property({ type: Boolean })\n button: boolean;\n\n @property({ type: String, attribute: false })\n currentChat = '';\n\n @property({ type: String })\n accept = ''; //e.g. \".xls,.xlsx\"\n\n @property({ type: String, attribute: false })\n endpoint = '/msgmedia/upload/';\n\n @property({ type: Boolean, attribute: false })\n uploading: boolean;\n\n // values = valid and uploaded attachments\n // errorValues = invalid and not-uploaded attachments\n @property({ type: Array, attribute: false })\n errorValues: Attachment[] = [];\n\n @property({ type: String })\n buttonName = 'Send';\n\n @property({ type: Boolean, attribute: false })\n buttonDisabled = true;\n\n @property({ type: String, attribute: false })\n buttonError = '';\n\n public constructor() {\n super();\n }\n\n public updated(changes: Map<string, any>): void {\n super.updated(changes);\n\n if (\n changes.has('currentChat') ||\n changes.has('values') ||\n changes.has('buttonError')\n ) {\n this.toggleButton();\n }\n }\n\n firstUpdated(): void {\n const completion = this.shadowRoot.querySelector(\n 'temba-completion'\n ) as Completion;\n if (completion) {\n window.setTimeout(() => {\n completion.click();\n }, 0);\n }\n }\n\n public reset(): void {\n this.currentChat = '';\n this.values = [];\n this.errorValues = [];\n this.buttonError = '';\n }\n\n private handleChatboxChange(evt: Event) {\n const completion = evt.target as Completion;\n const textInput = completion.textInputElement;\n this.currentChat = textInput.value;\n this.preventDefaults(evt);\n }\n\n private handleDragEnter(evt: DragEvent): void {\n this.highlight(evt);\n }\n\n private handleDragOver(evt: DragEvent): void {\n this.highlight(evt);\n }\n\n private handleDragLeave(evt: DragEvent): void {\n this.unhighlight(evt);\n }\n\n private handleDrop(evt: DragEvent): void {\n this.unhighlight(evt);\n\n const dt = evt.dataTransfer;\n if (dt) {\n const files = dt.files;\n this.uploadFiles(files);\n }\n }\n\n private preventDefaults(evt: Event): void {\n evt.preventDefault();\n evt.stopPropagation();\n }\n\n private highlight(evt: DragEvent): void {\n this.pendingDrop = true;\n this.preventDefaults(evt);\n }\n\n private unhighlight(evt: DragEvent): void {\n this.pendingDrop = false;\n this.preventDefaults(evt);\n }\n\n private handleAddAttachments(): void {\n this.dispatchEvent(new Event('change'));\n }\n\n private handleUploadFileChanged(evt: Event): void {\n const target = evt.target as HTMLInputElement;\n const files = target.files;\n this.uploadFiles(files);\n }\n\n public uploadFiles(files: FileList): void {\n let filesToUpload = [];\n if (this.values && this.values.length > 0) {\n //remove duplicate files that have already been uploaded\n filesToUpload = [...files].filter(file => {\n const index = this.values.findIndex(\n value => value.name === file.name && value.size === file.size\n );\n if (index === -1) {\n return file;\n }\n });\n } else {\n filesToUpload = [...files];\n }\n filesToUpload.map(fileToUpload => {\n this.uploadFile(fileToUpload);\n });\n }\n\n private uploadFile(file: File): void {\n this.uploading = true;\n\n const url = this.endpoint;\n const payload = new FormData();\n payload.append('file', file);\n postFormData(url, payload)\n .then((response: WebResponse) => {\n if (response.json.error) {\n this.addErrorValue(file, response.json.error);\n } else {\n const attachment = response.json as Attachment;\n if (attachment) {\n this.addValue(attachment);\n this.fireCustomEvent(CustomEventType.AttachmentAdded, attachment);\n }\n }\n })\n .catch((error: string) => {\n console.log(error);\n this.addErrorValue(file, error);\n })\n .finally(() => {\n this.uploading = false;\n });\n }\n\n private addErrorValue(file: File, error: string) {\n const errorValue = {\n uuid: Math.random().toString(36).slice(2, 6),\n content_type: file.type,\n type: file.type,\n name: file.name,\n url: file.name,\n size: file.size,\n error: error,\n };\n this.errorValues.push(errorValue);\n this.requestUpdate('errorValues');\n }\n public removeErrorValue(valueToRemove: any) {\n this.errorValues = this.errorValues.filter(\n (value: any) => value !== valueToRemove\n );\n this.requestUpdate('errorValues');\n }\n\n private handleRemoveAttachment(evt: Event): void {\n const target = evt.target as HTMLDivElement;\n\n const attachment = this.values.find(({ uuid }) => uuid === target.id);\n if (attachment) {\n this.removeValue(attachment);\n this.fireCustomEvent(CustomEventType.AttachmentRemoved, attachment);\n }\n const errorAttachment = this.errorValues.find(\n ({ uuid }) => uuid === target.id\n );\n if (errorAttachment) {\n this.removeErrorValue(errorAttachment);\n this.fireCustomEvent(CustomEventType.AttachmentRemoved, attachment);\n }\n }\n\n public toggleButton() {\n if (this.button) {\n if (this.buttonError && this.buttonError.length > 0) {\n this.buttonDisabled = true;\n } else {\n const chatboxEmpty = this.currentChat.trim().length === 0;\n const attachmentsEmpty = this.values.length === 0;\n if (this.chatbox && this.attachments) {\n this.buttonDisabled = chatboxEmpty && attachmentsEmpty;\n } else if (this.chatbox) {\n this.buttonDisabled = chatboxEmpty;\n } else if (this.attachments) {\n this.buttonDisabled = attachmentsEmpty;\n } else {\n this.buttonDisabled = true;\n }\n }\n }\n }\n\n private handleSendClick() {\n this.handleSend();\n }\n\n private handleSendEnter(evt: KeyboardEvent) {\n if (evt.key === 'Enter' && !evt.shiftKey) {\n const chat = evt.target as Completion;\n if (!chat.hasVisibleOptions()) {\n this.handleSend();\n }\n }\n }\n\n private handleSend() {\n if (!this.buttonDisabled) {\n this.buttonDisabled = true;\n const name = this.buttonName;\n this.fireCustomEvent(CustomEventType.ButtonClicked, { name });\n }\n }\n\n private handleSendBlur() {\n if (this.buttonError.length > 0) {\n this.buttonError = '';\n }\n }\n\n public render(): TemplateResult {\n return html`\n <div\n class=${getClasses({ container: true, highlight: this.pendingDrop })}\n @dragenter=\"${this.handleDragEnter}\"\n @dragover=\"${this.handleDragOver}\"\n @dragleave=\"${this.handleDragLeave}\"\n @drop=\"${this.handleDrop}\"\n >\n <div class=\"drop-mask\"><div>Upload Attachment</div></div>\n\n ${this.chatbox\n ? html`<div class=\"items chatbox\">${this.getChatbox()}</div>`\n : null}\n ${this.attachments\n ? html`<div class=\"items attachments\">${this.getAttachments()}</div>`\n : null}\n <div class=\"items actions\">${this.getActions()}</div>\n </div>\n `;\n }\n\n private getChatbox(): TemplateResult {\n return html` <temba-completion\n value=${this.currentChat}\n gsm\n textarea\n autogrow\n @change=${this.handleChatboxChange}\n @keydown=${this.handleSendEnter}\n placeholder=\"Write something here\"\n @blur=${this.handleSendBlur}\n >\n </temba-completion>`;\n }\n\n private getAttachments(): TemplateResult {\n return html`\n ${(this.values && this.values.length > 0) ||\n (this.errorValues && this.errorValues.length > 0)\n ? html` <div class=\"attachments-list\">\n ${this.values.map(attachment => {\n return html` <div class=\"attachment-item\">\n <div\n class=\"remove-item\"\n @click=\"${this.handleRemoveAttachment}\"\n >\n <temba-icon\n id=\"${attachment.uuid}\"\n name=\"${Icon.delete_small}\"\n ></temba-icon>\n </div>\n <div class=\"attachment-name\">\n <span\n title=\"${attachment.name} (${formatFileSize(\n attachment.size,\n 2\n )}) ${attachment.type}\"\n >${truncate(attachment.name, 25)}\n (${formatFileSize(attachment.size, 0)})\n ${formatFileType(attachment.type)}</span\n >\n </div>\n </div>`;\n })}\n ${this.errorValues.map(errorAttachment => {\n return html` <div class=\"attachment-item error\">\n <div\n class=\"remove-item error\"\n @click=\"${this.handleRemoveAttachment}\"\n >\n <temba-icon\n id=\"${errorAttachment.uuid}\"\n name=\"${Icon.delete_small}\"\n ></temba-icon>\n </div>\n <div class=\"attachment-name\">\n <span\n title=\"${errorAttachment.name} (${formatFileSize(\n 0,\n 0\n )}) - Attachment failed - ${errorAttachment.error}\"\n >${truncate(errorAttachment.name, 25)}\n (${formatFileSize(0, 0)}) - Attachment failed</span\n >\n </div>\n </div>`;\n })}\n </div>`\n : null}\n `;\n }\n\n private getActions(): TemplateResult {\n return html`\n <div class=\"actions-left\">\n ${this.attachments ? this.getUploader() : null}\n </div>\n <div class=\"actions-center\"></div>\n <div class=\"actions-right\">\n ${this.buttonError\n ? html`<div class=\"send-error\">${this.buttonError}</div>`\n : null}\n ${this.counter ? this.getCounter() : null}\n ${this.button ? this.getButton() : null}\n </div>\n `;\n }\n\n private getUploader(): TemplateResult {\n if (this.uploading) {\n return html`<temba-loading units=\"3\" size=\"12\"></temba-loading>`;\n } else {\n return html` <input\n type=\"file\"\n id=\"upload-files\"\n multiple\n accept=\"${this.accept}\"\n @change=\"${this.handleUploadFileChanged}\"\n />\n <label class=\"actions-left upload-label\" for=\"upload-files\">\n <temba-icon\n class=\"upload-icon\"\n name=\"${Icon.attachment}\"\n @click=\"${this.handleAddAttachments}\"\n clickable\n ></temba-icon>\n </label>`;\n }\n }\n\n private getCounter(): TemplateResult {\n return html`<temba-charcount text=\"${this.currentChat}\"></temba-charcount>`;\n }\n\n private getButton(): TemplateResult {\n return html` <temba-button\n id=\"send-button\"\n name=${this.buttonName}\n @click=${this.handleSendClick}\n ?disabled=${this.buttonDisabled}\n @blur=${this.handleSendBlur}\n ></temba-button>`;\n }\n}\n"]}