Skip to content

rosteleset/SmartYard-web

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

22 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

SmartYard-web

web-расширения для приложений

Часть функционала приложения, которая требует лишь диалогов с пользователем и не требует каких-то платформозависимых операций, мы решили реализовывать в виде Web расширений.

Реализованы 2 вида контроллеров:

  • WebViewController - простой web-контроллер, который используется в NavigationStack
  • WebPopupController - модальный контроллер в виде шторки, отображается поверх NavigationStack

Для интеграции с нативной частью приложения мы добавили несколько JS-расширений для WebKit:

  • function bearerToken() - возвращает bearer-token пользователя (данное расширение подключается нативно к WebKit до загрузки HTML)

Следующие 3 расширения используют механизм window.webkit.messageHandlers и для удобства обёрнуты в функции, реализация которых содержится в /js/lanta.js

  • function postLoadingStarted() - уведомляет нативное приложение о том, что началась загрузка динамического контента и надо скрыть страницу от пользователя под скелетон
  • function postLoadingFinished() - уведомляет нативное приложение о том, что весь динамический контент завершил свою загрузку и можно показать страницу пользователю
  • function postRefreshParent(timeout) - уведомляет нативное приложение о том, что надо обновить активный веб-контроллер в NavigationStack через timeout секунд

Переходы между контроллерами - версия 1 (старая)

Для WebViewController любой переход по ссылке в пределах домена текущей страницы приводит к тому, что создаётся новый WebViewController, который будет запушен в NavigationStack. подпись для кнопки "Назад" будет взята из title вызывающей страницы Если же у ссылки указан target=_blank, то будет открыт WebPopupController

Для WebPopupController любой переход по ссылке в пределах домена текущей страницы приводит к тому, что создаётся новый WebPopupController и заменяет текущий. Если же у ссылки в указан target=_blank, будет создан новый WebViewController и будет осуществлена замена активного контроллера в NavigationStack, при этом текущее модальное окно будет закрыто.

Переходы между контроллерами - версия 2 (новая)

Если у ссылки указан target=_blank, то ссылка всегда откроется в браузере. Если у ссылки схема отличная от http или https (например: "tel:" или "sberpay:"), то она всегда откроется системой.

Для всех остальных случаев выбор способа обработки перехода определяется по наличию в url якоря:

Для WebViewController:

  • #smart-yard-push - создаётся новый WebViewController, который будет запушен в NavigationStack. подпись для кнопки "Назад" будет взята из title вызывающей страницы
  • #smart-yard-popup - будет открыт WebPopupController
  • #smart-yard-replace - будет создан новый WebViewController и будет осуществлена замена видимого контроллера в NavigationStack.
  • #smart-yard-external - ссылка будет открыта в браузере (альтернатива target=_blank)

Для WebPopupController:

  • #smart-yard-push - шторка закрывается, создаётся новый WebViewController, который будет запушен в NavigationStack.
  • #smart-yard-popup - создаётся новый WebPopupController и заменяет текущий.
  • #smart-yard-replace - шторка закрывается, будет создан новый WebViewController и будет осуществлена замена видимого контроллера в NavigationStack.
  • #smart-yard-external - ссылка будет открыта в браузере (альтернатива target=_blank)
  • #smart-yard-close - шторка закрывается, а ссылка будет открыта в браузере (тут есть отличии от варианта с target=_blank, когда шторка останется активной)

Пример нашей реализации

Исходный код HTML/CSS/JS нашей реализации web-контроллеров на примере раздела бонусной программы вы можете посмотреть в папке bonuses.

Расширение добавляется в главное меню приложения через методы API методы приложения /ext/list и /ext/ext Порядковый номер пункта в главном меню задаётся в параметре order. Нативные пункты главного меню имеют номера 100, 200 и т.д. Соответственно, если хочетеся добавить элемент меню между 200 и 300 позицией, то достаточно задать web-расширению order=250

> POST /api/ext/list

