Сортировка в табличной части (таблица значений) формы управляемого приложения

30.05.23

Разработка - Работа с интерфейсом

Для табличной части (таблица значений) управляемой формы добавляем в контекстное меню две команды: сортировать по убыванию и возрастанию и две процедуры. И получаем возможность сортировки по любой колонке. Пример "Номенклатура.артикул" , "Номенклатура.артикул.код" и т.д.

В описанных ранее сортировках таблицы значений формы, надо обязательно указать название табличной части и поля для сортировки.  Для всех своих таблиц значения, размещаемых на форме, использую универсальный код, который работает для любой табличной части и любого поля. 

Для новой формы остаётся перенести команды сортировки, код подпрограмм и включаю команды в контекстное меню   табличной части формы.

И сортировка работает для любого поля.

 


 

Все реализовано достаточно просто, см.код.

// Сортировка

&НаКлиенте
Процедура сортировать(Команда)

      ТЭ_имя_объект=ТекущийЭлемент.Имя;
      ТЭ_поле=ТекущийЭлемент.ТекущийЭлемент.имя;

      имяКолонки=СтрЗаменить( ТЭ_поле,ТЭ_имя_объект,"");
      если команда.имя="СортировкаУ" тогда
           СортироватьНасервере(ТЭ_имя_объект,имяКолонки,"Убыв");
      иначе
          СортироватьНасервере(ТЭ_имя_объект,имяКолонки,"Возр");
      конецесли;
КонецПроцедуры


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


        пдТ= ТекущийЭлемент.ПутьКДанным;
        ПД= ТекущийЭлемент.ТекущийЭлемент.ПутьКДанным;
        пд= СтрЗаменить(ПД,пдТ+".","");
        // получим основную и дополнительную колонку
         основннаяКолонка=лев(пд,найти(пд,".")-1);
         допколонка=СтрЗаменить(пд,основннаяКолонка+".","");
         тз.Колонки.Добавить( имяКолонки);
        //заполним поля для сортировки
        для каждого элементаТЗ из тз цикл
            тзобъекта= элементаТз[основннаяКолонка] ;
            элементаТз[имяКолонки]=вычислить("тзобъекта."+допколонка);
         конеццикла;
         тз.Сортировать(имяКолонки+" "+ВидСортировки);
         индексТЗпослеСортировки=ТЗ.Индекс(строкаВтаблице);
         ТЗ_объекта.Загрузить(тз);
          //восстановим курсор на нужную строку
          позиция_стр= ТЗ_объекта.Получить( индексТЗпослеСортировки);
          ткэлемент.ТекущаяСтрока=позиция_стр.ПолучитьИдентификатор();
     конецесли;

     ЗначениеВРеквизитФормы(Эф,"Объект");
конецпроцедуры

// Сортировка

 

Сортировка Таблица значений.

См. также

Работа с интерфейсом Рабочее место Платформа 1С v8.3 Управляемые формы Конфигурации 1cv8 Платные (руб)

Богатый редактор картинок 1С предназначен для обработки изображений в режиме «Предприятие», с возможностью РИСОВАТЬ на них. Поддерживается работа как в обычных формах (толстый клиент) так и на управляемых формах (тонкий клиент). Обработка позволяет редактировать как картинки, хранимые в базе, так и графические файлы с диска на файловой системе. Помимо базовых функций (изменение размеров, преобразование формата, обрезание картинки, повороты и т.п.) – редактор имеет богатый набор инструментов для рисования. Доступна функция вставки изображения из буфера обмена. Объект может быть использован: на стороне клиента, на стороне сервера, из внешнего соединения. Обработка будет особенно полезна тем, кто вносит картинки в базу (изображения номенклатуры, фотографии физических лиц и т.п.). Функционал реализуется с использованием JavaScript и бесплатного ПО ImageMagick (без использования внешних компонент).

6000 руб.

16.01.2015    62998    44    59    

82

Работа с интерфейсом Программист Платформа 1С v8.3 Конфигурации 1cv8 1С:ERP Управление предприятием 2 Платные (руб)

Обработка предназначена для создания и управления дашбордами.

2400 руб.

29.06.2020    18870    26    6    

41

Работа с интерфейсом Программист Платформа 1С v8.3 Управляемые формы Конфигурации 1cv8 Россия Платные (руб)

Редактор графов в 1С - внешний отчет, который формирует графы на основе таблицы значений, используя рисунки табличного документа. Есть возможность добавления, редактирования объектов графа и выгрузки результата в таблицу значений.

