@c8y/ngx-components 1021.35.2 → 1021.37.5

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.
Files changed (74) hide show
  1. package/alarms/alarms-view.service.d.ts.map +1 -1
  2. package/context-dashboard/context-dashboard.component.d.ts.map +1 -1
  3. package/core/common/forOf.directive.d.ts.map +1 -1
  4. package/core/data-grid/data-grid.model.d.ts +2 -0
  5. package/core/data-grid/data-grid.model.d.ts.map +1 -1
  6. package/core/data-grid/data-grid.service.d.ts.map +1 -1
  7. package/core/file-picker/file-picker.component.d.ts +4 -2
  8. package/core/file-picker/file-picker.component.d.ts.map +1 -1
  9. package/core/forms/validation-pattern.d.ts +8 -0
  10. package/core/forms/validation-pattern.d.ts.map +1 -1
  11. package/device-profile/device-profile.component.d.ts +3 -1
  12. package/device-profile/device-profile.component.d.ts.map +1 -1
  13. package/device-profile/device-profile.service.d.ts +25 -0
  14. package/device-profile/device-profile.service.d.ts.map +1 -1
  15. package/device-profile/device-tab-profile/device-tab-profile.component.d.ts +7 -10
  16. package/device-profile/device-tab-profile/device-tab-profile.component.d.ts.map +1 -1
  17. package/esm2022/alarms/alarms-view.service.mjs +3 -1
  18. package/esm2022/alarms/alarms.component.mjs +2 -2
  19. package/esm2022/assets-navigator/asset-selector/asset-selector.component.mjs +3 -3
  20. package/esm2022/context-dashboard/context-dashboard.component.mjs +8 -2
  21. package/esm2022/core/common/forOf.directive.mjs +5 -1
  22. package/esm2022/core/data-grid/data-grid.model.mjs +1 -1
  23. package/esm2022/core/data-grid/data-grid.service.mjs +5 -2
  24. package/esm2022/core/file-picker/file-picker.component.mjs +7 -5
  25. package/esm2022/core/forms/validation-pattern.mjs +9 -1
  26. package/esm2022/core/navigator/navigator-bottom/navigator-bottom.component.mjs +3 -3
  27. package/esm2022/device-profile/device-profile.component.mjs +14 -10
  28. package/esm2022/device-profile/device-profile.service.mjs +58 -2
  29. package/esm2022/device-profile/device-tab-profile/device-tab-profile.component.mjs +40 -34
  30. package/esm2022/operations/operations-list/operations-list-item.component.mjs +14 -6
  31. package/esm2022/operations/operations-list/operations-list.component.mjs +13 -5
  32. package/esm2022/repository/configuration/list/configuration-detail.component.mjs +5 -4
  33. package/esm2022/repository/firmware/list/add-firmware-modal.component.mjs +5 -4
  34. package/esm2022/repository/software/list/add-software-modal.component.mjs +5 -4
  35. package/fesm2022/c8y-ngx-components-alarms.mjs +3 -1
  36. package/fesm2022/c8y-ngx-components-alarms.mjs.map +1 -1
  37. package/fesm2022/c8y-ngx-components-assets-navigator.mjs +2 -2
  38. package/fesm2022/c8y-ngx-components-assets-navigator.mjs.map +1 -1
  39. package/fesm2022/c8y-ngx-components-context-dashboard.mjs +6 -0
  40. package/fesm2022/c8y-ngx-components-context-dashboard.mjs.map +1 -1
  41. package/fesm2022/c8y-ngx-components-device-profile.mjs +107 -43
  42. package/fesm2022/c8y-ngx-components-device-profile.mjs.map +1 -1
  43. package/fesm2022/c8y-ngx-components-operations-operations-list.mjs +25 -9
  44. package/fesm2022/c8y-ngx-components-operations-operations-list.mjs.map +1 -1
  45. package/fesm2022/c8y-ngx-components-repository-configuration.mjs +10 -9
  46. package/fesm2022/c8y-ngx-components-repository-configuration.mjs.map +1 -1
  47. package/fesm2022/c8y-ngx-components-repository-firmware.mjs +4 -3
  48. package/fesm2022/c8y-ngx-components-repository-firmware.mjs.map +1 -1
  49. package/fesm2022/c8y-ngx-components-repository-software.mjs +4 -3
  50. package/fesm2022/c8y-ngx-components-repository-software.mjs.map +1 -1
  51. package/fesm2022/c8y-ngx-components.mjs +88 -72
  52. package/fesm2022/c8y-ngx-components.mjs.map +1 -1
  53. package/locales/de.po +10 -4
  54. package/locales/es.po +10 -4
  55. package/locales/fr.po +10 -4
  56. package/locales/ja_JP.po +10 -5
  57. package/locales/ko.po +10 -5
  58. package/locales/locales.pot +8 -2
  59. package/locales/nl.po +10 -4
  60. package/locales/pl.po +10 -4
  61. package/locales/pt_BR.po +10 -4
  62. package/locales/zh_CN.po +10 -5
  63. package/locales/zh_TW.po +10 -5
  64. package/operations/operations-list/operations-list-item.component.d.ts +3 -1
  65. package/operations/operations-list/operations-list-item.component.d.ts.map +1 -1
  66. package/operations/operations-list/operations-list.component.d.ts +1 -0
  67. package/operations/operations-list/operations-list.component.d.ts.map +1 -1
  68. package/package.json +1 -1
  69. package/repository/configuration/list/configuration-detail.component.d.ts +2 -1
  70. package/repository/configuration/list/configuration-detail.component.d.ts.map +1 -1
  71. package/repository/firmware/list/add-firmware-modal.component.d.ts +2 -1
  72. package/repository/firmware/list/add-firmware-modal.component.d.ts.map +1 -1
  73. package/repository/software/list/add-software-modal.component.d.ts +2 -1
  74. package/repository/software/list/add-software-modal.component.d.ts.map +1 -1
@@ -28,4 +28,4 @@ export var BuiltInActionType;
28
28
  /** Export built-in action. */
29
29
  BuiltInActionType["Export"] = "EXPORT";
30
30
  })(BuiltInActionType || (BuiltInActionType = {}));
