Автоматическое отключение неактивных веб клиентов

Программирование - Работа с интерфейсом

У вас организован доступ в базу через веб-клиент для посторонних лиц (веб-портал, веб-витрина, и т.д.), и вы испытываете проблему нехватки лицензий 1С из за того что пользователи оставляют открытыми вкладки с 1С не работая в них? Есть решение!

По мотивам дискуссии на партнерском форуме https://partners.v8.1c.ru/forum/topic/1620757
Публикация сделана для тех, у кого нет доступа на партнерский форум.

Внешние пользователи, после того как поработают в 1С, любят не закрывать вкладку, а просто переключаются на другие. 

Попытка решить проблему типовыми средствами, кажущимися подходящими для данной ситуации:

1. В параметрах информационной базы установить следующие значения:
Время засыпания пассивного сеанса - 600 сек.
Время завершения спящего сеанса  - 10 сек.

2. При публикации базы на веб сервере указать параметр "Время жизни соединения в сек." - 600

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

К сожалению, 1С не предоставляет штатных средств для определения времени последней активности пользователя.

Но есть работающий метод:

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

На "рабочем столе" 1С должна находиться как минимум одна форма. В эту форму нужно добавить Поле HTML документа. Для него можно установить минимальный размер, а цвет фона сделать равным цвету фона формы, чтобы его не было видно. Важное уточнение: поле должно иметь свойство Видимость = Истина, иначе оно попросту не будет создано на клиенте. 

Если на вашей форме нет даже самого малого свободного места, то можно создать две страницы (видимость переключателя = Ложь). На первую страницу переносятся все элементы формы, а на вторую Поле HTML документа. Затем при открытии формы нужно программно установить активной вторую страницу, а через 0.1 сек обратно первую. Если этого не сделать, то поле HTML физически создастся только после первой активации второй страницы, то есть, никогда (учитывая что переключатель невидим).

Затем в поле HTML загружаем javascript, который и будет выполнять всю работу. Доступ к родительскому окну осуществляется посредством идентификатора top.

<HTML>
<BODY>
    <SCRIPT>
        // Время завершения сеанса от последней активности пользователя, в миллисекундах.
        // Для 10-минутного тайм-аута следует установить 600000.
        var timeoutInMiliseconds = 5000; // 5 секунд.

        var onUserInactive = function() {
            // При наступлении тайм-аута переходим на пустую страницу, тем самым завершая сеанс 1С.
            top.location.href = 'about:blank';
        }

        // Первоначальный запуск обработчика ожидания.
        var wait = setTimeout(onUserInactive, timeoutInMiliseconds);

        var onUserActivity = function() {
            // При каждой активности пользователя перезапускаем обработчик ожидания.
            clearTimeout(wait);
            wait = setTimeout(onUserInactive, timeoutInMiliseconds);
        };

        // Подключаем обработчики отслеживания активности пользователя.
        top.document.onmousemove     = onUserActivity;
        top.document.onmouseover     = onUserActivity;
        top.document.mousedown         = onUserActivity;
        top.document.mouseup         = onUserActivity;
        top.document.onkeydown         = onUserActivity;
        top.document.onkeyup         = onUserActivity;
        top.document.focus             = onUserActivity;
        
        // Для этих панелей подключаемся отдельно именно к этим элементам, 
        // так как в обработчике установленном фирмой 1С установлен запрет на event propagation,
        // и top.document.onmousemove его не словит.
        top.document.getElementById('navigatorPanelArea').onmousemove = onUserActivity;
        top.document.getElementById('actionPanelArea').onmousemove = onUserActivity;

        // Ловим события движения мыши из вложенных фреймов (других полей html документа).  
        function bubbleIframeMouseMove(iframe) {
            try
            {
                // Save any previous handler
                var existingOnMouseMove = iframe.contentWindow.onmousemove;

                // Attach a new listener
                iframe.contentWindow.onmousemove = function(e) {
                    // Fire any existing listener 
                    if (existingOnMouseMove) existingOnMouseMove(e);

                    // Create a new event for the this window
                    var evt = document.createEvent("MouseEvents");

                    // We'll need this to offset the mouse move appropriately
                    var boundingClientRect = iframe.getBoundingClientRect();

                    // Initialize the event, copying exiting event values
                    // for the most part
                    evt.initMouseEvent(
                        "mousemove",
                        true, // bubbles
                        false, // not cancelable 
                        window,
                        e.detail,
                        e.screenX,
                        e.screenY,
                        e.clientX + boundingClientRect.left,
                        e.clientY + boundingClientRect.top,
                        e.ctrlKey,
                        e.altKey,
                        e.shiftKey,
                        e.metaKey,
                        e.button,
                        null // no related element
                    );

                    // Dispatch the mousemove event on the iframe element
                    iframe.dispatchEvent(evt);
                };
            }
            catch(err)
            {
            }
        }

        var iframes = top.document.getElementsByTagName('iframe');
        for (var i in iframes)
            bubbleIframeMouseMove(iframes[i]);
    </SCRIPT>