{
	"code": 200,
	"name": "OK",
	"message": "Хорошо",
	"data": [
		{
			"caption": "Бонусы",
			"icon": "data:image\/png;base64,iVBORw0KGgoAAAANSUhEUgAAAIAAAACACAYAAADDPmHLAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAADsQAAA7EB9YPtSQAAABl0RVh0U29mdHdhcmUAd3d3Lmlua3NjYXBlLm9yZ5vuPBoAAAS\/SURBVHic7d1LixxVGMbxv86IYJKOY8gipBMiCAYvAYlGRATBRZBIJqJRBEHXosnCL+BH0LhyIYgQ8LbxEoOoS92Il4WYBCGRmTEuFGUuIWrSjoszQcXpOtVVp+pU9\/P84CWL7pzzVp2ne05X90yDmZmZmZmZmamYyt1AxBZgGvgzdyMV9IFbgF2E\/leydjNG9gNvA78Bq2u1AnwGHAOeAm4nBKMrtgIHgBeAD4Gf+af3K\/UtcBS4Jk+L3TcDvMP\/T9ywugB8DrwMPE17odgA3Ac8D7wBnBuh51Xga+CmFvos5arcDay5HvgYuLPmOBeBb4Avga\/W\/v0OuFxxvGngNmDfWt0F3Er9H51zhBDN1RxnYhxntEfRJNQnSc7cBLiH\/IuRqx5McP5quTp3A4RNnaoncjfQhQDcn7uBjO7N3UAXAtCP3H6xlS6aEet9WytdFOjCq4DVyO3TwG5g77\/qDuC6hvsa1SXge8Irjyv1BfB75P9lXYNxCMB6PU6RNxTDFvuPde5b5fikxHbKZU0De0qMV7f2MNoFp1TH14gupC\/1I6TueF3rp1Fd2ARaRg6AOAdAnAMgzgEQ5wCIcwDE1X0NuhOYJXwcahfhuv6GmmNasQvAAvAD8AHwLjDfdhPbgVcIn7TJ\/Z66eg2AtwgPwFYcApYbOhhX9VoCDhasWxJHCYnLfbCu9WsAHBm6ejUdwos\/DjVghGeCspvAPnAK2Fh2YMtqmfB2+fnYHct+vPlF4O46HVmrrgU2A+\/H7ljmGWAncJbu\/xqZ\/deA8MpgoehOZS4EzRJf\/HngUaBHCFWbFdP18UetHvAwcCbS1xRh7Wo7SfGmYw64IcVEFcU2RV0fv6oZwrkv6u1EionORCZ5JMUkNagGAOAwxb2dTjHJUmSSTSkmqUE5AD2Ke1uKDVDmZ1zsIHN\/rrDp\/ib6+P1uoDgHQJwDIC7FX9TIvRHKbayP388A4hwAcQ6AuDIBWG68C2vKYuwOZQLwaYJGLI+PUgxyM\/Ar+T\/pUrXqyt1\/1fqF+F9fKW0H4ZOnix04MAeguBaBNym5+CmuY8dOctPXypuef6KPz68CxDkA4hwAcQ6AOAdAnAMgzgEQ5wCIcwDEOQDiHABxDoA4B0CcAyDOARDnAIhzAMQ5AOIcAHEOgDgHQJwDIM4BEOcAiHMAxDkA4hwAcQ6AOAdAnAMgzgEQ5wCIcwDEOQDiHABxDoA4B0CcAyDOARDnAIhzAMQ5AOIcAHEOgDgHQJwDIM4BEOcAiHMAxDkA4hwAcQ6AOAdAnAMgzgEQlyIAse8W3pRgDlWbI7fX\/l7nFAH4KXL7AwnmULU\/cvv5VrqIOEnxV5meBmYanD\/3V8c2ZQswH5n7RIPzl\/Ys8ZM0BxwGeg3MP2kB6AGPEV\/8VeCZupOl+N7bHcA5YCrBWFbeZeBGYKHOICn2APPAawnGsdG8Ss3Fh3TffL0dOIV3\/G1ZAnYT34BHpboO8CPwODBINJ4N9xfwJAkWvwlHCCGIbV5c1WoAPFd6NTI5SHiKyn2yJq0WgYdGWIestgIvAZfIf+LGvQbA68C2kVagpFSbwGH6wCxwgPCSpQ9sbHjOcbdC2N2fJVzoeY8Eu30zMzMzMzMzs78BQGIm4CmLbRMAAAAASUVORK5CYII=",
			"order": 250,
			"extId": "10001"
		}
	]
}
> POST /api/ext/ext HTTP/1.1
| {
| 	"extId": "10001"
| }

