Новичок новичку: как добавить программно кнопку на форму путем расширения

0. user1312100 147 13.05.21 11:04 Сейчас в теме
Рассказываю очень простым языком, как добавить программно кнопку в типовую конфигурацию.
Сам новичок в этом деле и рассказываю на "новичковом" языке.

Перейти к публикации

Комментарии
В избранное Подписаться на ответы Сортировка: Древо развёрнутое
Свернуть все
1. vaxhab 15 13.05.21 12:26 Сейчас в теме
Спасибо за напоминалку
Uncore; DrAku1a; +2 Ответить
2. BomjBandit 13.05.21 14:16 Сейчас в теме
Я думаю, что все-таки не хватает картинок, которые показывают результат работы кода.
Если ещё докапываться, то
ЭтаФорма.
лишнее, но мб новичкам нагляднее.
DrAku1a; Sla; +2 Ответить
3. tommadm 15.05.21 08:58 Сейчас в теме
Читайте внимательнее, написано автором: "Отступление: Так как мы будем вставлять новый элемент в туже форму, в которой находится код, то «ЭтаФорма» можно не указывать.".
adhocprog; DrAku1a; +2 Ответить
4. Bassgood 1356 18.05.21 09:52 Сейчас в теме
(2) А еще вместо "ЭтаФорма" (вы его не найдете в СП) требуется использовать более универсальный "ЭтотОбъект"
adhocprog; +1 Ответить
5. Bassgood 1356 18.05.21 10:57 Сейчас в теме
А зачем в принципе что-то добавлять в интерфейс через расширение программным способом, ведь расширения призваны облегчить доработку таких вещей путем интерактивных добавлений на форму новых команд, кнопок и обработчиков к ним?
6. user1312100 147 18.05.21 15:32 Сейчас в теме
(5)В статье написано зачем.
7. SuhoffGV 19.05.21 10:15 Сейчас в теме
(5) Например, если вам нужно добавить одинаковую кнопку в несколько разных форм. В типовых для этого есть спец процедура общего модуля вызываемая в ПриСозданииНаСервере для всех (почти) форм.
Bassgood; +1 Ответить
8. Bassgood 1356 19.05.21 11:06 Сейчас в теме
(7) Если команда добавляется в множество форм - согласен, но в случае добавления ее в пару форм - создавать их программно как-то не очень.
Если нужно добавить одну и ту же команду в несколько форм - можно создать в дереве конфы общую команду (или команду объекта метаданных), указать в ней нужные типы данных объектов (тип параметра команды), в формах которых ее следует отображать, ну и задать ее расположение на формах через группу команд
9. user1312100 147 19.05.21 14:02 Сейчас в теме
Ребят, мне удалить статью?
10. Bassgood 1356 20.05.21 12:02 Сейчас в теме
(9) Зачем, вы будете удалять все публикации, которые обсуждаются или критикуются сообществом? :)
11. user1312100 147 20.05.21 14:12 Сейчас в теме
Дак получается статья ненужная
12. BomjBandit 20.05.21 15:44 Сейчас в теме
(11) Нужная-нужная, никто в здравом уме не будет дорабатывать серьезные базы через расширения. Новые реквизиты создают в дереве метаданных, а их элементы на форму выводят программно. Можешь до кучи сделать статью по программному выводу таблиц, полей и их наглядную иерархию.
28. wertep 9 21.02.23 16:05 Сейчас в теме
(11) Очень даже нужная статья. Особенно сейчас, когда в линейке конфигураций ERP, КА, УТ появились общие модули "МодификацияКонфигурацииПереопределяемый" и через них можно обрабатывать события добавленных элементов. Еще не полностью функционал реализован, но начало положено.

