Как "перемотать" динамический список в начало при открытии? (или некоторые нюансы получения данных динамического списка)

В современных интерфейсах (новостях, социальных сетях и прочим) принято размещать новые данные сверху, более старые — внизу. К сожалению, в 1C сделано несколько по-другому — по умолчанию списки документов отсортированы по возрастанию даты и далее возможно два варианта: либо список при открытии проматывается в конец, либо список может запоминать «последнюю» текущую строку.

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

К счастью, это поведение можно исправить. Для этого «всего лишь» нужно получить реальный исполняемый запрос ДС, выполнить его и установить текущий элемент списка. Положение усложняется тем, что у ДС есть несколько видов поиска, которые недоступны через его настройки. К счастью, фирма 1C осознала эту недоработку и в «зазеркалье» некоторое время назад была опубликована статья, которая рассказывает о костыле, который был придуман специально для этого: https://wonderland.v8.1c.ru/blog/poluchenie-dannykh-dinamicheskogo-spiska/

Итак, пытаемся применить полученные знания и «копипастим код» из статьи, допустим в ПриОткрытии формы списка:

ИсполняемаяСхема = Элементы.Список.ПолучитьИсполняемуюСхемуКомпоновкиДанных();
ИсполняемыеНастройки = Элементы.Список.ПолучитьИсполняемыеНастройкиКомпоновкиДанных()
КомпоновщикМакета = Новый КомпоновщикМакетаКомпоновкиДанных();
МакетКомпоновки = КомпоновщикМакета.Выполнить(Схема, Настройки);

ПроцессорКомпоновки = Новый ПроцессорКомпоновкиДанных();
ПроцессорКомпоновки.Инициализировать(МакетКомпоновки);

ПроцессорВывода = Новый ПроцессорВыводаРезультатаКомпоновкиДанныхВКоллекциюЗначений;
Результат = ПроцессорВывода.Вывести(ПроцессорКомпоновки);
Элементы.Список.ТекущаяСтрока = Результат[0].Ссылка;

Запускаем и…

… видим, что 1C не сильно запаривается в составлении тестовых примеров. Копаемся в синтакс-помощнике, видим, что у метода Выполнить компоновщика макета есть параметр, который отвечает за «тип макета». Оставим это, ИМХО, усложнение, на совести разработчиков платформы, испоравляем строку на

ИсполняемаяСхема = Элементы.Список.ПолучитьИсполняемуюСхемуКомпоновкиДанных();
ИсполняемыеНастройки = Элементы.Список.ПолучитьИсполняемыеНастройкиКомпоновкиДанных()
КомпоновщикМакета = Новый КомпоновщикМакетаКомпоновкиДанных();
МакетКомпоновки = КомпоновщикМакета.Выполнить(Схема, Настройки,,,Тип("ГенераторМакетаКомпоновкиДанныхДляКоллекцииЗначений"));

ПроцессорКомпоновки = Новый ПроцессорКомпоновкиДанных();
ПроцессорКомпоновки.Инициализировать(МакетКомпоновки);

ПроцессорВывода = Новый ПроцессорВыводаРезультатаКомпоновкиДанныхВКоллекциюЗначений;
Результат = ПроцессорВывода.Вывести(ПроцессорКомпоновки);
Элементы.Список.ТекущаяСтрока = Результат[0].Ссылка;

Запускаем и…

… видим, что «оптимизация» запроса ДС действительно работает. Мы не выводим в ДС поле «Ссылка», и в результирующем запросе его нет. Немного поколдовав с отладчиком, видим, что выбранные поля в ДС задаются не на корневом уровне настроек, а внутри детальных записей, которые указаны без автополя. Слава богу, ДС имеет простые настройки и не надо всяких рекурсивных обходов для добавления выбранных полей во все ветки. Меняем код:

ИсполняемаяСхема = Элементы.Список.ПолучитьИсполняемуюСхемуКомпоновкиДанных();
ИсполняемыеНастройки = Элементы.Список.ПолучитьИсполняемыеНастройкиКомпоновкиДанных()

ВыбранноеПоле = ИсполняемыеНастройки.Структура[0].Выбор.Элементы.Добавить(Тип("ВыбранноеПолеКомпоновкиДанных"));
ВыбранноеПоле.Поле = Новый ПолеКомпоновкиДанных("Ссылка");

