@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.
- package/CHANGELOG.md +9 -0
- package/demo/index.html +22 -0
- package/dist/{973a6cc2.js → f3b7c2f1.js} +397 -157
- package/dist/index.js +397 -157
- package/dist/sw.js +1 -1
- package/dist/sw.js.map +1 -1
- package/dist/templates/components-body.html +1 -1
- package/dist/templates/components-head.html +1 -1
- package/out-tsc/src/button/Button.js +3 -3
- package/out-tsc/src/button/Button.js.map +1 -1
- package/out-tsc/src/charcount/CharCount.js +5 -4
- package/out-tsc/src/charcount/CharCount.js.map +1 -1
- package/out-tsc/src/completion/Completion.js +5 -0
- package/out-tsc/src/completion/Completion.js.map +1 -1
- package/out-tsc/src/compose/Compose.js +531 -0
- package/out-tsc/src/compose/Compose.js.map +1 -0
- package/out-tsc/src/contacts/ContactChat.js +73 -69
- package/out-tsc/src/contacts/ContactChat.js.map +1 -1
- package/out-tsc/src/contacts/ContactHistory.js +1 -1
- package/out-tsc/src/contacts/ContactHistory.js.map +1 -1
- package/out-tsc/src/contacts/events.js +1 -0
- package/out-tsc/src/contacts/events.js.map +1 -1
- package/out-tsc/src/interfaces.js +2 -0
- package/out-tsc/src/interfaces.js.map +1 -1
- package/out-tsc/src/textinput/TextInput.js +3 -1
- package/out-tsc/src/textinput/TextInput.js.map +1 -1
- package/out-tsc/src/utils/index.js +10 -0
- package/out-tsc/src/utils/index.js.map +1 -1
- package/out-tsc/src/vectoricon/index.js +1 -0
- package/out-tsc/src/vectoricon/index.js.map +1 -1
- package/out-tsc/temba-modules.js +2 -0
- package/out-tsc/temba-modules.js.map +1 -1
- package/out-tsc/test/temba-compose.test.js +432 -0
- package/out-tsc/test/temba-compose.test.js.map +1 -0
- package/out-tsc/test/temba-contact-chat.test.js +219 -47
- package/out-tsc/test/temba-contact-chat.test.js.map +1 -1
- package/out-tsc/test/temba-contact-history.test.js +7 -3
- package/out-tsc/test/temba-contact-history.test.js.map +1 -1
- package/out-tsc/test/utils.test.js +12 -8
- package/out-tsc/test/utils.test.js.map +1 -1
- package/package.json +1 -1
- package/screenshots/truth/compose/attachments-and-send-button.png +0 -0
- package/screenshots/truth/compose/attachments-no-send-button.png +0 -0
- package/screenshots/truth/compose/attachments-with-all-files-and-click-send.png +0 -0
- package/screenshots/truth/compose/attachments-with-all-files.png +0 -0
- package/screenshots/truth/compose/attachments-with-failure-files.png +0 -0
- package/screenshots/truth/compose/attachments-with-success-files-and-click-send.png +0 -0
- package/screenshots/truth/compose/attachments-with-success-files.png +0 -0
- package/screenshots/truth/compose/chatbox-attachments-counter-and-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-attachments-counter-no-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-attachments-no-counter-and-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-attachments-no-counter-no-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-counter-and-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-counter-no-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-no-counter-and-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-no-counter-no-send-button.png +0 -0
- package/screenshots/truth/compose/chatbox-no-text-attachments-with-all-files-and-click-send.png +0 -0
- package/screenshots/truth/compose/chatbox-no-text-attachments-with-all-files.png +0 -0
- package/screenshots/truth/compose/chatbox-no-text-attachments-with-failure-files.png +0 -0
- package/screenshots/truth/compose/chatbox-no-text-attachments-with-success-files-and-click-send.png +0 -0
- package/screenshots/truth/compose/chatbox-no-text-attachments-with-success-files.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-and-click-send.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-and-hit-enter.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-and-spaces.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-and-url.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-no-files-and-click-send.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-no-files-and-hit-enter.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-no-files.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-with-all-files-and-click-send.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-with-all-files-and-hit-enter.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-with-all-files.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-with-failure-files.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-with-success-files-and-click-send.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-with-success-files-and-hit-enter.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-attachments-with-success-files.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text-no-spaces.png +0 -0
- package/screenshots/truth/compose/chatbox-with-text.png +0 -0
- package/screenshots/truth/contacts/badges.png +0 -0
- package/screenshots/truth/contacts/compose-attachments-no-text-failure.png +0 -0
- package/screenshots/truth/contacts/compose-attachments-no-text-success.png +0 -0
- package/screenshots/truth/contacts/compose-text-and-attachments-failure-attachments.png +0 -0
- package/screenshots/truth/contacts/compose-text-and-attachments-failure-generic.png +0 -0
- package/screenshots/truth/contacts/compose-text-and-attachments-failure-text-and-attachments.png +0 -0
- package/screenshots/truth/contacts/compose-text-and-attachments-failure-text.png +0 -0
- package/screenshots/truth/contacts/compose-text-and-attachments-success.png +0 -0
- package/screenshots/truth/contacts/compose-text-no-attachments-failure.png +0 -0
- package/screenshots/truth/contacts/compose-text-no-attachments-success.png +0 -0
- package/screenshots/truth/contacts/contact-active-default.png +0 -0
- package/screenshots/truth/contacts/contact-active-show-chatbox.png +0 -0
- package/screenshots/truth/contacts/contact-active-ticket-closed-show-reopen-button.png +0 -0
- package/screenshots/truth/contacts/contact-active-ticket-open-show-chatbox.png +0 -0
- package/screenshots/truth/contacts/contact-archived-hide-chatbox.png +0 -0
- package/screenshots/truth/contacts/contact-archived-ticket-closed-hide-chatbox.png +0 -0
- package/screenshots/truth/contacts/contact-blocked-hide-chatbox.png +0 -0
- package/screenshots/truth/contacts/contact-stopped-hide-chatbox.png +0 -0
- package/screenshots/truth/contacts/history.png +0 -0
- package/screenshots/truth/counter/summary.png +0 -0
- package/screenshots/truth/counter/text.png +0 -0
- package/screenshots/truth/counter/unicode-variables.png +0 -0
- package/screenshots/truth/counter/unicode.png +0 -0
- package/screenshots/truth/counter/variable.png +0 -0
- package/src/button/Button.ts +3 -3
- package/src/charcount/CharCount.ts +5 -4
- package/src/completion/Completion.ts +4 -0
- package/src/compose/Compose.ts +584 -0
- package/src/contacts/ContactChat.ts +75 -78
- package/src/contacts/ContactHistory.ts +1 -1
- package/src/contacts/events.ts +1 -0
- package/src/interfaces.ts +2 -0
- package/src/textinput/TextInput.ts +3 -1
- package/src/utils/index.ts +12 -0
- package/src/vectoricon/index.ts +1 -0
- package/static/css/temba-components.css +6 -0
- package/temba-modules.ts +2 -0
- package/test/temba-compose.test.ts +633 -0
- package/test/temba-contact-chat.test.ts +354 -110
- package/test/temba-contact-history.test.ts +10 -5
- package/test/utils.test.ts +24 -9
- 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"]}
|