Еще наверное стоит добавить нюанс с подсказкой. У кнопки на форме подсказка отсутствует, просто нет соответствующего свойства. Чтобы у кнопки была подсказка, ее нужно добавлять к команде. Наступил на такую особенность.
user1648665; +1 Ответить
22. пользователь 22.10.21 13:16
Сообщение было скрыто модератором.
...
13. psa247 18 15.10.21 19:49 Сейчас в теме
Такой вопрос. Мне нужно не закрывая документ обновить подменю программно созданных кнопок командной панели.
Уточню свой вопрос. Можно ли динамически пересоздать кнопки помимо процедуры "ПриСозданииНаСервере"
14. user1312100 147 18.10.21 06:33 Сейчас в теме
(13) Я не пробовал, но, думаю, можно. Самое главное определиться с событием: пересоздание кнопок будет при нажатии других кнопок или при изменении реквизита - решайте сами... Попробуйте там вызвать код, а потом обновить форму.
15. psa247 18 18.10.21 06:57 Сейчас в теме
Доброго дня! Респект разработчикам, все получилось и несложно в итоге. Моя задача была - генерировать по событию (считай - в любой момент) заранее неизвестное количество кнопок подменю и чтобы 1С умела конкретно в каждом нажатии сделать что-то уникальное. Само обновление количества кнопок Я сделал процедуру наКлиенте, которая вызывает процедуру НаСервере, которая, соответственно вызывает процедуру создания кнопок подменю аналогично той, что приведена в примере. А уникальность действий обеспечиваю ГУИДами, которые передаю через "Команду". Если кому понадобится - приложу пример
17. user1312100 147 18.10.21 08:08 Сейчас в теме
16. user1312100 147 18.10.21 08:07 Сейчас в теме
Приведите пример процедуры, в которой идет обновление формы... А уж куда ее вставить люди решат сами.
18. psa247 18 18.10.21 08:13 Сейчас в теме
&НаКлиенте
Процедура ПРОФ_ПриОткрытииПосле(Отказ)
	ОбновитьКнопкиМеню();
	ПереключитьДоступностьКнопок(ЛОЖЬ);
КонецПроцедуры

&НаКлиенте
Процедура ПереключитьДоступностьКнопок(ИдетЗапись)
	ЭтаФорма.Элементы.ПРОФ_КнопкаОстановитьЗапись.Доступность = ИдетЗапись;
	ЭтаФорма.Элементы.ПРОФ_ВоспроизвестиЗапись.Доступность = НЕ ИдетЗапись;
	ЭтаФорма.Элементы.ПРОФ_УдалитьЗапись.Доступность = НЕ ИдетЗапись;
	ЭтаФорма.Элементы.ПРОФ_КнопкаНачатьЗапись.Доступность = НЕ ИдетЗапись;
КонецПроцедуры

&НаКлиенте
Процедура ОбновитьКнопкиМеню()
	ОбновитьКнопкиНаСервере();
	ПереключитьДоступностьКнопок(ЛОЖЬ);
КонецПроцедуры