КомпоновщикМакета = Новый КомпоновщикМакетаКомпоновкиДанных();
МакетКомпоновки = КомпоновщикМакета.Выполнить(Схема, Настройки,,,Тип("ГенераторМакетаКомпоновкиДанныхДляКоллекцииЗначений"));

ПроцессорКомпоновки = Новый ПроцессорКомпоновкиДанных();
ПроцессорКомпоновки.Инициализировать(МакетКомпоновки);

ПроцессорВывода = Новый ПроцессорВыводаРезультатаКомпоновкиДанныхВКоллекциюЗначений;
Результат = ПроцессорВывода.Вывести(ПроцессорКомпоновки);
Элементы.Список.ТекущаяСтрока = Результат[0].Ссылка;

Запускаем, и… Если у вас список спозиционировался на первой строке, то вам не просто повезло, а очень повезло. Смотрим реальный запрос с помощью

Сообщить(МакетКомпоновки.НаборыДанных[0].Запрос)

 видим, что в результирующем запросе отсутствуют сортировки, которые есть в настройках, об этом также сказал отладчик. Вообще отладчик — полезная штука 🙂

Ну что ж. Нужно править текст запроса. Заодно можно оптимизировать запрос, выбрав «первые 1». Тут на ум приходит объект «СхемаЗапроса», про который также писали в зазеркалье: https://wonderland.v8.1c.ru/blog/upravlyaemyy-konstruktor-zaprosa-i-obektnaya-model-skhemy-zaprosa/

Тут возможно два варианта — либо менять результирующий текст запроса, либо менять исполняемый запрос СКД. Я выбрал второй вариант, чтобы не заморачиваться с параметрами запроса (параметрами самого запроса ДС и отборами). Опять пишем код, потратив время на изучение СП и набив немного шишек с типами данных, получается что-то типа того

ИсполняемаяСхема = Элементы.Список.ПолучитьИсполняемуюСхемуКомпоновкиДанных();
ИсполняемыеНастройки = Элементы.Список.ПолучитьИсполняемыеНастройкиКомпоновкиДанных();

ВыбранноеПоле = ИсполняемыеНастройки.Структура[0].Выбор.Элементы.Добавить(Тип("ВыбранноеПолеКомпоновкиДанных"));
ВыбранноеПоле.Поле = Новый ПолеКомпоновкиДанных("Ссылка");

СхемаЗапроса = Новый СхемаЗапроса;
СхемаЗапроса.УстановитьТекстЗапроса(ИсполняемаяСхема.НаборыДанных[0].Запрос);
ПоследнийЗапрос = СхемаЗапроса.ПакетЗапросов.Получить(СхемаЗапроса.ПакетЗапросов.Количество()-1); // да, в 8.3.8 в ДС появились пакетные запросы
Для каждого ОператорВыбрать Из ПоследнийЗапрос.Операторы Цикл
ОператорВыбрать.КоличествоПолучаемыхЗаписей = 1; // выбрать первые 1
КонецЦикла;
Для каждого ЭлементПорядка Из ИсполняемыеНастройки.Порядок.Элементы Цикл
ПолеПорядка = "" + ЭлементПорядка.Поле;
ЭлементПорядкаСхемы = ПоследнийЗапрос.Порядок.Добавить(ПолеПорядка);
ЭлементПорядкаСхемы.Направление = ?(ЭлементПорядка.ТипУпорядочивания = НаправлениеСортировкиКомпоновкиДанных.Возр, НаправлениеПорядкаСхемыЗапроса.ПоВозрастанию, НаправлениеПорядкаСхемыЗапроса.ПоУбыванию);
КонецЦикла;

КомпоновщикМакета = Новый КомпоновщикМакетаКомпоновкиДанных();
МакетКомпоновки = КомпоновщикМакета.Выполнить(ИсполняемаяСхема, ИсполняемыеНастройки,,,Тип("ГенераторМакетаКомпоновкиДанныхДляКоллекцииЗначений"));

ПроцессорКомпоновки = Новый ПроцессорКомпоновкиДанных();
ПроцессорКомпоновки.Инициализировать(МакетКомпоновки);

ПроцессорВывода = Новый ПроцессорВыводаРезультатаКомпоновкиДанныхВКоллекциюЗначений;
Результат = ПроцессорВывода.Вывести(ПроцессорКомпоновки);
Элементы.Список.ТекущаяСтрока = Результат[0].Ссылка;

Опять запускаем, и…

Не волнуйтесь, мы уже в конце пути, осталось совсем немного :))

