UNPKG

@c8y/ngx-components

Version:

Angular modules for Cumulocity IoT applications

158 lines • 20.6 kB
import { ViewContext } from '@c8y/ngx-components'; import { find, forEach, map, startsWith, unary } from 'lodash-es'; import { ReplaySubject } from 'rxjs'; export var ViewContextLegacyParameter; (function (ViewContextLegacyParameter) { ViewContextLegacyParameter["Device"] = "deviceId"; ViewContextLegacyParameter["Group"] = "groupId"; ViewContextLegacyParameter["User"] = "userId"; ViewContextLegacyParameter["Application"] = "applicationId"; ViewContextLegacyParameter["Microservice"] = "applicationId"; ViewContextLegacyParameter["SubscribedApplications"] = "applicationId"; ViewContextLegacyParameter["Tenant"] = "tenantId"; ViewContextLegacyParameter["Service"] = "deviceId"; ViewContextLegacyParameter["Simulators"] = "deviceId"; // required to hook the Alarms tab to a Simulator view })(ViewContextLegacyParameter || (ViewContextLegacyParameter = {})); function c8yViewsProvider($routeProvider, c8yTabsProvider, c8yPathUtils) { 'ngInject'; const viewMap = {}; const contextViews = new ReplaySubject(); return { when, $get() { return { contextViews, when(path, cfg) { return when(path, cfg, true); }, getByPath, prefixWithSlash }; } }; /** * @ngdoc function * @name when * @methodOf c8y.ui.provider:c8yViewsProvider * * @description * Defines a view for given route. * If multiple views are defined for a single route then there will be a separate tab for each view available when user visits that route. * * @param path Target route. * @param cfg View configuration object with the following properties: * * - **name** - `string` - View's name (in case of multiple views at single route this will be displayed as tab's title). * - **priority** - `integer` - View's priority (in case of multiple views at single route this will determine the position of view's tab in the tabs stack). * - **icon** - `string` - Font Awesome icon name for the view (displayed on the tab's header). * - **showIf** - `function` - Function returning boolean value indicating whether to show a tab for the view or not. * - **templateUrl** - `string` - Path to the template to use for displaying the view. * * You can also provide other view options - the same as available for standard {@link https://docs.angularjs.org/api/ngRoute/provider/$routeProvider $routeProvider} in AngularJS. * * ```html * The following example demonstrates how to add a new view to device details route * (which will be displayed as a tab if other views are assigned to the same route): * <pre> * c8yViewsProvider.when('/device/:deviceId', { * name: 'Tracking', * templateUrl: ':::PLUGIN_PATH:::/views/index.html', * icon: 'crosshairs', * showIf: ['$routeParams', 'c8yDevices', function ($routeParams, c8yDevices) { * var deviceId = $routeParams.deviceId; * return c8yDevices.detailCached(deviceId).then(function (res) { * var device = res.data; * return device && (device.c8y_MotionTracking || device.c8y_Geofence); * }); * }] * }); * </pre> * ``` */ function when(path, cfg, runPhase) { const newPath = prefixWithSlash(path); cfg.resolve = cfg.resolve || {}; // eslint-disable-next-line no-underscore-dangle cfg.resolve.__c8y_locales = [ 'c8yLocales', c8yLocales => { return c8yLocales.initDone; } ]; let currentCfg = viewMap[newPath]; const originalPath = newPath; if (!cfg.name) { // console.warn('View name not defined'); } if (!currentCfg) { viewMap[newPath] = []; currentCfg = viewMap[newPath]; } const upgradedContext = Object.keys(ViewContext) .map(key => ({ key, isUpgrade: prefixWithSlash(ViewContext[key].replace('id', ViewContextLegacyParameter[key])) === path })) .find(({ isUpgrade }) => isUpgrade); if (upgradedContext) { currentCfg.push(cfg); cfg.path = newPath; const p = c8yPathUtils.appendSegment(originalPath.replace(path, ''), cfg.name); contextViews.next({ ...cfg, path: cfg.name ? p.substring(1) : '', contextKey: upgradedContext.key, runPhase }); cfg.showIf = undefined; if (cfg.name) { cfg.path = c8yPathUtils.appendSegment(originalPath, cfg.name); } } else { if (currentCfg.length === 1) { const [existingConfig] = currentCfg; existingConfig.path = c8yPathUtils.appendSegment(originalPath, existingConfig.name); existingConfig.tab = createTab(originalPath, existingConfig); $routeProvider.when(existingConfig.path, existingConfig); } currentCfg.push(cfg); cfg.path = newPath; if (currentCfg.length > 1) { cfg.path = c8yPathUtils.appendSegment(originalPath, cfg.name); createTab(originalPath, cfg); $routeProvider.when(prefixWithSlash(originalPath), { resolveRedirectTo($route, $q, c8yUiUtil, c8yTabs, gettextCatalog) { 'ngInject'; const sortedCurrentCfg = c8yTabsProvider.sortTabsViews(currentCfg, gettextCatalog); const params = $route.current.pathParams; return $q .all(map(sortedCurrentCfg, unary(c8yUiUtil.configureVisibility))) .then(views => { const first = find(views, 'show'); let url = first.path; forEach(params, (val, key) => { url = url.replace(`:${key}`, val); }); c8yTabs.redirectedViewPath = url; return url; }); } }); } } return $routeProvider.when(prefixWithSlash(cfg.path), cfg); } function getByPath(path) { return viewMap[prefixWithSlash(path)]; } function createTab(path, cfg) { c8yTabsProvider.addTab(path, cfg); } function prefixWithSlash(path) { const prefix = startsWith(path, '/') ? '' : '/'; return prefix + path; } } export { c8yViewsProvider }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"views.provider.js","sourceRoot":"","sources":["../../../../upgrade/ng1/views.provider.ts"],"names":[],"mappings":"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","sourcesContent":["import { ViewContext } from '@c8y/ngx-components';\nimport { find, forEach, map, startsWith, unary } from 'lodash-es';\nimport { ReplaySubject } from 'rxjs';\n\nexport enum ViewContextLegacyParameter {\n  Device = 'deviceId',\n  Group = 'groupId',\n  User = 'userId',\n  Application = 'applicationId',\n  Microservice = 'applicationId',\n  SubscribedApplications = 'applicationId',\n  Tenant = 'tenantId',\n  Service = 'deviceId', // use 'deviceId' as parameter name so that device views (Measurements, Events, Alarms) can be reused for service\n  Simulators = 'deviceId' // required to hook the Alarms tab to a Simulator view\n}\n\nfunction c8yViewsProvider($routeProvider, c8yTabsProvider, c8yPathUtils) {\n  'ngInject';\n\n  const viewMap = {};\n  const contextViews = new ReplaySubject();\n\n  return {\n    when,\n    $get() {\n      return {\n        contextViews,\n        when(path, cfg) {\n          return when(path, cfg, true);\n        },\n        getByPath,\n        prefixWithSlash\n      };\n    }\n  };\n\n  /**\n   * @ngdoc function\n   * @name when\n   * @methodOf c8y.ui.provider:c8yViewsProvider\n   *\n   * @description\n   * Defines a view for given route.\n   * If multiple views are defined for a single route then there will be a separate tab for each view available when user visits that route.\n   *\n   * @param path Target route.\n   * @param cfg View configuration object with the following properties:\n   *\n   * - **name** - `string` - View's name (in case of multiple views at single route this will be displayed as tab's title).\n   * - **priority** - `integer` - View's priority (in case of multiple views at single route this will determine the position of view's tab in the tabs stack).\n   * - **icon** - `string` - Font Awesome icon name for the view (displayed on the tab's header).\n   * - **showIf** - `function` - Function returning boolean value indicating whether to show a tab for the view or not.\n   * - **templateUrl** - `string` - Path to the template to use for displaying the view.\n   *\n   * You can also provide other view options - the same as available for standard {@link https://docs.angularjs.org/api/ngRoute/provider/$routeProvider $routeProvider} in AngularJS.\n   *\n   * ```html\n   * The following example demonstrates how to add a new view to device details route\n   * (which will be displayed as a tab if other views are assigned to the same route):\n   * <pre>\n   *   c8yViewsProvider.when('/device/:deviceId', {\n   *     name: 'Tracking',\n   *     templateUrl: ':::PLUGIN_PATH:::/views/index.html',\n   *     icon: 'crosshairs',\n   *     showIf: ['$routeParams', 'c8yDevices', function ($routeParams, c8yDevices) {\n   *       var deviceId = $routeParams.deviceId;\n   *       return c8yDevices.detailCached(deviceId).then(function (res) {\n   *         var device = res.data;\n   *         return device && (device.c8y_MotionTracking || device.c8y_Geofence);\n   *       });\n   *     }]\n   *   });\n   * </pre>\n   * ```\n   */\n  function when(path, cfg, runPhase) {\n    const newPath = prefixWithSlash(path);\n    cfg.resolve = cfg.resolve || {};\n    // eslint-disable-next-line no-underscore-dangle\n    cfg.resolve.__c8y_locales = [\n      'c8yLocales',\n      c8yLocales => {\n        return c8yLocales.initDone;\n      }\n    ];\n\n    let currentCfg = viewMap[newPath];\n    const originalPath = newPath;\n\n    if (!cfg.name) {\n      // console.warn('View name not defined');\n    }\n\n    if (!currentCfg) {\n      viewMap[newPath] = [];\n      currentCfg = viewMap[newPath];\n    }\n\n    const upgradedContext = Object.keys(ViewContext)\n      .map(key => ({\n        key,\n        isUpgrade:\n          prefixWithSlash(ViewContext[key].replace('id', ViewContextLegacyParameter[key])) === path\n      }))\n      .find(({ isUpgrade }) => isUpgrade);\n\n    if (upgradedContext) {\n      currentCfg.push(cfg);\n      cfg.path = newPath;\n\n      const p = c8yPathUtils.appendSegment(originalPath.replace(path, ''), cfg.name);\n      contextViews.next({\n        ...cfg,\n        path: cfg.name ? p.substring(1) : '',\n        contextKey: upgradedContext.key,\n        runPhase\n      });\n      cfg.showIf = undefined;\n      if (cfg.name) {\n        cfg.path = c8yPathUtils.appendSegment(originalPath, cfg.name);\n      }\n    } else {\n      if (currentCfg.length === 1) {\n        const [existingConfig] = currentCfg;\n        existingConfig.path = c8yPathUtils.appendSegment(originalPath, existingConfig.name);\n        existingConfig.tab = createTab(originalPath, existingConfig);\n        $routeProvider.when(existingConfig.path, existingConfig);\n      }\n\n      currentCfg.push(cfg);\n      cfg.path = newPath;\n\n      if (currentCfg.length > 1) {\n        cfg.path = c8yPathUtils.appendSegment(originalPath, cfg.name);\n        createTab(originalPath, cfg);\n\n        $routeProvider.when(prefixWithSlash(originalPath), {\n          resolveRedirectTo($route, $q, c8yUiUtil, c8yTabs, gettextCatalog) {\n            'ngInject';\n\n            const sortedCurrentCfg = c8yTabsProvider.sortTabsViews(currentCfg, gettextCatalog);\n            const params = $route.current.pathParams;\n\n            return $q\n              .all(map(sortedCurrentCfg, unary(c8yUiUtil.configureVisibility)))\n              .then(views => {\n                const first = find(views, 'show');\n                let url = first.path;\n                forEach(params, (val, key) => {\n                  url = url.replace(`:${key}`, val);\n                });\n                c8yTabs.redirectedViewPath = url;\n                return url;\n              });\n          }\n        });\n      }\n    }\n    return $routeProvider.when(prefixWithSlash(cfg.path), cfg);\n  }\n\n  function getByPath(path) {\n    return viewMap[prefixWithSlash(path)];\n  }\n\n  function createTab(path, cfg) {\n    c8yTabsProvider.addTab(path, cfg);\n  }\n\n  function prefixWithSlash(path) {\n    const prefix = startsWith(path, '/') ? '' : '/';\n    return prefix + path;\n  }\n}\n\nexport { c8yViewsProvider };\n"]}