31
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-grid.model.js","sourceRoot":"","sources":["../../../../core/data-grid/data-grid.model.ts"],"names":[],"mappings":"AA0KA,MAAM,CAAC,MAAM,sBAAsB,GAAG,EAAE,CAAC;AAEzC,4CAA4C;AAC5C,MAAM,CAAC,MAAM,mBAAmB,GAAG;IACjC,6CAA6C;IAC7C,IAAI,EAAE,CAAC;IAEP,gDAAgD;IAChD,OAAO,EAAE,CAAC;IAEV,mDAAmD;IACnD,YAAY,EAAE,IAAI;IAElB,kDAAkD;IAClD,WAAW,EAAE,IAAI;CAClB,CAAC;AAaF,yBAAyB;AACzB,MAAM,CAAN,IAAY,mBAMX;AAND,WAAY,mBAAmB;IAC7B,6DAA6D;IAC7D,mDAA4B,CAAA;IAE5B,6DAA6D;IAC7D,mDAA4B,CAAA;AAC9B,CAAC,EANW,mBAAmB,KAAnB,mBAAmB,QAM9B;AA6ND,mCAAmC;AACnC,MAAM,CAAN,IAAY,iBASX;AATD,WAAY,iBAAiB;IAC3B,4BAA4B;IAC5B,kCAAa,CAAA;IAEb,8BAA8B;IAC9B,sCAAiB,CAAA;IAEjB,8BAA8B;IAC9B,sCAAiB,CAAA;AACnB,CAAC,EATW,iBAAiB,KAAjB,iBAAiB,QAS5B","sourcesContent":["import { TemplateRef, Type } from '@angular/core';\nimport { FormGroup } from '@angular/forms';\nimport { ActivatedRoute } from '@angular/router';\nimport { IResultList } from '@c8y/client';\nimport { FormlyFieldConfig } from '@ngx-formly/core';\nimport { Observable } from 'rxjs';\nimport { ExtensionFactory } from '../common/extension-hooks';\nimport { GridConfigContext } from './data-grid-configuration.model';\n\n/** Describes the interface of the data grid. */\nexport interface DataGrid {\n  /** The list of columns. */\n  columns: Column[];\n\n  /** The list of items. */\n  rows: Row[];\n\n  /** Pagination object. */\n  pagination: Pagination;\n\n  /** A callback function to fetch server data. */\n  serverSideDataCallback: ServerSideDataCallback;\n\n  /** Whether items are selectable. */\n  selectable: boolean;\n\n  /** The name of the primary key property. */\n  selectionPrimaryKey: string;\n\n  /** Data grid display options. */\n  displayOptions: DisplayOptions;\n\n  /** Individual action controls. */\n  actionControls: ActionControl[];\n\n  /** Bulk action controls. */\n  bulkActionControls: BulkActionControl[];\n\n  /** Header action controls. */\n  headerActionControls?: HeaderActionControl[];\n}\n\n/** Describes a data grid column. */\nexport interface Column {\n  /** The name for the column. */\n  name: string | SpecialColumnName;\n  /** The column's header (if not given, `name` is used by default). */\n  header?: string;\n  /** The path in a row item to read the cell value from. */\n  path?: string;\n  /** Whether the column is displayed. */\n  visible?: boolean;\n  /** Whether the column can be moved to another position. */\n  positionFixed?: boolean;\n  /** Marks a user configured custom column */\n  custom?: boolean;\n  /** Whether the column can be resized. */\n  resizable?: boolean;\n  /** Optional type of data in the column (used to set data-type attribute, e.g. for additional styling). */\n  dataType?: ColumnDataType;\n  /** The column's size, e.g. \"40 px\". */\n  gridTrackSize?: string;\n  /** Additional CSS classes for the header cell. */\n  headerCSSClassName?: string | string[];\n  /** Additional CSS classes for data cells. */\n  cellCSSClassName?: string | string[];\n\n  /** Whether the column is sortable. */\n  sortable?: boolean;\n  /** Whether the sort order is ascending, descending or not specified. */\n  sortOrder?: SortOrder;\n  sortingConfig?: ColumnSortingConfig;\n\n  /** Whether the column is filterable. */\n  filterable?: boolean;\n  /** The string to search for or a function for client-side filtering. */\n  filterPredicate?: string | FilterPredicateFunction;\n  /** Custom filtering form renderer can set any value here and it can be used to build a query to the server. */\n  externalFilterQuery?: any;\n  filteringConfig?: FormlyColumnFilteringConfig;\n\n  /** Custom data cell renderer component. Inject `CellRendererContext` to get access to data value, item and column object. */\n  cellRendererComponent?: Type<any>;\n  /** Custom header cell renderer component. Inject `CellRendererContext` to get access to header value and column object. */\n  headerCellRendererComponent?: Type<any>;\n  /** Custom filtering form renderer component. Inject `FilteringFormRendererContext` to get access to column object and methods: applyFilter, resetFilter. */\n  filteringFormRendererComponent?: Type<any>;\n}\n\n/** Describes a column configuration. */\nexport interface ColumnConfig {\n  /** The name for the column. */\n  name?: string;\n\n  /** Whether the column is displayed. */\n  visible?: boolean;\n\n  /** Whether the sort order is ascending, descending or not specified. */\n  sortOrder?: SortOrder;\n\n  /** The settings of filter in a column. */\n  filter?: Filter;\n}\n\n/** Describes a custom column configuration. */\nexport interface CustomColumnConfig extends ColumnConfig {\n  /** JSON path to the managed object property to be displayed */\n  path: string;\n\n  /** Column header title */\n  header: string;\n\n  /** Flag to identify custom columns */\n  custom: boolean;\n}\n\n/** Describes the settings of filter in a column. */\nexport interface Filter {\n  /** Custom filtering form renderer can set any value here and it can be used to build a query to the server. */\n  externalFilterQuery?: any;\n}\n\n/** Describes an object with data grid configuration. */\nexport interface GridConfig {\n  /** The configuration objects for all the columns. */\n  columns: ColumnConfig[];\n\n  /** Pagination object. */\n  pagination: Pagination;\n}\n\n/**  */\nexport const enum SpecialColumnName {\n  /** Column with checkbox. */\n  Checkbox = 'checkbox',\n\n  /** Column with radio button. */\n  RadioButton = 'radio-button',\n\n  /** Column with row actions. */\n  Actions = 'actions'\n}\n\n/**  */\nexport const enum ColumnDataType {\n  /** Column with icon. */\n  Icon = 'icon',\n\n  /** Numeric column. */\n  Numeric = 'numeric',\n\n  /** Column with short text. */\n  TextShort = 'text-short',\n\n  /** Column with long text. */\n  TextLong = 'text-long'\n}\n\n/** Classes for column data record. */\nexport const enum ColumnDataRecordClassName {\n  /** An icon. */\n  Icon = 'data-record-icon',\n\n  /** A header. */\n  Header = 'data-record-header',\n\n  /** Default - empty. */\n  Default = ''\n}\n\nexport const minColumnGridTrackSize = 80;\n\n/** Maps column types to relative widths. */\nexport const ratiosByColumnTypes = {\n  /** The width ration for icon column type. */\n  icon: 1,\n\n  /** The width ration for numeric column type. */\n  numeric: 1,\n\n  /** The width ration for short text column type. */\n  'text-short': 1.67,\n\n  /** The width ration for long text column type. */\n  'text-long': 3.33\n};\n\n/** Sorting order: ascending, descending, or not specified. */\nexport type SortOrder = 'asc' | 'desc' | '';\n\n/**\n * A filter predicate function.\n * @param item The current item to be checked.\n * @param path The property path configured in the current column.\n * @returns The boolean value indicating whether the item matches a condition or not.\n */\nexport type FilterPredicateFunction = (item: object, path: string) => boolean;\n\n/** Filtering actions. */\nexport enum FilteringActionType {\n  /** Action invoked when filter settings are to be applied. */\n  ApplyFilter = 'APPLY_FILTER',\n\n  /** Action invoked when filter settings are to be cleared. */\n  ResetFilter = 'RESET_FILTER'\n}\n\nexport interface FormlyColumnFilteringConfig {\n  /** Defines a FieldConfig configuration for Formly.  */\n  fields?: FormlyFieldConfig[];\n  /** Defines a JSON schema for filtering config model. Can be used instead of `fields` */\n  schema?: object;\n  /** Defines a model that will be applied to rendered form. */\n  model?: object;\n  /** Defines FormGroup instance */\n  formGroup?: FormGroup;\n  /**\n   * Transforms a filtering config model to an array of partial filter chip objects.\n   * @param model An object with defined structure (e.g. by schema).\n   * @returns An array of partial filter chip objects, each containing at least `displayValue` and the actual `value` to be updated.\n   * Optionally, other properties from `FilterChip` are allowed, for example, a custom `remove` callback to be executed when a chip is being removed.\n   */\n  generateChips?: (model: any) => PartialFilterChipGenerationType[];\n  /**\n   * Transforms a filtering config model (e.g. coming from schema form component) to a query object.\n   * However, using schema form component is not necessary.\n   * Model can be defined arbitrarily but must converted to a valid query object.\n   * @param model An object with defined structure (e.g. by schema).\n   * @returns A query object to be used to generate a query string (QueryUtils).\n   */\n  getFilter: (model: any) => any;\n}\n\nexport interface ColumnSortingConfig {\n  pathSortingConfigs: PathSortingConfig[];\n}\n\nexport interface PathSortingConfig {\n  path: string;\n  sortOrderModifier?: SortOrderModifier;\n}\n\nexport const enum SortOrderModifier {\n  Keep,\n  Invert\n}\n\n/** Describes an item in the row. */\nexport interface Row {\n  /** The unique identifier of the item. */\n  id: string;\n\n  /** Any item property. */\n  [key: string]: any;\n}\n\n/** Describes a pagination object. */\nexport interface Pagination {\n  /** The number of the current page. */\n  currentPage?: number;\n\n  /** The number of the next page. */\n  nextPage?: number;\n\n  /** The number of items on a single page. */\n  pageSize: number;\n}\n\n/**\n * A callback function to fetch server data.\n * @param dataSourceModifier The current data source modifier.\n * @returns Returns the result from server.\n */\nexport type ServerSideDataCallback = (\n  dataSourceModifier: DataSourceModifier\n) => ServerSideDataResult | Promise<ServerSideDataResult> | Observable<ServerSideDataResult>;\n\n/** Describes a data source modifier for requesting server data. */\nexport interface DataSourceModifier {\n  /** The list of columns. */\n  columns: Column[];\n\n  /** Text to search. */\n  searchText: string;\n\n  /** Pagination object. */\n  pagination: Pagination;\n\n  /** Selection info object. */\n  selection: {\n    /** Whether the selection is enabled. */\n    enabled: boolean;\n\n    /** The name of the primary key. */\n    primaryKey: string;\n  };\n}\n\n/** Describes a result from server with data and additional statistics. */\nexport type ServerSideDataResult = IResultList<object> & {\n  /** The real grand total number of items (the whole dataset). */\n  size: number;\n\n  /** The number of items after filtering applied to the whole dataset (subset). */\n  filteredSize: number;\n\n  /** The list of ids of all filtered items (subset). */\n  filteredDataIds?: string[];\n};\n\n/** Describes an object with data source statistics. */\nexport interface DataSourceStats {\n  /** The real grand total number of items (the whole dataset). */\n  size: number;\n\n  /** The number of items after filtering applied to the whole dataset (subset). */\n  filteredSize: number;\n\n  /** The number of the current page. */\n  currentPage: number;\n\n  /** The number of the next page. */\n  nextPage?: number;\n\n  /** The number of items currently shown in the list view, a.k.a. current page (what the user sees on screen). */\n  currentPageSize: number;\n\n  /** The number of items on the first page.  */\n  firstPageSize: number;\n}\n\n/** Describes data grid display options. */\nexport interface DisplayOptions {\n  /** Show or hide zebra-striping in the table */\n  striped: boolean;\n\n  /** Show or hide each cell border */\n  bordered: boolean;\n\n  /** Show or hide the grid header */\n  gridHeader: boolean;\n\n  /** Show or hide filter label in the grid header */\n  filter: boolean;\n\n  /** Enable to display a background color on hover */\n  hover: boolean;\n}\n\n/** Describes a data grid action control for individual item. */\nexport interface ActionControl {\n  /** The type of the action, predefined or custom. */\n  type: BuiltInActionType | string;\n\n  /** The label for the action button. */\n  text?: string;\n\n  /** The icon for the action button. */\n  icon?: string;\n\n  /** The icon classes for the action button icon. */\n  iconClasses?: string;\n\n  /** Action icon will be visible on hover only */\n  showOnHover?: boolean;\n\n  /**\n   * A callback function.\n   * @param item The item to perform the action on.\n   * @param reload The function to call, if you want to reload the grid.\n   */\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  callback: ((item: Row, reload: () => void) => void) | Function;\n\n  /**\n   * Determines if the action should be shown for given item (if not defined, the action will be shown always).\n   * @param item The item for which the action is supposed to be performed.\n   */\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  showIf?: ((item: Row & string[]) => boolean | Promise<boolean> | Observable<boolean>) | Function;\n\n  /**\n   * Defines the order in which action controls appear. Higher value means earlier position.\n   * Controls without priority are handled with priority = 0. If you want to place an action after\n   * an action without priority, negative numbers can be used.\n   */\n  priority?: number;\n\n  [key: string]: any;\n}\n\n/** Describes a data grid action control for multiple items. */\nexport interface BulkActionControl extends ActionControl {\n  /**\n   * A callback function.\n   * @param selectedItemIds The list of ids of the selected items.\n   * @param reload The function to call, if you want to reload the grid.\n   */\n  callback(selectedItemIds: string[], reload: () => void): void;\n\n  /**\n   * Determines if the action should be shown for given selected items (if not defined, the action will be shown always).\n   * @param selectedItemIds The list of ids of the selected items for which the action is supposed to be performed.\n   */\n  showIf?(selectedItemIds: string[]): boolean;\n}\n\n/** Describes a data grid action control for the header. */\nexport interface HeaderActionControl extends ActionControl {\n  /**\n   * Custom template can be provided. If no template is provided,\n   * the default headerActionControl template is used.\n   */\n  template?: TemplateRef<any>;\n\n  /**\n   * A callback function.\n   */\n  callback(): void;\n\n  /**\n   * Determines if the action should be shown.\n   */\n  showIf?(): boolean;\n}\n\n/** Predefined types of actions. */\nexport enum BuiltInActionType {\n  /** Edit built-in action. */\n  Edit = 'EDIT',\n\n  /** Delete built-in action. */\n  Delete = 'DELETE',\n\n  /** Export built-in action. */\n  Export = 'EXPORT'\n}\n\n/**\n * Allows to attach action controls to specified grids.\n */\nexport interface ActionControlHook {\n  /**\n   * The action control(s) that will be attached to the grid.\n   */\n  actionControls: ActionControl | ActionControl[];\n  /**\n   * A functions that determines to which grid(s) the action control(s) will be attached.\n   * Not providing a function results in the actions control(s) not being attached to any grid.\n   */\n  matchesGrid: (\n    route: ActivatedRoute,\n    context?: GridConfigContext\n  ) => boolean | Promise<boolean> | Observable<boolean>;\n}\n\n/**\n * Factory to implement in order to add action controls from an extension.\n */\nexport type ActionControlFactory = ExtensionFactory<ActionControlHook>;\n\n/**\n * Defines an interface for a filter chip object, which represents a filter applied to a column.\n */\nexport interface FilterChip {\n  /**\n   * The actual value that the chip will hold.\n   * It can be just a string or a more complex object like\n   * {id: 123, name: Test, value: test}.\n   */\n  value: any;\n\n  /**\n   * The value that the chip will display in the dropdown.\n   */\n  displayValue: string;\n\n  /**\n   * Optional label that will be displayed if filter chips are separated into multiple groups.\n   */\n  label?: string;\n\n  /** An optional property that holds the path to the chip value in the externalFilterQuery. */\n  path?: string[];\n\n  /**\n   * An optional property that holds the form value taken from the column configuration.\n   */\n  externalFilterQuery?: any;\n\n  /**\n   * An optional property that holds the formly filtering configuration for the column that this filter is applied to.\n   */\n  filteringConfig?: FormlyColumnFilteringConfig;\n\n  /** An optional property that holds a string to search for. */\n  filterPredicate?: string;\n\n  /**\n   * The name of the column that this filter is applied to.\n   */\n  columnName: string | SpecialColumnName;\n\n  /**\n   * An optional function to remove this filter from the column.\n   * If not provided will set default removal function that will reduce the external query by removing the object that is resolved by the path set in the FilterChip.\n   * @returns An object defined by the `FilterChip` type, including the `columnName` and  updated `externalFilterQuery` or 'filterPredicate' properties.\n   * The returned object may also include additional properties defined by the `FilterChip` type as partials.\n   */\n  remove?: () => PartialFilterChipRemovalType;\n}\n\n/**\n * Represents a partial filter chip with either 'columnName' and 'externalFilterQuery' or 'columnName' and 'filterPredicate'.\n * @typedef {Object} PartialFilterChipRemovalType\n * @property {string} columnName - The column name for the filter chip.\n * @property {any} [externalFilterQuery] - The external filter query for the chip used for complex filters.\n * @property {any} [filterPredicate] - The filtering predicate for the chip used for simple filters with client-side filtering.\n * @property {...FilterChip} [optionalProperties] - Optional properties from the FilterChip type.\n */\nexport type PartialFilterChipRemovalType = (\n  | Required<Pick<FilterChip, 'columnName' | 'externalFilterQuery'>>\n  | Required<Pick<FilterChip, 'columnName' | 'filterPredicate'>>\n) &\n  Partial<FilterChip>;\n\n/**\n * Represents a partial filter chip with required properties 'displayValue' and 'value'.\n * @typedef {Object} PartialFilterChipGenerationType\n * @property {string} displayValue - The display value for the filter chip.\n * @property {any} value - The value for the filter chip.\n * @property {...FilterChip} [optionalProperties] - Optional properties from the FilterChip type.\n */\nexport type PartialFilterChipGenerationType = Required<Pick<FilterChip, 'displayValue' | 'value'>> &\n  Partial<FilterChip>;\n\n/**\n * Defines an interface for a mapper function that maps a `FormlyFieldConfig` to a `FilterChip`.\n */\nexport interface FilterMapper {\n  /**\n   * The type or types of the `FormlyFieldConfig` that this mapper applies to.\n   */\n  fieldType: string[] | string;\n\n  /**\n   * Maps a `FormlyFieldConfig` and a `FilterChip` to an `Observable` of a `FilterChip`.\n   *\n   * This function extends the provided `FilterChip` object with additional properties based on the information in the `FormlyFieldConfig`.\n   * The mapping process depends on the specific implementation of the `FilterMapper`.\n   *\n   * @param field - The `FormlyFieldConfig` to map.\n   * @param filter - The `FilterChip` to map.\n   * @returns An `Observable` that emits the mapped `FilterChip`.\n   * If undefined is returned an error is thrown and the chip is not visualized.\n   *\n   * ```typescript\n   * // Example implementation of `map` function in a `FilterMapper`\n   * map(field: FormlyFieldConfig, filter: FilterChip): Observable<FilterChip> {\n   *   if (this.fieldType.includes(field.type.toString())) {\n   *     return toObservable(field.templateOptions.options).pipe(\n   *       map(options => {\n   *         const object = options.find(option => option.value === get(filter.externalFilterQuery, filter.path));\n   *         return {\n   *           ...filter,\n   *           value: object,\n   *           displayValue: object?.label\n   *         };\n   *       })\n   *     );\n   *   }\n   * }\n   * ```\n   */\n  map: (field: FormlyFieldConfig, filter: FilterChip) => Observable<FilterChip>;\n}\n"]}
31
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-grid.model.js","sourceRoot":"","sources":["../../../../core/data-grid/data-grid.model.ts"],"names":[],"mappings":"AA4KA,MAAM,CAAC,MAAM,sBAAsB,GAAG,EAAE,CAAC;AAEzC,4CAA4C;AAC5C,MAAM,CAAC,MAAM,mBAAmB,GAAG;IACjC,6CAA6C;IAC7C,IAAI,EAAE,CAAC;IAEP,gDAAgD;IAChD,OAAO,EAAE,CAAC;IAEV,mDAAmD;IACnD,YAAY,EAAE,IAAI;IAElB,kDAAkD;IAClD,WAAW,EAAE,IAAI;CAClB,CAAC;AAaF,yBAAyB;AACzB,MAAM,CAAN,IAAY,mBAMX;AAND,WAAY,mBAAmB;IAC7B,6DAA6D;IAC7D,mDAA4B,CAAA;IAE5B,6DAA6D;IAC7D,mDAA4B,CAAA;AAC9B,CAAC,EANW,mBAAmB,KAAnB,mBAAmB,QAM9B;AA6ND,mCAAmC;AACnC,MAAM,CAAN,IAAY,iBASX;AATD,WAAY,iBAAiB;IAC3B,4BAA4B;IAC5B,kCAAa,CAAA;IAEb,8BAA8B;IAC9B,sCAAiB,CAAA;IAEjB,8BAA8B;IAC9B,sCAAiB,CAAA;AACnB,CAAC,EATW,iBAAiB,KAAjB,iBAAiB,QAS5B","sourcesContent":["import { TemplateRef, Type } from '@angular/core';\nimport { FormGroup } from '@angular/forms';\nimport { ActivatedRoute } from '@angular/router';\nimport { IResultList } from '@c8y/client';\nimport { FormlyFieldConfig } from '@ngx-formly/core';\nimport { Observable } from 'rxjs';\nimport { ExtensionFactory } from '../common/extension-hooks';\nimport { GridConfigContext } from './data-grid-configuration.model';\n\n/** Describes the interface of the data grid. */\nexport interface DataGrid {\n  /** The list of columns. */\n  columns: Column[];\n\n  /** The list of items. */\n  rows: Row[];\n\n  /** Pagination object. */\n  pagination: Pagination;\n\n  /** A callback function to fetch server data. */\n  serverSideDataCallback: ServerSideDataCallback;\n\n  /** Whether items are selectable. */\n  selectable: boolean;\n\n  /** The name of the primary key property. */\n  selectionPrimaryKey: string;\n\n  /** Data grid display options. */\n  displayOptions: DisplayOptions;\n\n  /** Individual action controls. */\n  actionControls: ActionControl[];\n\n  /** Bulk action controls. */\n  bulkActionControls: BulkActionControl[];\n\n  /** Header action controls. */\n  headerActionControls?: HeaderActionControl[];\n}\n\n/** Describes a data grid column. */\nexport interface Column {\n  /** The name for the column. */\n  name: string | SpecialColumnName;\n  /** The column's header (if not given, `name` is used by default). */\n  header?: string;\n  /** The path in a row item to read the cell value from. */\n  path?: string;\n  /** Whether the column is displayed. */\n  visible?: boolean;\n  /** Whether the column can be moved to another position. */\n  positionFixed?: boolean;\n  /** Marks a user configured custom column */\n  custom?: boolean;\n  /** Whether the column can be resized. */\n  resizable?: boolean;\n  /** Optional type of data in the column (used to set data-type attribute, e.g. for additional styling). */\n  dataType?: ColumnDataType;\n  /** The column's size, e.g. \"40 px\". */\n  gridTrackSize?: string;\n  /** Additional CSS classes for the header cell. */\n  headerCSSClassName?: string | string[];\n  /** Additional CSS classes for data cells. */\n  cellCSSClassName?: string | string[];\n\n  /** Whether the column is sortable. */\n  sortable?: boolean;\n  /** Whether the sort order is ascending, descending or not specified. */\n  sortOrder?: SortOrder;\n  sortingConfig?: ColumnSortingConfig;\n\n  /** Whether the column is filterable. */\n  filterable?: boolean;\n  /** The string to search for or a function for client-side filtering. */\n  filterPredicate?: string | FilterPredicateFunction;\n  /** Custom filtering form renderer can set any value here and it can be used to build a query to the server. */\n  externalFilterQuery?: any;\n  filteringConfig?: FormlyColumnFilteringConfig;\n\n  /** Custom data cell renderer component. Inject `CellRendererContext` to get access to data value, item and column object. */\n  cellRendererComponent?: Type<any>;\n  /** Custom header cell renderer component. Inject `CellRendererContext` to get access to header value and column object. */\n  headerCellRendererComponent?: Type<any>;\n  /** Custom filtering form renderer component. Inject `FilteringFormRendererContext` to get access to column object and methods: applyFilter, resetFilter. */\n  filteringFormRendererComponent?: Type<any>;\n}\n\n/** Describes a column configuration. */\nexport interface ColumnConfig {\n  /** The name for the column. */\n  name?: string;\n\n  /** Whether the column is displayed. */\n  visible?: boolean;\n\n  /** Whether the sort order is ascending, descending or not specified. */\n  sortOrder?: SortOrder;\n\n  /** The settings of filter in a column. */\n  filter?: Filter;\n}\n\n/** Describes a custom column configuration. */\nexport interface CustomColumnConfig extends ColumnConfig {\n  /** JSON path to the managed object property to be displayed */\n  path: string;\n\n  /** Column header title */\n  header: string;\n\n  /** Flag to identify custom columns */\n  custom: boolean;\n}\n\n/** Describes the settings of filter in a column. */\nexport interface Filter {\n  /** Custom filtering form renderer can set any value here and it can be used to build a query to the server. */\n  externalFilterQuery?: any;\n  /** An optional property that holds a string to search for. */\n  filterPredicate?: string | FilterPredicateFunction;\n}\n\n/** Describes an object with data grid configuration. */\nexport interface GridConfig {\n  /** The configuration objects for all the columns. */\n  columns: ColumnConfig[];\n\n  /** Pagination object. */\n  pagination: Pagination;\n}\n\n/**  */\nexport const enum SpecialColumnName {\n  /** Column with checkbox. */\n  Checkbox = 'checkbox',\n\n  /** Column with radio button. */\n  RadioButton = 'radio-button',\n\n  /** Column with row actions. */\n  Actions = 'actions'\n}\n\n/**  */\nexport const enum ColumnDataType {\n  /** Column with icon. */\n  Icon = 'icon',\n\n  /** Numeric column. */\n  Numeric = 'numeric',\n\n  /** Column with short text. */\n  TextShort = 'text-short',\n\n  /** Column with long text. */\n  TextLong = 'text-long'\n}\n\n/** Classes for column data record. */\nexport const enum ColumnDataRecordClassName {\n  /** An icon. */\n  Icon = 'data-record-icon',\n\n  /** A header. */\n  Header = 'data-record-header',\n\n  /** Default - empty. */\n  Default = ''\n}\n\nexport const minColumnGridTrackSize = 80;\n\n/** Maps column types to relative widths. */\nexport const ratiosByColumnTypes = {\n  /** The width ration for icon column type. */\n  icon: 1,\n\n  /** The width ration for numeric column type. */\n  numeric: 1,\n\n  /** The width ration for short text column type. */\n  'text-short': 1.67,\n\n  /** The width ration for long text column type. */\n  'text-long': 3.33\n};\n\n/** Sorting order: ascending, descending, or not specified. */\nexport type SortOrder = 'asc' | 'desc' | '';\n\n/**\n * A filter predicate function.\n * @param item The current item to be checked.\n * @param path The property path configured in the current column.\n * @returns The boolean value indicating whether the item matches a condition or not.\n */\nexport type FilterPredicateFunction = (item: object, path: string) => boolean;\n\n/** Filtering actions. */\nexport enum FilteringActionType {\n  /** Action invoked when filter settings are to be applied. */\n  ApplyFilter = 'APPLY_FILTER',\n\n  /** Action invoked when filter settings are to be cleared. */\n  ResetFilter = 'RESET_FILTER'\n}\n\nexport interface FormlyColumnFilteringConfig {\n  /** Defines a FieldConfig configuration for Formly.  */\n  fields?: FormlyFieldConfig[];\n  /** Defines a JSON schema for filtering config model. Can be used instead of `fields` */\n  schema?: object;\n  /** Defines a model that will be applied to rendered form. */\n  model?: object;\n  /** Defines FormGroup instance */\n  formGroup?: FormGroup;\n  /**\n   * Transforms a filtering config model to an array of partial filter chip objects.\n   * @param model An object with defined structure (e.g. by schema).\n   * @returns An array of partial filter chip objects, each containing at least `displayValue` and the actual `value` to be updated.\n   * Optionally, other properties from `FilterChip` are allowed, for example, a custom `remove` callback to be executed when a chip is being removed.\n   */\n  generateChips?: (model: any) => PartialFilterChipGenerationType[];\n  /**\n   * Transforms a filtering config model (e.g. coming from schema form component) to a query object.\n   * However, using schema form component is not necessary.\n   * Model can be defined arbitrarily but must converted to a valid query object.\n   * @param model An object with defined structure (e.g. by schema).\n   * @returns A query object to be used to generate a query string (QueryUtils).\n   */\n  getFilter: (model: any) => any;\n}\n\nexport interface ColumnSortingConfig {\n  pathSortingConfigs: PathSortingConfig[];\n}\n\nexport interface PathSortingConfig {\n  path: string;\n  sortOrderModifier?: SortOrderModifier;\n}\n\nexport const enum SortOrderModifier {\n  Keep,\n  Invert\n}\n\n/** Describes an item in the row. */\nexport interface Row {\n  /** The unique identifier of the item. */\n  id: string;\n\n  /** Any item property. */\n  [key: string]: any;\n}\n\n/** Describes a pagination object. */\nexport interface Pagination {\n  /** The number of the current page. */\n  currentPage?: number;\n\n  /** The number of the next page. */\n  nextPage?: number;\n\n  /** The number of items on a single page. */\n  pageSize: number;\n}\n\n/**\n * A callback function to fetch server data.\n * @param dataSourceModifier The current data source modifier.\n * @returns Returns the result from server.\n */\nexport type ServerSideDataCallback = (\n  dataSourceModifier: DataSourceModifier\n) => ServerSideDataResult | Promise<ServerSideDataResult> | Observable<ServerSideDataResult>;\n\n/** Describes a data source modifier for requesting server data. */\nexport interface DataSourceModifier {\n  /** The list of columns. */\n  columns: Column[];\n\n  /** Text to search. */\n  searchText: string;\n\n  /** Pagination object. */\n  pagination: Pagination;\n\n  /** Selection info object. */\n  selection: {\n    /** Whether the selection is enabled. */\n    enabled: boolean;\n\n    /** The name of the primary key. */\n    primaryKey: string;\n  };\n}\n\n/** Describes a result from server with data and additional statistics. */\nexport type ServerSideDataResult = IResultList<object> & {\n  /** The real grand total number of items (the whole dataset). */\n  size: number;\n\n  /** The number of items after filtering applied to the whole dataset (subset). */\n  filteredSize: number;\n\n  /** The list of ids of all filtered items (subset). */\n  filteredDataIds?: string[];\n};\n\n/** Describes an object with data source statistics. */\nexport interface DataSourceStats {\n  /** The real grand total number of items (the whole dataset). */\n  size: number;\n\n  /** The number of items after filtering applied to the whole dataset (subset). */\n  filteredSize: number;\n\n  /** The number of the current page. */\n  currentPage: number;\n\n  /** The number of the next page. */\n  nextPage?: number;\n\n  /** The number of items currently shown in the list view, a.k.a. current page (what the user sees on screen). */\n  currentPageSize: number;\n\n  /** The number of items on the first page.  */\n  firstPageSize: number;\n}\n\n/** Describes data grid display options. */\nexport interface DisplayOptions {\n  /** Show or hide zebra-striping in the table */\n  striped: boolean;\n\n  /** Show or hide each cell border */\n  bordered: boolean;\n\n  /** Show or hide the grid header */\n  gridHeader: boolean;\n\n  /** Show or hide filter label in the grid header */\n  filter: boolean;\n\n  /** Enable to display a background color on hover */\n  hover: boolean;\n}\n\n/** Describes a data grid action control for individual item. */\nexport interface ActionControl {\n  /** The type of the action, predefined or custom. */\n  type: BuiltInActionType | string;\n\n  /** The label for the action button. */\n  text?: string;\n\n  /** The icon for the action button. */\n  icon?: string;\n\n  /** The icon classes for the action button icon. */\n  iconClasses?: string;\n\n  /** Action icon will be visible on hover only */\n  showOnHover?: boolean;\n\n  /**\n   * A callback function.\n   * @param item The item to perform the action on.\n   * @param reload The function to call, if you want to reload the grid.\n   */\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  callback: ((item: Row, reload: () => void) => void) | Function;\n\n  /**\n   * Determines if the action should be shown for given item (if not defined, the action will be shown always).\n   * @param item The item for which the action is supposed to be performed.\n   */\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  showIf?: ((item: Row & string[]) => boolean | Promise<boolean> | Observable<boolean>) | Function;\n\n  /**\n   * Defines the order in which action controls appear. Higher value means earlier position.\n   * Controls without priority are handled with priority = 0. If you want to place an action after\n   * an action without priority, negative numbers can be used.\n   */\n  priority?: number;\n\n  [key: string]: any;\n}\n\n/** Describes a data grid action control for multiple items. */\nexport interface BulkActionControl extends ActionControl {\n  /**\n   * A callback function.\n   * @param selectedItemIds The list of ids of the selected items.\n   * @param reload The function to call, if you want to reload the grid.\n   */\n  callback(selectedItemIds: string[], reload: () => void): void;\n\n  /**\n   * Determines if the action should be shown for given selected items (if not defined, the action will be shown always).\n   * @param selectedItemIds The list of ids of the selected items for which the action is supposed to be performed.\n   */\n  showIf?(selectedItemIds: string[]): boolean;\n}\n\n/** Describes a data grid action control for the header. */\nexport interface HeaderActionControl extends ActionControl {\n  /**\n   * Custom template can be provided. If no template is provided,\n   * the default headerActionControl template is used.\n   */\n  template?: TemplateRef<any>;\n\n  /**\n   * A callback function.\n   */\n  callback(): void;\n\n  /**\n   * Determines if the action should be shown.\n   */\n  showIf?(): boolean;\n}\n\n/** Predefined types of actions. */\nexport enum BuiltInActionType {\n  /** Edit built-in action. */\n  Edit = 'EDIT',\n\n  /** Delete built-in action. */\n  Delete = 'DELETE',\n\n  /** Export built-in action. */\n  Export = 'EXPORT'\n}\n\n/**\n * Allows to attach action controls to specified grids.\n */\nexport interface ActionControlHook {\n  /**\n   * The action control(s) that will be attached to the grid.\n   */\n  actionControls: ActionControl | ActionControl[];\n  /**\n   * A functions that determines to which grid(s) the action control(s) will be attached.\n   * Not providing a function results in the actions control(s) not being attached to any grid.\n   */\n  matchesGrid: (\n    route: ActivatedRoute,\n    context?: GridConfigContext\n  ) => boolean | Promise<boolean> | Observable<boolean>;\n}\n\n/**\n * Factory to implement in order to add action controls from an extension.\n */\nexport type ActionControlFactory = ExtensionFactory<ActionControlHook>;\n\n/**\n * Defines an interface for a filter chip object, which represents a filter applied to a column.\n */\nexport interface FilterChip {\n  /**\n   * The actual value that the chip will hold.\n   * It can be just a string or a more complex object like\n   * {id: 123, name: Test, value: test}.\n   */\n  value: any;\n\n  /**\n   * The value that the chip will display in the dropdown.\n   */\n  displayValue: string;\n\n  /**\n   * Optional label that will be displayed if filter chips are separated into multiple groups.\n   */\n  label?: string;\n\n  /** An optional property that holds the path to the chip value in the externalFilterQuery. */\n  path?: string[];\n\n  /**\n   * An optional property that holds the form value taken from the column configuration.\n   */\n  externalFilterQuery?: any;\n\n  /**\n   * An optional property that holds the formly filtering configuration for the column that this filter is applied to.\n   */\n  filteringConfig?: FormlyColumnFilteringConfig;\n\n  /** An optional property that holds a string to search for. */\n  filterPredicate?: string;\n\n  /**\n   * The name of the column that this filter is applied to.\n   */\n  columnName: string | SpecialColumnName;\n\n  /**\n   * An optional function to remove this filter from the column.\n   * If not provided will set default removal function that will reduce the external query by removing the object that is resolved by the path set in the FilterChip.\n   * @returns An object defined by the `FilterChip` type, including the `columnName` and  updated `externalFilterQuery` or 'filterPredicate' properties.\n   * The returned object may also include additional properties defined by the `FilterChip` type as partials.\n   */\n  remove?: () => PartialFilterChipRemovalType;\n}\n\n/**\n * Represents a partial filter chip with either 'columnName' and 'externalFilterQuery' or 'columnName' and 'filterPredicate'.\n * @typedef {Object} PartialFilterChipRemovalType\n * @property {string} columnName - The column name for the filter chip.\n * @property {any} [externalFilterQuery] - The external filter query for the chip used for complex filters.\n * @property {any} [filterPredicate] - The filtering predicate for the chip used for simple filters with client-side filtering.\n * @property {...FilterChip} [optionalProperties] - Optional properties from the FilterChip type.\n */\nexport type PartialFilterChipRemovalType = (\n  | Required<Pick<FilterChip, 'columnName' | 'externalFilterQuery'>>\n  | Required<Pick<FilterChip, 'columnName' | 'filterPredicate'>>\n) &\n  Partial<FilterChip>;\n\n/**\n * Represents a partial filter chip with required properties 'displayValue' and 'value'.\n * @typedef {Object} PartialFilterChipGenerationType\n * @property {string} displayValue - The display value for the filter chip.\n * @property {any} value - The value for the filter chip.\n * @property {...FilterChip} [optionalProperties] - Optional properties from the FilterChip type.\n */\nexport type PartialFilterChipGenerationType = Required<Pick<FilterChip, 'displayValue' | 'value'>> &\n  Partial<FilterChip>;\n\n/**\n * Defines an interface for a mapper function that maps a `FormlyFieldConfig` to a `FilterChip`.\n */\nexport interface FilterMapper {\n  /**\n   * The type or types of the `FormlyFieldConfig` that this mapper applies to.\n   */\n  fieldType: string[] | string;\n\n  /**\n   * Maps a `FormlyFieldConfig` and a `FilterChip` to an `Observable` of a `FilterChip`.\n   *\n   * This function extends the provided `FilterChip` object with additional properties based on the information in the `FormlyFieldConfig`.\n   * The mapping process depends on the specific implementation of the `FilterMapper`.\n   *\n   * @param field - The `FormlyFieldConfig` to map.\n   * @param filter - The `FilterChip` to map.\n   * @returns An `Observable` that emits the mapped `FilterChip`.\n   * If undefined is returned an error is thrown and the chip is not visualized.\n   *\n   * ```typescript\n   * // Example implementation of `map` function in a `FilterMapper`\n   * map(field: FormlyFieldConfig, filter: FilterChip): Observable<FilterChip> {\n   *   if (this.fieldType.includes(field.type.toString())) {\n   *     return toObservable(field.templateOptions.options).pipe(\n   *       map(options => {\n   *         const object = options.find(option => option.value === get(filter.externalFilterQuery, filter.path));\n   *         return {\n   *           ...filter,\n   *           value: object,\n   *           displayValue: object?.label\n   *         };\n   *       })\n   *     );\n   *   }\n   * }\n   * ```\n   */\n  map: (field: FormlyFieldConfig, filter: FilterChip) => Observable<FilterChip>;\n}\n"]}
@@ -43,7 +43,8 @@ export class DataGridService {
43
43
  const customColumns = config.columns
44
44
  .filter(col => col.custom)
45
45
  .map((col) => new CustomColumn(col));
46
- const allColumns = [...columns, ...customColumns];
46
+ // Avoid config properties to be applied to the original column object by creating shallow copy of columns
47
+ const allColumns = [...columns.map(col => ({ ...col })), ...customColumns];
47
48
  noConfigColumns = allColumns.filter(col => !config.columns.some(configCol => col.name === configCol.name));
48
49
  config.columns.forEach(({ visible, name, sortOrder, filter }) => {
49
50
  const columnToReorder = allColumns.find(col => col.name === name);
@@ -52,6 +53,8 @@ export class DataGridService {
52
53
  columnToReorder.sortOrder = sortOrder;
53
54
  columnToReorder.externalFilterQuery =
54
55
  columnToReorder.externalFilterQuery ?? filter?.externalFilterQuery;
56
+ columnToReorder.filterPredicate =
57
+ columnToReorder.filterPredicate ?? filter?.filterPredicate;
55
58
  reOrderedColumns.push(columnToReorder);
56
59
  }
57
60
  });
@@ -97,4 +100,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
97
100
  providedIn: 'root'
98
101
  }]
