@opengis/admin 0.4.33 → 0.4.35
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +97 -97
 - package/dist/{add-page-OMFiDtpe.js → add-page-GMWte58m.js} +1 -1
 - package/dist/{admin-interface-CbuB55Fp.js → admin-interface-TLM61JNF.js} +23 -26
 - package/dist/{admin-view-CAmlI6sN.js → admin-view-BKD-rKqQ.js} +28 -33
 - package/dist/admin.js +1 -1
 - package/dist/admin.umd.cjs +54 -54
 - package/dist/assets/logo.svg +41 -41
 - package/dist/{card-view-DCL2Y7Hd.js → card-view-CwkJ9Qg2.js} +1 -1
 - package/dist/{edit-page-CEyRUTE5.js → edit-page-C_igWB2a.js} +1 -1
 - package/dist/{import-file-BLgU6k4_.js → import-file-2iB1izw1.js} +9654 -9556
 - package/dist/{profile-page-B82nN_fd.js → profile-page-Cuy6hkl8.js} +1 -1
 - package/dist/style.css +1 -1
 - package/module/settings/card/admin.accounts.table/index.yml +7 -7
 - package/module/settings/card/admin.accounts.table/rules.hbs +18 -18
 - package/module/settings/card/admin.accounts.table/users.hbs +13 -13
 - package/module/settings/card/admin.routes.table/groups.hbs +11 -11
 - package/module/settings/card/admin.routes.table/users.hbs +16 -16
 - package/module/settings/cls/core.actions.json +17 -17
 - package/module/settings/cls/core.scope.json +13 -13
 - package/module/settings/cls/properties.site_status.json +13 -13
 - package/module/settings/cls/properties.widget_status.json +13 -13
 - package/module/settings/cls/yes_no.json +11 -11
 - package/module/settings/form/admin.accounts.form.json +13 -13
 - package/module/settings/form/admin.properties.form.json +15 -15
 - package/module/settings/form/admin.roles.form.json +21 -21
 - package/module/settings/form/admin.user_properties.form.json +15 -15
 - package/module/settings/form/admin.user_roles_card.form.json +13 -13
 - package/module/settings/interface/admin.properties.json +4 -4
 - package/module/settings/interface/admin.roles.json +4 -4
 - package/module/settings/interface/admin.routes.json +4 -4
 - package/module/settings/interface/admin.users.json +4 -4
 - package/module/settings/select/core.routes.sql +1 -1
 - package/module/settings/select/core.user_mentioned.sql +1 -1
 - package/module/settings/select/core.user_uid.sql +1 -1
 - package/module/settings/table/admin.properties.table.json +39 -39
 - package/module/settings/table/admin.user_properties.table.json +34 -34
 - package/package.json +2 -2
 - package/server/helpers/core/coalesce.js +7 -7
 - package/server/helpers/core/select.js +48 -48
 - package/server/helpers/core/token.js +18 -18
 - package/server/helpers/list/buttonHelper.js +21 -21
 - package/server/helpers/list/utils/button.js +5 -5
 - package/server/helpers/temp/contentList.js +58 -58
 - package/server/helpers/temp/ifCond.js +101 -101
 - package/server/helpers/utils/button.js +5 -5
 - package/server/helpers/utils/mdToHTML.js +17 -17
 - package/server/plugins/cron.js +10 -10
 - package/server/plugins/docs.js +28 -28
 - package/server/plugins/hook.js +4 -4
 - package/server/routes/calendar/controllers/calendar.data.js +124 -124
 - package/server/routes/calendar/index.mjs +7 -7
 - package/server/routes/notifications/controllers/readNotifications.js +18 -18
 - package/server/routes/notifications/controllers/testEmail.js +35 -35
 - package/server/routes/notifications/controllers/userNotifications.js +53 -53
 - package/server/routes/notifications/hook/onWidgetSet.js +56 -56
 - package/server/routes/notifications/index.mjs +26 -26
 - package/server/routes/root.mjs +3 -3
 - package/server/routes/user/controllers/user.cls.id.js +14 -14
 - package/server/routes/user/controllers/user.cls.js +72 -72
 - package/server/routes/user/controllers/user.info.js +17 -17
 - package/server/templates/cls/itree.recrzone_category.json +73 -73
 - package/server/templates/cls/test.json +9 -9
 - package/server/templates/form/admin.user_cls.data.form.json +49 -49
 - package/server/templates/form/admin.user_group_rel.form.json +21 -21
 - package/server/templates/form/form-user-pass.json +10 -10
 - package/server/templates/form/form-user_group.json +39 -39
 - package/server/templates/form/form-users.json +156 -156
 - package/server/templates/form/user_group_access.form.json +22 -22
 - package/server/templates/select/account_id.json +2 -2
 - package/server/templates/table/gis.dataset.table.json +43 -43
 - package/server/templates/table/management.user_group.table.json +112 -112
 - package/server/templates/table/management.users.table.json +126 -126
 - package/server/utils/addNotification.js +21 -21
 - package/server/utils/sendNotification.js +89 -89
 
| 
         @@ -1,26 +1,26 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            import { addHook } from '@opengis/fastify-table/utils.js';
         
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            // api
         
     | 
| 
       4 
     | 
    
         
            -
            import testEmail from './controllers/testEmail.js';
         
     | 
| 
       5 
     | 
    
         
            -
            import readNotifications from './controllers/readNotifications.js'; // mark as read
         
     | 
| 
       6 
     | 
    
         
            -
            import userNotifications from './controllers/userNotifications.js'; // check all, backend pagination
         
     | 
| 
       7 
     | 
    
         
            -
             
     | 
| 
       8 
     | 
    
         
            -
            // hook
         
     | 
| 
       9 
     | 
    
         
            -
            import onWidgetSet from './hook/onWidgetSet.js'; // send notification on comment
         
     | 
| 
       10 
     | 
    
         
            -
             
     | 
| 
       11 
     | 
    
         
            -
            // funcs
         
     | 
| 
       12 
     | 
    
         
            -
            // import addNotification from '../notifications/funcs/addNotification.js'; // add to db
         
     | 
| 
       13 
     | 
    
         
            -
            // import notification from '../notifications/funcs/sendNotification.js'; // send notification
         
     | 
| 
       14 
     | 
    
         
            -
             
     | 
| 
       15 
     | 
    
         
            -
            import { notificationSchema, emailSchema } from './schema.js';
         
     | 
| 
       16 
     | 
    
         
            -
             
     | 