&НаСервере
Процедура ОбновитьКнопкиНаСервере()
	ГУИД = Объект.Ссылка.УникальныйИдентификатор();
	ИмяКаталога = ПолучитьПутьХраненияЗаписей()+ГУИД;
	КаталогНаДиске = Новый Файл(ИмяКаталога);
	Если КаталогНаДиске.Существует() = ЛОЖЬ Тогда Возврат; КонецЕсли;

	Файлы = НайтиФайлы(ИмяКаталога, "*.mp3", ЛОЖЬ);
	// Удаляем все кнопки
	
	Для Каждого Тек ИЗ ЭтаФорма.Элементы.ПРОФ_ВоспроизвестиЗапись.ПодчиненныеЭлементы Цикл
		ЭтаФорма.Элементы.Удалить(Тек);			
	КонецЦикла;
	Для Каждого Тек ИЗ ЭтаФорма.Элементы.ПРОФ_УдалитьЗапись.ПодчиненныеЭлементы Цикл
		ЭтаФорма.Элементы.Удалить(Тек);			
	КонецЦикла;
	
	Для Каждого Тек ИЗ Файлы Цикл
		Струк = ПолучитьСведенияОФайле(Тек.ИмяБезРасширения);
		ИмяКнопки = СтрЗаменить(Тек.ИмяБезРасширения, "~", "_");
		ИмяКнопки = СтрЗаменить(ИмяКнопки, "-", "_");
		ИмяКнопки = Струк.ГУИД;
		
		ЗаголовокКнопки = "Запись от " + Струк.ДатаЗаписи+", время: "+Струк.Продолжительность+" "+Струк.Автор;
		
		НайденныйЭлементФормы = ЭтаФорма.Элементы.Найти("П_" + ИмяКнопки);
		Если НайденныйЭлементФормы <> Неопределено  Тогда
            ЭтаФорма.Элементы.Удалить(НайденныйЭлементФормы);
        КонецЕсли;		
		
		Если ЭтаФорма.Команды.Найти("П_" + ИмяКнопки) = Неопределено Тогда
			НоваяКоманда = ЭтаФорма.Команды.Добавить("П_" + ИмяКнопки);
	        НоваяКоманда.Действие = "ПроигратьЗапись";
			НоваяКоманда.Заголовок = ЗаголовокКнопки;
		КонецЕсли;		
			
		НоваяКнопка = Элементы.Добавить("П_" + ИмяКнопки, Тип("КнопкаФормы"), ЭтаФорма.Элементы.ПРОФ_ВоспроизвестиЗапись);
		НоваяКнопка.Вид = ВидКнопкиФормы.КнопкаКоманднойПанели;
		НоваяКнопка.ИмяКоманды = "П_" + ИмяКнопки;
		НоваяКнопка.Заголовок = ЗаголовокКнопки;
		НоваяКнопка.ТолькоВоВсехДействиях = ЛОЖЬ;  
		НоваяКнопка.Пометка = ЛОЖЬ;
		
		
		ЗаголовокКнопки = "Удалить запись от " + Струк.ДатаЗаписи +", время: "+Струк.Продолжительность+" "+Струк.Автор;
		
		НайденныйЭлементФормы = ЭтаФорма.Элементы.Найти("У_" + ИмяКнопки);
		Если НайденныйЭлементФормы <> Неопределено  Тогда
            ЭтаФорма.Элементы.Удалить(НайденныйЭлементФормы);
        КонецЕсли;		
		
		Если ЭтаФорма.Команды.Найти("У_" + ИмяКнопки) = Неопределено Тогда
			НоваяКоманда = ЭтаФорма.Команды.Добавить("У_" + ИмяКнопки);
	        НоваяКоманда.Действие = "УдалитьЗапись";
			НоваяКоманда.Заголовок = ЗаголовокКнопки;
		КонецЕсли;
		
		НоваяКнопка = Элементы.Добавить("У_" + ИмяКнопки, Тип("КнопкаФормы"), ЭтаФорма.Элементы.ПРОФ_УдалитьЗапись);
		НоваяКнопка.Вид = ВидКнопкиФормы.КнопкаКоманднойПанели;
		НоваяКнопка.ИмяКоманды = "У_" + ИмяКнопки;
		НоваяКнопка.Заголовок = ЗаголовокКнопки;
		НоваяКнопка.ТолькоВоВсехДействиях = ЛОЖЬ;
		НоваяКнопка.Пометка = ЛОЖЬ;
	КонецЦикла;
	
КонецПроцедуры

&НаСервере
Процедура ПРОФ_ПриСозданииНаСервереПосле(Отказ, СтандартнаяОбработка)
	ОбновитьКнопкиНаСервере();
КонецПроцедуры

&НаСервере
Функция ПолучитьСведенияОФайле(Стр)
	Струк = Новый Структура;
	Струк.Вставить("ДатаЗаписи", Неопределено);
	Струк.Вставить("Автор", Неопределено);
	Струк.Вставить("ГУИД", Неопределено);
	Струк.Вставить("Продолжительность", Неопределено);
	
	// Формат имени файла: ДатаЗаписи_Продолжительность_Автор_ГУИД
	Струк.ДатаЗаписи = ПрочитатьДоСледПалки(0, 1, Стр);
	Струк.Продолжительность = ПрочитатьДоСледПалки(1, 2, Стр);
	Струк.Автор = ПрочитатьДоСледПалки(2, 3, Стр);
	Струк.ГУИД = ПрочитатьДоСледПалки(3, 4, Стр);
	
	Если Струк.ГУИД <> Неопределено Тогда
		Струк.ГУИД = СтрЗаменить(Струк.ГУИД, "-", "_");
	КонецЕсли;
	
	Возврат(Струк);