Немного гугла, яндекса и наблюдений за ДС с различными вариантами сортировки и становится понятно, что это «дополнительное поле порядка 2» очень похоже на УИД. Ну что ж, немного плохого кода (я действительно не знаю, как из «ДополнительноеПолеПорядка2» получить «Ссылка»):

Для каждого ЭлементПорядка Из ИсполняемыеНастройки.Порядок.Элементы Цикл
ПолеПорядка = "" + ЭлементПорядка.Поле;
Попытка
ЭлементПорядкаСхемы = ПоследнийЗапрос.Порядок.Добавить(ПолеПорядка);
Исключение
ЭлементПорядкаСхемы = ПоследнийЗапрос.Порядок.Добавить("Ссылка");
КонецПопытки
ЭлементПорядкаСхемы.Направление = ?(ЭлементПорядка.ТипУпорядочивания = НаправлениеСортировкиКомпоновкиДанных.Возр, НаправлениеПорядкаСхемыЗапроса.ПоВозрастанию, НаправлениеПорядкаСхемыЗапроса.ПоУбыванию);
КонецЦикла;

Запускаем и… бинго! Тут должен быть скриншот, но я вам его не покажу. В общем, для ДС с основной таблицей документов — работает, испытайте некоторое чувство удовлетворения от того, что заставили 1C делать то, что вам надо, самостоятельно.

Задания на дом: Сделать то же, но без использования процессора компоновки (вы же уже знаете, как получить настоящий текст запроса). Сделать то же для независимого периодического РС или регистра накопления. Ответить на вопрос, можно ли использовать не исполняемую схему и исполняемые настройки, а настройки из реквизита формы? Добавить самосортировку по убыванию даты. Погоревать о том, что нет события на интерактивное изменение настроек ДС. Сделать вариант для ДС с группировкой (кстати, так не надо делать, посмотрите профайлером, почему).

Надеюсь, вы почерпнули немного полезного в этой статье и найдете не только такое узкое применение для работы с программным выполнением СКД, работой со схемой запроса и прочими плюшками (внимательные читатели заметили комментарии с упоминанием некоторых из них).

ЗЫ: А теперь выкинте весь код и внимательно посмотрите палитру свойств, там есть подозрительный пункт:

который делает то же самое, но без всего того, что написано выше. Но это же не интересно, правда? Иначе как бы вы узнали про исполняеую схему и прочие плюшки, не понабивали бы шишек раскрывая для себя немного внутреннего устройства?