1500 руб.

06.10.2020    10241    7    7    

10

Работа с интерфейсом Программист Стажер Платформа 1С v8.3 Бесплатно (free)

Это инструкция по дизайну форм в среде 1С. Гайд охватывает рекомендации и стандарты для оптимизации пользовательского интерфейса. В гайде содержатся указания по использованию элементов интерфейса, включая как основные, так и продвинутые аспекты. Предоставляются также примеры и антипримеры для наглядного понимания принципов дизайна

20.08.2024    16922    mrXoxot    43    

121

Работа с интерфейсом Программист Платформа 1С v8.3 Бесплатно (free)

Пример простого и симпатичного прогресс-бара в динамическом списке, без картинок, используя редактирование запроса.

27.05.2024    7457    smielka    37    

100

Работа с интерфейсом Платформа 1С v8.3 1С:Бухгалтерия 3.0 1С:Управление торговлей 11 1С:Зарплата и Управление Персоналом 3.x 1С:Управление нашей фирмой 3.0 Бесплатно (free)

Добавьте новогоднего настроения! Расширение создает декорацию в виде гирлянды на некоторых формах объектов.

27.12.2023    14836    921    elcoan    47    

117

Инструментарий разработчика Работа с интерфейсом Программист Платформа 1С v8.3 Конфигурации 1cv8 Абонемент ($m)

Подходит для создания web-страниц для замены управляемых форм 1С, красивых отчетов, интерфейса мобильного приложения на платформе 1С и для простых страниц веб-сайтов.

2 стартмани

10.04.2023    11926    162    acces969    31    

124
Комментарии
Подписаться на ответы Инфостарт бот Сортировка: Древо развёрнутое
Свернуть все
1. ixijixi 1913 30.05.23 16:03 Сейчас в теме
У Вас опечатка, нужно Возр
Прикрепленные файлы:
2. sandr13 35 01.06.23 12:58 Сейчас в теме
Вообще-то для возрастания - "Возр" можно не указывать.
3. Niks27 29.06.23 04:25 Сейчас в теме
В УНФ сделал так:

&НаКлиенте
Процедура РД_СортироватьВместо(Команда)	
	ТЭ_имя_объект = ТекущийЭлемент.Имя;
	ТЭ_поле = ТекущийЭлемент.ТекущийЭлемент.имя;
	
	ИмяКолонки = СтрЗаменить( ТЭ_поле,ТЭ_имя_объект,"");
	Если Команда.Имя = "РД_СортироватьУ" тогда
		СортироватьНаСервере(ТЭ_имя_объект,имяКолонки,"Убыв");
	иначе
		СортироватьНаСервере(ТЭ_имя_объект,имяКолонки,"Возр");
	Конецесли;
КонецПроцедуры

&НаСервере
Процедура СортироватьНаСервере(имяОбъекта,имяКолонки,ВидСортировки)
 
	ТЗ_объекта = ЭтаФорма.Объект.Запасы;
	тз = ТЗ_объекта.Выгрузить();
	колонкасортировки = тз.Колонки.Найти(ИмяКолонки);
	Если Колонкасортировки <> Неопределено Тогда
		ТЗ_объекта.Сортировать(имяКолонки+" "+ВидСортировки);
	Иначе
		//запомним текущую строку
		Ткэлемент = элементы[имяОбъекта];
		Позиция_стр = Тз_объекта.НайтиПоИдентификатору(ткэлемент.ТекущаяСтрока);
		Идекс_позиции = Тз_объекта.индекс(позиция_стр);
		СтрокаВтаблице = тз.Получить(Идекс_позиции);
		
		ПД = имяКолонки;
		// получим основную и дополнительную колонку
		основннаяКолонка=лев(пд,найти(пд,"_")-1);
		допколонка=СтрЗаменить(пд,основннаяКолонка+"_","");
		тз.Колонки.Добавить( имяКолонки);
		//заполним поля для сортировки
		Для Каждого элементаТЗ из тз Цикл
			тзобъекта = элементаТз[основннаяКолонка] ;
			элементаТз[ИмяКолонки] = вычислить("тзобъекта." + допколонка);
		КонецЦикла;
		тз.Сортировать(имяКолонки + " " + ВидСортировки);
		индексТЗпослеСортировки = ТЗ.Индекс(строкаВтаблице);
		ТЗ_объекта.Загрузить(тз);
		//восстановим курсор на нужную строку
		позиция_стр = ТЗ_объекта.Получить( индексТЗпослеСортировки);
		ткэлемент.ТекущаяСтрока=позиция_стр.ПолучитьИдентификатор();
	КонецЕсли;  
	