КонецФункции
Показать
19. psa247 18 18.10.21 08:14 Сейчас в теме
Без проблем, кто не разберется - переспросит
20. user1312100 147 18.10.21 08:36 Сейчас в теме
(19)Я правильно понял, что обновлять\перечитывать форму для отображения новых кнопок не нужно?
21. psa247 18 18.10.21 08:46 Сейчас в теме
Да. Я вызываю каждый раз процедуру, которая удаляет все кнопки подменю. Потом заново их пересоздает. Обновление формы не требуется. Открывать/закрывать документ, соответственно, тоже
23. admnnts 01.05.22 08:44 Сейчас в теме
Ожидал от статьи немного большего, т.к. она называется "как добавить программно кнопку на форму путем расширения".
Зачем форму тащить в расширение, если можно добавить через дабавление в расширение процедуры &После("ПриСозданииНаСервере") и тут пиши кнопку программно.
Вот теперь мой искомый вопрос, как заставить эту кнопку выполнять процедуру при нажатии, никак не получается всякими способами, подскажите, пожалуйста, что не так делаю.

При этом кроме самого справочника ничего не добавлялось в расширение. Не формы, не реквизиты. Все описывается в общем модуле - МодификацияКонфигурацииПереопределяемый.
вот код:
&После("ПриСозданииНаСервере")
Процедура ДопЭлементыПрограммно_ПриСозданииНаСервере(Форма, Отказ, СтандартнаяОбработка)
	Если Форма.ИмяФормы = "Справочник.КлючиАналитикиУчетаНоменклатуры.Форма.ФормаЭлемента" Тогда
		
		ПолеНоменклатураКод77 			= Форма.Элементы.Вставить("НоменклатураКод77", Тип("ПолеФормы"));
		ПолеНоменклатураКод77.Заголовок	= "Код 7.7";
		ПолеНоменклатураКод77.Вид		= ВидПоляФормы.ПолеНадписи;
			
		НоменклатураКод77 = Форма.Объект.Номенклатура;
		ПолеНоменклатураКод77.ПутьКДанным	= "Объект.Номенклатура.КодИз77";
			
		//СсылкаНоменклатураКод77				= Форма.Элементы.Вставить("СсылкаКод77", Тип("ДекорацияФормы"));
		//СсылкаНоменклатураКод77.Заголовок 	= "Скопировать код";
		//СсылкаНоменклатураКод77.Гиперссылка	= Истина;
		//СсылкаНоменклатураКод77.УстановитьДействие("Нажатие", "ДопЭлементыПрограммно_ВыполнитьКопированиеКода77");
		
		
		КомандаНоменклатураКод77			= Форма.Команды.Добавить("СкописроватьКод");
		КомандаНоменклатураКод77.Заголовок	= "Скопировать код";
		КомандаНоменклатураКод77.Действие	= "ДопЭлементыПрограммно_ВыполнитьКопированиеКода77";  
		
		СсылкаНоменклатураКод77				= Форма.Элементы.Вставить("СсылкаКод77", Тип("КнопкаФормы"));
		СсылкаНоменклатураКод77.ИмяКоманды	= "СкописроватьКод";
		СсылкаНоменклатураКод77.Вид			= ВидКнопкиФормы.Гиперссылка;
		
	КонецЕсли;

КонецПроцедуры

&НаКлиенте
Процедура ДопЭлементыПрограммно_ВыполнитьКопированиеКода77()

	Сообщить("Должно произойти копирование кода");

КонецПроцедуры // ВыполнитьКопированиеКода77()
Показать
26. wrooom 150 07.09.22 12:43 Сейчас в теме
(23) Нашли решение этой задачи? У меня всегда видна кнопка через "МодификацияКонфигурацииПереопределяемый.ПриСозданииНаСервере", но чтоб кнопка выполняло какое-то действие не получается.
29. wertep 9 21.02.23 16:23 Сейчас в теме
(26) На данный момент через этот механизм можно выполнять действия на клиенте. В действия команды записать вызов "Подключаемый_ВыполнитьПереопределяемуюКоманду".
Команда.Действие = "Подключаемый_ВыполнитьПереопределяемуюКоманду"


И потом перехватить это в модуле "МодификацияКонфигурацииКлиентПереопределяемый".
&После("ВыполнитьПереопределяемуюКоманду")
Процедура Префикс_ВыполнитьПереопределяемуюКоманду(Форма, Команда, ДополнительныеПараметры) Экспорт
	
	// Обработчики команд модифицированных форм
	Если Форма.ИмяФормы = "ПолноеИмяМодифицированнойФормы" Тогда
		
		Если Команда.Имя = "ИмяКоманды" Тогда
			// Обработка команды 
			ОбщегоНазначенияКлиент.СообщитьПользователю("Выполнена команда");
		КонецЕсли;
		
	КонецЕсли;
	