99
102
  }], ctorParameters: () => [{ type: i1.UserPreferencesService }] });
100
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-grid.service.js","sourceRoot":"","sources":["../../../../core/data-grid/data-grid.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { QueriesUtil } from '@c8y/client';\nimport { assign, forEach, get, identity, transform } from 'lodash-es';\nimport { from, isObservable, Observable, of } from 'rxjs';\nimport { map, share, take, withLatestFrom } from 'rxjs/operators';\nimport { UserPreferencesService } from '../common/user-preferences/user-preferences.service';\nimport { CustomColumn } from './column/custom.column';\nimport { Column, CustomColumnConfig, GridConfig } from './data-grid.model';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class DataGridService {\n  protected DEFAULT_PAGE_SIZE = 20;\n\n  protected queriesUtil: QueriesUtil;\n\n  constructor(protected userPreferencesService: UserPreferencesService) {\n    this.queriesUtil = new QueriesUtil();\n  }\n\n  clearConfig(key: string) {\n    localStorage.removeItem(key);\n  }\n\n  getConfig$(key: string): Observable<GridConfig> {\n    return this.userPreferencesService.get(key).pipe(\n      map(\n        config =>\n          config || {\n            columns: [],\n            pagination: { pageSize: this.DEFAULT_PAGE_SIZE, currentPage: 1 }\n          }\n      )\n    );\n  }\n\n  saveConfig$(config: GridConfig, key: string): Observable<GridConfig> {\n    return from(this.userPreferencesService.set(key, config));\n  }\n\n  getUserConfiguredColumns$(columns: Column[] | Observable<Column[]>, storageKey?: string) {\n    return this.getConfig$(storageKey).pipe(\n      withLatestFrom(\n        isObservable(columns) ? columns : (of(columns) as unknown as Observable<Column[]>)\n      ),\n      map(([config, cols]) => this.applyConfigToColumns(config, cols, storageKey)),\n      take(1),\n      share()\n    );\n  }\n\n  getQueryObj(columns: Column[], defaultFilter = {}): any {\n    return transform(columns, (query, column) => this.extendQueryByColumn(query, column), {\n      __filter: {},\n      __orderby: [],\n      ...defaultFilter\n    });\n  }\n\n  applyConfigToColumns(config: GridConfig, columns: Column[], storageKey?: string): Column[] {\n    if (config?.columns?.length > 0) {\n      const reOrderedColumns = [];\n      let noConfigColumns = [];\n      try {\n        const customColumns = config.columns\n          .filter(col => (col as CustomColumnConfig).custom)\n          .map((col: CustomColumnConfig) => new CustomColumn(col));\n\n        const allColumns = [...columns, ...customColumns];\n\n        noConfigColumns = allColumns.filter(\n          col => !config.columns.some(configCol => col.name === configCol.name)\n        );\n        config.columns.forEach(({ visible, name, sortOrder, filter }) => {\n          const columnToReorder = allColumns.find(col => col.name === name);\n          if (columnToReorder) {\n            columnToReorder.visible = visible;\n            columnToReorder.sortOrder = sortOrder;\n            columnToReorder.externalFilterQuery =\n              columnToReorder.externalFilterQuery ?? filter?.externalFilterQuery;\n            reOrderedColumns.push(columnToReorder);\n          }\n        });\n      } catch (ex) {\n        this.clearConfig(storageKey);\n      }\n      return [...reOrderedColumns, ...noConfigColumns];\n    }\n    return columns;\n  }\n\n  private extendQueryByColumn(query: any, column: Column): void {\n    if (column.filterable && column.externalFilterQuery) {\n      const getFilter = column.filteringConfig.getFilter || identity;\n      const queryObj = getFilter(column.externalFilterQuery);\n\n      if (queryObj.__or) {\n        query.__filter.__and = query.__filter.__and || [];\n        query.__filter.__and.push(queryObj);\n      } else if (queryObj.__and && get(query, '__filter.__and')) {\n        queryObj.__and.map(obj => query.__filter.__and.push(obj));\n      } else {\n        assign(query.__filter, queryObj);\n      }\n    }\n\n    if (column.sortable && column.sortOrder) {\n      const cs = {};\n      forEach(column.sortingConfig.pathSortingConfigs, pathSortingConfig => {\n        cs[pathSortingConfig.path] =\n          (column.sortOrder === 'asc' ? 1 : -1) * (pathSortingConfig.sortOrderModifier || 1);\n      });\n      query.__orderby.push(cs);\n    }\n    return query;\n  }\n}\n"]}
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-grid.service.js","sourceRoot":"","sources":["../../../../core/data-grid/data-grid.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { QueriesUtil } from '@c8y/client';\nimport { assign, forEach, get, identity, transform } from 'lodash-es';\nimport { from, isObservable, Observable, of } from 'rxjs';\nimport { map, share, take, withLatestFrom } from 'rxjs/operators';\nimport { UserPreferencesService } from '../common/user-preferences/user-preferences.service';\nimport { CustomColumn } from './column/custom.column';\nimport { Column, CustomColumnConfig, GridConfig } from './data-grid.model';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class DataGridService {\n  protected DEFAULT_PAGE_SIZE = 20;\n\n  protected queriesUtil: QueriesUtil;\n\n  constructor(protected userPreferencesService: UserPreferencesService) {\n    this.queriesUtil = new QueriesUtil();\n  }\n\n  clearConfig(key: string) {\n    localStorage.removeItem(key);\n  }\n\n  getConfig$(key: string): Observable<GridConfig> {\n    return this.userPreferencesService.get(key).pipe(\n      map(\n        config =>\n          config || {\n            columns: [],\n            pagination: { pageSize: this.DEFAULT_PAGE_SIZE, currentPage: 1 }\n          }\n      )\n    );\n  }\n\n  saveConfig$(config: GridConfig, key: string): Observable<GridConfig> {\n    return from(this.userPreferencesService.set(key, config));\n  }\n\n  getUserConfiguredColumns$(columns: Column[] | Observable<Column[]>, storageKey?: string) {\n    return this.getConfig$(storageKey).pipe(\n      withLatestFrom(\n        isObservable(columns) ? columns : (of(columns) as unknown as Observable<Column[]>)\n      ),\n      map(([config, cols]) => this.applyConfigToColumns(config, cols, storageKey)),\n      take(1),\n      share()\n    );\n  }\n\n  getQueryObj(columns: Column[], defaultFilter = {}): any {\n    return transform(columns, (query, column) => this.extendQueryByColumn(query, column), {\n      __filter: {},\n      __orderby: [],\n      ...defaultFilter\n    });\n  }\n\n  applyConfigToColumns(config: GridConfig, columns: Column[], storageKey?: string): Column[] {\n    if (config?.columns?.length > 0) {\n      const reOrderedColumns = [];\n      let noConfigColumns = [];\n      try {\n        const customColumns = config.columns\n          .filter(col => (col as CustomColumnConfig).custom)\n          .map((col: CustomColumnConfig) => new CustomColumn(col));\n\n        // Avoid config properties to be applied to the original column object by creating shallow copy of columns\n        const allColumns = [...columns.map(col => ({ ...col })), ...customColumns];\n\n        noConfigColumns = allColumns.filter(\n          col => !config.columns.some(configCol => col.name === configCol.name)\n        );\n        config.columns.forEach(({ visible, name, sortOrder, filter }) => {\n          const columnToReorder = allColumns.find(col => col.name === name);\n          if (columnToReorder) {\n            columnToReorder.visible = visible;\n            columnToReorder.sortOrder = sortOrder;\n            columnToReorder.externalFilterQuery =\n              columnToReorder.externalFilterQuery ?? filter?.externalFilterQuery;\n            columnToReorder.filterPredicate =\n              columnToReorder.filterPredicate ?? filter?.filterPredicate;\n            reOrderedColumns.push(columnToReorder);\n          }\n        });\n      } catch (ex) {\n        this.clearConfig(storageKey);\n      }\n      return [...reOrderedColumns, ...noConfigColumns];\n    }\n    return columns;\n  }\n\n  private extendQueryByColumn(query: any, column: Column): void {\n    if (column.filterable && column.externalFilterQuery) {\n      const getFilter = column.filteringConfig.getFilter || identity;\n      const queryObj = getFilter(column.externalFilterQuery);\n\n      if (queryObj.__or) {\n        query.__filter.__and = query.__filter.__and || [];\n        query.__filter.__and.push(queryObj);\n      } else if (queryObj.__and && get(query, '__filter.__and')) {\n        queryObj.__and.map(obj => query.__filter.__and.push(obj));\n      } else {\n        assign(query.__filter, queryObj);\n      }\n    }\n\n    if (column.sortable && column.sortOrder) {\n      const cs = {};\n      forEach(column.sortingConfig.pathSortingConfigs, pathSortingConfig => {\n        cs[pathSortingConfig.path] =\n          (column.sortOrder === 'asc' ? 1 : -1) * (pathSortingConfig.sortOrderModifier || 1);\n      });\n      query.__orderby.push(cs);\n    }\n    return query;\n  }\n}\n"]}
@@ -1,6 +1,7 @@
1
- import { Component, Output, Input, EventEmitter, ViewChild } from '@angular/core';
2
- import { DropAreaComponent } from '../drop-area/drop-area.component';
1
+ import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core';
3
2
  import { isEmpty } from 'lodash-es';
3
+ import { DropAreaComponent } from '../drop-area/drop-area.component';
4
+ import { ValidationPattern } from '../forms/validation-pattern';
4
5
  import * as i0 from "@angular/core";
5
6
  import * as i1 from "@angular/forms";
6
7
  import * as i2 from "../forms/form-group.component";
@@ -28,6 +29,7 @@ export class FilePickerComponent {
28
29
  this.maxAllowedFiles = Infinity;
29
30
  this.uploadChoice = 'uploadBinary';
30
31
  this.config = { maxlength: 2048 };
32
+ this.ValidationPattern = ValidationPattern;
31
33
  }
32
34
  /**
33
35
  * @ignore
@@ -81,11 +83,11 @@ export class FilePickerComponent {
81
83
  return !isEmpty(this.fileUrlPopover);
82
84
  }
83
85
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: FilePickerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
84
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: FilePickerComponent, selector: "c8y-file-picker", inputs: { maxAllowedFiles: "maxAllowedFiles", uploadChoice: "uploadChoice", fileUrl: "fileUrl", fileBinary: "fileBinary", config: "config", filePickerIndex: "filePickerIndex", fileUrlPopover: "fileUrlPopover" }, outputs: { onFilesPicked: "onFilesPicked" }, viewQueries: [{ propertyName: "dropArea", first: true, predicate: DropAreaComponent, descendants: true, static: true }], ngImport: i0, template: "<div class=\"form-group\">\n <label title=\"{{ 'Upload a binary' | translate }}\" class=\"c8y-radio radio-inline\">\n <input\n #radio\n type=\"radio\"\n value=\"uploadBinary\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearInputFromUrl()\"\n />\n <span></span>\n <span>{{ 'Upload a binary' | translate }}</span>\n </label>\n <label\n title=\"{{ 'Provide a file path' | translate }}\"\n class=\"c8y-radio radio-inline m-l-8\"\n data-cy=\"file-picker--file-path-input\"\n >\n <input\n #radio\n type=\"radio\"\n value=\"uploadUrl\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearSelectedFiles()\"\n />\n <span></span>\n <span>\n {{ 'Provide a file path' | translate }}\n </span>\n <button\n class=\"btn-help\"\n type=\"button\"\n [attr.aria-label]=\"'Help' | translate\"\n popover=\"{{ fileUrlPopover | translate }}\"\n placement=\"top\"\n triggers=\"focus\"\n container=\"body\"\n placement=\"top\"\n *ngIf=\"isPopoverUsed()\"\n ></button>\n </label>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadBinary'\">\n <c8y-form-group class=\"m-0\">\n <c8y-drop-area\n class=\"drop-area-sm\"\n (dropped)=\"onFileDropped($event)\"\n [title]=\"'Drop file or click to browse' | translate\"\n [attr.aria-label]=\"'Drop file or click to browse' | translate\"\n [maxAllowedFiles]=\"maxAllowedFiles\"\n [files]=\"droppedFiles\"\n ></c8y-drop-area>\n </c8y-form-group>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadUrl'\">\n <c8y-form-group class=\"m-0\">\n <div class=\"m-b-4 p-b-8\">\n <div class=\"input-group\">\n <span class=\"input-group-addon\">\n <i c8yIcon=\"globe\"></i>\n </span>\n <input\n type=\"text\"\n class=\"form-control\"\n name=\"fileUrl\"\n data-cy=\"file-picker--fileUrl\"\n [(ngModel)]=\"fileUrl\"\n (ngModelChange)=\"onFileUrlChange($event)\"\n placeholder=\"{{ 'e.g.' | translate }} http://example.com/binary.zip\"\n maxlength=\"{{ config.maxlength }}\"\n required\n />\n </div>\n </div>\n </c8y-form-group>\n</div>\n", dependencies: [{ kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.RadioControlValueAccessor, selector: "input[type=radio][formControlName],input[type=radio][formControl],input[type=radio][ngModel]", inputs: ["name", "formControlName", "value"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i1.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i2.FormGroupComponent, selector: "c8y-form-group", inputs: ["hasError", "hasWarning", "hasSuccess", "novalidation", "status"] }, { kind: "directive", type: i3.RequiredInputPlaceholderDirective, selector: "input[required], input[formControlName]" }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.DropAreaComponent, selector: "c8y-drop-area", inputs: ["formControl", "title", "message", "icon", "loadingMessage", "forceHideList", "alwaysShow", "clickToOpen", "loading", "progress", "maxAllowedFiles", "files", "maxFileSizeInMegaBytes", "accept"], outputs: ["dropped"] }, { kind: "directive", type: i6.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i7.PopoverDirective, selector: "[popover]", inputs: ["adaptivePosition", "boundariesElement", "popover", "popoverContext", "popoverTitle", "placement", "outsideClick", "triggers", "container", "containerClass", "isOpen", "delay"], outputs: ["onShown", "onHidden"], exportAs: ["bs-popover"] }, { kind: "pipe", type: i8.C8yTranslatePipe, name: "translate" }] }); }
86
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: FilePickerComponent, selector: "c8y-file-picker", inputs: { maxAllowedFiles: "maxAllowedFiles", uploadChoice: "uploadChoice", fileUrl: "fileUrl", fileBinary: "fileBinary", config: "config", filePickerIndex: "filePickerIndex", fileUrlPopover: "fileUrlPopover" }, outputs: { onFilesPicked: "onFilesPicked" }, viewQueries: [{ propertyName: "dropArea", first: true, predicate: DropAreaComponent, descendants: true, static: true }], ngImport: i0, template: "<div class=\"form-group\">\n <label title=\"{{ 'Upload a binary' | translate }}\" class=\"c8y-radio radio-inline\">\n <input\n #radio\n type=\"radio\"\n value=\"uploadBinary\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearInputFromUrl()\"\n />\n <span></span>\n <span>{{ 'Upload a binary' | translate }}</span>\n </label>\n <label\n title=\"{{ 'Provide a file path' | translate }}\"\n class=\"c8y-radio radio-inline m-l-8\"\n data-cy=\"file-picker--file-path-input\"\n >\n <input\n #radio\n type=\"radio\"\n value=\"uploadUrl\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearSelectedFiles()\"\n />\n <span></span>\n <span>\n {{ 'Provide a file path' | translate }}\n </span>\n <button\n class=\"btn-help\"\n type=\"button\"\n [attr.aria-label]=\"'Help' | translate\"\n popover=\"{{ fileUrlPopover | translate }}\"\n placement=\"top\"\n triggers=\"focus\"\n container=\"body\"\n placement=\"top\"\n *ngIf=\"isPopoverUsed()\"\n ></button>\n </label>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadBinary'\">\n <c8y-form-group class=\"m-0\">\n <c8y-drop-area\n class=\"drop-area-sm\"\n (dropped)=\"onFileDropped($event)\"\n [title]=\"'Drop file or click to browse' | translate\"\n [attr.aria-label]=\"'Drop file or click to browse' | translate\"\n [maxAllowedFiles]=\"maxAllowedFiles\"\n [files]=\"droppedFiles\"\n ></c8y-drop-area>\n </c8y-form-group>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadUrl'\">\n <c8y-form-group class=\"m-0\">\n <div class=\"m-b-4 p-b-8\">\n <div class=\"input-group\">\n <span class=\"input-group-addon\">\n <i c8yIcon=\"globe\"></i>\n </span>\n <input\n type=\"text\"\n class=\"form-control\"\n name=\"fileUrl\"\n data-cy=\"file-picker--fileUrl\"\n [(ngModel)]=\"fileUrl\"\n (ngModelChange)=\"onFileUrlChange($event)\"\n placeholder=\"{{ 'e.g.' | translate }} http://example.com/binary.zip\"\n maxlength=\"{{ config.maxlength }}\"\n required\n [pattern]=\"ValidationPattern.rules.noWhiteSpaceOnly.pattern\"\n />\n </div>\n </div>\n </c8y-form-group>\n</div>\n", dependencies: [{ kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.RadioControlValueAccessor, selector: "input[type=radio][formControlName],input[type=radio][formControl],input[type=radio][ngModel]", inputs: ["name", "formControlName", "value"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i1.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i1.PatternValidator, selector: "[pattern][formControlName],[pattern][formControl],[pattern][ngModel]", inputs: ["pattern"] }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i2.FormGroupComponent, selector: "c8y-form-group", inputs: ["hasError", "hasWarning", "hasSuccess", "novalidation", "status"] }, { kind: "directive", type: i3.RequiredInputPlaceholderDirective, selector: "input[required], input[formControlName]" }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.DropAreaComponent, selector: "c8y-drop-area", inputs: ["formControl", "title", "message", "icon", "loadingMessage", "forceHideList", "alwaysShow", "clickToOpen", "loading", "progress", "maxAllowedFiles", "files", "maxFileSizeInMegaBytes", "accept"], outputs: ["dropped"] }, { kind: "directive", type: i6.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i7.PopoverDirective, selector: "[popover]", inputs: ["adaptivePosition", "boundariesElement", "popover", "popoverContext", "popoverTitle", "placement", "outsideClick", "triggers", "container", "containerClass", "isOpen", "delay"], outputs: ["onShown", "onHidden"], exportAs: ["bs-popover"] }, { kind: "pipe", type: i8.C8yTranslatePipe, name: "translate" }] }); }
85
87
  }
86
88
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: FilePickerComponent, decorators: [{
87
89
  type: Component,
88
- args: [{ selector: 'c8y-file-picker', template: "<div class=\"form-group\">\n <label title=\"{{ 'Upload a binary' | translate }}\" class=\"c8y-radio radio-inline\">\n <input\n #radio\n type=\"radio\"\n value=\"uploadBinary\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearInputFromUrl()\"\n />\n <span></span>\n <span>{{ 'Upload a binary' | translate }}</span>\n </label>\n <label\n title=\"{{ 'Provide a file path' | translate }}\"\n class=\"c8y-radio radio-inline m-l-8\"\n data-cy=\"file-picker--file-path-input\"\n >\n <input\n #radio\n type=\"radio\"\n value=\"uploadUrl\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearSelectedFiles()\"\n />\n <span></span>\n <span>\n {{ 'Provide a file path' | translate }}\n </span>\n <button\n class=\"btn-help\"\n type=\"button\"\n [attr.aria-label]=\"'Help' | translate\"\n popover=\"{{ fileUrlPopover | translate }}\"\n placement=\"top\"\n triggers=\"focus\"\n container=\"body\"\n placement=\"top\"\n *ngIf=\"isPopoverUsed()\"\n ></button>\n </label>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadBinary'\">\n <c8y-form-group class=\"m-0\">\n <c8y-drop-area\n class=\"drop-area-sm\"\n (dropped)=\"onFileDropped($event)\"\n [title]=\"'Drop file or click to browse' | translate\"\n [attr.aria-label]=\"'Drop file or click to browse' | translate\"\n [maxAllowedFiles]=\"maxAllowedFiles\"\n [files]=\"droppedFiles\"\n ></c8y-drop-area>\n </c8y-form-group>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadUrl'\">\n <c8y-form-group class=\"m-0\">\n <div class=\"m-b-4 p-b-8\">\n <div class=\"input-group\">\n <span class=\"input-group-addon\">\n <i c8yIcon=\"globe\"></i>\n </span>\n <input\n type=\"text\"\n class=\"form-control\"\n name=\"fileUrl\"\n data-cy=\"file-picker--fileUrl\"\n [(ngModel)]=\"fileUrl\"\n (ngModelChange)=\"onFileUrlChange($event)\"\n placeholder=\"{{ 'e.g.' | translate }} http://example.com/binary.zip\"\n maxlength=\"{{ config.maxlength }}\"\n required\n />\n </div>\n </div>\n </c8y-form-group>\n</div>\n" }]
90
+ args: [{ selector: 'c8y-file-picker', template: "<div class=\"form-group\">\n <label title=\"{{ 'Upload a binary' | translate }}\" class=\"c8y-radio radio-inline\">\n <input\n #radio\n type=\"radio\"\n value=\"uploadBinary\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearInputFromUrl()\"\n />\n <span></span>\n <span>{{ 'Upload a binary' | translate }}</span>\n </label>\n <label\n title=\"{{ 'Provide a file path' | translate }}\"\n class=\"c8y-radio radio-inline m-l-8\"\n data-cy=\"file-picker--file-path-input\"\n >\n <input\n #radio\n type=\"radio\"\n value=\"uploadUrl\"\n name=\"uploadChoice-{{filePickerIndex}}\"\n [(ngModel)]=\"uploadChoice\"\n (click)=\"clearSelectedFiles()\"\n />\n <span></span>\n <span>\n {{ 'Provide a file path' | translate }}\n </span>\n <button\n class=\"btn-help\"\n type=\"button\"\n [attr.aria-label]=\"'Help' | translate\"\n popover=\"{{ fileUrlPopover | translate }}\"\n placement=\"top\"\n triggers=\"focus\"\n container=\"body\"\n placement=\"top\"\n *ngIf=\"isPopoverUsed()\"\n ></button>\n </label>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadBinary'\">\n <c8y-form-group class=\"m-0\">\n <c8y-drop-area\n class=\"drop-area-sm\"\n (dropped)=\"onFileDropped($event)\"\n [title]=\"'Drop file or click to browse' | translate\"\n [attr.aria-label]=\"'Drop file or click to browse' | translate\"\n [maxAllowedFiles]=\"maxAllowedFiles\"\n [files]=\"droppedFiles\"\n ></c8y-drop-area>\n </c8y-form-group>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadUrl'\">\n <c8y-form-group class=\"m-0\">\n <div class=\"m-b-4 p-b-8\">\n <div class=\"input-group\">\n <span class=\"input-group-addon\">\n <i c8yIcon=\"globe\"></i>\n </span>\n <input\n type=\"text\"\n class=\"form-control\"\n name=\"fileUrl\"\n data-cy=\"file-picker--fileUrl\"\n [(ngModel)]=\"fileUrl\"\n (ngModelChange)=\"onFileUrlChange($event)\"\n placeholder=\"{{ 'e.g.' | translate }} http://example.com/binary.zip\"\n maxlength=\"{{ config.maxlength }}\"\n required\n [pattern]=\"ValidationPattern.rules.noWhiteSpaceOnly.pattern\"\n />\n </div>\n </div>\n </c8y-form-group>\n</div>\n" }]
89
91
  }], propDecorators: { dropArea: [{
90
92
  type: ViewChild,
91
93
  args: [DropAreaComponent, { static: true }]
@@ -106,4 +108,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
106
108
  }], fileUrlPopover: [{
107
109
  type: Input
108
110
  }] } });
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-picker.component.js","sourceRoot":"","sources":["../../../../core/file-picker/file-picker.component.ts","../../../../core/file-picker/file-picker.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,YAAY,EAAE,SAAS,EAAU,MAAM,eAAe,CAAC;AAC1F,OAAO,EAAe,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AAElF,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;;;;;;;;;;AAGpC;;;;;;;;;;;GAWG;AAMH,MAAM,OAAO,mBAAmB;IAJhC;QAMY,kBAAa,GAA8B,IAAI,YAAY,EAAE,CAAC;QAC/D,oBAAe,GAAG,QAAQ,CAAC;QAC3B,iBAAY,GAAiC,cAAc,CAAC;QAG5D,WAAM,GAAqB,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;KAqEzD;IAxDC;;OAEG;IACH,QAAQ;QACN,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC,YAAY,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACxC,CAAC;IACH,CAAC;IAED;;OAEG;IACH,aAAa,CAAC,YAA2B;QACvC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC;YACtB,YAAY;SACb,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,iBAAiB;QACf,OAAO,IAAI,CAAC,OAAO,CAAC;QACpB,IAAI,CAAC,UAAU,GAAG;YAChB,GAAG,EAAE,SAAS;SACf,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,kBAAkB;QAChB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;QACzB,IAAI,CAAC,UAAU,GAAG;YAChB,YAAY,EAAE,SAAS;SACxB,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,MAAM;QACpB,IAAI,CAAC,UAAU,GAAG;YAChB,GAAG,EAAE,MAAM;SACZ,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,aAAa;QACX,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IACvC,CAAC;8GA3EU,mBAAmB;kGAAnB,mBAAmB,kWACnB,iBAAiB,8DCxB9B,ozEA+EA;;2FDxDa,mBAAmB;kBAJ/B,SAAS;+BACE,iBAAiB;8BAIqB,QAAQ;sBAAvD,SAAS;uBAAC,iBAAiB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBACpC,aAAa;sBAAtB,MAAM;gBACE,eAAe;sBAAvB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACG,UAAU;sBAAlB,KAAK;gBACG,MAAM;sBAAd,KAAK;gBAIG,eAAe;sBAAvB,KAAK;gBAIG,cAAc;sBAAtB,KAAK","sourcesContent":["import { Component, Output, Input, EventEmitter, ViewChild, OnInit } from '@angular/core';\nimport { DroppedFile, DropAreaComponent } from '../drop-area/drop-area.component';\nimport { PickedFiles } from './file-picker.model';\nimport { isEmpty } from 'lodash-es';\nimport { FilePickerConfig } from './file-picker-config.model';\n\n/**\n * User can upload a binary directly or use an URL.\n *\n * ## Usage\n *\n * ```html\n * <div>\n * <c8y-file-picker [maxAllowedFiles]=\"1\" (onFilesPicked)=\"onFile($event)\">\n * </c8y-file-picker>\n * </div>\n * ```\n */\n\n@Component({\n  selector: 'c8y-file-picker',\n  templateUrl: './file-picker.component.html'\n})\nexport class FilePickerComponent implements OnInit {\n  @ViewChild(DropAreaComponent, { static: true }) dropArea: DropAreaComponent;\n  @Output() onFilesPicked: EventEmitter<PickedFiles> = new EventEmitter();\n  @Input() maxAllowedFiles = Infinity;\n  @Input() uploadChoice: 'uploadBinary' | 'uploadUrl' = 'uploadBinary';\n  @Input() fileUrl: string;\n  @Input() fileBinary: DroppedFile;\n  @Input() config: FilePickerConfig = { maxlength: 2048 };\n  /**\n   * Used only when displaying multiple file pickers in the same view.\n   */\n  @Input() filePickerIndex: number;\n  /**\n   * Additional string to be displayed in a popover.\n   */\n  @Input() fileUrlPopover: string;\n\n  droppedFiles: DroppedFile[];\n  private fileToSave: PickedFiles;\n\n  /**\n   * @ignore\n   */\n  ngOnInit(): void {\n    if (this.fileBinary) {\n      this.droppedFiles = [this.fileBinary];\n    }\n  }\n\n  /**\n   * Triggered by dropped file in component and pass it into drop-area component.\n   */\n  onFileDropped(droppedFiles: DroppedFile[]): void {\n    this.onFilesPicked.emit({\n      droppedFiles\n    });\n  }\n\n  /**\n   * Triggered when user changes upload choice, from upload url to upload binary.\n   */\n  clearInputFromUrl(): void {\n    delete this.fileUrl;\n    this.fileToSave = {\n      url: undefined\n    };\n    this.onFilesPicked.emit(this.fileToSave);\n  }\n\n  /**\n   * Triggered when user changes upload choice, to avoid cumulation of droppedFiles.\n   */\n  clearSelectedFiles(): void {\n    this.dropArea.onDelete();\n    this.fileToSave = {\n      droppedFiles: undefined\n    };\n    this.onFilesPicked.emit(this.fileToSave);\n  }\n\n  /**\n   * Triggered when user puts binary's url to upload.\n   */\n  onFileUrlChange(urlStr): void {\n    this.fileToSave = {\n      url: urlStr\n    };\n    this.onFilesPicked.emit(this.fileToSave);\n  }\n\n  /**\n   * Checks if there is popover to display.\n   */\n  isPopoverUsed() {\n    return !isEmpty(this.fileUrlPopover);\n  }\n}\n","<div class=\"form-group\">\n  <label title=\"{{ 'Upload a binary' | translate }}\" class=\"c8y-radio radio-inline\">\n    <input\n      #radio\n      type=\"radio\"\n      value=\"uploadBinary\"\n      name=\"uploadChoice-{{filePickerIndex}}\"\n      [(ngModel)]=\"uploadChoice\"\n      (click)=\"clearInputFromUrl()\"\n    />\n    <span></span>\n    <span>{{ 'Upload a binary' | translate }}</span>\n  </label>\n  <label\n    title=\"{{ 'Provide a file path' | translate }}\"\n    class=\"c8y-radio radio-inline m-l-8\"\n    data-cy=\"file-picker--file-path-input\"\n  >\n    <input\n      #radio\n      type=\"radio\"\n      value=\"uploadUrl\"\n      name=\"uploadChoice-{{filePickerIndex}}\"\n      [(ngModel)]=\"uploadChoice\"\n      (click)=\"clearSelectedFiles()\"\n    />\n    <span></span>\n    <span>\n      {{ 'Provide a file path' | translate }}\n    </span>\n    <button\n      class=\"btn-help\"\n      type=\"button\"\n      [attr.aria-label]=\"'Help' | translate\"\n      popover=\"{{ fileUrlPopover | translate }}\"\n      placement=\"top\"\n      triggers=\"focus\"\n      container=\"body\"\n      placement=\"top\"\n      *ngIf=\"isPopoverUsed()\"\n    ></button>\n  </label>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadBinary'\">\n  <c8y-form-group class=\"m-0\">\n    <c8y-drop-area\n      class=\"drop-area-sm\"\n      (dropped)=\"onFileDropped($event)\"\n      [title]=\"'Drop file or click to browse' | translate\"\n      [attr.aria-label]=\"'Drop file or click to browse' | translate\"\n      [maxAllowedFiles]=\"maxAllowedFiles\"\n      [files]=\"droppedFiles\"\n    ></c8y-drop-area>\n  </c8y-form-group>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadUrl'\">\n  <c8y-form-group class=\"m-0\">\n    <div class=\"m-b-4 p-b-8\">\n      <div class=\"input-group\">\n        <span class=\"input-group-addon\">\n          <i c8yIcon=\"globe\"></i>\n        </span>\n        <input\n          type=\"text\"\n          class=\"form-control\"\n          name=\"fileUrl\"\n          data-cy=\"file-picker--fileUrl\"\n          [(ngModel)]=\"fileUrl\"\n          (ngModelChange)=\"onFileUrlChange($event)\"\n          placeholder=\"{{ 'e.g.' | translate }} http://example.com/binary.zip\"\n          maxlength=\"{{ config.maxlength }}\"\n          required\n        />\n      </div>\n    </div>\n  </c8y-form-group>\n</div>\n"]}
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-picker.component.js","sourceRoot":"","sources":["../../../../core/file-picker/file-picker.component.ts","../../../../core/file-picker/file-picker.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1F,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,iBAAiB,EAAe,MAAM,kCAAkC,CAAC;AAClF,OAAO,EAAE,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;;;;;;;;;;AAIhE;;;;;;;;;;;GAWG;AAMH,MAAM,OAAO,mBAAmB;IAJhC;QAMY,kBAAa,GAA8B,IAAI,YAAY,EAAE,CAAC;QAC/D,oBAAe,GAAG,QAAQ,CAAC;QAC3B,iBAAY,GAAiC,cAAc,CAAC;QAG5D,WAAM,GAAqB,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;QAUxD,sBAAiB,GAAG,iBAAiB,CAAC;KA4DvC;IAxDC;;OAEG;IACH,QAAQ;QACN,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC,YAAY,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACxC,CAAC;IACH,CAAC;IAED;;OAEG;IACH,aAAa,CAAC,YAA2B;QACvC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC;YACtB,YAAY;SACb,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,iBAAiB;QACf,OAAO,IAAI,CAAC,OAAO,CAAC;QACpB,IAAI,CAAC,UAAU,GAAG;YAChB,GAAG,EAAE,SAAS;SACf,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,kBAAkB;QAChB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;QACzB,IAAI,CAAC,UAAU,GAAG;YAChB,YAAY,EAAE,SAAS;SACxB,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,MAAM;QACpB,IAAI,CAAC,UAAU,GAAG;YAChB,GAAG,EAAE,MAAM;SACZ,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,aAAa;QACX,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IACvC,CAAC;8GA5EU,mBAAmB;kGAAnB,mBAAmB,kWACnB,iBAAiB,8DCzB9B,83EAgFA;;2FDxDa,mBAAmB;kBAJ/B,SAAS;+BACE,iBAAiB;8BAIqB,QAAQ;sBAAvD,SAAS;uBAAC,iBAAiB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBACpC,aAAa;sBAAtB,MAAM;gBACE,eAAe;sBAAvB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACG,UAAU;sBAAlB,KAAK;gBACG,MAAM;sBAAd,KAAK;gBAIG,eAAe;sBAAvB,KAAK;gBAIG,cAAc;sBAAtB,KAAK","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output, ViewChild } from '@angular/core';\nimport { isEmpty } from 'lodash-es';\nimport { DropAreaComponent, DroppedFile } from '../drop-area/drop-area.component';\nimport { ValidationPattern } from '../forms/validation-pattern';\nimport { FilePickerConfig } from './file-picker-config.model';\nimport { PickedFiles } from './file-picker.model';\n\n/**\n * User can upload a binary directly or use an URL.\n *\n * ## Usage\n *\n * ```html\n * <div>\n * <c8y-file-picker [maxAllowedFiles]=\"1\" (onFilesPicked)=\"onFile($event)\">\n * </c8y-file-picker>\n * </div>\n * ```\n */\n\n@Component({\n  selector: 'c8y-file-picker',\n  templateUrl: './file-picker.component.html'\n})\nexport class FilePickerComponent implements OnInit {\n  @ViewChild(DropAreaComponent, { static: true }) dropArea: DropAreaComponent;\n  @Output() onFilesPicked: EventEmitter<PickedFiles> = new EventEmitter();\n  @Input() maxAllowedFiles = Infinity;\n  @Input() uploadChoice: 'uploadBinary' | 'uploadUrl' = 'uploadBinary';\n  @Input() fileUrl: string;\n  @Input() fileBinary: DroppedFile;\n  @Input() config: FilePickerConfig = { maxlength: 2048 };\n  /**\n   * Used only when displaying multiple file pickers in the same view.\n   */\n  @Input() filePickerIndex: number;\n  /**\n   * Additional string to be displayed in a popover.\n   */\n  @Input() fileUrlPopover: string;\n\n  ValidationPattern = ValidationPattern;\n  droppedFiles: DroppedFile[];\n  private fileToSave: PickedFiles;\n\n  /**\n   * @ignore\n   */\n  ngOnInit(): void {\n    if (this.fileBinary) {\n      this.droppedFiles = [this.fileBinary];\n    }\n  }\n\n  /**\n   * Triggered by dropped file in component and pass it into drop-area component.\n   */\n  onFileDropped(droppedFiles: DroppedFile[]): void {\n    this.onFilesPicked.emit({\n      droppedFiles\n    });\n  }\n\n  /**\n   * Triggered when user changes upload choice, from upload url to upload binary.\n   */\n  clearInputFromUrl(): void {\n    delete this.fileUrl;\n    this.fileToSave = {\n      url: undefined\n    };\n    this.onFilesPicked.emit(this.fileToSave);\n  }\n\n  /**\n   * Triggered when user changes upload choice, to avoid cumulation of droppedFiles.\n   */\n  clearSelectedFiles(): void {\n    this.dropArea.onDelete();\n    this.fileToSave = {\n      droppedFiles: undefined\n    };\n    this.onFilesPicked.emit(this.fileToSave);\n  }\n\n  /**\n   * Triggered when user puts binary's url to upload.\n   */\n  onFileUrlChange(urlStr): void {\n    this.fileToSave = {\n      url: urlStr\n    };\n    this.onFilesPicked.emit(this.fileToSave);\n  }\n\n  /**\n   * Checks if there is popover to display.\n   */\n  isPopoverUsed() {\n    return !isEmpty(this.fileUrlPopover);\n  }\n}\n","<div class=\"form-group\">\n  <label title=\"{{ 'Upload a binary' | translate }}\" class=\"c8y-radio radio-inline\">\n    <input\n      #radio\n      type=\"radio\"\n      value=\"uploadBinary\"\n      name=\"uploadChoice-{{filePickerIndex}}\"\n      [(ngModel)]=\"uploadChoice\"\n      (click)=\"clearInputFromUrl()\"\n    />\n    <span></span>\n    <span>{{ 'Upload a binary' | translate }}</span>\n  </label>\n  <label\n    title=\"{{ 'Provide a file path' | translate }}\"\n    class=\"c8y-radio radio-inline m-l-8\"\n    data-cy=\"file-picker--file-path-input\"\n  >\n    <input\n      #radio\n      type=\"radio\"\n      value=\"uploadUrl\"\n      name=\"uploadChoice-{{filePickerIndex}}\"\n      [(ngModel)]=\"uploadChoice\"\n      (click)=\"clearSelectedFiles()\"\n    />\n    <span></span>\n    <span>\n      {{ 'Provide a file path' | translate }}\n    </span>\n    <button\n      class=\"btn-help\"\n      type=\"button\"\n      [attr.aria-label]=\"'Help' | translate\"\n      popover=\"{{ fileUrlPopover | translate }}\"\n      placement=\"top\"\n      triggers=\"focus\"\n      container=\"body\"\n      placement=\"top\"\n      *ngIf=\"isPopoverUsed()\"\n    ></button>\n  </label>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadBinary'\">\n  <c8y-form-group class=\"m-0\">\n    <c8y-drop-area\n      class=\"drop-area-sm\"\n      (dropped)=\"onFileDropped($event)\"\n      [title]=\"'Drop file or click to browse' | translate\"\n      [attr.aria-label]=\"'Drop file or click to browse' | translate\"\n      [maxAllowedFiles]=\"maxAllowedFiles\"\n      [files]=\"droppedFiles\"\n    ></c8y-drop-area>\n  </c8y-form-group>\n</div>\n\n<div [hidden]=\"uploadChoice !== 'uploadUrl'\">\n  <c8y-form-group class=\"m-0\">\n    <div class=\"m-b-4 p-b-8\">\n      <div class=\"input-group\">\n        <span class=\"input-group-addon\">\n          <i c8yIcon=\"globe\"></i>\n        </span>\n        <input\n          type=\"text\"\n          class=\"form-control\"\n          name=\"fileUrl\"\n          data-cy=\"file-picker--fileUrl\"\n          [(ngModel)]=\"fileUrl\"\n          (ngModelChange)=\"onFileUrlChange($event)\"\n          placeholder=\"{{ 'e.g.' | translate }} http://example.com/binary.zip\"\n          maxlength=\"{{ config.maxlength }}\"\n          required\n          [pattern]=\"ValidationPattern.rules.noWhiteSpaceOnly.pattern\"\n        />\n      </div>\n    </div>\n  </c8y-form-group>\n</div>\n"]}
@@ -43,6 +43,8 @@ export class ValidationPattern {
43
43
  user: gettext('Username must not contain spaces nor slashes ("/", "\\") nor ("+"), (":"), ("$") signs.'),
44
44
  httpUrl: gettext('Must be a valid HTTP(S) URL.'),
45
45
  colonedHexNumber: gettext('Must be a valid hexadecimal number. Must contain only the following characters: 0-9, a-f, A-F, :.'),
46
+ noWhiteSpaceOnly: gettext('This field must not contain only whitespaces'),
47
+ noWhiteSpaceAtBeginning: gettext('Name must not contain white spaces at the beginning or ending.'),
46
48
  noDots: gettext('Dots not allowed.'),
47
49
  integer: gettext('Invalid type, expected integer.'),
48
50
  simpleJsonPath: gettext('Must be a valid JSON path'),
@@ -87,6 +89,12 @@ export class ValidationPattern {
87
89
  loginAlias: {
88
90
  pattern: /^[^\\/$:+]*$/
89
91
  },
92
+ noWhiteSpaceOnly: {
93
+ pattern: /^(?!\s*$).+/
94
+ },
95
+ noWhiteSpaceAtBeginning: {
96
+ pattern: /^[^\s].+[^\s]$/
97
+ },
90
98
  noDots: {
91
99
  pattern: /^[^.]*$/
92
100
  },
@@ -105,4 +113,4 @@ export class ValidationPattern {
105
113
  return pattern;
106
114
  }
107
115
  }
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation-pattern.js","sourceRoot":"","sources":["../../../../core/forms/validation-pattern.ts"],"names":[],"mappings":"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","sourcesContent":["// tslint:disable: max-line-length\nimport { gettext } from '../i18n/gettext';\n\nexport interface ValidationRules {\n  pattern?: RegExp;\n  message?: string;\n  maxLength?: number;\n  minLength?: number;\n}\n\nexport enum DatapointLibraryValidationErrors {\n  IS_REQUIRED = 'required',\n  PATTERN = 'pattern',\n  SHOULD_CONTAIN_NUMBER = 'number',\n  SHOULD_BE_DEFINED = 'should-be-defined',\n  GREATER_THAN_SCALE_MAX = 'greater-than-scale-max',\n  LESS_THAN_SCALE_MIN = 'less-than-scale-min',\n  GREATER_THAN_RANGE_MAX = 'greater-than-range-max',\n  LESS_THAN_RANGE_MIN = 'less-than-range-min',\n  MIN_ACTIVE_COUNT = 'minActiveCount',\n  MAX_ACTIVE_COUNT = 'maxActiveCount'\n}\n\n// @dynamic\nexport class ValidationPattern {\n  static readonly messages = {\n    deviceId: gettext('Device ID must not contain spaces or slashes (\"/\").'),\n    domain: gettext(\n      'Only lowercase letters, digits and hyphens allowed in the first part of the URI. ' +\n        'Must start with a letter; hyphens only allowed in the middle. Must be a valid URI.'\n    ),\n    email: gettext('Invalid email address.'),\n    emails: gettext('Invalid email addresses.'),\n    internationalPhoneNumber: gettext(\n      'Must be a valid phone number (only digits, spaces, slashes (\"/\"), dashes (\"-\"), and plus (\"+\") allowed, for example: +49 9 876 543 210).'\n    ),\n    loginAlias: gettext(\n      'Login alias must not contain slashes (\"/\", \"\\\\\") nor (\"+\"), (\":\"), (\"$\") signs.'\n    ),\n    invalidColor: gettext('Invalid color \"{{ currentValue }}\".'),\n    max: gettext('Max value: {{ max }}'),\n    min: gettext('Min value: {{ min }}'),\n    minLength: gettext('Should have at least {{ minLength }} characters.'),\n    minlength: gettext('Should have at least {{ requiredLength }} characters.'),\n    maxLength: gettext('Should have at most {{ maxLength }} characters.'),\n    maxlength: gettext('Should have at most {{ requiredLength }} characters.'),\n    pattern: gettext('Does not match pattern {{ requiredPattern }}.'),\n    opcuaBrowsePath: gettext('Invalid OPC UA URI.'),\n    password: gettext(\n      'Password must have at least 8 characters and no more than 32 and can only contain letters, numbers and following symbols: `~!@#$%^&*()_|+-=?;:\\'\",.<>{}[]\\\\/'\n    ),\n    passwordConfirm: gettext('Passwords do not match.'),\n    passwordStrength: gettext('Password is not strong enough, use a stronger password.'),\n    passwordStrengthChecklist: gettext(\n      'Password is not strong enough, use a stronger password. Check the requirements listed on the right.'\n    ),\n    required: gettext('This field is required.'),\n    tenantId: gettext(\n      'May contain lowercase letters, digits, and hyphens. ' +\n        'Must start with a letter; hyphens only allowed in the middle; minimum 2 characters.'\n    ),\n    urlSegment: gettext('Slash, single or double dots and white space not allowed.'),\n    user: gettext(\n      'Username must not contain spaces nor slashes (\"/\", \"\\\\\") nor (\"+\"), (\":\"), (\"$\") signs.'\n    ),\n    httpUrl: gettext('Must be a valid HTTP(S) URL.'),\n    colonedHexNumber: gettext(\n      'Must be a valid hexadecimal number. Must contain only the following characters: 0-9, a-f, A-F, :.'\n    ),\n    noDots: gettext('Dots not allowed.'),\n    integer: gettext('Invalid type, expected integer.'),\n    simpleJsonPath: gettext('Must be a valid JSON path'),\n    [DatapointLibraryValidationErrors.SHOULD_CONTAIN_NUMBER]: gettext(\n      'This field must contain a number.'\n    ),\n    [DatapointLibraryValidationErrors.SHOULD_BE_DEFINED]: gettext('Value must be defined.'),\n    [DatapointLibraryValidationErrors.GREATER_THAN_SCALE_MAX]: gettext(\n      'Value must be less than scale maximum.'\n    ),\n    [DatapointLibraryValidationErrors.LESS_THAN_SCALE_MIN]: gettext(\n      'Value must be greater than scale minimum.'\n    ),\n    [DatapointLibraryValidationErrors.GREATER_THAN_RANGE_MAX]: gettext(\n      'Value must be less than respective maximum.'\n    ),\n    [DatapointLibraryValidationErrors.LESS_THAN_RANGE_MIN]: gettext(\n      'Value must be greater than respective minimum.'\n    )\n  } as const;\n\n  static readonly rules = {\n    deviceId: {\n      pattern: /^[^\\s/]*$/\n    },\n    domain: {\n      pattern: /^[a-z]+[a-z0-9-]*[a-z0-9]+\\.{1}.+$/\n    },\n    groupName: {\n      maxLength: 254\n    },\n    hostname: {\n      pattern:\n        /^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9-]*[a-zA-Z0-9])\\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9-]*[A-Za-z0-9])$/\n    },\n    ip: {\n      pattern:\n        /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/\n    },\n    opcuaBrowsePath: {\n      pattern: /^(opc.tcp|http|https):\\/\\/[^ \"]+$/\n    },\n    password: {\n      pattern: /^[a-zA-Z0-9`~!@#$%^&*()_|+\\-=?;:'\",.<>{}[\\]\\\\/]{8,32}$/\n    },\n    tenantId: {\n      pattern: /^[a-z]+[a-z0-9-]*[a-z0-9]+$/\n    },\n    user: {\n      pattern: /^[^\\\\/\\s$:+]*$/\n    },\n    urlSegment: {\n      pattern: /^(([.]{1,2})|([/])+|(([/]*)([.]{1,2})([/]*))|((.*)([/]|[ ]+)(.*))|(\\s+))$/\n    },\n    loginAlias: {\n      pattern: /^[^\\\\/$:+]*$/\n    },\n    noDots: {\n      pattern: /^[^.]*$/\n    },\n    httpUrl: {\n      pattern: /^(https?):\\/\\/.*$/\n    },\n    colonedHexNumber: {\n      pattern: /^[0-9a-fA-F:]+$/\n    },\n    integer: {\n      pattern: /^-?\\d+$/\n    }\n  };\n\n  static get(key: string) {\n    const pattern = this.rules[key];\n    return pattern as ValidationRules;\n  }\n}\n"]}
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation-pattern.js","sourceRoot":"","sources":["../../../../core/forms/validation-pattern.ts"],"names":[],"mappings":"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","sourcesContent":["// tslint:disable: max-line-length\nimport { gettext } from '../i18n/gettext';\n\nexport interface ValidationRules {\n  pattern?: RegExp;\n  message?: string;\n  maxLength?: number;\n  minLength?: number;\n}\n\nexport enum DatapointLibraryValidationErrors {\n  IS_REQUIRED = 'required',\n  PATTERN = 'pattern',\n  SHOULD_CONTAIN_NUMBER = 'number',\n  SHOULD_BE_DEFINED = 'should-be-defined',\n  GREATER_THAN_SCALE_MAX = 'greater-than-scale-max',\n  LESS_THAN_SCALE_MIN = 'less-than-scale-min',\n  GREATER_THAN_RANGE_MAX = 'greater-than-range-max',\n  LESS_THAN_RANGE_MIN = 'less-than-range-min',\n  MIN_ACTIVE_COUNT = 'minActiveCount',\n  MAX_ACTIVE_COUNT = 'maxActiveCount'\n}\n\n// @dynamic\nexport class ValidationPattern {\n  static readonly messages = {\n    deviceId: gettext('Device ID must not contain spaces or slashes (\"/\").'),\n    domain: gettext(\n      'Only lowercase letters, digits and hyphens allowed in the first part of the URI. ' +\n        'Must start with a letter; hyphens only allowed in the middle. Must be a valid URI.'\n    ),\n    email: gettext('Invalid email address.'),\n    emails: gettext('Invalid email addresses.'),\n    internationalPhoneNumber: gettext(\n      'Must be a valid phone number (only digits, spaces, slashes (\"/\"), dashes (\"-\"), and plus (\"+\") allowed, for example: +49 9 876 543 210).'\n    ),\n    loginAlias: gettext(\n      'Login alias must not contain slashes (\"/\", \"\\\\\") nor (\"+\"), (\":\"), (\"$\") signs.'\n    ),\n    invalidColor: gettext('Invalid color \"{{ currentValue }}\".'),\n    max: gettext('Max value: {{ max }}'),\n    min: gettext('Min value: {{ min }}'),\n    minLength: gettext('Should have at least {{ minLength }} characters.'),\n    minlength: gettext('Should have at least {{ requiredLength }} characters.'),\n    maxLength: gettext('Should have at most {{ maxLength }} characters.'),\n    maxlength: gettext('Should have at most {{ requiredLength }} characters.'),\n    pattern: gettext('Does not match pattern {{ requiredPattern }}.'),\n    opcuaBrowsePath: gettext('Invalid OPC UA URI.'),\n    password: gettext(\n      'Password must have at least 8 characters and no more than 32 and can only contain letters, numbers and following symbols: `~!@#$%^&*()_|+-=?;:\\'\",.<>{}[]\\\\/'\n    ),\n    passwordConfirm: gettext('Passwords do not match.'),\n    passwordStrength: gettext('Password is not strong enough, use a stronger password.'),\n    passwordStrengthChecklist: gettext(\n      'Password is not strong enough, use a stronger password. Check the requirements listed on the right.'\n    ),\n    required: gettext('This field is required.'),\n    tenantId: gettext(\n      'May contain lowercase letters, digits, and hyphens. ' +\n        'Must start with a letter; hyphens only allowed in the middle; minimum 2 characters.'\n    ),\n    urlSegment: gettext('Slash, single or double dots and white space not allowed.'),\n    user: gettext(\n      'Username must not contain spaces nor slashes (\"/\", \"\\\\\") nor (\"+\"), (\":\"), (\"$\") signs.'\n    ),\n    httpUrl: gettext('Must be a valid HTTP(S) URL.'),\n    colonedHexNumber: gettext(\n      'Must be a valid hexadecimal number. Must contain only the following characters: 0-9, a-f, A-F, :.'\n    ),\n    noWhiteSpaceOnly: gettext('This field must not contain only whitespaces'),\n    noWhiteSpaceAtBeginning: gettext(\n      'Name must not contain white spaces at the beginning or ending.'\n    ),\n    noDots: gettext('Dots not allowed.'),\n    integer: gettext('Invalid type, expected integer.'),\n    simpleJsonPath: gettext('Must be a valid JSON path'),\n    [DatapointLibraryValidationErrors.SHOULD_CONTAIN_NUMBER]: gettext(\n      'This field must contain a number.'\n    ),\n    [DatapointLibraryValidationErrors.SHOULD_BE_DEFINED]: gettext('Value must be defined.'),\n    [DatapointLibraryValidationErrors.GREATER_THAN_SCALE_MAX]: gettext(\n      'Value must be less than scale maximum.'\n    ),\n    [DatapointLibraryValidationErrors.LESS_THAN_SCALE_MIN]: gettext(\n      'Value must be greater than scale minimum.'\n    ),\n    [DatapointLibraryValidationErrors.GREATER_THAN_RANGE_MAX]: gettext(\n      'Value must be less than respective maximum.'\n    ),\n    [DatapointLibraryValidationErrors.LESS_THAN_RANGE_MIN]: gettext(\n      'Value must be greater than respective minimum.'\n    )\n  } as const;\n\n  static readonly rules = {\n    deviceId: {\n      pattern: /^[^\\s/]*$/\n    },\n    domain: {\n      pattern: /^[a-z]+[a-z0-9-]*[a-z0-9]+\\.{1}.+$/\n    },\n    groupName: {\n      maxLength: 254\n    },\n    hostname: {\n      pattern:\n        /^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9-]*[a-zA-Z0-9])\\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9-]*[A-Za-z0-9])$/\n    },\n    ip: {\n      pattern:\n        /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/\n    },\n    opcuaBrowsePath: {\n      pattern: /^(opc.tcp|http|https):\\/\\/[^ \"]+$/\n    },\n    password: {\n      pattern: /^[a-zA-Z0-9`~!@#$%^&*()_|+\\-=?;:'\",.<>{}[\\]\\\\/]{8,32}$/\n    },\n    tenantId: {\n      pattern: /^[a-z]+[a-z0-9-]*[a-z0-9]+$/\n    },\n    user: {\n      pattern: /^[^\\\\/\\s$:+]*$/\n    },\n    urlSegment: {\n      pattern: /^(([.]{1,2})|([/])+|(([/]*)([.]{1,2})([/]*))|((.*)([/]|[ ]+)(.*))|(\\s+))$/\n    },\n    loginAlias: {\n      pattern: /^[^\\\\/$:+]*$/\n    },\n    noWhiteSpaceOnly: {\n      pattern: /^(?!\\s*$).+/\n    },\n    noWhiteSpaceAtBeginning: {\n      pattern: /^[^\\s].+[^\\s]$/\n    },\n    noDots: {\n      pattern: /^[^.]*$/\n    },\n    httpUrl: {\n      pattern: /^(https?):\\/\\/.*$/\n    },\n    colonedHexNumber: {\n      pattern: /^[0-9a-fA-F:]+$/\n    },\n    integer: {\n      pattern: /^-?\\d+$/\n    }\n  };\n\n  static get(key: string) {\n    const pattern = this.rules[key];\n    return pattern as ValidationRules;\n  }\n}\n"]}
@@ -11,12 +11,12 @@ export class NavigatorBottomComponent {
11
11
  this.hasHidePoweredSet$ = this.bottomService.hasHidePoweredSet();
12
12
  }
13
13
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavigatorBottomComponent, deps: [{ token: i1.NavigatorBottomService }], target: i0.ɵɵFactoryTarget.Component }); }
14
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: NavigatorBottomComponent, isStandalone: true, selector: "c8y-navigator-bottom", host: { classAttribute: "navigator-slot-bottom" }, ngImport: i0, template: "<div\n class=\"d-flex gap-8 {{!(hasCustomNavigatorLogoSet$ | async) ? 'c8y-logo' : ''}}\"\n *ngIf=\"!(hasCustomNavigatorLogoSet$ | async); else branded\"\n>\n <p\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n class=\"powered-by text-left a-s-end\"\n translate\n >\n powered by\n </p>\n <a href=\"https://cumulocity.com\" target=\"_blank\" title=\"Cumulocity\" class=\"tenant-brand\"></a>\n</div>\n<ng-template #branded>\n <div class=\"logo-spacer\"></div>\n <div class=\"tenant-brand\"></div>\n <p\n class=\"powered-by text-center a-s-center\"\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n translate\n >\n powered by <a href=\"https://cumulocity.com\" target=\"_blank\">Cumulocity</a>\n </p>\n <div class=\"logo-spacer\"></div>\n</ng-template>\n", dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }] }); }
14
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: NavigatorBottomComponent, isStandalone: true, selector: "c8y-navigator-bottom", host: { classAttribute: "navigator-slot-bottom" }, ngImport: i0, template: "<div\n class=\"d-flex gap-8 {{!(hasCustomNavigatorLogoSet$ | async) ? 'c8y-logo' : ''}}\"\n *ngIf=\"!(hasCustomNavigatorLogoSet$ | async); else branded\"\n>\n <p\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n class=\"powered-by text-left a-s-end\"\n translate\n >\n powered by`KEEP_ORIGINAL`\n </p>\n <a href=\"https://cumulocity.com\" target=\"_blank\" title=\"Cumulocity\" class=\"tenant-brand\"></a>\n</div>\n<ng-template #branded>\n <div class=\"logo-spacer\"></div>\n <div class=\"tenant-brand\"></div>\n <p\n class=\"powered-by text-center a-s-center\"\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n translate\n >\n powered by <a href=\"https://cumulocity.com\" target=\"_blank\">Cumulocity</a>`KEEP_ORIGINAL`\n </p>\n <div class=\"logo-spacer\"></div>\n</ng-template>\n", dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }] }); }
15
15
  }
