(() => { let Tag = new Proxy({}, { get(target, prop) { return (prop => (options = {}) => { if (!(options instanceof Object)) { throw 'Bad options'; } let el = document.createElement(prop); let processNestedObjects = (el, data, name, func = (el, name, data, key) => el[name][key] = data[key]) => { for (let key of Object.keys(data)) { func(el, name, data, key); } }; for (let name of Object.keys(options)) { if (name == 'attributes' && options[name] instanceof Object) { processNestedObjects(el, options[name], name, (el, name, data, key) => el.setAttribute(key, data[key])) continue; } if (name == 'dataset' && options[name] instanceof Object) { processNestedObjects(el, options[name], name); continue; } if (name == 'parent' && options[name] instanceof Element) { options[name].append(el); continue; } if (name == 'children' && options[name] instanceof Array) { processNestedObjects(el, options[name], name, (el, name, data, key) => (data[key] instanceof Element) ? el.append(data[key]) : null); continue; } if (name == 'style' && options[name] instanceof Object) { processNestedObjects(el, options[name], name); continue; } el[name] = options[name]; } return el; })(prop); } }); function htmlEncode(value) { let a = document.createElement('div'); a.innerText = value; return a.innerText; } // Параметры которые берутся из настроек let styleUrl = 'https://api.doctorznaet.online/web/css/chat-bots/main.css'; let name = 'Доктор Знает'; let avatarUrl = 'https://doctorznaet.online/static/logo_small.png'; let apiUrl = 'https://api.doctorznaet.online/chat-bots/widget?id=11'; let botId = '11'; let vkUrl = '#@vkUrl'; let tgUrl = '#@tgUrl'; let recaptchaPublic = 'undefined'; let mainColor = '#1894bf'; // ------------------------------------- if (vkUrl == '#@vkUrl') { vkUrl = null; } if (tgUrl == '#@tgUrl') { tgUrl = null; } if (mainColor == '#@color') { mainColor = null; } if (avatarUrl == '#@avatar') { avatarUrl = ''; } // Достаем сессию из localStorage // Если ее нет, то создаем let chatId = window.localStorage.getItem('__chat__widget'); if (!chatId) { // Генерируем id чата chatId = ((base = 'QWERTYUIOPASDFGHJKLZXCVBNMqwertyuiopasdfghjklzxcvbnm1234567890') => [...Array(20)] .map(_ => base[Math.random() * base.length | 0]).join('') )(); window.localStorage.setItem('__chat__widget', chatId); } // Достаем историю переписки из localStorage let history = window.localStorage.getItem('__chat__widget_history'); if (!history) { history = []; } else { history = JSON.parse(history); } let styles = Tag.link({ href: styleUrl, rel: 'stylesheet' }); document.head.append(styles); // После загрузки стилей добавляем контейнер чата styles.onload = () => { let fontStyles = Tag.link({ href: 'https://fonts.googleapis.com/css2?family=Mulish:ital,wght@0,200;0,300;0,400;0,500;0,600;0,700;0,800;0,900;1,200;1,300;1,400;1,500;1,600;1,700;1,800;1,900&display=swap', rel: 'stylesheet' }); document.head.append(fontStyles); // Добавляем шрифты fontStyles.onload = () => { console.log('styles loaded'); let firstTime = true; // Клик по кнопке отправить let sendClick = (messageInput, errorBox, messageContainer, buttonsContainer, self, token = null, isEmpty = false, buttonText = null) => function() { messageInput.value = messageInput.value.trim(); let text = buttonText ?? messageInput.value; if (text.length == 0 && !isEmpty) { // Ошибка если сообщение пустое if (errorBox.innerText.indexOf('Нельзя отправить пустое сообщение.') === -1) { errorBox.insertAdjacentHTML('beforeend', ' Нельзя отправить пустое сообщение.'); } return; } buttonsContainer.innerHTML = null; errorBox.innerHTML = null; sendMessage(text).then(response => response.json()).then(response => { // Когда пришел ответ на наше сообщение // Если текст не пришел if (!response.text) { return; } if (!isEmpty) { history.push({ // Добавляем наше сообщение в историю isMy: true, text: text }); } let wrap = messageContainer.parentElement; let pos = wrap.scrollHeight - wrap.clientHeight; let flag = false; // Устанавливаем флажок для корректировки позиции скролла if (pos <= wrap.scrollTop + 20) { flag = true; } history.push({ // Добавляем сообщение бота в историю isMy: false, text: response.text }); window.localStorage.setItem('__chat__widget_history', JSON.stringify(history)); // Сохраняем историю messageContainer.append(Tag.div({ // Добавляем сообщение className: 'chat__message', innerHTML: `
${response.text}
` })); // Добавляем кнопки buttonsContainer.innerHTML = null; for (let button of response.buttons ?? []) { let el = Tag.button({ innerText: button.text, onclick: (buttonsContainer => () => { verifyCaptchaAndSend(false, button.text); buttonsContainer.innerHTML = null; })(buttonsContainer) }); buttonsContainer.append(el); } // Корректируем скролл если нужно if (flag) { wrap.scroll(0, wrap.scrollHeight); } }).then(response => { // Если too many requests if (response && response.status == 429) { if (errorBox.innerText.indexOf('Слишком много запросов, попробуйте позже.') === -1) { errorBox.insertAdjacentHTML('beforeend', ' Слишком много запросов, попробуйте позже.'); } } }); if (!isEmpty) { messageContainer.append(Tag.div({ className: 'chat__message chat__message_my', innerHTML: `
${text}
` })); messageContainer.parentElement.scroll(0, messageContainer.parentElement.scrollHeight); messageInput.value = null; } }; // Отправка сообщения function sendMessage(text) { if (!chatId) { chatId = window.localStorage.getItem('__chat__widget'); if (!chatId) { // Генерируем id чата chatId = ((base = 'QWERTYUIOPASDFGHJKLZXCVBNMqwertyuiopasdfghjklzxcvbnm1234567890') => [...Array(20)] .map(_ => base[Math.random() * base.length | 0]).join('') )(); window.localStorage.setItem('__chat__widget', chatId); } } let data = new FormData(); let payload = { message: { text: text, chat: { id: chatId } } }; data.append('json', JSON.stringify(payload)); return fetch(`${apiUrl}`, { method: 'POST', body: data }); } let verifyCaptchaAndSend = (isEmpty = false, buttonText = null) => { sendClick(messageInput, errorBox, messageContainer, buttonsContainer, sendBtn, null, isEmpty, buttonText)(); } // Создание элемента let chatContainer = Tag.div({ className: 'chat__wrap __display-none', innerHTML: `
${htmlEncode(name)}
` }); // Объявляем элементы let messageInput = chatContainer.querySelector('.chat__input'); let sendBtn = chatContainer.querySelector('.chat__send'); let errorBox = chatContainer.querySelector('.chat__error'); let messageContainer = chatContainer.querySelector('.chat__messages'); let buttonsContainer = chatContainer.querySelector('.chat__buttons'); // Клик по кнопке свернуть чат chatContainer.querySelector('.chat__hide').onclick = (chatContainer => function() { chatContainer.classList.toggle('__display-none'); chatContainer.parentElement.querySelector('.chat__show').classList.toggle('__display-none'); })(chatContainer); // Клик по кнопке закрыть чат /*chatContainer.querySelector('.chat__close').onclick = (chatContainer => function() { chatContainer.parentElement.remove(); })(chatContainer);*/ // Рисуем сообщения из истории (Сообщения которые уже были отправлены) for (let message of history) { messageContainer.append(Tag.div({ className: `chat__message ${message.isMy ? 'chat__message_my' : ''}`, innerHTML: `
${message.text}
` })); } // Клик на enter для отправки messageInput.onkeydown = (sendBtn => function(event) { if (event.key === 'Enter') { sendBtn.click(); event.preventDefault(); } })(sendBtn); // Эвент при вводе messageInput.oninput = (errorBox => function() { if (this.value.length > 300) { // Если длинное сообщение выводим ошибку if (errorBox.innerText.indexOf('Длина сообщения не должна превышать 300 символов.') !== -1) { return; } errorBox.insertAdjacentHTML('beforeend', ' Длина сообщения не должна превышать 300 символов.'); return; } else { // Иначе очищаем поле для ошибок errorBox.innerText = null; } })(errorBox); // Клик по кнопке отправить sendBtn.onclick = () => { verifyCaptchaAndSend(); } let chatWrap = Tag.div({ className: 'chat__popup', innerHTML: ` ` }); if (mainColor) { chatWrap.style.setProperty('--color-main', mainColor); } // Клик по кнопке показать чат let chatBtn = chatWrap.querySelector('.chat__show'); chatBtn.onclick = (chatMessages => () => { // Получаем приветственное сообщение let shouldScroll = false; if (firstTime) { firstTime = false; shouldScroll = true; if (!history || (history instanceof Array && !history.length)) { // Отправляем приветственное сообщение verifyCaptchaAndSend(true); } } chatBtn.classList.toggle('__display-none'); chatContainer.classList.toggle('__display-none'); if (shouldScroll) { chatMessages.scroll(0, chatMessages.scrollHeight); } })(chatContainer.querySelector('.chat__messages-wrap')); // Контейнер для капчи Tag.div({ id: `__ks-widget__${recaptchaPublic}`, parent: document.body, }); chatWrap.append(chatContainer); document.body.append(chatWrap); /*setTimeout((chatBtn => () => { if (window.document.documentElement.clientWidth > 960) { chatBtn.click(); } })(chatBtn), 250);*/ } } })(); console.log(JSON.parse('{"name":"Доктор Знает","avatar":"https://doctorznaet.online/static/logo_small.png","apiUrl":"https://api.doctorznaet.online/chat-bots/widget?id=11","botId":11,"vkUrl":"#@vkUrl","tgUrl":"#@tgUrl","recaptcha":"undefined","color":"#1894bf"}'));