КонецПроцедуры
Показать
4. Serg2000mr 685 19.08.23 12:18 Сейчас в теме
Можно проще:

&НаКлиенте
Процедура Сортировать(Команда)
	
	Коллекция = ТекущийЭлемент;
	Колонка = Коллекция.ТекущийЭлемент;
	
	Если Команда.Имя="СортировкаУ" тогда
		СортироватьНаСервере(Коллекция.Имя, Колонка.Имя, "Убыв");
	Иначе
		СортироватьНаСервере(Коллекция.Имя, Колонка.Имя, "Возр");
	КонецЕсли;
	
КонецПроцедуры

&НаСервере
Процедура СортироватьНаСервере(КоллекцияИмя, КолонкаИмя, НаправлениеСортировки) 
	
	КоллекцияПутьКДанным = Элементы[КоллекцияИмя].ПутьКДанным;
	КолонкаПутьКДанным = СтрРазделить(Элементы[КолонкаИмя].ПутьКДанным, ".", Ложь)[1];
	
	ЭтотОбъект[КоллекцияПутьКДанным].Сортировать(КолонкаПутьКДанным + " " + НаправлениеСортировки);
	
КонецПроцедуры
Показать


Или вообще без кода:
в редакторе формы Команды - СтандартныеКоманды - Ищем нужную таблицу - Перетягиваем нужные стандартные команды в контекстное меню.
Прикрепленные файлы:
atom_1C; cheshirshik; slige; mentozavr; Дмитрий2254; user1316759; Летяга; klaus38; Niks27; +9 Ответить
5. kembrik 10 04.10.23 13:55 Сейчас в теме
(4) Вариант отличный, но стандартные команды не будут работать если мы притащили в таблицу реквизиты "через точку", например дату и номер документа который рядом есть по ссылке
shard; nuno21; +2 Ответить
11. cheshirshik 70 06.11.24 10:43 Сейчас в теме
(4) Спасибо. Без кода - это самый оптимальный способ.
Serg2000mr; +1 Ответить
6. shard 281 12.01.24 16:09 Сейчас в теме
Прошу прощения за мой англицкий, но пришлось наваять такое (но такова политика партии):
&AtClient
Procedure SortBL(Command)
	  CurrRow = CurrentItem.CurrentRow;
      if Command.Name="SortDESC" then
          SortBLAtServer(CurrentItem.Name,CurrentItem.CurrentItem.Name,"DESC", CurrRow);
      else
          SortBLAtServer(CurrentItem.Name,CurrentItem.CurrentItem.Name,"ASC",  CurrRow);
      endif;
      CurrentItem.CurrentRow = CurrRow;
EndProcedure

&AtServer
Procedure SortBLAtServer(ItemName,ColumnName,val DirSorting, CurrRow)
	ItemForm = FormAttributeToValue("Object");
	TableObject = ItemForm[ItemName];
	VT = TableObject.Unload();
	ColumnSort = VT.Columns.Find(ColumnName);
	CurrRowID = VT[CurrRow];
	if ColumnSort<>Undefined then
		TableObject.Sort(ColumnName+" "+DirSorting);
	else
		AttibuteNameArray = StrSplit(Items[ItemName].ChildItems.Find(ColumnName).DataPath,".");
		AttibuteNameArray.Delete(0);
		AttibuteNameArray.Delete(0);
		MainValueName = AttibuteNameArray[0];
		ValuesArray = VT.UnloadColumn(MainValueName);
		AttibuteName = StrConcat(AttibuteNameArray,"_");
		AttibuteNameArray.Delete(0);
		VT.Columns.Add(AttibuteName);
		MapValues = ObjectsAttributeValues(ValuesArray, StrConcat(AttibuteNameArray,"."));
		AttibuteNameInMap = StrConcat(AttibuteNameArray,"");
		for each str in VT do
			str[AttibuteName] = MapValues[str[MainValueName]][AttibuteNameInMap];
		enddo;
		VT.Sort(AttibuteName+" "+DirSorting);
		TableObject.Load(VT);
	endif;
	CurrRow = VT.IndexOf(CurrRowID);
	ValueToFormAttribute(ItemForm,"Object");
EndProcedure