| 
       17 
     | 
    
         
            -
            export default async function route(fastify) {
         
     | 
| 
       18 
     | 
    
         
            -
             
     | 
| 
       19 
     | 
    
         
            -
              // fastify.decorate('addNotification', addNotification);
         
     | 
| 
       20 
     | 
    
         
            -
              // fastify.decorate('notification', notification);
         
     | 
| 
       21 
     | 
    
         
            -
             
     | 
| 
       22 
     | 
    
         
            -
              fastify.get('/test-email', { config: { policy: ['user'] }, schema: emailSchema }, testEmail);
         
     | 
| 
       23 
     | 
    
         
            -
              fastify.get('/notification', { config: { policy: ['user'] }, schema: notificationSchema }, userNotifications);
         
     | 
| 
       24 
     | 
    
         
            -
              fastify.get('/notification-read/:id?', { config: { policy: ['user'] }, schema: notificationSchema }, readNotifications);
         
     | 
| 
       25 
     | 
    
         
            -
              addHook('onWidgetSet', onWidgetSet);
         
     | 
| 
       26 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            import { addHook } from '@opengis/fastify-table/utils.js';
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            // api
         
     | 
| 
      
 4 
     | 
    
         
            +
            import testEmail from './controllers/testEmail.js';
         
     | 
| 
      
 5 
     | 
    
         
            +
            import readNotifications from './controllers/readNotifications.js'; // mark as read
         
     | 
| 
      
 6 
     | 
    
         
            +
            import userNotifications from './controllers/userNotifications.js'; // check all, backend pagination
         
     | 
| 
      
 7 
     | 
    
         
            +
             
     | 
| 
      
 8 
     | 
    
         
            +
            // hook
         
     | 
| 
      
 9 
     | 
    
         
            +
            import onWidgetSet from './hook/onWidgetSet.js'; // send notification on comment
         
     | 
| 
      
 10 
     | 
    
         
            +
             
     | 
| 
      
 11 
     | 
    
         
            +
            // funcs
         
     | 
| 
      
 12 
     | 
    
         
            +
            // import addNotification from '../notifications/funcs/addNotification.js'; // add to db
         
     | 
| 
      
 13 
     | 
    
         
            +
            // import notification from '../notifications/funcs/sendNotification.js'; // send notification
         
     | 
| 
      
 14 
     | 
    
         
            +
             
     | 
| 
      
 15 
     | 
    
         
            +
            import { notificationSchema, emailSchema } from './schema.js';
         
     | 
| 
      
 16 
     | 
    
         
            +
             
     | 
| 
      
 17 
     | 
    
         
            +
            export default async function route(fastify) {
         
     | 
| 
      
 18 
     | 
    
         
            +
             
     | 
| 
      
 19 
     | 
    
         
            +
              // fastify.decorate('addNotification', addNotification);
         
     | 
| 
      
 20 
     | 
    
         
            +
              // fastify.decorate('notification', notification);
         
     | 
| 
      
 21 
     | 
    
         
            +
             
     | 
| 
      
 22 
     | 
    
         
            +
              fastify.get('/test-email', { config: { policy: ['user'] }, schema: emailSchema }, testEmail);
         
     | 
| 
      
 23 
     | 
    
         
            +
              fastify.get('/notification', { config: { policy: ['user'] }, schema: notificationSchema }, userNotifications);
         
     | 
| 
      
 24 
     | 
    
         
            +
              fastify.get('/notification-read/:id?', { config: { policy: ['user'] }, schema: notificationSchema }, readNotifications);
         
     | 
| 
      
 25 
     | 
    
         
            +
              addHook('onWidgetSet', onWidgetSet);
         
     | 
| 
      
 26 
     | 
    
         
            +
            }
         
     | 
    
        package/server/routes/root.mjs
    CHANGED
    
    | 
         @@ -1,3 +1,3 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            export default async function (fastify, opts) {
         
     | 
| 
       2 
     | 
    
         
            -
              fastify.get(`/test`, () => { return { test: true } });
         
     | 
| 
       3 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            export default async function (fastify, opts) {
         
     | 
| 
      
 2 
     | 
    
         
            +
              fastify.get(`/test`, () => { return { test: true } });
         
     | 
| 
      
 3 
     | 
    
         
            +
            }
         
     | 
| 
         @@ -1,14 +1,14 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            import userCls from './user.cls.js';
         
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            export default async function userClsId(req) {
         
     | 
| 
       4 
     | 
    
         
            -
              const { id } = req.params || {};
         
     | 
| 
       5 
     | 
    
         
            -
              const res = await userCls(req);
         
     | 
| 
       6 
     | 
    
         
            -
              if (req.query?.sql || res?.error) {
         
     | 
| 
       7 
     | 
    
         
            -
                return res;
         
     | 
| 
       8 
     | 
    
         
            -
              }
         
     | 
| 
       9 
     | 
    
         
            -
              const { rows = [] } = res?.message || {};
         
     | 
| 
       10 
     | 
    
         
            -
              if (!rows?.length) {
         
     | 
| 
       11 
     | 
    
         
            -
                return { message: `cls not found: ${id}`, status: 404 };
         
     | 
| 
       12 
     | 
    
         
            -
              }
         
     | 
| 
       13 
     | 
    
         
            -
              return { message: rows?.[0], status: 200 };
         
     | 
| 
       14 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            import userCls from './user.cls.js';
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            export default async function userClsId(req) {
         
     | 
| 
      
 4 
     | 
    
         
            +
              const { id } = req.params || {};
         
     | 
| 
      
 5 
     | 
    
         
            +
              const res = await userCls(req);
         
     | 
| 
      
 6 
     | 
    
         
            +
              if (req.query?.sql || res?.error) {
         
     | 
| 
      
 7 
     | 
    
         
            +
                return res;
         
     | 
| 
      
 8 
     | 
    
         
            +
              }
         
     | 
| 
      
 9 
     | 
    
         
            +
              const { rows = [] } = res?.message || {};
         
     | 
| 
      
 10 
     | 
    
         
            +
              if (!rows?.length) {
         
     | 
| 
      
 11 
     | 
    
         
            +
                return { message: `cls not found: ${id}`, status: 404 };
         
     | 
| 
      
 12 
     | 
    
         
            +
              }
         
     | 
| 
      
 13 
     | 
    
         
            +
              return { message: rows?.[0], status: 200 };
         
     | 
| 
      
 14 
     | 
    
         
            +
            }
         
     | 
| 
         @@ -1,72 +1,72 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            import { readFile } from 'node:fs/promises';
         
     | 
| 
       2 
     | 
    
         
            -
            import { getTemplatePath, pgClients } from '@opengis/fastify-table/utils.js';
         
     | 
| 
       3 
     | 
    
         
            -
             
     | 
| 
       4 
     | 
    
         
            -
            export default async function userCls(req) {
         
     | 
| 
       5 
     | 
    
         
            -
              const {
         
     | 
| 
       6 
     | 
    
         
            -
                pg = pgClients.client, params = {}, query = {}, user = {},
         
     | 
| 
       7 
     | 
    
         
            -
              } = req;
         
     | 
| 
       8 
     | 
    
         
            -
              const { uid } = user || {};
         
     | 
| 
       9 
     | 
    
         
            -
             
     | 
| 
       10 
     | 
    
         
            -
              if (!uid) {
         
     | 
| 
       11 
     | 
    
         
            -
                return { message: 'access restricted', status: 403 };
         
     | 
| 
       12 
     | 
    
         
            -
              }
         
     | 
| 
       13 
     | 
    
         
            -
             
     | 
| 
       14 
     | 
    
         
            -
              if (query.type && !['json', 'sql'].includes(query.type)) {
         
     | 
| 
       15 
     | 
    
         
            -
                return { message: 'param type is invalid', status: 400 };
         
     | 
| 
       16 
     | 
    
         
            -
              }
         
     | 
| 
       17 
     | 
    
         
            -
             
     | 
| 
       18 
     | 
    
         
            -
              const q = `select user_clsid as id, name, type, 
         
     | 
| 
       19 
     | 
    
         
            -
              case when type='json' then (
         
     | 
| 
       20 
     | 
    
         
            -
                  ${params?.id
         
     | 
| 
       21 
     | 
    
         
            -
                  ? `(with recursive rows as (
         
     | 
| 
       22 
     | 
    
         
            -
                    select user_clsid, code as id, name as text, icon, color, parent
         
     | 
| 
       23 
     | 
    
         
            -
                    from admin.user_cls a
         
     | 
| 
       24 
     | 
    
         
            -
                    where name=u.name
         
     | 
| 
       25 
     | 
    
         
            -
                    union all
         
     | 
| 
       26 
     | 
    
         
            -
                    select a.user_clsid, a.code, a.name, a.icon, a.color, a.parent
         
     | 
| 
       27 
     | 
    
         
            -
                    from admin.user_cls a
         
     | 
| 
       28 
     | 
    
         
            -
                    join rows b on a.parent=b.text
         
     | 
| 
       29 
     | 
    
         
            -
                  ) select json_agg(row_to_json(q)) from rows q where text<>u.name
         
     | 
| 
       30 
     | 
    
         
            -
                 )`
         
     | 
| 
       31 
     | 
    
         
            -
                  : 'select count(*)::int from admin.user_cls where parent=u.name'} ) else null end as children, 
         
     | 
| 
       32 
     | 
    
         
            -
                  case when type='sql' then data else null end as sql from admin.user_cls u
         
     | 
| 
       33 
     | 
    
         
            -
                where (case when type='json' then parent is null else true end)
         
     | 
| 
       34 
     | 
    
         
            -
                and uid=(select uid from admin.users where $1 in (login,uid) limit 1)
         
     | 
| 
       35 
     | 
    
         
            -
                and parent is null and ${query.type ? `type='${query.type}'` : '1=1'} and ${params?.id ? 'u.name=$2' : '1=1'}
         
     | 
| 
       36 
     | 
    
         
            -
                order by cdate desc`;
         
     | 
| 
       37 
     | 
    
         
            -
             
     | 
| 
       38 
     | 
    
         
            -
              if (query?.sql) return q;
         
     | 
| 
       39 
     | 
    
         
            -
             
     | 
| 
       40 
     | 
    
         
            -
             
     | 
| 
       41 
     | 
    
         
            -
              const { rows = [] } = await pg.query(q, [uid, params.id].filter((el) => el));
         
     | 
| 
       42 
     | 
    
         
            -
             
     | 
| 
       43 
     | 
    
         
            -
              rows.forEach((row) => {
         
     | 
| 
       44 
     | 
    
         
            -
                if (row.type === 'sql') delete row.children;
         
     | 
| 
       45 
     | 
    
         
            -
                if (row.type === 'json') { delete row.sql; Object.assign(row, { children: row.children || (params?.id ? [] : 0) }); }
         
     | 
| 
       46 
     | 
    
         
            -
              });
         
     | 
| 
       47 
     | 
    
         
            -
             
     | 
| 
       48 
     | 
    
         
            -
              const clsList = query.type === 'sql' ? [] : getTemplatePath('cls'); // skip cls if type sql
         
     | 
| 
       49 
     | 
    
         
            -
              const selectList = query.type === 'json' ? [] : getTemplatePath('select'); // skip sql if type json
         
     | 
| 
       50 
     | 
    
         
            -
             
     | 
| 
       51 
     | 
    
         
            -
              const userClsNames = rows.map((el) => el.name);
         
     | 
| 
       52 
     | 
    
         
            -
              const selectNames = selectList.map((el) => el[0]);
         
     | 
| 
       53 
     | 
    
         
            -
             
     | 
| 
       54 
     | 
    
         
            -
              const arr = (clsList || []).concat(selectList || []).map((el) => ({ name: el[0], path: el[1], type: el[2] }))
         
     | 
| 
       55 
     | 
    
         
            -
                ?.filter((el) => (params?.id ? el.name === params?.id : true))
         
     | 
| 
       56 
     | 
    
         
            -
                ?.filter((el) => ['sql', 'json'].includes(el?.type) && !userClsNames.includes(el.name))
         
     | 
| 
       57 
     | 
    
         
            -
                ?.filter((el) => (el?.type === 'json' ? !selectNames?.includes(el?.name) : true));
         
     | 
| 
       58 
     | 
    
         
            -
             
     | 
| 
       59 
     | 
    
         
            -
              const res = await Promise.all(arr?.map(async (el) => {
         
     | 
| 
       60 
     | 
    
         
            -
                const type = { json: 'cls', sql: 'select' }[el.type] || el.type;
         
     | 
| 
       61 
     | 
    
         
            -
                // const clsData = await getSelect(type, el.name);
         
     | 
| 
       62 
     | 
    
         
            -
                const str = await readFile(el.path, 'utf-8');
         
     | 
| 
       63 
     | 
    
         
            -
                const clsData = type === 'cls' ? JSON.parse(str) : str;
         
     | 
| 
       64 
     | 
    
         
            -
                if (type === 'cls') {
         
     | 
| 
       65 
     | 
    
         
            -
                  const children = params?.id ? clsData : clsData?.length || 0;
         
     | 
| 
       66 
     | 
    
         
            -
                  return { name: el.name, type: el.type, children };
         
     | 
| 
       67 
     | 
    
         
            -
                }
         
     | 
| 
       68 
     | 
    
         
            -
                return { name: el.name, type: el.type, sql: clsData?.sql || clsData };
         
     | 
| 
       69 
     | 
    
         
            -
              }));
         
     | 
| 
       70 
     | 
    
         
            -
             
     | 
| 
       71 
     | 
    
         
            -
              return { message: { rows: rows.concat(res) }, status: 200 };
         
     | 
| 
       72 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            import { readFile } from 'node:fs/promises';
         
     | 
| 
      
 2 
     | 
    
         
            +
            import { getTemplatePath, pgClients } from '@opengis/fastify-table/utils.js';
         
     | 
| 
      
 3 
     | 
    
         
            +
             
     | 
| 
      
 4 
     | 
    
         
            +
            export default async function userCls(req) {
         
     | 
| 
      
 5 
     | 
    
         
            +
              const {
         
     | 
| 
      
 6 
     | 
    
         
            +
                pg = pgClients.client, params = {}, query = {}, user = {},
         
     | 
| 
      
 7 
     | 
    
         
            +
              } = req;
         
     | 
| 
      
 8 
     | 
    
         
            +
              const { uid } = user || {};
         
     | 
| 
      
 9 
     | 
    
         
            +
             
     | 
| 
      
 10 
     | 
    
         
            +
              if (!uid) {
         
     | 
| 
      
 11 
     | 
    
         
            +
                return { message: 'access restricted', status: 403 };
         
     | 
| 
      
 12 
     | 
    
         
            +
              }
         
     | 
| 
      
 13 
     | 
    
         
            +
             
     | 
| 
      
 14 
     | 
    
         
            +
              if (query.type && !['json', 'sql'].includes(query.type)) {
         
     | 
| 
      
 15 
     | 
    
         
            +
                return { message: 'param type is invalid', status: 400 };
         
     | 
| 
      
 16 
     | 
    
         
            +
              }
         
     | 
| 
      
 17 
     | 
    
         
            +
             
     | 
| 
      
 18 
     | 
    
         
            +
              const q = `select user_clsid as id, name, type, 
         
     | 
| 
      
 19 
     | 
    
         
            +
              case when type='json' then (
         
     | 
| 
      
 20 
     | 
    
         
            +
                  ${params?.id
         
     | 
| 
      
 21 
     | 
    
         
            +
                  ? `(with recursive rows as (
         
     | 
| 
      
 22 
     | 
    
         
            +
                    select user_clsid, code as id, name as text, icon, color, parent
         
     | 
| 
      
 23 
     | 
    
         
            +
                    from admin.user_cls a
         
     | 
| 
      
 24 
     | 
    
         
            +
                    where name=u.name
         
     | 
| 
      
 25 
     | 
    
         
            +
                    union all
         
     | 
| 
      
 26 
     | 
    
         
            +
                    select a.user_clsid, a.code, a.name, a.icon, a.color, a.parent
         
     | 
| 
      
 27 
     | 
    
         
            +
                    from admin.user_cls a
         
     | 
| 
      
 28 
     | 
    
         
            +
                    join rows b on a.parent=b.text
         
     | 
| 
      
 29 
     | 
    
         
            +
                  ) select json_agg(row_to_json(q)) from rows q where text<>u.name
         
     | 
| 
      
 30 
     | 
    
         
            +
                 )`
         
     | 
| 
      
 31 
     | 
    
         
            +
                  : 'select count(*)::int from admin.user_cls where parent=u.name'} ) else null end as children, 
         
     | 
| 
      
 32 
     | 
    
         
            +
                  case when type='sql' then data else null end as sql from admin.user_cls u
         
     | 
| 
      
 33 
     | 
    
         
            +
                where (case when type='json' then parent is null else true end)
         
     | 
| 
      
 34 
     | 
    
         
            +
                and uid=(select uid from admin.users where $1 in (login,uid) limit 1)
         
     | 
| 
      
 35 
     | 
    
         
            +
                and parent is null and ${query.type ? `type='${query.type}'` : '1=1'} and ${params?.id ? 'u.name=$2' : '1=1'}
         
     | 
| 
      
 36 
     | 
    
         
            +
                order by cdate desc`;
         
     | 
| 
      
 37 
     | 
    
         
            +
             
     | 
| 
      
 38 
     | 
    
         
            +
              if (query?.sql) return q;
         
     | 
| 
      
 39 
     | 
    
         
            +
             
     | 
| 
      
 40 
     | 
    
         
            +
             
     | 
| 
      
 41 
     | 
    
         
            +
              const { rows = [] } = await pg.query(q, [uid, params.id].filter((el) => el));
         
     | 
| 
      
 42 
     | 
    
         
            +
             
     | 
| 
      
 43 
     | 
    
         
            +
              rows.forEach((row) => {
         
     | 
| 
      
 44 
     | 
    
         
            +
                if (row.type === 'sql') delete row.children;
         
     | 
| 
      
 45 
     | 
    
         
            +
                if (row.type === 'json') { delete row.sql; Object.assign(row, { children: row.children || (params?.id ? [] : 0) }); }
         
     | 
| 
      
 46 
     | 
    
         
            +
              });
         
     | 
| 
      
 47 
     | 
    
         
            +
             
     | 
| 
      
 48 
     | 
    
         
            +
              const clsList = query.type === 'sql' ? [] : getTemplatePath('cls'); // skip cls if type sql
         
     | 
| 
      
 49 
     | 
    
         
            +
              const selectList = query.type === 'json' ? [] : getTemplatePath('select'); // skip sql if type json
         
     | 
| 
      
 50 
     | 
    
         
            +
             
     | 
| 
      
 51 
     | 
    
         
            +
              const userClsNames = rows.map((el) => el.name);
         
     | 
| 
      
 52 
     | 
    
         
            +
              const selectNames = selectList.map((el) => el[0]);
         
     | 
| 
      
 53 
     | 
    
         
            +
             
     | 
| 
      
 54 
     | 
    
         
            +
              const arr = (clsList || []).concat(selectList || []).map((el) => ({ name: el[0], path: el[1], type: el[2] }))
         
     | 
| 
      
 55 
     | 
    
         
            +
                ?.filter((el) => (params?.id ? el.name === params?.id : true))
         
     | 
| 
      
 56 
     | 
    
         
            +
                ?.filter((el) => ['sql', 'json'].includes(el?.type) && !userClsNames.includes(el.name))
         
     | 
| 
      
 57 
     | 
    
         
            +
                ?.filter((el) => (el?.type === 'json' ? !selectNames?.includes(el?.name) : true));
         
     | 
| 
      
 58 
     | 
    
         
            +
             
     | 
| 
      
 59 
     | 
    
         
            +
              const res = await Promise.all(arr?.map(async (el) => {
         
     | 
| 
      
 60 
     | 
    
         
            +
                const type = { json: 'cls', sql: 'select' }[el.type] || el.type;
         
     | 
| 
      
 61 
     | 
    
         
            +
                // const clsData = await getSelect(type, el.name);
         
     | 
| 
      
 62 
     | 
    
         
            +
                const str = await readFile(el.path, 'utf-8');
         
     | 
| 
      
 63 
     | 
    
         
            +
                const clsData = type === 'cls' ? JSON.parse(str) : str;
         
     | 
| 
      
 64 
     | 
    
         
            +
                if (type === 'cls') {
         
     | 
| 
      
 65 
     | 
    
         
            +
                  const children = params?.id ? clsData : clsData?.length || 0;
         
     | 
| 
      
 66 
     | 
    
         
            +
                  return { name: el.name, type: el.type, children };
         
     | 
| 
      
 67 
     | 
    
         
            +
                }
         
     | 
| 
      
 68 
     | 
    
         
            +
                return { name: el.name, type: el.type, sql: clsData?.sql || clsData };
         
     | 
| 
      
 69 
     | 
    
         
            +
              }));
         
     | 
| 
      
 70 
     | 
    
         
            +
             
     | 
| 
      
 71 
     | 
    
         
            +
              return { message: { rows: rows.concat(res) }, status: 200 };
         
     | 
| 
      
 72 
     | 
    
         
            +
            }
         
     | 
| 
         @@ -1,17 +1,17 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            export default async function userInfo({
         
     | 
| 
       2 
     | 
    
         
            -
              pg, user = {},
         
     | 
| 
       3 
     | 
    
         
            -
            }) {
         
     | 
| 
       4 
     | 
    
         
            -
              const { uid } = user || {};
         
     | 
| 
       5 
     | 
    
         
            -
             
     | 
| 
       6 
     | 
    
         
            -
              if (!uid) {
         
     | 
| 
       7 
     | 
    
         
            -
                return { message: 'access restricted', status: 403 };
         
     | 
| 
       8 
     | 
    
         
            -
              }
         
     | 
| 
       9 
     | 
    
         
            -
             
     | 
| 
       10 
     | 
    
         
            -
              const data = await pg.query(`select user_name, sur_name, father_name, user_rnokpp, user_type, email, login from admin.users 
         
     | 
| 
       11 
     | 
    
         
            -
              where uid=$1`, [uid]).then((res) => res.rows?.[0] || {});
         
     | 
| 
       12 
     | 
    
         
            -
             
     | 
| 
       13 
     | 
    
         
            -
              const { notifications = 0 } = await pg.query(`select count(*)::int as notifications from crm.notifications 
         
     | 
| 
       14 
     | 
    
         
            -
                where addressee_id=$1 and read is not true`, [uid]).then((res) => res.rows?.[0] || {});
         
     | 
| 
       15 
     | 
    
         
            -
              return { uid, ...data, notifications };
         
     | 
| 
       16 
     | 
    
         
            -
             
     | 
| 
       17 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            export default async function userInfo({
         
     | 
| 
      
 2 
     | 
    
         
            +
              pg, user = {},
         
     | 
| 
      
 3 
     | 
    
         
            +
            }) {
         
     | 
| 
      
 4 
     | 
    
         
            +
              const { uid } = user || {};
         
     | 
| 
      
 5 
     | 
    
         
            +
             
     | 
| 
      
 6 
     | 
    
         
            +
              if (!uid) {
         
     | 
| 
      
 7 
     | 
    
         
            +
                return { message: 'access restricted', status: 403 };
         
     | 
| 
      
 8 
     | 
    
         
            +
              }
         
     | 
| 
      
 9 
     | 
    
         
            +
             
     | 
| 
      
 10 
     | 
    
         
            +
              const data = await pg.query(`select user_name, sur_name, father_name, user_rnokpp, user_type, email, login from admin.users 
         
     | 
| 
      
 11 
     | 
    
         
            +
              where uid=$1`, [uid]).then((res) => res.rows?.[0] || {});
         
     | 
| 
      
 12 
     | 
    
         
            +
             
     | 
| 
      
 13 
     | 
    
         
            +
              const { notifications = 0 } = await pg.query(`select count(*)::int as notifications from crm.notifications 
         
     | 
| 
      
 14 
     | 
    
         
            +
                where addressee_id=$1 and read is not true`, [uid]).then((res) => res.rows?.[0] || {});
         
     | 
| 
      
 15 
     | 
    
         
            +
              return { uid, ...data, notifications };
         
     | 
| 
      
 16 
     | 
    
         
            +
             
     | 
| 
      
 17 
     | 
    
         
            +
            }
         
     | 
| 
         @@ -1,74 +1,74 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            [
         
     | 
| 
       2 
     | 
    
         
            -
                {
         
     | 
| 
       3 
     | 
    
         
            -
                    "id": "12",
         
     | 
| 
       4 
     | 
    
         
            -
                    "text": "Розділювальна смуга",
         
     | 
| 
       5 
     | 
    
         
            -
                    "color": "#85754e",
         
     | 
| 
       6 
     | 
    
         
            -
                    "en": "Dividing line"
         
     | 
| 
       7 
     | 
    
         
            -
                },
         
     | 
| 
       8 
     | 
    
         
            -
                {
         
     | 
| 
       9 
     | 
    
         
            -
                    "id": "6",
         
     | 
| 
       10 
     | 
    
         
            -
                    "text": "Бульвар",
         
     | 
| 
       11 
     | 
    
         
            -
                    "color": "#00703c",
         
     | 
| 
       12 
     | 
    
         
            -
                    "en": "Boulevard"
         
     | 
| 
       13 
     | 
    
         
            -
                },
         
     | 
| 
       14 
     | 
    
         
            -
                {
         
     | 
| 
       15 
     | 
    
         
            -
                    "id": "1",
         
     | 
| 
       16 
     | 
    
         
            -
                    "text": "Вуличні насадження",
         
     | 
| 
       17 
     | 
    
         
            -
                    "color": "#8fbc8f",
         
     | 
| 
       18 
     | 
    
         
            -
                    "en": "Street plantings"
         
     | 
| 
       19 
     | 
    
         
            -
                },
         
     | 
| 
       20 
     | 
    
         
            -
                {
         
     | 
| 
       21 
     | 
    
         
            -
                    "id": "2",
         
     | 
| 
       22 
     | 
    
         
            -
                    "text": "Парк",
         
     | 
| 
       23 
     | 
    
         
            -
                    "color": "#a2cd5a",
         
     | 
| 
       24 
     | 
    
         
            -
                    "en": "Park"
         
     | 
| 
       25 
     | 
    
         
            -
                },
         
     | 
| 
       26 
     | 
    
         
            -
                {
         
     | 
| 
       27 
     | 
    
         
            -
                    "id": "3",
         
     | 
| 
       28 
     | 
    
         
            -
                    "text": "Сад",
         
     | 
| 
       29 
     | 
    
         
            -
                    "color": "#5e8c31",
         
     | 
| 
       30 
     | 
    
         
            -
                    "en": "Garden"
         
     | 
| 
       31 
     | 
    
         
            -
                },
         
     | 
| 
       32 
     | 
    
         
            -
                {
         
     | 
| 
       33 
     | 
    
         
            -
                    "id": "4",
         
     | 
| 
       34 
     | 
    
         
            -
                    "text": "Сквер",
         
     | 
| 
       35 
     | 
    
         
            -
                    "color": "#00ced1",
         
     | 
| 
       36 
     | 
    
         
            -
                    "en": "Square"
         
     | 
| 
       37 
     | 
    
         
            -
                },
         
     | 
| 
       38 
     | 
    
         
            -
                {
         
     | 
| 
       39 
     | 
    
         
            -
                    "id": "9",
         
     | 
| 
       40 
     | 
    
         
            -
                    "text": "Рекреаційна зона",
         
     | 
| 
       41 
     | 
    
         
            -
                    "color": "#e1a95f",
         
     | 
| 
       42 
     | 
    
         
            -
                    "en": "Recreational zone"
         
     | 
| 
       43 
     | 
    
         
            -
                },
         
     | 
| 
       44 
     | 
    
         
            -
                {
         
     | 
| 
       45 
     | 
    
         
            -
                    "id": "10",
         
     | 
| 
       46 
     | 
    
         
            -
                    "text": "Санітарно-захисна зона",
         
     | 
| 
       47 
     | 
    
         
            -
                    "color": "#6e7b8b",
         
     | 
| 
       48 
     | 
    
         
            -
                    "en": "Sanitary protection zone"
         
     | 
| 
       49 
     | 
    
         
            -
                },
         
     | 
| 
       50 
     | 
    
         
            -
                {
         
     | 
| 
       51 
     | 
    
         
            -
                    "id": "11",
         
     | 
| 
       52 
     | 
    
         
            -
                    "text": "Зелені насадження прибудинкової території",
         
     | 
| 
       53 
     | 
    
         
            -
                    "color": "#ace1af",
         
     | 
| 
       54 
     | 
    
         
            -
                    "en": "Green plantations in local areas"
         
     | 
| 
       55 
     | 
    
         
            -
                },
         
     | 
| 
       56 
     | 
    
         
            -
                {
         
     | 
| 
       57 
     | 
    
         
            -
                    "id": "5",
         
     | 
| 
       58 
     | 
    
         
            -
                    "text": "Прибережні зелені насадження",
         
     | 
| 
       59 
     | 
    
         
            -
                    "color": "#698b69",
         
     | 
| 
       60 
     | 
    
         
            -
                    "en": "Coastal green plantations"
         
     | 
| 
       61 
     | 
    
         
            -
                },
         
     | 
| 
       62 
     | 
    
         
            -
                {
         
     | 
| 
       63 
     | 
    
         
            -
                    "id": "7",
         
     | 
| 
       64 
     | 
    
         
            -
                    "text": "Лісопарк",
         
     | 
| 
       65 
     | 
    
         
            -
                    "color": "#6e8b3d",
         
     | 
| 
       66 
     | 
    
         
            -
                    "en": "Forest park"
         
     | 
| 
       67 
     | 
    
         
            -
                },
         
     | 
| 
       68 
     | 
    
         
            -
                {
         
     | 
| 
       69 
     | 
    
         
            -
                    "id": "8",
         
     | 
| 
       70 
     | 
    
         
            -
                    "text": "Міський ліс",
         
     | 
| 
       71 
     | 
    
         
            -
                    "color": "#4a5d23",
         
     | 
| 
       72 
     | 
    
         
            -
                    "en": "Urban forest"
         
     | 
| 
       73 
     | 
    
         
            -
                }
         
     | 
| 
      
 1 
     | 
    
         
            +
            [
         
     | 
| 
      
 2 
     | 
    
         
            +
                {
         
     | 
| 
      
 3 
     | 
    
         
            +
                    "id": "12",
         
     | 
| 
      
 4 
     | 
    
         
            +
                    "text": "Розділювальна смуга",
         
     | 
| 
      
 5 
     | 
    
         
            +
                    "color": "#85754e",
         
     | 
| 
      
 6 
     | 
    
         
            +
                    "en": "Dividing line"
         
     | 
| 
      
 7 
     | 
    
         
            +
                },
         
     | 
| 
      
 8 
     | 
    
         
            +
                {
         
     | 
| 
      
 9 
     | 
    
         
            +
                    "id": "6",
         
     | 
| 
      
 10 
     | 
    
         
            +
                    "text": "Бульвар",
         
     | 
| 
      
 11 
     | 
    
         
            +
                    "color": "#00703c",
         
     | 
| 
      
 12 
     | 
    
         
            +
                    "en": "Boulevard"
         
     | 
| 
      
 13 
     | 
    
         
            +
                },
         
     | 
| 
      
 14 
     | 
    
         
            +
                {
         
     | 
| 
      
 15 
     | 
    
         
            +
                    "id": "1",
         
     | 
| 
      
 16 
     | 
    
         
            +
                    "text": "Вуличні насадження",
         
     | 
| 
      
 17 
     | 
    
         
            +
                    "color": "#8fbc8f",
         
     | 
| 
      
 18 
     | 
    
         
            +
                    "en": "Street plantings"
         
     | 
| 
      
 19 
     | 
    
         
            +
                },
         
     | 
| 
      
 20 
     | 
    
         
            +
                {
         
     | 
| 
      
 21 
     | 
    
         
            +
                    "id": "2",
         
     | 
| 
      
 22 
     | 
    
         
            +
                    "text": "Парк",
         
     | 
| 
      
 23 
     | 
    
         
            +
                    "color": "#a2cd5a",
         
     | 
| 
      
 24 
     | 
    
         
            +
                    "en": "Park"
         
     | 
| 
      
 25 
     | 
    
         
            +
                },
         
     | 
| 
      
 26 
     | 
    
         
            +
                {
         
     | 
| 
      
 27 
     | 
    
         
            +
                    "id": "3",
         
     | 
| 
      
 28 
     | 
    
         
            +
                    "text": "Сад",
         
     | 
| 
      
 29 
     | 
    
         
            +
                    "color": "#5e8c31",
         
     | 
| 
      
 30 
     | 
    
         
            +
                    "en": "Garden"
         
     | 
| 
      
 31 
     | 
    
         
            +
                },
         
     | 
| 
      
 32 
     | 
    
         
            +
                {
         
     | 
| 
      
 33 
     | 
    
         
            +
                    "id": "4",
         
     | 
| 
      
 34 
     | 
    
         
            +
                    "text": "Сквер",
         
     | 
| 
      
 35 
     | 
    
         
            +
                    "color": "#00ced1",
         
     | 
| 
      
 36 
     | 
    
         
            +
                    "en": "Square"
         
     | 
| 
      
 37 
     | 
    
         
            +
                },
         
     | 
| 
      
 38 
     | 
    
         
            +
                {
         
     | 
| 
      
 39 
     | 
    
         
            +
                    "id": "9",
         
     | 
| 
      
 40 
     | 
    
         
            +
                    "text": "Рекреаційна зона",
         
     | 
| 
      
 41 
     | 
    
         
            +
                    "color": "#e1a95f",
         
     | 
| 
      
 42 
     | 
    
         
            +
                    "en": "Recreational zone"
         
     | 
| 
      
 43 
     | 
    
         
            +
                },
         
     | 
| 
      
 44 
     | 
    
         
            +
                {
         
     | 
| 
      
 45 
     | 
    
         
            +
                    "id": "10",
         
     | 
| 
      
 46 
     | 
    
         
            +
                    "text": "Санітарно-захисна зона",
         
     | 
| 
      
 47 
     | 
    
         
            +
                    "color": "#6e7b8b",
         
     | 
| 
      
 48 
     | 
    
         
            +
                    "en": "Sanitary protection zone"
         
     | 
| 
      
 49 
     | 
    
         
            +
                },
         
     | 
| 
      
 50 
     | 
    
         
            +
                {
         
     | 
| 
      
 51 
     | 
    
         
            +
                    "id": "11",
         
     | 
| 
      
 52 
     | 
    
         
            +
                    "text": "Зелені насадження прибудинкової території",
         
     | 
| 
      
 53 
     | 
    
         
            +
                    "color": "#ace1af",
         
     | 
| 
      
 54 
     | 
    
         
            +
                    "en": "Green plantations in local areas"
         
     | 
| 
      
 55 
     | 
    
         
            +
                },
         
     | 
| 
      
 56 
     | 
    
         
            +
                {
         
     | 
| 
      
 57 
     | 
    
         
            +
                    "id": "5",
         
     | 
| 
      
 58 
     | 
    
         
            +
                    "text": "Прибережні зелені насадження",
         
     | 
| 
      
 59 
     | 
    
         
            +
                    "color": "#698b69",
         
     | 
| 
      
 60 
     | 
    
         
            +
                    "en": "Coastal green plantations"
         
     | 
| 
      
 61 
     | 
    
         
            +
                },
         
     | 
| 
      
 62 
     | 
    
         
            +
                {
         
     | 
| 
      
 63 
     | 
    
         
            +
                    "id": "7",
         
     | 
| 
      
 64 
     | 
    
         
            +
                    "text": "Лісопарк",
         
     | 
| 
      
 65 
     | 
    
         
            +
                    "color": "#6e8b3d",
         
     | 
| 
      
 66 
     | 
    
         
            +
                    "en": "Forest park"
         
     | 
| 
      
 67 
     | 
    
         
            +
                },
         
     | 
| 
      
 68 
     | 
    
         
            +
                {
         
     | 
| 
      
 69 
     | 
    
         
            +
                    "id": "8",
         
     | 
| 
      
 70 
     | 
    
         
            +
                    "text": "Міський ліс",
         
     | 
| 
      
 71 
     | 
    
         
            +
                    "color": "#4a5d23",
         
     | 
| 
      
 72 
     | 
    
         
            +
                    "en": "Urban forest"
         
     | 
| 
      
 73 
     | 
    
         
            +
                }
         
     | 
| 
       74 
74 
     | 
    
         
             
            ]
         
     | 
| 
         @@ -1,10 +1,10 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            [
         
     | 
| 
       2 
     | 
    
         
            -
                {
         
     | 
| 
       3 
     | 
    
         
            -
                    "id": 1,
         
     | 
| 
       4 
     | 
    
         
            -
                    "text": "test"
         
     | 
| 
       5 
     | 
    
         
            -
                },
         
     | 
| 
       6 
     | 
    
         
            -
                {
         
     | 
| 
       7 
     | 
    
         
            -
                    "id": 2,
         
     | 
| 
       8 
     | 
    
         
            -
                    "text": "test2"
         
     | 
| 
       9 
     | 
    
         
            -
                }
         
     | 
| 
      
 1 
     | 
    
         
            +
            [
         
     | 
| 
      
 2 
     | 
    
         
            +
                {
         
     | 
| 
      
 3 
     | 
    
         
            +
                    "id": 1,
         
     | 
| 
      
 4 
     | 
    
         
            +
                    "text": "test"
         
     | 
| 
      
 5 
     | 
    
         
            +
                },
         
     | 
| 
      
 6 
     | 
    
         
            +
                {
         
     | 
| 
      
 7 
     | 
    
         
            +
                    "id": 2,
         
     | 
| 
      
 8 
     | 
    
         
            +
                    "text": "test2"
         
     | 
| 
      
 9 
     | 
    
         
            +
                }
         
     | 
| 
       10 
10 
     | 
    
         
             
            ]
         
     | 
| 
         @@ -1,50 +1,50 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            {
         
     | 
| 
       2 
     | 
    
         
            -
                "label_style": "vertical",
         
     | 
| 
       3 
     | 
    
         
            -
                "schema": {
         
     | 
| 
       4 
     | 
    
         
            -
                    "code": {
         
     | 
| 
       5 
     | 
    
         
            -
                        "type": "Text",
         
     | 
| 
       6 
     | 
    
         
            -
                        "ua": "Код",
         
     | 
| 
       7 
     | 
    
         
            -
                        "ru": "Код",
         
     | 
| 
       8 
     | 
    
         
            -
                        "en": "Code",
         
     | 
| 
       9 
     | 
    
         
            -
                        "placeholder": {
         
     | 
| 
       10 
     | 
    
         
            -
                            "ua": "Код",
         
     | 
| 
       11 
     | 
    
         
            -
                            "ru": "Код",
         
     | 
| 
       12 
     | 
    
         
            -
                            "en": "Code"
         
     | 
| 
       13 
     | 
    
         
            -
                        },
         
     | 
| 
       14 
     | 
    
         
            -
                        "validators": [
         
     | 
| 
       15 
     | 
    
         
            -
                            "required"
         
     | 
| 
       16 
     | 
    
         
            -
                        ],
         
     | 
| 
       17 
     | 
    
         
            -
                        "col": 6
         
     | 
| 
       18 
     | 
    
         
            -
                    },        
         
     | 
| 
       19 
     | 
    
         
            -
                    "name": {
         
     | 
| 
       20 
     | 
    
         
            -
                        "type": "Text",
         
     | 
| 
       21 
     | 
    
         
            -
                        "ua": "Назва",
         
     | 
| 
       22 
     | 
    
         
            -
                        "ru": "Название",
         
     | 
| 
       23 
     | 
    
         
            -
                        "en": "Name",
         
     | 
| 
       24 
     | 
    
         
            -
                        "placeholder": {
         
     | 
| 
       25 
     | 
    
         
            -
                            "ua": "Назва",
         
     | 
| 
       26 
     | 
    
         
            -
                            "ru": "Название",
         
     | 
| 
       27 
     | 
    
         
            -
                            "en": "Name"
         
     | 
| 
       28 
     | 
    
         
            -
                        },
         
     | 
| 
       29 
     | 
    
         
            -
                        "validators": [
         
     | 
| 
       30 
     | 
    
         
            -
                            "required"
         
     | 
| 
       31 
     | 
    
         
            -
                        ],
         
     | 
| 
       32 
     | 
    
         
            -
                        "col": 6
         
     | 
| 
       33 
     | 
    
         
            -
                    },
         
     | 
| 
       34 
     | 
    
         
            -
                    "icon": {
         
     | 
| 
       35 
     | 
    
         
            -
                        "type": "File",
         
     | 
| 
       36 
     | 
    
         
            -
                        "multi": false,
         
     | 
| 
       37 
     | 
    
         
            -
                        "ua": "Іконка",
         
     | 
| 
       38 
     | 
    
         
            -
                        "ru": "Иконка",
         
     | 
| 
       39 
     | 
    
         
            -
                        "en": "Icon",
         
     | 
| 
       40 
     | 
    
         
            -
                        "col": 6
         
     | 
| 
       41 
     | 
    
         
            -
                    },
         
     | 
| 
       42 
     | 
    
         
            -
                    "color": {
         
     | 
| 
       43 
     | 
    
         
            -
                        "type": "ColorPicker",
         
     | 
| 
       44 
     | 
    
         
            -
                        "ua": "Колір",
         
     | 
| 
       45 
     | 
    
         
            -
                        "ru": "Цвет",
         
     | 
| 
       46 
     | 
    
         
            -
                        "en": "Color",
         
     | 
| 
       47 
     | 
    
         
            -
                        "col": 6
         
     | 
| 
       48 
     | 
    
         
            -
                    }        
         
     | 
| 
       49 
     | 
    
         
            -
                }
         
     | 
| 
      
 1 
     | 
    
         
            +
            {
         
     | 
| 
      
 2 
     | 
    
         
            +
                "label_style": "vertical",
         
     | 
| 
      
 3 
     | 
    
         
            +
                "schema": {
         
     | 
| 
      
 4 
     | 
    
         
            +
                    "code": {
         
     | 
| 
      
 5 
     | 
    
         
            +
                        "type": "Text",
         
     | 
| 
      
 6 
     | 
    
         
            +
                        "ua": "Код",
         
     | 
| 
      
 7 
     | 
    
         
            +
                        "ru": "Код",
         
     | 
| 
      
 8 
     | 
    
         
            +
                        "en": "Code",
         
     | 
| 
      
 9 
     | 
    
         
            +
                        "placeholder": {
         
     | 
| 
      
 10 
     | 
    
         
            +
                            "ua": "Код",
         
     | 
| 
      
 11 
     | 
    
         
            +
                            "ru": "Код",
         
     | 
| 
      
 12 
     | 
    
         
            +
                            "en": "Code"
         
     | 
| 
      
 13 
     | 
    
         
            +
                        },
         
     | 
| 
      
 14 
     | 
    
         
            +
                        "validators": [
         
     | 
| 
      
 15 
     | 
    
         
            +
                            "required"
         
     | 
| 
      
 16 
     | 
    
         
            +
                        ],
         
     | 
| 
      
 17 
     | 
    
         
            +
                        "col": 6
         
     | 
| 
      
 18 
     | 
    
         
            +
                    },        
         
     | 
| 
      
 19 
     | 
    
         
            +
                    "name": {
         
     | 
| 
      
 20 
     | 
    
         
            +
                        "type": "Text",
         
     | 
| 
      
 21 
     | 
    
         
            +
                        "ua": "Назва",
         
     | 
| 
      
 22 
     | 
    
         
            +
                        "ru": "Название",
         
     | 
| 
      
 23 
     | 
    
         
            +
                        "en": "Name",
         
     | 
| 
      
 24 
     | 
    
         
            +
                        "placeholder": {
         
     | 
| 
      
 25 
     | 
    
         
            +
                            "ua": "Назва",
         
     | 
| 
      
 26 
     | 
    
         
            +
                            "ru": "Название",
         
     | 
| 
      
 27 
     | 
    
         
            +
                            "en": "Name"
         
     | 
| 
      
 28 
     | 
    
         
            +
                        },
         
     | 
| 
      
 29 
     | 
    
         
            +
                        "validators": [
         
     | 
| 
      
 30 
     | 
    
         
            +
                            "required"
         
     | 
| 
      
 31 
     | 
    
         
            +
                        ],
         
     | 
| 
      
 32 
     | 
    
         
            +
                        "col": 6
         
     | 
| 
      
 33 
     | 
    
         
            +
                    },
         
     | 
| 
      
 34 
     | 
    
         
            +
                    "icon": {
         
     | 
| 
      
 35 
     | 
    
         
            +
                        "type": "File",
         
     | 
| 
      
 36 
     | 
    
         
            +
                        "multi": false,
         
     | 
| 
      
 37 
     | 
    
         
            +
                        "ua": "Іконка",
         
     | 
| 
      
 38 
     | 
    
         
            +
                        "ru": "Иконка",
         
     | 
| 
      
 39 
     | 
    
         
            +
                        "en": "Icon",
         
     | 
| 
      
 40 
     | 
    
         
            +
                        "col": 6
         
     | 
| 
      
 41 
     | 
    
         
            +
                    },
         
     | 
| 
      
 42 
     | 
    
         
            +
                    "color": {
         
     | 
| 
      
 43 
     | 
    
         
            +
                        "type": "ColorPicker",
         
     | 
| 
      
 44 
     | 
    
         
            +
                        "ua": "Колір",
         
     | 
| 
      
 45 
     | 
    
         
            +
                        "ru": "Цвет",
         
     | 
| 
      
 46 
     | 
    
         
            +
                        "en": "Color",
         
     | 
| 
      
 47 
     | 
    
         
            +
                        "col": 6
         
     | 
| 
      
 48 
     | 
    
         
            +
                    }        
         
     | 
| 
      
 49 
     | 
    
         
            +
                }
         
     | 
| 
       50 
50 
     | 
    
         
             
            }
         
     | 
| 
         @@ -1,22 +1,22 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            {
         
     | 
| 
       2 
     | 
    
         
            -
                "schema": {
         
     | 
| 
       3 
     | 
    
         
            -
                    "user_uid": {
         
     | 
| 
       4 
     | 
    
         
            -
                        "type": "Autocomplete",
         
     | 
| 
       5 
     | 
    
         
            -
                        "data": "user_id",
         
     | 
| 
       6 
     | 
    
         
            -
                        "ua": "Користувач",
         
     | 
| 
       7 
     | 
    
         
            -
                        "validators": [
         
     | 
| 
       8 
     | 
    
         
            -
                            "required"
         
     | 
| 
       9 
     | 
    
         
            -
                        ]
         
     | 
| 
       10 
     | 
    
         
            -
                    },
         
     | 
| 
       11 
     | 
    
         
            -
                    "user_access": {
         
     | 
| 
       12 
     | 
    
         
            -
                        "type": "Select2",
         
     | 
| 
       13 
     | 
    
         
            -
                        "data": "user_access",
         
     | 
| 
       14 
     | 
    
         
            -
                        "placeholder": "Згідно налаштувань інтерфейсів",
         
     | 
| 
       15 
     | 
    
         
            -
                        "ua": "Доступ"
         
     | 
| 
       16 
     | 
    
         
            -
                    },
         
     | 
| 
       17 
     | 
    
         
            -
                    "expiration": {
         
     | 
| 
       18 
     | 
    
         
            -
                        "type": "DatePicker",
         
     | 
| 
       19 
     | 
    
         
            -
                        "ua": "Закінчення терміну дії доступу до групи"
         
     | 
| 
       20 
     | 
    
         
            -
                    }
         
     | 
| 
       21 
     | 
    
         
            -
                }
         
     | 
| 
      
 1 
     | 
    
         
            +
            {
         
     | 
| 
      
 2 
     | 
    
         
            +
                "schema": {
         
     | 
| 
      
 3 
     | 
    
         
            +
                    "user_uid": {
         
     | 
| 
      
 4 
     | 
    
         
            +
                        "type": "Autocomplete",
         
     | 
| 
      
 5 
     | 
    
         
            +
                        "data": "user_id",
         
     | 
| 
      
 6 
     | 
    
         
            +
                        "ua": "Користувач",
         
     | 
| 
      
 7 
     | 
    
         
            +
                        "validators": [
         
     | 
| 
      
 8 
     | 
    
         
            +
                            "required"
         
     | 
| 
      
 9 
     | 
    
         
            +
                        ]
         
     | 
| 
      
 10 
     | 
    
         
            +
                    },
         
     | 
| 
      
 11 
     | 
    
         
            +
                    "user_access": {
         
     | 
| 
      
 12 
     | 
    
         
            +
                        "type": "Select2",
         
     | 
| 
      
 13 
     | 
    
         
            +
                        "data": "user_access",
         
     | 
| 
      
 14 
     | 
    
         
            +
                        "placeholder": "Згідно налаштувань інтерфейсів",
         
     | 
| 
      
 15 
     | 
    
         
            +
                        "ua": "Доступ"
         
     | 
| 
      
 16 
     | 
    
         
            +
                    },
         
     | 
| 
      
 17 
     | 
    
         
            +
                    "expiration": {
         
     | 
| 
      
 18 
     | 
    
         
            +
                        "type": "DatePicker",
         
     | 
| 
      
 19 
     | 
    
         
            +
                        "ua": "Закінчення терміну дії доступу до групи"
         
     | 
| 
      
 20 
     | 
    
         
            +
                    }
         
     | 
| 
      
 21 
     | 
    
         
            +
                }
         
     | 
| 
       22 
22 
     | 
    
         
             
            }
         
     | 
| 
         @@ -1,11 +1,11 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            {
         
     | 
| 
       2 
     | 
    
         
            -
              "login":{"type":"Text","ua":"Логін","ru":"Логин"}, 
         
     | 
| 
       3 
     | 
    
         
            -
              "password":{"type":"Password","ua":"Пароль","ru":"Пароль","validators": [
         
     | 
| 
       4 
     | 
    
         
            -
                            {
         
     | 
| 
       5 
     | 
    
         
            -
                                "type": "regexp",
         
     | 
| 
       6 
     | 
    
         
            -
                                "regexp": "^.{8,}$",
         
     | 
| 
       7 
     | 
    
         
            -
                                "flags": "gm",
         
     | 
| 
       8 
     | 
    
         
            -
                                "message": "Пароль повинен бути більше 8 символів"
         
     | 
| 
       9 
     | 
    
         
            -
                            }
         
     | 
| 
       10 
     | 
    
         
            -
                        ]}
         
     | 
| 
      
 1 
     | 
    
         
            +
            {
         
     | 
| 
      
 2 
     | 
    
         
            +
              "login":{"type":"Text","ua":"Логін","ru":"Логин"}, 
         
     | 
| 
      
 3 
     | 
    
         
            +
              "password":{"type":"Password","ua":"Пароль","ru":"Пароль","validators": [
         
     | 
| 
      
 4 
     | 
    
         
            +
                            {
         
     | 
| 
      
 5 
     | 
    
         
            +
                                "type": "regexp",
         
     | 
| 
      
 6 
     | 
    
         
            +
                                "regexp": "^.{8,}$",
         
     | 
| 
      
 7 
     | 
    
         
            +
                                "flags": "gm",
         
     | 
| 
      
 8 
     | 
    
         
            +
                                "message": "Пароль повинен бути більше 8 символів"
         
     | 
| 
      
 9 
     | 
    
         
            +
                            }
         
     | 
| 
      
 10 
     | 
    
         
            +
                        ]}
         
     | 
| 
       11 
11 
     | 
    
         
             
            }
         
     |