16
16
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: NavigatorBottomComponent, decorators: [{
17
17
  type: Component,
18
18
  args: [{ selector: 'c8y-navigator-bottom', host: {
19
19
  class: 'navigator-slot-bottom'
20
- }, standalone: true, imports: [AsyncPipe, NgIf, C8yTranslateDirective], template: "<div\n class=\"d-flex gap-8 {{!(hasCustomNavigatorLogoSet$ | async) ? 'c8y-logo' : ''}}\"\n *ngIf=\"!(hasCustomNavigatorLogoSet$ | async); else branded\"\n>\n <p\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n class=\"powered-by text-left a-s-end\"\n translate\n >\n powered by\n </p>\n <a href=\"https://cumulocity.com\" target=\"_blank\" title=\"Cumulocity\" class=\"tenant-brand\"></a>\n</div>\n<ng-template #branded>\n <div class=\"logo-spacer\"></div>\n <div class=\"tenant-brand\"></div>\n <p\n class=\"powered-by text-center a-s-center\"\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n translate\n >\n powered by <a href=\"https://cumulocity.com\" target=\"_blank\">Cumulocity</a>\n </p>\n <div class=\"logo-spacer\"></div>\n</ng-template>\n" }]
20
+ }, standalone: true, imports: [AsyncPipe, NgIf, C8yTranslateDirective], template: "<div\n class=\"d-flex gap-8 {{!(hasCustomNavigatorLogoSet$ | async) ? 'c8y-logo' : ''}}\"\n *ngIf=\"!(hasCustomNavigatorLogoSet$ | async); else branded\"\n>\n <p\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n class=\"powered-by text-left a-s-end\"\n translate\n >\n powered by`KEEP_ORIGINAL`\n </p>\n <a href=\"https://cumulocity.com\" target=\"_blank\" title=\"Cumulocity\" class=\"tenant-brand\"></a>\n</div>\n<ng-template #branded>\n <div class=\"logo-spacer\"></div>\n <div class=\"tenant-brand\"></div>\n <p\n class=\"powered-by text-center a-s-center\"\n *ngIf=\"!(hasHidePoweredSet$ | async)\"\n translate\n >\n powered by <a href=\"https://cumulocity.com\" target=\"_blank\">Cumulocity</a>`KEEP_ORIGINAL`\n </p>\n <div class=\"logo-spacer\"></div>\n</ng-template>\n" }]
21
21
  }], ctorParameters: () => [{ type: i1.NavigatorBottomService }] });