&AtServerNoContext
Function ObjectsAttributeValues(RefArray, AttributeNames) Export
	
	AttributeValues = New Map;
	If RefArray.Count() = 0 Then
		Return AttributeValues;
	EndIf;
	
	Query = New Query;
	Query.Text =
		"SEL ECT
		|	Ref AS Ref, " + AttributeNames + " FR OM " + RefArray[0].Metadata().FullName() + " AS Table
		| WHERE Table.Ref IN (&RefArray)";
	Query.SetParameter("RefArray", RefArray);
	
	Selection = Query.Execute().Select();
	While Selection.Next() Do
		Result = New Structure(StrReplace(AttributeNames,".",""));
		FillPropertyValues(Result, Selection);
		AttributeValues[Selection.Ref] = Result;
	EndDo;
	
	Return AttributeValues;
	
EndFunction
Показать

Имя команды в условие подставлять свое. Поддерживает сортировку по колонке реквизита реквизита. Значения реквизитов для сортировки читаются не в цикле. Наверное можно было и весь контекст формы на сервер не гонять.
7. shard 281 13.01.24 00:49 Сейчас в теме
(6) недооценил коварство коллег: колонки могут быть содержащие собственно реквизит табличной части, содержащие реквизит реквизита табличной части добавленные в конфигураторе и добавленные в режиме предприятия через изменение формы. Исправил код для обработки описанных трех случаев, добавил комментарии:
&AtClient
Procedure SortBL(Command)
	if not CurrentItem.CurrentData=Undefined then 
		CurrentItem.CurrentRow = SortBLAtServer(CurrentItem.Name,CurrentItem.CurrentItem.Name,
			?(StrFind(Command.Name,"DESC")>0,"DESC","ASC"), CurrentItem.CurrentData.LineNumber-1);
	endif;
EndProcedure

&AtServer
Function SortBLAtServer(ItemName, ColumnName, DirSorting, CurrRow)
	ItemForm = FormAttributeToValue("Object");
	TableObject = ItemForm[ItemName];
	
	ColumnFormField = Items[ItemName].ChildItems.Find(ColumnName); // для проверки на случай сортировки по колонкам реквизитов реквизитов, добавленным в конфигураторе
	if ColumnFormField=Undefined then // случай сортировки колонки, содержащую реквизит реквизита табличной части и добавленную через изменение формы в предприятии
		AttibuteNameArray = StrSplit(StrReplace(ColumnName,ItemName,""),"_"); // здесь разделитель "_" и дополнительно нужно убрать имя табличного поля из начала имени колонки
	else
		AttibuteNameArray = StrSplit(ColumnFormField.DataPath,".");
		if AttibuteNameArray.Count()=3 then // случай сортировки колонки, содержащую реквизит табличной части, их всегда 3: объект, имя табличного поля и имя нужного реквизита
			CurrRowID = TableObject[CurrRow]; // запомнили текущую выделенную строку
			TableObject.Sort(AttibuteNameArray[2]+" "+DirSorting);
			ValueToFormAttribute(ItemForm,"Object");
			return TableObject.IndexOf(CurrRowID); // вернем позицию выделенной ранее строке
		else // случай сортировки колонки, содержащую реквизит реквизита табличной части и добавленную в конфигураторе
			AttibuteNameArray = StrSplit(ColumnFormField.DataPath,".");
			AttibuteNameArray.Delete(0); // удаляем элемент "объект"
			AttibuteNameArray.Delete(0); // удаляем элемент с именем табличного поля
		EndIf;
	endif;
	
	VT = TableObject.Unload(); // создаем вспомогательную таблицу
	VT.Columns.Add("AdditionFieldForSorting"); // интерактивная сортировка всегда по одной колонке, имя назначим служебное
	CurrRowID = VT[CurrRow]; // запомнили текущую выделенную строку. Во вспомогательной таблице, чтобы ИД строки был от сортируемой таблицы
	MainValueName = AttibuteNameArray[0]; // имя реквизита табличной части, значения реквизитов которого нужно сортировать
	AttibuteNameArray.Delete(0); // удаляем элемент, реквизиты которого получаем
	
	ValuesArray = VT.UnloadColumn(MainValueName); // формируем массив ссылок для получения значений реквизита и формируем соответствие ссылка-значение
	Query = New Query("SEL ECT Ref, " + StrConcat(AttibuteNameArray,".") + " AS AttributeName FR OM " + ValuesArray[0].Metadata().FullName() + " AS T WHERE T.Ref IN (&RefArray)");
	Query.SetParameter("RefArray", ValuesArray);
 	Selection = Query.Execute().Select();
	MapValues = New Map;
	While Selection.Next() Do
		MapValues.Insert(Selection.Ref, Selection.AttributeName);
	EndDo;

	for each str in VT do // заполняем значения для сортировки
		str.AdditionFieldForSorting = MapValues[str[MainValueName]]; 
	enddo;               
	
	VT.Sort("AdditionFieldForSorting "+DirSorting); // сортировка по служебной колонке
	TableObject.Load(VT);
	ValueToFormAttribute(ItemForm,"Object");
	return VT.IndexOf(CurrRowID); // вернем позицию выделенной ранее строке