</BODY>
</HTML>


По истечении отведенного времени, сеанс веб-клиента завершится (останется пустая вкладка браузера).
Впрочем, если есть открытые формы с признаком Модифицированность = Истина (звездочка в конце заголовка формы), то сеанс не завершается автоматически, а происходит автоматическая активация вкладки 1С и отображение вопроса "Покинуть эту страницу?". Это типовое поведение, сделанное чтобы не потерять введенные данные, присутствующее и в тонком клиенте при закрытии приложения ("Данные были изменены. Сохранить изменения?").

Полная реализация метода в прикрепленной обработке.

Рекомендую ознакомиться с другой моей публикацией
Javascript и 1С. Кросс-платформенное взаимодействие //1c-soft.it-terminal.ru/public/398366/

Скачать файлы

Наименование Файл Версия Размер
НеактивностьВебКлиента
.epf 7,89Kb
20.07.17
5
.epf 7,89Kb 5 Скачать

См. также

Комментарии
1. Igor Gorbachev (Leits) 3 20.07.17 16:35 Сейчас в теме
я не до конца понял на какую форму нужно выводить HTML поле? там же их много(например Корзина, Рабочий стол, различные журналы заказы, РНК )
2. Виталий Чебан (VitaliyCeban) 298 20.07.17 16:56 Сейчас в теме
(1) На любую, которая остается открытой на всём протяжении работы. То есть на одну из форм из "рабочей области начальной страницы".
3. Глеб Зломанов (Glebis) 7 21.07.17 12:56 Сейчас в теме
Статья очень нужная.

Не понял фразу, "Затем в поле HTML загружаем javascript,". У этого элемента куча событий, но никакой загрузки в свойствах элемента нет.

Подойдет этот способ для обычных форм?
4. Виталий Чебан (VitaliyCeban) 298 21.07.17 13:19 Сейчас в теме
(3) Код из публикации помещается в макет с типом ТекстовыйДокумент.
Затем в ПриСозданииНаСервере выполняете
ПолеHTML = ПолучитьОбщийМакет("МакетJavascript").ПолучитьТекст();

ПолеHTML это имя реквизита формы с типом "Строка", к которому привязано поле формы с видом "Поле HTML документа". Это вся загрузка.

Обычные формы в веб-клиенте не доступны, соответственно, данный метод к ним неприменим. Замечу, что и при управляемых формах метод работает только в веб-клиенте, но не в тонком/толстом.
5. Глеб Зломанов (Glebis) 7 21.07.17 14:14 Сейчас в теме
Передо мной стоит задача: отключать сессии неактивных пользователей. Ваша статья натолкнула меня на мысль - определять период "не активности" по времени без движения мыши.
Вопрос: можно ли добавив на обычную или управляемую форму (не веб интерфейс) рабочего стола элемент "ПолеHTMLДокумента" событием onmousemove сбрасывать таймер обработчика ожидания, который закрывает текущую сессию?
6. Виталий Чебан (VitaliyCeban) 298 21.07.17 14:41 Сейчас в теме
(5) Нет, невозможно, так как в толстом/тонком клиенте omousemove будет отрабатывать движения мыши только внутри Поля HTML документа.

Вам должно помочь http://infostart.ru/public/20593/ или http://infostart.ru/public/16441/
Оставьте свое сообщение