22
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmF2aWdhdG9yLWJvdHRvbS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9jb3JlL25hdmlnYXRvci9uYXZpZ2F0b3ItYm90dG9tL25hdmlnYXRvci1ib3R0b20uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vY29yZS9uYXZpZ2F0b3IvbmF2aWdhdG9yLWJvdHRvbS9uYXZpZ2F0b3ItYm90dG9tLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUMsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDcEUsT0FBTyxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUNsRCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxZQUFZLENBQUM7OztBQVduRCxNQUFNLE9BQU8sd0JBQXdCO0lBSW5DLFlBQW9CLGFBQXFDO1FBQXJDLGtCQUFhLEdBQWIsYUFBYSxDQUF3QjtRQUh6RCwrQkFBMEIsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLHlCQUF5QixFQUFFLENBQUM7UUFDNUUsdUJBQWtCLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO0lBRUEsQ0FBQzs4R0FKbEQsd0JBQXdCO2tHQUF4Qix3QkFBd0IsbUlDZHJDLDB3QkF5QkEsdUNEYlksU0FBUyw4Q0FBRSxJQUFJLDZGQUFFLHFCQUFxQjs7MkZBRXJDLHdCQUF3QjtrQkFUcEMsU0FBUzsrQkFDRSxzQkFBc0IsUUFFMUI7d0JBQ0osS0FBSyxFQUFFLHVCQUF1QjtxQkFDL0IsY0FDVyxJQUFJLFdBQ1AsQ0FBQyxTQUFTLEVBQUUsSUFBSSxFQUFFLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBOYXZpZ2F0b3JCb3R0b21TZXJ2aWNlIH0gZnJvbSAnLi9uYXZpZ2F0b3ItYm90dG9tLnNlcnZpY2UnO1xuaW1wb3J0IHsgQXN5bmNQaXBlLCBOZ0lmIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEM4eVRyYW5zbGF0ZURpcmVjdGl2ZSB9IGZyb20gJy4uLy4uL2kxOG4nO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjOHktbmF2aWdhdG9yLWJvdHRvbScsXG4gIHRlbXBsYXRlVXJsOiAnLi9uYXZpZ2F0b3ItYm90dG9tLmNvbXBvbmVudC5odG1sJyxcbiAgaG9zdDoge1xuICAgIGNsYXNzOiAnbmF2aWdhdG9yLXNsb3QtYm90dG9tJ1xuICB9LFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQXN5bmNQaXBlLCBOZ0lmLCBDOHlUcmFuc2xhdGVEaXJlY3RpdmVdXG59KVxuZXhwb3J0IGNsYXNzIE5hdmlnYXRvckJvdHRvbUNvbXBvbmVudCB7XG4gIGhhc0N1c3RvbU5hdmlnYXRvckxvZ29TZXQkID0gdGhpcy5ib3R0b21TZXJ2aWNlLmhhc0N1c3RvbU5hdmlnYXRvckxvZ29TZXQoKTtcbiAgaGFzSGlkZVBvd2VyZWRTZXQkID0gdGhpcy5ib3R0b21TZXJ2aWNlLmhhc0hpZGVQb3dlcmVkU2V0KCk7XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSBib3R0b21TZXJ2aWNlOiBOYXZpZ2F0b3JCb3R0b21TZXJ2aWNlKSB7fVxufVxuIiwiPGRpdlxuICBjbGFzcz1cImQtZmxleCBnYXAtOCB7eyEoaGFzQ3VzdG9tTmF2aWdhdG9yTG9nb1NldCQgfCBhc3luYykgPyAnYzh5LWxvZ28nIDogJyd9fVwiXG4gICpuZ0lmPVwiIShoYXNDdXN0b21OYXZpZ2F0b3JMb2dvU2V0JCB8IGFzeW5jKTsgZWxzZSBicmFuZGVkXCJcbj5cbiAgPHBcbiAgICAqbmdJZj1cIiEoaGFzSGlkZVBvd2VyZWRTZXQkIHwgYXN5bmMpXCJcbiAgICBjbGFzcz1cInBvd2VyZWQtYnkgdGV4dC1sZWZ0IGEtcy1lbmRcIlxuICAgIHRyYW5zbGF0ZVxuICA+XG4gICAgcG93ZXJlZCBieVxuICA8L3A+XG4gIDxhIGhyZWY9XCJodHRwczovL2N1bXVsb2NpdHkuY29tXCIgdGFyZ2V0PVwiX2JsYW5rXCIgdGl0bGU9XCJDdW11bG9jaXR5XCIgY2xhc3M9XCJ0ZW5hbnQtYnJhbmRcIj48L2E+XG48L2Rpdj5cbjxuZy10ZW1wbGF0ZSAjYnJhbmRlZD5cbiAgPGRpdiBjbGFzcz1cImxvZ28tc3BhY2VyXCI+PC9kaXY+XG4gIDxkaXYgY2xhc3M9XCJ0ZW5hbnQtYnJhbmRcIj48L2Rpdj5cbiAgPHBcbiAgICBjbGFzcz1cInBvd2VyZWQtYnkgdGV4dC1jZW50ZXIgYS1zLWNlbnRlclwiXG4gICAgKm5nSWY9XCIhKGhhc0hpZGVQb3dlcmVkU2V0JCB8IGFzeW5jKVwiXG4gICAgdHJhbnNsYXRlXG4gID5cbiAgICBwb3dlcmVkIGJ5IDxhIGhyZWY9XCJodHRwczovL2N1bXVsb2NpdHkuY29tXCIgdGFyZ2V0PVwiX2JsYW5rXCI+Q3VtdWxvY2l0eTwvYT5cbiAgPC9wPlxuICA8ZGl2IGNsYXNzPVwibG9nby1zcGFjZXJcIj48L2Rpdj5cbjwvbmctdGVtcGxhdGU+XG4iXX0=
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmF2aWdhdG9yLWJvdHRvbS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9jb3JlL25hdmlnYXRvci9uYXZpZ2F0b3ItYm90dG9tL25hdmlnYXRvci1ib3R0b20uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vY29yZS9uYXZpZ2F0b3IvbmF2aWdhdG9yLWJvdHRvbS9uYXZpZ2F0b3ItYm90dG9tLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUMsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDcEUsT0FBTyxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUNsRCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxZQUFZLENBQUM7OztBQVduRCxNQUFNLE9BQU8sd0JBQXdCO0lBSW5DLFlBQW9CLGFBQXFDO1FBQXJDLGtCQUFhLEdBQWIsYUFBYSxDQUF3QjtRQUh6RCwrQkFBMEIsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLHlCQUF5QixFQUFFLENBQUM7UUFDNUUsdUJBQWtCLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO0lBRUEsQ0FBQzs4R0FKbEQsd0JBQXdCO2tHQUF4Qix3QkFBd0IsbUlDZHJDLHd5QkF5QkEsdUNEYlksU0FBUyw4Q0FBRSxJQUFJLDZGQUFFLHFCQUFxQjs7MkZBRXJDLHdCQUF3QjtrQkFUcEMsU0FBUzsrQkFDRSxzQkFBc0IsUUFFMUI7d0JBQ0osS0FBSyxFQUFFLHVCQUF1QjtxQkFDL0IsY0FDVyxJQUFJLFdBQ1AsQ0FBQyxTQUFTLEVBQUUsSUFBSSxFQUFFLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBOYXZpZ2F0b3JCb3R0b21TZXJ2aWNlIH0gZnJvbSAnLi9uYXZpZ2F0b3ItYm90dG9tLnNlcnZpY2UnO1xuaW1wb3J0IHsgQXN5bmNQaXBlLCBOZ0lmIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEM4eVRyYW5zbGF0ZURpcmVjdGl2ZSB9IGZyb20gJy4uLy4uL2kxOG4nO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjOHktbmF2aWdhdG9yLWJvdHRvbScsXG4gIHRlbXBsYXRlVXJsOiAnLi9uYXZpZ2F0b3ItYm90dG9tLmNvbXBvbmVudC5odG1sJyxcbiAgaG9zdDoge1xuICAgIGNsYXNzOiAnbmF2aWdhdG9yLXNsb3QtYm90dG9tJ1xuICB9LFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQXN5bmNQaXBlLCBOZ0lmLCBDOHlUcmFuc2xhdGVEaXJlY3RpdmVdXG59KVxuZXhwb3J0IGNsYXNzIE5hdmlnYXRvckJvdHRvbUNvbXBvbmVudCB7XG4gIGhhc0N1c3RvbU5hdmlnYXRvckxvZ29TZXQkID0gdGhpcy5ib3R0b21TZXJ2aWNlLmhhc0N1c3RvbU5hdmlnYXRvckxvZ29TZXQoKTtcbiAgaGFzSGlkZVBvd2VyZWRTZXQkID0gdGhpcy5ib3R0b21TZXJ2aWNlLmhhc0hpZGVQb3dlcmVkU2V0KCk7XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSBib3R0b21TZXJ2aWNlOiBOYXZpZ2F0b3JCb3R0b21TZXJ2aWNlKSB7fVxufVxuIiwiPGRpdlxuICBjbGFzcz1cImQtZmxleCBnYXAtOCB7eyEoaGFzQ3VzdG9tTmF2aWdhdG9yTG9nb1NldCQgfCBhc3luYykgPyAnYzh5LWxvZ28nIDogJyd9fVwiXG4gICpuZ0lmPVwiIShoYXNDdXN0b21OYXZpZ2F0b3JMb2dvU2V0JCB8IGFzeW5jKTsgZWxzZSBicmFuZGVkXCJcbj5cbiAgPHBcbiAgICAqbmdJZj1cIiEoaGFzSGlkZVBvd2VyZWRTZXQkIHwgYXN5bmMpXCJcbiAgICBjbGFzcz1cInBvd2VyZWQtYnkgdGV4dC1sZWZ0IGEtcy1lbmRcIlxuICAgIHRyYW5zbGF0ZVxuICA+XG4gICAgcG93ZXJlZCBieWBLRUVQX09SSUdJTkFMYFxuICA8L3A+XG4gIDxhIGhyZWY9XCJodHRwczovL2N1bXVsb2NpdHkuY29tXCIgdGFyZ2V0PVwiX2JsYW5rXCIgdGl0bGU9XCJDdW11bG9jaXR5XCIgY2xhc3M9XCJ0ZW5hbnQtYnJhbmRcIj48L2E+XG48L2Rpdj5cbjxuZy10ZW1wbGF0ZSAjYnJhbmRlZD5cbiAgPGRpdiBjbGFzcz1cImxvZ28tc3BhY2VyXCI+PC9kaXY+XG4gIDxkaXYgY2xhc3M9XCJ0ZW5hbnQtYnJhbmRcIj48L2Rpdj5cbiAgPHBcbiAgICBjbGFzcz1cInBvd2VyZWQtYnkgdGV4dC1jZW50ZXIgYS1zLWNlbnRlclwiXG4gICAgKm5nSWY9XCIhKGhhc0hpZGVQb3dlcmVkU2V0JCB8IGFzeW5jKVwiXG4gICAgdHJhbnNsYXRlXG4gID5cbiAgICBwb3dlcmVkIGJ5IDxhIGhyZWY9XCJodHRwczovL2N1bXVsb2NpdHkuY29tXCIgdGFyZ2V0PVwiX2JsYW5rXCI+Q3VtdWxvY2l0eTwvYT5gS0VFUF9PUklHSU5BTGBcbiAgPC9wPlxuICA8ZGl2IGNsYXNzPVwibG9nby1zcGFjZXJcIj48L2Rpdj5cbjwvbmctdGVtcGxhdGU+XG4iXX0=