{
	"code": 200,
	"name": "OK",
	"message": "Хорошо",
	"data": {
		"basePath": "https:\/\/dm.lanta.me\/app_static\/bonuses\/",
		"code": "<содержимое файла /bonuses/index.html>"
	}
}

Обращение к мобильному API

<html>
    <body>
        pizza.html

	    <script>
	
			if (typeof bearerToken != 'function' && Android) {
				bearerToken = function() {
					return Android.bearerToken();
				}
			}
	
			if (typeof bearerToken != 'function') {
				bearerToken = function() {
					return (new URL(document.location.href)).searchParams.get('token');
				}
			}
	
			alert(bearerToken());
	
			fetch('https://ВАШ_СЕРВЕР/mobile/address/offices', {
				method: 'POST',
				headers: {
					'Authorization': 'Bearer ' + bearerToken(),
					'Accept': 'application/json',
					'Content-Type': 'application/json'
				},
				body: JSON.stringify({
					a: 1,
					b: 'Textual content'
				}),
			}).
			then(res => { return res.json(); }).
			then(data => { alert(JSON.stringify(data)); }).
			catch(error => { alert(error); });
	    </script>
    </body>
</html>

Добавление своего расширения

  1. Создать или отредактировать файл /opt/rbt/server/mobile/custom/ext.php
    <?php
    
     /**
      * @api {post} /ext/ext расширение
      * @apiVersion 1.0.0
      * @apiDescription **[нет верстки]**
      *
      * @apiGroup Ext
      *
      * @apiHeader {String} authorization токен авторизации
      *
      * @apiParam {String} extId идентификатор расширения
      * @apiParam {Object[]} params параметры передаваемые в расширение
      * @apiParam {String} params.id идентификатор
      * @apiParam {String} params.value значение
      *
      * @apiSuccess {Object} - страничка которую надо отобразить во вьюшке
      * @apiSuccess {String} -.basePath базовый путь (от которго должна была загрузиться страница)
      * @apiSuccess {String} -.code html страница
      * @apiSuccess {Number=1,2} [-.version=1] версия реализации web-расширений (1 - по умолчанию)
      */
    
     auth();
    
     switch ($postdata['extId']) {
         case "10002":
             response(200, [
                 "basePath" => "https://ВАШ_СЕРВЕР/static/portal/pizza.html",
                 "code" => @file_get_contents("/opt/rbt/static/portal/pizza.html"),
             ]);
             break;
    
         default:
             response(404);
             break;
     }
    
  2. Создать или отредактировать файл /opt/rbt/server/mobile/custom/list.php
    <?php
    
    /**
     * @api {post} /ext/list список глобальных расширений (меню)
     * @apiVersion 1.0.0
     * @apiDescription **[нет верстки]**
     *
     * @apiGroup Ext
     *
     * @apiHeader {String} authorization токен авторизации
     *
     * @apiSuccess {Object[]} - массив объектов
     * @apiSuccess {String} -.caption имя расширения (для отображения)
     * @apiSuccess {String} -.icon иконка расширения (svg)
     * @apiSuccess {String} -.extId идентификатор расширения
     * @apiSuccess {Number} -.order порядок следования (вес)
     * @apiSuccess {String="t","f"} [-.highlight="f"] "подсветка" (красная точка)
     */
    
    auth();
    
    response(200, [
        [
            "caption" => "Pizza!",
            "icon" => "",
            "order" => 50,
            "extId" => "10002",
            "highlight" => "f",
        ],
    ]);
    
  3. Создать сам файл расширения /opt/rbt/static/portal/pizza.html (пример см. выше). Файлы можно класть в каталоги /opt/rbt/static/portal и /opt/rbt/static/custom (они в .gitignore)

About

web-расширения для приложений / web-extensions for SmartYard mobile app

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •