Часть 1: 1С Поле HTML документа — Кнопки.
Часть 2: 1С Поле HTML документа — Обмен.
Часть 3: 1С Поле HTML документа — Список.
Часть 4: 1С Поле HTML документа — React Список.
Рассмотрим простой пример использования списка, созданного при помощи HTML, CSS и JavaScript, с возможностью загрузки данных из 1С. За основу будет взята база данных из записи Анализ данных и прогнозирование в 1С — Общая статистика, реализация будет выполнена во внешней обработке.
1. Добавлен макет HTML.
Во внешнюю обработку добавлен макет HTML, содержащий следующий код:
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>1С Поле HTML документа - Список</title> <style> * { box-sizing: border-box; margin: 0; padding: 0; } body { background-color: rgb(248, 249, 251); font-family: Arial, Helvetica, sans-serif; color: #2E3440; } /* Элемент списка */ .item { position: relative; background-color: #ECEFF4; border-radius: 5px; padding: 3px; margin: 3px; margin-bottom: 3px; cursor: pointer; } .item:hover, .item:hover > div { background-color: #E5E9F0; } .item:active, .item:active > div { background-color: #D8DEE9; } .item:hover .rating { opacity: 1; } /* Наименование элемента списка */ .itemName { display: inline-block; width: 85%; font-size: 12px; font-weight: bold; } /* Данные элемента списка */ .itemData div { display: inline-block; font-weight: normal; font-size: 10px; } /* Звезда рейтинга */ .rating { position: absolute; top: 20%; right: 10px; opacity: 0.5; } .yellowStar { display: inline-block; width: 24px; height: 24px; background-image: url(''); } </style> </head> <body> <script defer> // Переменная для данных элемента, из тега "data-*. let itemId = undefined; // Элемент body текушей страницы. let body = document.querySelector("body"); // Функция добавление нового элемента списка. function addItem(id, itemName, itemData1, itemData2, itemData3) { // Создаем новый элемент списка. let newItem = document.createElement("div"); newItem.className = "item"; newItem.dataset.id = id; // Формируем структуру элемента списка, заполненного данными. newItem.innerHTML = `<div class="itemName">${itemName}</div> <div class="itemData"> <div>${itemData1}</div> <div>▪ ${itemData2}</div> <div>▪ ${itemData3}</div> </div> <div class="rating"> <div class="yellowStar"></div> </div>`; // Добавляем новый элемент на страницу. body.append(newItem); // Добавляем событие нажатия на элемент. newItem.addEventListener("click", (event) => { // Сохраняем данные элемента в переменной для хранения. itemId = event.currentTarget.dataset.id; }); }; // Обработка события прокрутки, уведомить платформу 1С о необходимости загрузить новые элементы. window.addEventListener("scroll", () => { if (window.pageYOffset + document.documentElement.clientHeight >= document.documentElement.scrollHeight) { // console.log("scroll end"); } }); // Получение данных элемента, по которому было совершенно нажатие. function getItemId() { return itemId; }; </script> </body> </html>
Пояснение кода:
Существует особенность работы события «scroll» для элементов Поле HTML, был добавлен обработчик события фиксации прокрутки списка до конца, для автоматической загрузки новых элементов списка, но при работе в 1С он не срабатывал, привязка к элементу «document.body», так же не дало результата.
// Обработка события прокрутки, уведомить платформу 1С о необходимости загрузить новые элементы. window.addEventListener("scroll", () => { if (window.pageYOffset + document.documentElement.clientHeight >= document.documentElement.scrollHeight) { // console.log("scroll end"); } });
Изображение звездочки, используется в виде строки, закодированной в Base64. Более подробно можно прочесть по адресу: Data URL и Храните мелкие картинки в CSS.
.yellowStar { display: inline-block; width: 24px; height: 24px; background-image: url('data:image/png;base64, Закодированная строка); }
Макет HTML был сверстан с учетом использования атрибута «data-*» тега «div», в котором хранится текущий идентификатор записи, для последующей обработки события нажатия. В текущей версии платформы 1С, не удалось получить на прямую данные из dataset атрибута, видимо ограничение взаимодействие между платформой 1С и web движком «WebKit». Для обхода сложившейся ситуации, на элементы с классом «item», добавлен обработчик нажатия, который получает необходимые данные и сохраняет их в переменной «itemId». Для получения данных добавлена функция «getItemId», которая просто возвращает значение переменной «itemId».
// Переменная для данных элемента, из тега "data-*. let itemId = undefined; // Элемент body текушей страницы. let body = document.querySelector("body"); // Функция добавление нового элемента списка. function addItem(id, itemName, itemData1, itemData2, itemData3) { // Создаем новый элемент списка. let newItem = document.createElement("div"); newItem.className = "item"; newItem.dataset.id = id; // Формируем структуру элемента списка, заполненного данными. newItem.innerHTML = `<div class="itemName">${itemName}</div> <div class="itemData"> <div>${itemData1}</div> <div>▪ ${itemData2}</div> <div>▪ ${itemData3}</div> </div> <div class="rating"> <div class="yellowStar"></div> </div>`; // Добавляем новый элемент на страницу. body.append(newItem); // Добавляем событие нажатия на элемент. newItem.addEventListener("click", (event) => { // Сохраняем данные элемента в переменной для хранения. itemId = event.currentTarget.dataset.id; }); };
2. Добавлены реквизиты формы.
На форму обработки, добавлены два реквизита «ПолеHTMLСписок» тип строка, вид «Поле HTML документа», и текстовый документ «ТекстовыеДанные», для простого вывода данных элемента списка, при нажатии, а именно идентификатора элемента.
3. Добавлены обработчики событий.
Загрузка макета HTML, осуществляется в процедуре «ПриСозданииНаСервере»:
&НаСервере Процедура ПриСозданииНаСервере(Отказ, СтандартнаяОбработка) ОбъектОбработки = РеквизитФормыВЗначение("Объект"); ПолеHTMLСписок = ОбъектОбработки.ПолучитьМакет("МакетHTMLСписок").ПолучитьТекст(); КонецПроцедуры
Для элемента формы «ПолеHTMLСписок» добавлены обработчики «ДокументСформирован» и «ПриНажатии», со следующей реализацией:
&НаКлиенте Процедура ПолеHTMLСписокДокументСформирован(Элемент) Если НЕ Элементы.ПолеHTMLСписок.Документ.readyState = "complete" Тогда Возврат; КонецЕсли; ОкноHTML = Элементы.ПолеHTMLСписок.Документ.defaultView; Если ОкноHTML = Неопределено Тогда Возврат; КонецЕсли; МассивПродаж = ДанныеПродаж(); Для каждого Продажа Из МассивПродаж Цикл // addItem(id, itemName, itemData1, itemData2, itemData3); ОкноHTML.addItem(Продажа.ИдентификаторСчетФактуры, Продажа.ВидНоменклатуры, Продажа.ТипКлиента, Продажа.Пол, Продажа.РейтингКлиента); КонецЦикла; КонецПроцедуры &НаКлиенте Процедура ПолеHTMLСписокПриНажатии(Элемент, ДанныеСобытия, СтандартнаяОбработка) СтандартнаяОбработка = Ложь; Если ДанныеСобытия.Event.type = "click" И ДанныеСобытия.Element.tagName = "DIV" Тогда ОкноHTML = Элементы.ПолеHTMLСписок.Документ.defaultView; Если ОкноHTML = Неопределено Тогда Возврат; КонецЕсли; ИдентификаторЭлемента = ОкноHTML.getItemId(); ТекстовыеДанные.ДобавитьСтроку(ИдентификаторЭлемента); КонецЕсли; КонецПроцедуры
💾 Внешняя обработка ПолеHTMLДокументаСписок.zip.