25 Comments

  1. MiniMuk



    + еще настроить список -> сортировка плохие варианты?

    Reply
  2. Fragster

    (1) MiniMuk, да, так можно, но на самом деле статья не про то 🙂

    Reply
  3. alxarz

    (2) а читая анонс можно подумать что именно про это 🙂

    Reply
  4. MiniMuk

    (2) за ссылку отдельное спасибо.

    Reply
  5. Fragster

    (3) alxarz, статья про то, что в 1с есть много разных прикольных возможностей, но сложный путь не всегда верный.

    Reply
  6. Fragster

    (4) кстати, в зазеркалье подписаться на рассылку новых статей можно

    Reply
  7. BigBoss

    Я начиная читая статью, подумал что автор не знает про тот пункт в палитре свойств. Статья интересная

    Reply
  8. Fragster

    (3) alxarz, дополнил заголовок статьи

    Reply
  9. Fragster

    эй! куда подевался код!

    Reply
  10. Fragster

    (9)+ восстановил

    Reply
  11. Danil.Potapov

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

    Reply
  12. Fragster

    (11) Danil.Potapov, раздражает, что не всё, что делает пользователь доступно программно (даже состояние, не говоря уж о самих действиях), и не на все действия есть события. Например весьма пригодилось бы событие на изменение пользовательских настроек ДС. Или подписка на изменение определенного типа данных (например для ДС без основной таблицы). И оповещение об изменении данных с сервера для всех клиентов (реактивность).

    Те же Список.КомпоновщикНастроек.ПолучитьНастройки() не содержат то, что пользователь отобрал через «найти».

    Reply
  13. Danil.Potapov

    (12) да, точно, сколько лет прошло а воз и ныне там.

    Reply
  14. starik-2005

    Это только с ДС с основной таблицей? А как быть с ДС произвольным? Есть там возможность спозиционироваться на чем-нибудь, отличном от начала или окончания?

    Reply
  15. Fragster

    (14) starik-2005, там текущая строка похожа на порядковый номер строки в результате компоновки. т.е. для позиционирования на произвольной строке нужно получить позицию в полной выборке. Ну а для первой — 1, для последней — количество(*)

    Reply
  16. Yashazz

    Сначала было собрался ругаться, что автор не знает азы, но, к счастью, это оказался такой грамотный стёб. Зачёт) Вещи, в целом, очевидные, но полезные.

    Вообще, не зря эту хрень прозвали «демонический список». Постоянно сюрпризы))

    Reply
  17. Dementor

    Статья интересна в плане «у меня есть какой-то код — давайте заставим его работать».

    У Вас немного неправильное восприятие сайта Зазеркалья. Это НЕ документация по платформе. Это анонсы функциональности, которые разработчики сейчас у себя внутри тестируют и планируют когда-то включить в рабочие сборки платформ. То, что параметров не хватает, или указаны лишние, или у функций могут оказаться совсем другие названия — это нормальные рабочие моменты. Поэтому, каждый раз на форуме, когда кто-то выдает подобные вашим претензии к примерам, то им вежливо объясняют, что их путь должен последовать на its.1c.ru . Раз даже до смешного дошло — один разработчик жаловался на неработающую фичу, про которую он прочитал в Зазеркалье, но которую на тот момент еще даже в тестовую сборку не включили 🙂

    Reply
  18. Fragster

    (17) Dementor, к сожалению, поиск по its.1c.ru фраз «ПолучитьИсполняемуюСхемуКомпоновкиДанных» или «Получить Исполняемую Схему Компоновки Данных» не приводит к ни к какому результату. Равно как и нет информации про отсутствие сортировки в «реальном» запросе СКД или про «ДополнительноеПолеПорядка».

    Reply
  19. Dementor

    (18) Fragster, странно. Почему же у меня получилось? http://its.1c.ru/db/v839doc#bookmark:dev:TI000001476

    Кстати, ошибка не у разработчиков с неправильным примером, а у вас. Тип генератора макета по-умолчанию установлен в значение для вывода в табличный документ. И в примере происходит вывод как раз в табличный документ! Чуть ниже ссылка для желающих вывести данные в таблицу значений — http://its.1c.ru/db/v839doc#bookmark:dev:TI000000626 и там уже правильно переопределен генератор макета.

    О Зазеркалье повторюсь — это не документация и даже не блог разработчиков конфигураций. Это просто анонсы, что бы иметь представление о том, что будет реализовано в ближайших релизах. К примеру, я сегодня полистал красивые новые диаграммы — молодцы, но когда новый функционал будет в платформе, то я буду готов, что настройки назовут немного по другому, что-то не сделают всего обещанного или что-то возможно удивят и добавят нового, не описанного в этой свежей статье.

    Reply
  20. Fragster

    (19) Dementor, повторюсь, увидев анонс в зазеркалье, я взял пример кода из него. И там именно вывод в коллекцию значений. То, что на ИТС ты смог найти документацию — поздравляю. Но по почему-то по названию ключевых функций оно не ищется. За ссылку, конечно, спасибо (кстати, пример там уже с табличным документом, в отличии от зазеркалья).

    Я рассматриваю зазеркалье как сборник интересных особенностей платформы, с приблизительными примерами и указанием, когда оно (будет) реализовано. Не всегда хватает сил прочитать (и запомнить) v8update, да и самих примеров (с картинками 🙂 ) он не содержит.

    Про диаграммы — да, реализовано кое что, особенно порадовало условное оформление для СКД. Но сильно востребованные вещи типа логарифмической шкалы или двух шкал — не так и не реализованы, так что продолжим пользоваться полем HTML документа.

    Reply
  21. Fragster

    (19) Dementor, я понял, как получилось найти информацию. Поиск по ИТС запоминает раздел. Из-за этого у меня результат оказался пуст, пока не переключишься в «разработка и администрирование».

    Reply
  22. Fragster

    (19) предложил на партнерском вставлять ссылку на ИТС в статью в зазеркалье. Пока думают.

    Reply
  23. it@contlog.ru

    Или кривокод или что?

    Смотрю СхемаЗапроса модифицирует запрос, но дальше нигде не используется?

    Reply
  24. Fragster

    (23) Согласен, пропустил установку запроса обратно в схему. Но глобально это ни на что не влияет 🙂

    Даже на сообщаемые сведения про «дополнительное поле порядка»

    Радует, что хоть кто-то внимательно это прочитал 🙂

    Reply
  25. Hatson

    Трололоша )))

    Reply

Leave a Comment

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