КонецПроцедуры
Показать


С выполнением кода на сервере пока все плохо. Только через костыли подключаемых команд.
24. user1312100 147 11.05.22 10:24 Сейчас в теме
Вот реально работающий пример, добавляющий кнопку на стандартную форму движений документа в БП 3.0
&НаСервере
Процедура UK_ПриСозданииНаСервереПосле(Отказ, СтандартнаяОбработка)
	//Создаем Команду
	НоваяКоманда = ЭтаФорма.Команды.Добавить("ДД");
	НоваяКоманда.Действие = "ВыполнитьКоманду_UK";//Имя процедуры
	НоваяКоманда.Заголовок = "ДобавитьДвижение";
	//Создаем Кнопку перед указанным
	НоваяКнопка = ЭтаФорма.Элементы.Вставить("КнопкаВНачале", Тип("КнопкаФормы"), ЭтаФорма, ЭтаФорма.ПодчиненныеЭлементы[0]);
	НоваяКнопка.ИмяКоманды = "ДД";
	//скрываем кнопку и показываем только для реализации и для услуг
	НоваяКнопка.Видимость = Ложь;
	Если  ТипЗнч(ДокументДвижений) = Тип("ДокументСсылка.РеализацияТоваровУслуг") Тогда
		Если ДокументДвижений.ВидОперации = Перечисления.ВидыОперацийРеализацияТоваров.Услуги Тогда
			НоваяКнопка.Видимость = Истина
		КонецЕсли;
	КонецЕсли;	
КонецПроцедуры

&НаКлиенте
Процедура ВыполнитьКоманду_UK(Команда)
	
    РучнаяКорректировка = Истина;
	//Модифицированность = Истина;
	ВыполнитьДействияПриИзмененииРучнойКорректировки();
	Кол = ХозрасчетныйНаборЗаписей.Количество();
	Стр = ХозрасчетныйНаборЗаписей[0];
	Нов = ХозрасчетныйНаборЗаписей.Добавить();
		
	Нов.Активность = Истина;
	Нов.ВалютаДт = Null;
	Нов.ВалютаКт = Null;
	Нов.ВалютнаяСуммаДт = Null;
	Нов.ВалютнаяСуммаКт = Null;
	Нов.ВалютныйДтДоступность = Ложь;
	Нов.ВалютныйКтДоступность = Ложь;
	Нов.КоличественныйДтДоступность = Ложь;
	Нов.КоличественныйКтДоступность = Ложь;
	Нов.КоличествоДт = Null;
	Нов.КоличествоКт = Null;
	Нов.НадписьВР = "ВР:";
	Нов.НадписьКоличествоДт = "";
	Нов.НадписьКоличествоКт = "";
	Нов.НадписьНУ = "НУ:";
	Нов.НадписьПР = "ПР:";
	Нов.НеКорректироватьСтоимостьАвтоматически = Ложь;
	Нов.Организация = Стр.Организация;
	Нов.Период = Стр.Период;
	Нов.ПодразделениеДт = Null;
	Нов.ПодразделениеДтДоступность = Ложь;
	Нов.ПодразделениеКт = Null;
	Нов.ПодразделениеКтДоступность = Ложь;
	Нов.Регистратор = Стр.Регистратор;
	Нов.Содержание = Стр.Содержание;
	Нов.СубконтоДт1 = Стр.СубконтоКт1;
	СубконтоДт1Доступность = Истина;
	Нов.СубконтоДт2 = Стр.СубконтоКт2;
	Нов.СубконтоДт2Доступность = Истина;
	Нов.СубконтоДт3 = Неопределено;
	Нов.СубконтоДт3Доступность = Ложь;
	Нов.СубконтоКт1 = Стр.СубконтоДт1;
	СубконтоКт1Доступность = Истина;
	Нов.СубконтоКт2 = Стр.СубконтоДт2;
	СубконтоКт2Доступность = Истина;
	Нов.СубконтоКт3 = Стр.Регистратор;
	СубконтоКт3Доступность = Истина;
	Нов.Сумма = ?(Кол > 1, Стр.СуммаНУКт, Стр.Сумма);
	Нов.СуммаВРДт = Стр.СуммаВРДт;
	Нов.СуммаВРКт = Стр.СуммаВРКт;
	Нов.СуммаНУДт = ?(Кол > 1, Стр.СуммаНУКт, Стр.СуммаНУДт);
	Нов.СуммаНУКт = Стр.СуммаНУКт;
	Нов.СуммаПРДт = Стр.СуммаПРДт;
	Нов.СуммаПРКт = Стр.СуммаПРКт;
	Нов.УточнениеПериода = Стр.УточнениеПериода;
	
	Нов.СчетДт = ПолучитьПланСчетов_UK("91.02");
	ОбработатьИзменениеСчета_UK("Дт", Нов);
	Нов.СчетКт = ПолучитьПланСчетов_UK("76.06");
	ОбработатьИзменениеСчета_UK("Кт", Нов);
		