EndFunction
Показать

Название команды должно содержать ASC или DESC, иначе следует переписать определение направления сортировки в функции SortBL
cleaner_it; Egovigor; +2 Ответить
8. fixin 4273 01.09.24 11:28 Сейчас в теме
В коде есть ошибки, но для сортировки по артикулу взял за образец, поправил:
&НаКлиенте
Процедура СортироватьПоАртикулу(Команда)
	//Объект.Цены.Сортировать("Номенклатура.Артикул");
	СортироватьНасервере("Цены", "Номенклатура.Артикул", "Возр");
КонецПроцедуры


&НаСервере
Процедура сортироватьНасервере(имяОбъекта,имяКолонки,ВидСортировки)
	//https://infostart.ru/1c/articles/336446/
    ЭФ= РеквизитФормыВЗначение("Объект");
    ТЗ_объекта=ЭФ[имяОбъекта];
    тз= ТЗ_объекта.Выгрузить();
    колонкасортировки=тз.Колонки.Найти(имяКолонки);
    если колонкасортировки<>Неопределено тогда
        ТЗ_объекта.Сортировать(имяКолонки+" "+ВидСортировки);
    иначе
        //запомним текущую строку
		//ткэлемент=элементы[имяОбъекта];
		//позиция_стр=Тз_объекта.НайтиПоИдентификатору(ткэлемент.ТекущаяСтрока);
		//Идекс_позиции=Тз_объекта.индекс(позиция_стр);
		// строкаВтаблице=тз.Получить(Идекс_позиции);


		//пдТ= ТекущийЭлемент.ПутьКДанным;
		//ПД= ТекущийЭлемент.ТекущийЭлемент.ПутьКДанным;
		//пд= СтрЗаменить(ПД,пдТ+".","");
        // получим основную и дополнительную колонку
		пд = ИмяКолонки;
         основннаяКолонка=лев(пд,найти(пд,".")-1);
         допколонка=СтрЗаменить(пд,основннаяКолонка+".","");
         тз.Колонки.Добавить( допколонка);
        //заполним поля для сортировки
        для каждого элементаТЗ из тз цикл
            тзобъекта= элементаТз[основннаяКолонка] ;
            элементаТз[допколонка]=вычислить("тзобъекта."+допколонка);
         конеццикла;
         тз.Сортировать(допколонка+" "+ВидСортировки);
         //индексТЗпослеСортировки=ТЗ.Индекс(строкаВтаблице);
         ТЗ_объекта.Загрузить(тз);
		  ////восстановим курсор на нужную строку
		  //позиция_стр= ТЗ_объекта.Получить( индексТЗпослеСортировки);
		  //ткэлемент.ТекущаяСтрока=позиция_стр.ПолучитьИдентификатор();
     конецесли;

     ЗначениеВРеквизитФормы(Эф,"Объект");
конецпроцедуры



Показать


Уже, кстати, второй раз пытаюсь этот пример записпользовать, оба раза поправляю. Решил написать рабочую версию в комментарии.
9. MrKondr 13.09.24 09:46 Сейчас в теме
(8) От души, очень помог! Что удивительно, наивный я именно сегодня столкнулся с невозможностью сортировки по колонкам, отсутствующим в метаданных. И мне нужно было именно поле Номенклатура.Артикул. Максимальный копипаст, даже как-то не по себе)

Единственное разбил клиентскую часть на 2 кнопки/команды
&НаКлиенте
Процедура СортироватьПоАртикулуВозр(Команда)
    СортироватьНасервере("Запасы", "Номенклатура.Артикул", "Возр");
КонецПроцедуры


&НаКлиенте
Процедура СортироватьПоАртикулуУбыв(Команда)
    СортироватьНасервере("Запасы", "Номенклатура.Артикул", "Убыв");
КонецПроцедуры
Показать
10. fixin 4273 13.09.24 09:59 Сейчас в теме
Оставьте свое сообщение