From ea2dcc03fc5c06e082779a9677e7a6e51c6600e1 Mon Sep 17 00:00:00 2001 From: wakadakawaka Date: Thu, 17 Apr 2025 00:42:23 +0500 Subject: [PATCH] =?UTF-8?q?=D0=B4=D0=BE=D0=B1=D0=B0=D0=B2=D0=BB=D0=B5?= =?UTF-8?q?=D0=BD=D0=B8=D0=B5=20=D1=81=D1=82=D0=B0=D1=82=D0=B5=D0=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Длительная многопоточная операция.md | 150 ++++++++++++++++++ .../Длительная однопоточная операция.md | 127 +++++++++++++++ .../Настройки/Внешняя печатная форма.md | 4 +- .../Подключение к подсистемам/Дата запрета изменений.md | 54 +++++++ .../БСП и типовые/Механизмы печати/Макет печатной формы.md | 20 +++ .../Механизмы печати/Пример вывода настроек печати.md | 148 +++++++++++++++++ .../Подключение БСП к собственной конфигурации.md | 26 +++ 7 files changed, 528 insertions(+), 1 deletion(-) create mode 100644 src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная многопоточная операция.md create mode 100644 src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная однопоточная операция.md create mode 100644 src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Интеграция собственных объектов/Подключение к подсистемам/Дата запрета изменений.md create mode 100644 src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Макет печатной формы.md create mode 100644 src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Пример вывода настроек печати.md create mode 100644 src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Подключение БСП к собственной конфигурации.md diff --git a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная многопоточная операция.md b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная многопоточная операция.md new file mode 100644 index 0000000..80ddb5a --- /dev/null +++ b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная многопоточная операция.md @@ -0,0 +1,150 @@ +# Определение длительной операции +Многопоточные операции применяются в случаях обработки большого массива данных — например, для изменения реквизитов номенклатуры, обработки больших табличных документов, массового добавления объектов и пр. +В типовых конфигурациях есть механизм для создания операций, выполняющихся в нескольких потоках — этот механизм определен в общих модулях `«Длительные операции»` +Для примера возьмем алгоритм, изменяющий реквизиты объектов справочника `Номенклатура`. Реализуем в общем модуле процедуру, которая будет вызываться в многопоточном режиме: +```bsl +Процедура ИзменитьРеквизитыНоменклатуры(НаборЗаписей) Экспорт + + Если Не Пользователи.ЭтоПолноправныйПользователь(Пользователи.ТекущийПользователь()) Тогда + ТекстСообщения = "Это действие может выполнить только администратор"; + ОбщегоНазначения.СообщитьПользователю(ТекстСообщения); + Возврат; + КонецЕсли; + + Для Каждого Элемент Из НаборЗаписей Цикл + + Об = Элемент.Номенклатура.ПолучитьОбъект(); + Если Об.Заблокирован() Тогда + Продолжить; + КонецЕсли; + + Попытка + Об.Заблокировать(); + Исключение + ТекстСообщения = СтрШаблон("Не удалось изменить %1. Объект заблокирован", Элемент.Номенклатура); + ОбщегоНазначения.СообщитьПользователю(ТекстСообщения); + Продолжить; + КонецПопытки; + + Об.ИспользоватьУпаковки = Истина; + Об.НаборУпаковок = Элемент.НаборУпаковок; + Попытка + Об.Записать(); + Исключение + ТекстСообщения = СтрШаблон("Не удалось изменить %1. Ошибка записи объекта", Элемент.Номенклатура); + ОбщегоНазначения.СообщитьПользователю(ТекстСообщения); + Продолжить; + КонецПопытки; + + КонецЦикла; + +КонецПроцедуры +``` + +# Определение клиентских вызовов +![image.png](https://sinenikolsky.ru/s/QEWSKmHn7HKGJX4/download?path=%2F2025%2F04%2F16&files=0f68e5a7-bd92-4e6a-9d6b-4972c5ed1f44.png) +Команда `«Установить новый набор»` выполняет разделение строк ТЧ на равные порции, т.е. создает список наборов, которые будут передаваться в процедуру из шага 1 (Функция `«СписокНаборовДанныхДляДлительнойОперации«`). Дополнительно инициализируются настройки ожидания и параметры выполняемой процедуры (Функция `«ДлительныеОперации«`). Также указывается процедура, которая будет выполнена после завершения всех потоков длительной операции +```bsl +&НаКлиенте +Процедура УстановитьНовыйНабор(Команда) + + ПараметрыОжидания = ДлительныеОперацииКлиент.ПараметрыОжидания(ЭтотОбъект); + ПараметрыОжидания.ВыводитьОкноОжидания = Истина; + ПараметрыОжидания.ВыводитьСообщения = Истина; + ПараметрыОжидания.ОповещениеПользователя.Показать = Истина; + + СписокНаборов = СписокНаборовДанныхДляДлительнойОперации(); + + МногопоточнаяОперация = ДлительнаяОперация(УникальныйИдентификатор, СписокНаборов); + + ОписаниеОповещения = Новый ОписаниеОповещения("ВыполнитьПоЗавершениюМногопоточнойОперации", ЭтотОбъект); + + ДлительныеОперацииКлиент.ОжидатьЗавершение(МногопоточнаяОперация, ОписаниеОповещения, ПараметрыОжидания); + +КонецПроцедуры + +&НаКлиенте +Функция СписокНаборовДанныхДляДлительнойОперации() + + НаборЗаписей = Новый Массив; + + КоличествоЗаписейНаПоток = Цел(Объект.СписокНоменклатуры.Количество() / КоличествоПотоков) + 1; + + СписокНаборов = Новый Массив; + + Сч = 1; + Для Каждого Строка Из Объект.СписокНоменклатуры Цикл + + Если Не Строка.НовыйИспользоватьУпаковки Тогда + Сообщить("Не добавлена строка " + Строка.НомерСтроки); + Продолжить; + КонецЕсли; + + ПараметрыСтроки = Новый Структура; + ПараметрыСтроки.Вставить("Номенклатура", Строка.Номенклатура); + ПараметрыСтроки.Вставить("ИспользоватьУпаковки", Истина); + ПараметрыСтроки.Вставить("НаборУпаковок", Строка.НовыйНаборУпаковок); + + НаборЗаписей.Добавить(ПараметрыСтроки); + Сч = Сч + 1; + НужноДобавитьНабор = Истина; + + Если Сч % КоличествоЗаписейНаПоток = 0 Тогда + СписокНаборов.Добавить(НаборЗаписей); + НаборЗаписей = Новый Массив; + НужноДобавитьНабор = Ложь; + КонецЕсли; + + КонецЦикла; + + Если НужноДобавитьНабор Тогда + СписокНаборов.Добавить(НаборЗаписей); + КонецЕсли; + + КоличествоЗаписей = 0; + + Для Каждого Массив Из СписокНаборов Цикл + + КоличествоЗаписей = КоличествоЗаписей + Массив.Количество(); + + КонецЦикла; + + Возврат СписокНаборов; + +КонецФункции + +&НаКлиенте +Процедура ВыполнитьПоЗавершениюМногопоточнойОперации(Задание, ДополнительныеПараметры) Экспорт + + Если Задание = Неопределено Тогда + Возврат; + КонецЕсли; + + Сообщить(Задание.Статус); + +КонецПроцедуры +``` +В функции длительная операция необходимо правильно сформировать параметры для метода `ДлительныеОперации.ВыполнитьПроцедуруВНесколькоПотоков`. Первым аргументом передаем полное имя экспортного метода, который нужно выполнять, вторым — структуру `ДлительныеОперации.ПараметрыВыполненияВФоне`, а третьим — правильно оформленный набор параметров. + +**Набор параметров** — это соответствие, ключ — произвольный тип, значение — МАССИВ параметров, которые нужно передать в вызываемый метод. Количество параметров — **до 7 шт**. В примере метод `«ИзменитьРеквизитыНоменклатуры»` принимает только один параметр, соответственно в значение записи набора параметров мы передаем обернутое в массив значение параметра (`«ОбщегоНазначенияКлиентСервер.ЗначениеВМассиве(Набор)»`) +```bsl +&НаСервере +Функция ДлительнаяОперация(УИД, СписокНаборовДанных) + + ПараметрыВыполнения = ДлительныеОперации.ПараметрыВыполненияВФоне(УИД); + ПараметрыВыполнения.НаименованиеФоновогоЗадания = "Многопоточное изменение реквизитов номенклатуры"; + ПараметрыВыполнения.ЗапуститьВФоне = Истина; + + Сч = 1; + НаборПараметровПроцедуры = Новый Соответствие; + Для Каждого Набор Из СписокНаборовДанных Цикл + НаборПараметровПроцедуры.Вставить("Набор №" + Сч, ОбщегоНазначенияКлиентСервер.ЗначениеВМассиве(Набор)); + Сч = Сч + 1; + КонецЦикла; + + Возврат ДлительныеОперации.ВыполнитьПроцедуруВНесколькоПотоков( + "Расш1_ДлительныеОперации.ИзменитьРеквизитыНоменклатуры", + ПараметрыВыполнения, НаборПараметровПроцедуры); + +КонецФункции +``` \ No newline at end of file diff --git a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная однопоточная операция.md b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная однопоточная операция.md new file mode 100644 index 0000000..2390093 --- /dev/null +++ b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Длительные операции и выполнение в фоне/Длительная однопоточная операция.md @@ -0,0 +1,127 @@ +# Схема выполнения длительной серверной операции +Механизм длительных операций БСП необходим для выполнения тяжелых операций без блокирования окна клиентского приложения. По умолчанию платформа блокирует клиентский интерфейс до тех пор, пока серверная операция не будет завершена. Подсистема "Длительные операции" позволяет обойти эту проблему +![diagramma-bez-nazvaniya.drawio.png](https://sinenikolsky.ru/s/QEWSKmHn7HKGJX4/download?path=%2F2025%2F04%2F16&files=fe7a4397-bffe-44ca-be28-f3f909684212.png) + +## Создание длительной операции +Длительную операцию необходимо размещать в **экспортной** функции. Рекомендуется выносить их в общий модуль. Например, для примера создаем длительную функцию в общем модуле `МоиДлительныеОперации` +```bsl +Функция МояДлительнаяОперацияНаСервере(ВремяОжидания) Экспорт + + // Имитация длительной операции + КонечнаяДата = ТекущаяДатаСеанса() + ВремяОжидания; + Пока ТекущаяДатаСеанса() < КонечнаяДата Цикл + + //=================================================================================================== + + // Алгоритмы операции, Очень много страшного тяжелого кода + + //=================================================================================================== + + + //=================================================================================================== + // Относится к механизму "Длительные операции" + // Расчет прогресса для сообщения клиенту + СекундОсталось = КонечнаяДата - ТекущаяДатаСеанса(); + Прогресс = -(Окр((СекундОсталось / ВремяОжидания) * 100, 0)) + 100; + // Сообщение клиенту + ДлительныеОперации.СообщитьПрогресс(Прогресс, "Выполняюсь, осталось " + (100 - Прогресс) + "%"); + //=================================================================================================== + + КонецЦикла; + + // Любой тип, доступный на клиенте. + // Можно, например, возвращать какой-то сложный табличный документ, или ссылку на какой-нибудь объект. + Возврат "Все выполнилось хорошо"; + +КонецФункции +``` + +## Настройка модуля формы + +Так как пользователь взаимодействует с формами, очевидно, что инициализация процесса начинается именно с нее. Для примера создадим внешнюю обработку с одним числовым реквизитом и командой. Планируется запускать длительную операцию на то количество секунд, которое введет пользователь +Как показано на схеме, при нажатии на кнопку на стороне клиента должны сформироваться параметры ожидания. +`Параметры ожидания` — это структура с определенными ключами, на основании которой будут установлены настройки длительной операции. Параметры формируются функцией общего модуля `ДлительныеОперацииКлиент.ПараметрыОжидания`. Описание можно посмотреть в общем модуле. Пример реализации: +```bsl +&НаКлиенте +Процедура ВыполнитьВФоне(Команда) + + // Аргументом передается форма, с которой запускается длительное задание + ПараметрыОжидания = ДлительныеОперацииКлиент.ПараметрыОжидания(ЭтотОбъект); + ПараметрыОжидания.ТекстСообщения = "Выполняется длительная операция."; + ПараметрыОжидания.ВыводитьПрогрессВыполнения = Истина; + ПараметрыОжидания.Интервал = 2; + + // Структура, которая передается в качестве настроек оповещения при завершении длительной операции + ПараметрыОповещения = Новый Структура("НавигационнаяСсылка, Пояснение, Картинка, Важное" + , "" + , "" + , Неопределено + , Ложь); + + ПараметрыОповещения.Вставить("Показать", Истина); + ПараметрыОповещения.Вставить("Текст", "Синхронизация завершена"); + + ПараметрыОжидания.ОповещениеПользователя = ПараметрыОповещения; + + // Инициализация запуска длительной операции. + МояДлительнаяОперация = ДлительнаяОперация(УникальныйИдентификатор, "МояДлительнаяОперацияНаСервере", ДлительностьОжидания); + + // Определение действия после завершения длительной операции + ОписаниеОповещения = Новый ОписаниеОповещения("ПослеЗавершенияВыполненияДлительнойФункции", ЭтаФорма); + + // Запуск ожидания завершения + ДлительныеОперацииКлиент.ОжидатьЗавершение(МояДлительнаяОперация + , ОписаниеОповещения + , ПараметрыОжидания); + + +КонецПроцедуры +``` +В этом методе есть вызов функции `ДлительнаяОперация`, собственно и запускает выполнение серверной функции. Она возвращает объект, позволяющий отслеживать выполнение операции с клиента в методе `ДлительныеОперацииКлиент.ОжидатьЗавершение` +```bsl +&НаСервереБезКонтекста +Функция ДлительнаяОперация(УИДЭтойФормы, ИмяФункции, ДлительностьОжидания) + + // Указываем идентификатор владельца задания, + // полный путь до тяжелой операции и дополнительные параметры (до 7 шт), + // которые необходимо передать в тяжелую операцию + Операция = ДлительныеОперации.ВыполнитьФункцию(УИДЭтойФормы + , "МоиДлительныеОперации." + ИмяФункции + , ДлительностьОжидания); + + Возврат Операция; + +КонецФункции +``` +Также в качестве параметра ожидания на клиенте было передано `ОписаниеОповещения`, которое определяет поведение после завершения выполнения длительной операции. +> ВАЖНО! Длительная операция может завершиться как удачно, так и нет, и это нужно учитывать +```bsl +&НаКлиенте +Процедура ПослеЗавершенияВыполненияДлительнойФункции(Результат, ДополнительныеПараметры) Экспорт + + // В аргумент "Результат" принимается структура с определенными ключами, среди которых + // наиболее интересны Статус, АдресРезультата, КраткоеПредставлениеОшибки + // + //Если что-то пошло не так, то результат будет Неопределено + Если Результат = Неопределено Тогда + Сообщить("Ошибка выполнения"); + Возврат; + КонецЕсли; + + // Если результат не равер Неопределено, то анализируется статус + // Варианты статусов - "Выполняется", "Выполнено", "Ошибка", "Отменено" + Если Результат.Статус = "Выполнено" Тогда + // Возвращаемое значение длительной операции помещается во временное хранилище + // и передается на клиент в качестве адреса + Сообщение = ПолучитьИзВременногоХранилища(Результат.АдресРезультата); + ИначеЕсли Результат.Статус = "Ошибка" Тогда + Сообщение = "Ошибка выполнения: " + + Символы.ПС + + Результат.КраткоеПредставлениеОшибки; + Иначе + Сообщение = "Неопределенный результат выполнения длительной операции. См. журнал регистрации"; + КонецЕсли; + + Сообщить(Сообщение); + +``` \ No newline at end of file diff --git a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Дополнительные отчеты и обработки/Настройки/Внешняя печатная форма.md b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Дополнительные отчеты и обработки/Настройки/Внешняя печатная форма.md index ae65f46..5c917f0 100644 --- a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Дополнительные отчеты и обработки/Настройки/Внешняя печатная форма.md +++ b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Дополнительные отчеты и обработки/Настройки/Внешняя печатная форма.md @@ -154,4 +154,6 @@ Теперь можно заниматься отладкой печати БЕЗ добавления нового элемента в справочник `«ДополнительныеОтчетыОбработки»`. Для этого мы рисовали форму в обработке ![image.png]() -![image.png]() \ No newline at end of file +![image.png]() + +А после добавления в справочник у нас автоматом нарисуется нужная **команда печати** для тех элементов, которые мы указали в массиве назначений (Если на форме используются подключаемые команды, конечно. Просто так ничего не нарисуется) \ No newline at end of file diff --git a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Интеграция собственных объектов/Подключение к подсистемам/Дата запрета изменений.md b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Интеграция собственных объектов/Подключение к подсистемам/Дата запрета изменений.md new file mode 100644 index 0000000..9533779 --- /dev/null +++ b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Интеграция собственных объектов/Подключение к подсистемам/Дата запрета изменений.md @@ -0,0 +1,54 @@ +# Общие модули +В процедуре `ЗаполнитьИсточникиДанныхДляПроверкиЗапретаИзменения` общего модуля `ДатыЗапретаИзмененияПереопределяемый` дополнить: +```bsl +// Позволяет задать таблицы и поля объектов для проверки запрета изменения данных. +// Для добавления нового источника в ИсточникиДанных см. ДатыЗапретаИзменения.ДобавитьСтроку. +// +// Вызывается из процедуры ИзменениеЗапрещено общего модуля ДатыЗапретаИзменения, +// используемой в подписке на событие ПередЗаписью объекта для проверки наличия +// запретов и отказа от изменений запрещенного объекта. +// +// Параметры: +// ИсточникиДанных - ТаблицаЗначений: +// * Таблица - Строка - полное имя объекта метаданных, +// например Метаданные.Документы.ПриходнаяНакладная.ПолноеИмя(). +// * ПолеДаты - Строка - имя реквизита объекта или табличной части, +// например: "Дата", "Товары.ДатаОтгрузки". +// * Раздел - Строка - имя раздела дат запрета, указанного в +// процедуре ПриЗаполненииРазделовДатЗапретаИзменения (см. выше). +// * ПолеОбъекта - Строка - имя реквизита объекта или реквизита табличной части, +// например: "Организация", "Товары.Склад". +// +Процедура ЗаполнитьИсточникиДанныхДляПроверкиЗапретаИзменения(ИсточникиДанных) + + ДатыЗапретаИзменения.ДобавитьСтроку(ИсточникиДанных, + Метаданные.Документы.<МойДокумент>.ПолноеИмя(), + "Дата"); + +КонецПроцедуры +``` +# Настройка формы +В обработчике `ПриЧтенииНаСервере` +```bsl +&НаСервере +Процедура ПриЧтенииНаСервере(ТекущийОбъект) + //СтандартныеПодсистемы.ДатыЗапретаИзменения + ДатыЗапретаИзменения.ОбъектПриЧтенииНаСервере(ЭтотОбъект, ТекущийОбъект); + //Конец СтандартныеПодсистемы.ДатыЗапретаИзменения +КонецПроцедуры +``` + +## Запрет создания и удаления объектов в закрытом периоде +### Добавление +дополняем подписки на события +`ПроверитьДатуЗапретаИзмененияПередЗаписьюДокумента` //Для документов +`ПроверитьДатуЗапретаИзмененияПередЗаписьюНабораЗаписей` //Для регистров +- `ИСТОЧНИК` — новые объекты, которые нужно проверять +- `Событие` — ПередЗаписью +- `Обработчик` — Процедура общего модуля `ДатыЗапретаИзменения.ПроверитьДатуЗапретаИзмененияПередЗаписьюДокумента` для документов или `ДатыЗапретаИзменения.ПроверитьДатуЗапретаИзмененияПередЗаписьюНабораЗаписей` для регистров +### Удаление +дополняем подписки на события +`ПроверитьДатуЗапретаИзмененияПередУдалением` +- `ИСТОЧНИК` — новые объекты, которые нужно проверять +- `Событие` — ПередУдалением +- `Обработчик` — Процедура общего модуля `ДатыЗапретаИзменения.ПроверитьДатуЗапретаИзмененияПередУдалением` \ No newline at end of file diff --git a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Макет печатной формы.md b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Макет печатной формы.md new file mode 100644 index 0000000..1d011b3 --- /dev/null +++ b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Макет печатной формы.md @@ -0,0 +1,20 @@ +```bsl +//Возвращает макет печатной формы по полному пути к макету. +//Если конфигурация рассчитана на несколько языков, в ней могут быть предусмотрены несколько макетов для этих языков: +// - ПФ_DOC_СчетНаОплату_ru +// - ПФ_DOC_СчетНаОплату_en +// - и т.п. +// В этом случае при поиске нужного макета учитывается следующий приоритет: +// 1) на языке, указанном в параметре КодЯзыка, +// 2) на языке конфигурации (ОбщегоНазначения.КодОсновногоЯзыка()), +// 3) без указания языка. +// +// Параметры +// ПутьКМакету - Строка - полный путь к макету в формате: +// "Документ.<ИмяДокумента>.<ИмяМакета>" +// "Обработка.<ИмяОбработки>.<ИмяМакета>" +// "ОбщийМакет.<ИмяМакета>". +// КодЯзыка - Строка - язык, на котором требуется получить макет. +// Состоит из кода языка по ISO 639-1 и, опционально, кода страны по ISO 3166-1, разделенных символом подчеркивания. Примеры: "en", "en_US", "en_GB", "ru", "ru_RU". +Макет = УправлениеПечатью.МакетПечатнойФормы(ПутьКМакету, КодЯзыка) +``` \ No newline at end of file diff --git a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Пример вывода настроек печати.md b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Пример вывода настроек печати.md new file mode 100644 index 0000000..e9de72c --- /dev/null +++ b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Механизмы печати/Пример вывода настроек печати.md @@ -0,0 +1,148 @@ +--- +title: Вывод настроек печати перед печатью на примере дополнительной внешней ПФ акта сверки взаиморасчетов +--- + +# Сведения о внешней обработке +```bsl +Функция СведенияОВнешнейОбработке() Экспорт + + // Версия БСП, на которой велась разработка - 3.1.9.302 + + ВерсияБСП = СтандартныеПодсистемыСервер.ВерсияБиблиотеки(); + + ПараметрыРегистрации = ДополнительныеОтчетыИОбработки.СведенияОВнешнейОбработке(ВерсияБСП); + ПараметрыРегистрации.Вид = ДополнительныеОтчетыИОбработкиКлиентСервер.ВидОбработкиПечатнаяФорма(); + + МассивНазначений = Новый Массив; + МассивНазначений.Добавить("Документ.СверкаВзаиморасчетов"); + + ПараметрыРегистрации.Назначение = МассивНазначений; + ПараметрыРегистрации.Наименование = "Акт сверки взаимных расчетов (развернутое сальдо)"; + ПараметрыРегистрации.БезопасныйРежим = Ложь; + ПараметрыРегистрации.Версия = "1.0.1.1"; + ПараметрыРегистрации.Информация = "Акт сверки взаимных расчетов с развернутым сальдо"; + + ПараметрыРегистрации.Команды.Колонки.Добавить("ПроверкаПроведенияПередПечатью", Новый ОписаниеТипов("Булево")); + + Команда = ПараметрыРегистрации.Команды.Добавить(); + Команда.Представление = "Акт сверки взаимных расчетов (развернутое сальдо)"; + Команда.Идентификатор = " СН_АктСверкиВзаимныхРасчетовРазвернутоеСальдо"; + Команда.Использование = ДополнительныеОтчетыИОбработкиКлиентСервер.ТипКомандыВызовКлиентскогоМетода(); // ВАЖНО + Команда.ПоказыватьОповещение = Истина; + Команда.Модификатор = "ПечатьMXL"; + Команда.ПроверкаПроведенияПередПечатью = Истина; + + Возврат ПараметрыРегистрации; + +КонецФункции +``` +# Формы внешней обработки +В создаем 2 формы — основная форма для отладки, и форма настроек печати для вызова перед печатью. Форму настроек печати просто копируем из типовой конфигурации. Основная форма будет открываться только при явном открытии файла внешней обработки. +Добавляем на основную форму реквизит с типом необходимого документа и команду «Печать» +![image-12.png](https://sinenikolsky.ru/s/QEWSKmHn7HKGJX4/download?path=%2F2025%2F04%2F16&files=5d781369-bfa1-4db3-8dcf-e4abd6c57bf2.png) +![image-13.png](https://sinenikolsky.ru/s/QEWSKmHn7HKGJX4/download?path=%2F2025%2F04%2F16&files=d87c1613-5ee7-4843-b61e-d73af33a7ee4.png) +## Основная форма +В модуле основной формы реализуем интерфейсный метод «Печать», который будет искать механизм БСП. В нем передаем нужные параметры и открываем форму настроек. +```bsl +&НаКлиенте +Процедура Печать(ИдентификаторКоманды, МассивОбъектов) Экспорт + + ПараметрыФормы = Новый Структура; + ПараметрыФормы.Вставить("ИдентификаторКоманды", ИдентификаторКоманды); + ПараметрыФормы.Вставить("МассивОбъектов", МассивОбъектов); + + ОткрытьФорму("ВнешняяОбработка.<ИмяВнешнейОбработки>.Форма.НастройкаПечати",ПараметрыФормы); + +КонецПроцедуры + +&НаКлиенте +Процедура ТестПечати(Команда) + + ПараметрыФормы = Новый Структура; + ПараметрыФормы.Вставить("ИдентификаторКоманды", ""); + МассивОбъектов = Новый Массив; + МассивОбъектов.Добавить(СверкаВзаимныхРасчетов); + ПараметрыФормы.Вставить("МассивОбъектов", МассивОбъектов); + + ОткрытьФорму("ВнешняяОбработка.<ИмяВнешнейОбработки>.Форма.НастройкаПечати",ПараметрыФормы); +КонецПроцедуры +``` +## Форма настроек печати +В скопированную форму настроек (в моем случае форма настроек документа «Сверка взаиморасчетов») добавляем реквизит `«Объект»`, с типом `«(ВнешняяОбработка.<ИмяВнешнейОбработки>)»`. Этот реквизит нужен для обращения к методам модуля объекта внешней обработки. + +Также добавляем реквизит с типом `«Список значений»`, куда будут записываться ссылки на документы, которые необходимо напечатать. В модуле формы `при создании на сервере` необходимо прописать заполнение этого реквизита +![image-14.png](https://sinenikolsky.ru/s/QEWSKmHn7HKGJX4/download?path=%2F2025%2F04%2F16&files=1bce3405-6d36-49ae-9a6a-26da857b8de7.png) +```bsl +&НаСервере +Процедура ПриСозданииНаСервере(Отказ, СтандартнаяОбработка) + + ... + // типовой код + ... + + МассивОбъектов = Новый Массив; + Если Параметры.Свойство("МассивОбъектов", МассивОбъектов) Тогда + + Для Каждого ОбъектПечати Из МассивОбъектов Цикл + СписокОбъектовПечати.Добавить(ОбъектПечати); + КонецЦикла; + + КонецЕсли; + +КонецПроцедуры +``` +В модуле формы настроек реализуем вызов метода `Печать` с клиента +```bsl +&НаКлиенте +Процедура ПечатьНаКлиенте() + + ТабДок = ПечатьНаСервере(); + КоллекцияПечатныхФорм = УправлениеПечатьюКлиент.НоваяКоллекцияПечатныхФорм("ПФ_MXL_АктСверкиВзаимныхРасчетовРазвернутоеСальдо"); + КоллекцияПечатныхФорм[0].ТабличныйДокумент = ТабДок; + КоллекцияПечатныхФорм[0].Экземпляров=1; + КоллекцияПечатныхФорм[0].СинонимМакета = "Акт сверки взаимных расчетов (развернутое сальдо)"; + УправлениеПечатьюКлиент.ПечатьДокументов(КоллекцияПечатныхФорм, Неопределено, ЭтотОбъект); + +КонецПроцедуры + +&НаСервере +Функция ПечатьНаСервере() + + ОбработкаОбъект = РеквизитФормыВЗначение("Объект"); + Возврат ОбработкаОбъект.ПечатьАктаСверкиВзаимныхРасчетовСРазвернутымСальдо(СписокОбъектовПечати.ВыгрузитьЗначения(), Новый СписокЗначений); + +КонецФункции +``` +В основной команде вызываем новый метод `ПечатьНаКлиенте()`. В моем случае это команда `«Сохранить»`, у которой я явно поменял заголовок на «Печать» +```bsl +&НаКлиенте +Процедура Сохранить(Команда) + + НастройкиПечати = НастройкиПечатиПоУмолчанию(); + ЗаполнитьЗначенияСвойств(НастройкиПечати, ЭтотОбъект); + СохранитьНастройкиНаСервере(НастройкиПечати); + Закрыть(НастройкиПечати); + + ПечатьНаКлиенте(); + +КонецПроцедуры +``` + +# Модуль объекта обработки +В модуле объекта внешней обработки реализуем методы печати +```bsl +Процедура Печать(МассивОбъектов, КоллекцияПечатныхФорм, ОбъектыПечати, ПараметрыВывода) Экспорт + + ИдентификаторКомандыПечати = "СН_АктСверкиВзаимныхРасчетовРазвернутоеСальдо"; + + ПечатнаяФорма = УправлениеПечатью.СведенияОПечатнойФорме(КоллекцияПечатныхФорм, ИдентификаторКомандыПечати); + Если ПечатнаяФорма <> Неопределено Тогда + ПечатнаяФорма.ТабличныйДокумент = СформироватьПечатнуюФормуАктСверкиВзаиморасчетов(МассивОбъектов, ОбъектыПечати); + ПечатнаяФорма.СинонимМакета = "Акт сверки взаимных расчетов (развернутое сальдо)"; + КонецЕсли; + + ФормированиеПечатныхФорм.ЗаполнитьПараметрыОтправки(ПараметрыВывода.ПараметрыОтправки, МассивОбъектов, КоллекцияПечатныхФорм); + +КонецПроцедуры +``` +Дальше — все как обычно. Получаем нужные данные, формируем ТабДок и т.д. В примере все это вызывается методом `СформироватьПечатнуюФормуАктСверкиВзаиморасчетов` \ No newline at end of file diff --git a/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Подключение БСП к собственной конфигурации.md b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Подключение БСП к собственной конфигурации.md new file mode 100644 index 0000000..e5c3e38 --- /dev/null +++ b/src/content/docs/knowledge-base/Примеры, сниппеты/БСП и типовые/Подключение БСП к собственной конфигурации.md @@ -0,0 +1,26 @@ +# Для полной интеграции БСП, не частичной +1. Создаем пустую конфигурацию +2. Запускаем в пользовательском режиме +3. Открываем обработку ПервоеВнедрениеБСП.epf из папки шаблона БСП (`%AppData%\1C\1cv8\tmplts\1c\SSL\3_1_9_144\ExtFiles`) +4. В обработке ничего не выбираем, оставляем только базовую функциональность +5. Сохраняем в файл +6. Сравниваем и объединяем нашу пустую конфигурацию с конфигурацией из шаблона БСП `1Cv8.cf`. Полную замену не делаем, ставим на поддержку (опционально) +В окне снимаем все флажки и загружаем настройки, сохраненные в п.5 +1. Назначаем имя, поставщика и версию конфигурации +2. Копируем общий модуль `ОбновлениеИнформационнойБазыБСП`, переименовываем (`ОбновлениеИнформационнойБазыСигма`, например). В нем редактируем процедуру `ПриДобавленииПодсистемы` и удаляем код из всех остальных методов модуля +```bsl +Процедура ПриДобавленииПодсистемы(Описание) Экспорт + + Описание.Имя = "Сигма"; // ИмяКонфигурации + Описание.Версия = "1.0.1.1"; //ВерсияКонфигурации + //Описание.ИдентификаторИнтернетПоддержки = "SSL"; + //Описание.РежимВыполненияОтложенныхОбработчиков = "Параллельно"; + //Описание.ПараллельноеОтложенноеОбновлениеСВерсии = "2.3.3.0"; + //Описание.ЗаполнятьДанныеНовыхПодсистемПриПереходеСДругойПрограммы = Истина; + +КонецПроцедуры +``` +9. В модуле `ПодсистемыКонфигурацииПереопределяемый` в методе `ПриДобавленииПодсистем` пишем +```bsl +МодулиПодсистем.Добавить("ОбновлениеИнформационнойБазыСигма") +``` \ No newline at end of file