КонецПроцедуры

Функция ПолучитьПланСчетов_UK (Стр)
	Возврат ПланыСчетов.Хозрасчетный.НайтиПоКоду(Стр);
КонецФункции

&НаКлиенте
Процедура ОбработатьИзменениеСчета_UK(ДтКт, ТекущиеДанные)

	//ТекущиеДанные = ДтКт;
	
	ПоляФормы = Новый Структура("Субконто1,Субконто2,Субконто3");
	ПоляФормы.Субконто1 = "ХозрасчетныйСубконто" + ДтКт + "1";
	ПоляФормы.Субконто2 = "ХозрасчетныйСубконто" + ДтКт + "2";
	ПоляФормы.Субконто3 = "ХозрасчетныйСубконто" + ДтКт + "3";
	БухгалтерскийУчетКлиентСервер.ПриВыбореСчета(ТекущиеДанные["Счет" + ДтКт], ЭтаФорма, ПоляФормы, Неопределено, Истина, Ложь);
	
	ПоляОбъекта = Новый Структура("Субконто1,Субконто2,Субконто3,Подразделение,Валютный,Количественный,Организация");
	ПоляОбъекта.Субконто1      = "Субконто" + ДтКт + "1";
	ПоляОбъекта.Субконто2      = "Субконто" + ДтКт + "2";
	ПоляОбъекта.Субконто3      = "Субконто" + ДтКт + "3";
	ПоляОбъекта.Подразделение  = "Подразделение" + ДтКт;
	ПоляОбъекта.Валютный       = "Валютный" + ДтКт;
	ПоляОбъекта.Количественный = "Количественный" + ДтКт;
	ПоляОбъекта.Организация    = Объект.Организация;
	БухгалтерскийУчетКлиентСервер.ПриИзмененииСчета(ТекущиеДанные["Счет" + ДтКт], ТекущиеДанные, ПоляОбъекта, Истина,, Ложь);
	
	ИзменитьПараметрыВыбораПолейСубконто(ЭтаФорма, ДтКт);
	
	ЗаполнитьНадписиВПроводке(ТекущиеДанные);
	
КонецПроцедуры
Показать
user751607; +1 Ответить
25. a-m-gv 35 08.06.22 14:13 Сейчас в теме
Ситуацию 2 стоит дополнить, например:
Элементы.Переместить(Элт,Элементы.ГруппаШапка,Элементы.ГруппаВидаОплаты);

//Элт - вновь созданный элемент,
// переместили в ГруппуШапка, перед Группой Вида оплаты
adhocprog; +1 Ответить
27. 1c-asu 03.12.22 21:23 Сейчас в теме
Спасибо большое. Очень пригодилось для Универсального отчета.
30. MaratM 27.04.23 13:33 Сейчас в теме
Спасибо за статью. Так бы 1с документацию писало, цены бы ей не было.
Оставьте свое сообщение
Вакансии
Программист/тестировщик
Москва
зарплата от 130 000 руб. до 150 000 руб.
Полный день

Ведущий разработчик 1С / Team lead отдела разработки 1С
Москва
зарплата от 300 000 руб. до 300 000 руб.
Полный день

Программист 1С
Москва
зарплата от 130 000 руб. до 150 000 руб.
Полный день

Бизнес-аналитик
Москва
зарплата от 130 000 руб. до 150 000 руб.
Полный день

Ведущий консультант 1С:ERP
Москва
зарплата от 200 000 руб.
Полный день