@c8y/ngx-components 1021.60.0 → 1021.62.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/alarms/alarm-details.component.d.ts +0 -4
- package/alarms/alarm-details.component.d.ts.map +1 -1
- package/core/authentication/new-password.component.d.ts +6 -3
- package/core/authentication/new-password.component.d.ts.map +1 -1
- package/core/common/interval-based-reload.abstract.d.ts +12 -1
- package/core/common/interval-based-reload.abstract.d.ts.map +1 -1
- package/core/common/permissions.service.d.ts +2 -0
- package/core/common/permissions.service.d.ts.map +1 -1
- package/core/common/tenant-ui.service.d.ts +9 -2
- package/core/common/tenant-ui.service.d.ts.map +1 -1
- package/core/forms/phone-validation.directive.d.ts +3 -6
- package/core/forms/phone-validation.directive.d.ts.map +1 -1
- package/core/router/router.module.d.ts.map +1 -1
- package/device-provisioned-certificates/device-provisioned-certificates.service.d.ts.map +1 -1
- package/esm2022/alarms/alarm-details.component.mjs +10 -10
- package/esm2022/core/authentication/new-password.component.mjs +26 -5
- package/esm2022/core/common/interval-based-reload.abstract.mjs +50 -2
- package/esm2022/core/common/permissions.service.mjs +3 -1
- package/esm2022/core/common/tenant-ui.service.mjs +28 -6
- package/esm2022/core/forms/phone-validation.directive.mjs +18 -19
- package/esm2022/core/router/router.module.mjs +1 -5
- package/esm2022/core/user/user-edit.component.mjs +1 -1
- package/esm2022/device-provisioned-certificates/device-provisioned-certificates.service.mjs +8 -3
- package/esm2022/tenants/custom-properties/custom-properties.component.mjs +25 -5
- package/esm2022/tenants/existing-tenant.guard.mjs +18 -0
- package/esm2022/tenants/index.mjs +3 -1
- package/esm2022/tenants/support-user-access/support-user-access.component.mjs +31 -0
- package/esm2022/tenants/tenant-form/tenant-form-inputs-definitions.mjs +116 -0
- package/esm2022/tenants/tenant-form/tenant-form.component.mjs +294 -0
- package/esm2022/tenants/tenant-limits/tenant-limits-definitions.mjs +10 -17
- package/esm2022/tenants/tenant-limits/tenant-limits.component.mjs +42 -29
- package/esm2022/tenants/tenant-list/tenant-list.component.mjs +17 -8
- package/esm2022/tenants/tenants.model.mjs +30 -1
- package/esm2022/tenants/tenants.module.mjs +57 -22
- package/esm2022/translation-editor/data/translation-store.service.mjs +11 -8
- package/esm2022/translation-editor/index.mjs +22 -5
- package/esm2022/translation-editor/lazy/advanced-translation-editor/advanced-translation-editor.component.mjs +136 -0
- package/esm2022/translation-editor/lazy/index.mjs +2 -1
- package/esm2022/translation-editor/translation-editor-tab-factory.service.mjs +40 -0
- package/esm2022/translation-editor/translation-editor.constants.mjs +4 -0
- package/esm2022/upgrade/upgraded-services/index.mjs +2 -1
- package/esm2022/upgrade/upgraded-services/tenant-policies.service.mjs +11 -0
- package/esm2022/upgrade/upgraded-services/upgraded-services.module.mjs +12 -3
- package/esm2022/widgets/implementations/alarms/alarm-widget-alarms-reload.component.mjs +9 -7
- package/esm2022/widgets/implementations/datapoints-table/datapoints-table-view/datapoints-reload/datapoints-reload.component.mjs +9 -7
- package/fesm2022/c8y-ngx-components-alarms.mjs +5 -5
- package/fesm2022/c8y-ngx-components-alarms.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-device-provisioned-certificates.mjs +7 -2
- package/fesm2022/c8y-ngx-components-device-provisioned-certificates.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-tenants.mjs +603 -82
- package/fesm2022/c8y-ngx-components-tenants.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-translation-editor-data.mjs +9 -6
- package/fesm2022/c8y-ngx-components-translation-editor-data.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-translation-editor-lazy.mjs +127 -2
- package/fesm2022/c8y-ngx-components-translation-editor-lazy.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-translation-editor.mjs +58 -4
- package/fesm2022/c8y-ngx-components-translation-editor.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-upgrade-upgraded-services.mjs +22 -3
- package/fesm2022/c8y-ngx-components-upgrade-upgraded-services.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-widgets-implementations-alarms.mjs +8 -6
- package/fesm2022/c8y-ngx-components-widgets-implementations-alarms.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-widgets-implementations-datapoints-table.mjs +8 -6
- package/fesm2022/c8y-ngx-components-widgets-implementations-datapoints-table.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components.mjs +499 -413
- package/fesm2022/c8y-ngx-components.mjs.map +1 -1
- package/locales/de.po +17 -25
- package/locales/es.po +17 -25
- package/locales/fr.po +17 -25
- package/locales/ja_JP.po +17 -25
- package/locales/ko.po +17 -25
- package/locales/locales.pot +34 -24
- package/locales/nl.po +17 -25
- package/locales/pl.po +17 -25
- package/locales/pt_BR.po +17 -25
- package/locales/zh_CN.po +17 -25
- package/locales/zh_TW.po +17 -25
- package/package.json +1 -1
- package/tenants/custom-properties/custom-properties.component.d.ts +4 -2
- package/tenants/custom-properties/custom-properties.component.d.ts.map +1 -1
- package/tenants/existing-tenant.guard.d.ts +13 -0
- package/tenants/existing-tenant.guard.d.ts.map +1 -0
- package/tenants/index.d.ts +2 -0
- package/tenants/index.d.ts.map +1 -1
- package/tenants/support-user-access/support-user-access.component.d.ts +18 -0
- package/tenants/support-user-access/support-user-access.component.d.ts.map +1 -0
- package/tenants/tenant-form/tenant-form-inputs-definitions.d.ts +113 -0
- package/tenants/tenant-form/tenant-form-inputs-definitions.d.ts.map +1 -0
- package/tenants/tenant-form/tenant-form.component.d.ts +79 -0
- package/tenants/tenant-form/tenant-form.component.d.ts.map +1 -0
- package/tenants/tenant-limits/tenant-limits-definitions.d.ts +11 -15
- package/tenants/tenant-limits/tenant-limits-definitions.d.ts.map +1 -1
- package/tenants/tenant-limits/tenant-limits.component.d.ts +9 -7
- package/tenants/tenant-limits/tenant-limits.component.d.ts.map +1 -1
- package/tenants/tenant-list/tenant-list.component.d.ts +6 -3
- package/tenants/tenant-list/tenant-list.component.d.ts.map +1 -1
- package/tenants/tenants.model.d.ts +31 -0
- package/tenants/tenants.model.d.ts.map +1 -1
- package/tenants/tenants.module.d.ts +4 -2
- package/tenants/tenants.module.d.ts.map +1 -1
- package/translation-editor/data/translation-store.service.d.ts +3 -2
- package/translation-editor/data/translation-store.service.d.ts.map +1 -1
- package/translation-editor/index.d.ts.map +1 -1
- package/translation-editor/lazy/advanced-translation-editor/advanced-translation-editor.component.d.ts +28 -0
- package/translation-editor/lazy/advanced-translation-editor/advanced-translation-editor.component.d.ts.map +1 -0
- package/translation-editor/lazy/index.d.ts +1 -0
- package/translation-editor/lazy/index.d.ts.map +1 -1
- package/translation-editor/translation-editor-tab-factory.service.d.ts +22 -0
- package/translation-editor/translation-editor-tab-factory.service.d.ts.map +1 -0
- package/translation-editor/translation-editor.constants.d.ts +4 -0
- package/translation-editor/translation-editor.constants.d.ts.map +1 -0
- package/upgrade/upgraded-services/index.d.ts +1 -0
- package/upgrade/upgraded-services/index.d.ts.map +1 -1
- package/upgrade/upgraded-services/tenant-policies.service.d.ts +11 -0
- package/upgrade/upgraded-services/tenant-policies.service.d.ts.map +1 -0
- package/upgrade/upgraded-services/upgraded-services.module.d.ts.map +1 -1
- package/widgets/implementations/alarms/alarm-widget-alarms-reload.component.d.ts +7 -2
- package/widgets/implementations/alarms/alarm-widget-alarms-reload.component.d.ts.map +1 -1
- package/widgets/implementations/datapoints-table/datapoints-table-view/datapoints-reload/datapoints-reload.component.d.ts +7 -2
- package/widgets/implementations/datapoints-table/datapoints-table-view/datapoints-reload/datapoints-reload.component.d.ts.map +1 -1
|
@@ -1,3 +1,32 @@
|
|
|
1
1
|
import { InjectionToken } from '@angular/core';
|
|
2
2
|
export const TENANTS_MODULE_CONFIG = new InjectionToken('TenantsModuleConfig');
|
|
3
|
-
|
|
3
|
+
export const PRODUCT_EXPERIENCE_TENANT_MANAGEMENT = {
|
|
4
|
+
EVENTS: {
|
|
5
|
+
TENANT_MANAGEMENT: 'tenantManagement'
|
|
6
|
+
},
|
|
7
|
+
COMPONENTS: {
|
|
8
|
+
TENANT_LIST: 'tenant-list',
|
|
9
|
+
TENANT_FORM: 'tenant-form',
|
|
10
|
+
TENANT_CUSTOM_PROPERTIES: 'tenant-custom-properties',
|
|
11
|
+
TENANT_LIMITS: 'tenant-limits'
|
|
12
|
+
},
|
|
13
|
+
ACTIONS: {
|
|
14
|
+
TENANT_CREATION_INITIALIZED: 'tenantCreationInitialized',
|
|
15
|
+
TENANT_CREATION_STARTED_FILLING: 'tenantCreationStartedFilling',
|
|
16
|
+
TENANT_CREATION_SAVED: 'tenantCreationSaved',
|
|
17
|
+
TENANT_PROPERTIES_OPENED: 'tenantPropertiesOpened',
|
|
18
|
+
TENANT_PROPERTIES_STARTED_CHANGING: 'tenantPropertiesStartedChanging',
|
|
19
|
+
TENANT_PROPERTIES_SAVED: 'tenantPropertiesSaved',
|
|
20
|
+
TENANT_CUSTOM_PROPERTIES_OPENED: 'tenantCustomPropertiesOpened',
|
|
21
|
+
TENANT_CUSTOM_PROPERTIES_STARTED_CHANGING: 'tenantCustomPropertiesStartedChanging',
|
|
22
|
+
TENANT_CUSTOM_PROPERTIES_SAVED: 'tenantCustomPropertiesSaved',
|
|
23
|
+
TENANT_LIMITS_OPENED: 'tenantLimitsOpened',
|
|
24
|
+
TENANT_LIMITS_STARTED_CHANGING: 'tenantLimitsStartedChanging',
|
|
25
|
+
TENANT_LIMITS_SAVED: 'tenantLimitsSaved'
|
|
26
|
+
},
|
|
27
|
+
RESULTS: {
|
|
28
|
+
SUCCESS: 'success',
|
|
29
|
+
FAILURE: 'failure'
|
|
30
|
+
}
|
|
31
|
+
};
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVuYW50cy5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RlbmFudHMvdGVuYW50cy5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRy9DLE1BQU0sQ0FBQyxNQUFNLHFCQUFxQixHQUFHLElBQUksY0FBYyxDQUFzQixxQkFBcUIsQ0FBQyxDQUFDO0FBZ0JwRyxNQUFNLENBQUMsTUFBTSxvQ0FBb0MsR0FBRztJQUNsRCxNQUFNLEVBQUU7UUFDTixpQkFBaUIsRUFBRSxrQkFBa0I7S0FDdEM7SUFDRCxVQUFVLEVBQUU7UUFDVixXQUFXLEVBQUUsYUFBYTtRQUMxQixXQUFXLEVBQUUsYUFBYTtRQUMxQix3QkFBd0IsRUFBRSwwQkFBMEI7UUFDcEQsYUFBYSxFQUFFLGVBQWU7S0FDL0I7SUFDRCxPQUFPLEVBQUU7UUFDUCwyQkFBMkIsRUFBRSwyQkFBMkI7UUFDeEQsK0JBQStCLEVBQUUsOEJBQThCO1FBQy9ELHFCQUFxQixFQUFFLHFCQUFxQjtRQUM1Qyx3QkFBd0IsRUFBRSx3QkFBd0I7UUFDbEQsa0NBQWtDLEVBQUUsaUNBQWlDO1FBQ3JFLHVCQUF1QixFQUFFLHVCQUF1QjtRQUNoRCwrQkFBK0IsRUFBRSw4QkFBOEI7UUFDL0QseUNBQXlDLEVBQUUsdUNBQXVDO1FBQ2xGLDhCQUE4QixFQUFFLDZCQUE2QjtRQUM3RCxvQkFBb0IsRUFBRSxvQkFBb0I7UUFDMUMsOEJBQThCLEVBQUUsNkJBQTZCO1FBQzdELG1CQUFtQixFQUFFLG1CQUFtQjtLQUN6QztJQUNELE9BQU8sRUFBRTtRQUNQLE9BQU8sRUFBRSxTQUFTO1FBQ2xCLE9BQU8sRUFBRSxTQUFTO0tBQ25CO0NBQ08sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBOYXZpZ2F0b3JOb2RlIH0gZnJvbSAnQGM4eS9uZ3gtY29tcG9uZW50cyc7XG5cbmV4cG9ydCBjb25zdCBURU5BTlRTX01PRFVMRV9DT05GSUcgPSBuZXcgSW5qZWN0aW9uVG9rZW48VGVuYW50c01vZHVsZUNvbmZpZz4oJ1RlbmFudHNNb2R1bGVDb25maWcnKTtcblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIG9iamVjdCBmb3IgYFRlbmFudHNNb2R1bGVgLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFRlbmFudHNNb2R1bGVDb25maWcge1xuICAvKipcbiAgICogQWxsb3dzIGZvciBoaWRpbmcgb3IgY3VzdG9taXppbmcgXCJTdWJ0ZW5hbnRzXCIgbmF2aWdhdG9yIG5vZGU6XG4gICAqIC0gYHN1YnRlbmFudHNOYXZpZ2F0b3JOb2RlOiB0cnVlYCAtIHNob3dzIHRoZSBkZWZhdWx0IG5hdmlnYXRvciBub2RlIChkZWZhdWx0KSxcbiAgICogLSBgc3VidGVuYW50c05hdmlnYXRvck5vZGU6IGZhbHNlYCAtIGhpZGVzIHRoZSBuYXZpZ2F0b3Igbm9kZSxcbiAgICogLSBgc3VidGVuYW50c05hdmlnYXRvck5vZGU6IHsgbGFiZWw6ICdNeSBzdWJ0ZW5hbnRzJyB9YCAtIG92ZXJyaWRlcyBkZWZhdWx0IG5hdmlnYXRvciBub2RlLFxuICAgKiAtIGBzdWJ0ZW5hbnRzTmF2aWdhdG9yTm9kZTogbmV3IE5hdmlnYXRvck5vZGUoeyBsYWJlbDogJ015IHN1YnRlbmFudHMnIH0pYCAtIG92ZXJyaWRlcyBkZWZhdWx0IG5hdmlnYXRvciBub2RlLlxuICAgKi9cbiAgc3VidGVuYW50c05hdmlnYXRvck5vZGU/OiBib29sZWFuIHwgUGFydGlhbDxOYXZpZ2F0b3JOb2RlPjtcbn1cblxuZXhwb3J0IGNvbnN0IFBST0RVQ1RfRVhQRVJJRU5DRV9URU5BTlRfTUFOQUdFTUVOVCA9IHtcbiAgRVZFTlRTOiB7XG4gICAgVEVOQU5UX01BTkFHRU1FTlQ6ICd0ZW5hbnRNYW5hZ2VtZW50J1xuICB9LFxuICBDT01QT05FTlRTOiB7XG4gICAgVEVOQU5UX0xJU1Q6ICd0ZW5hbnQtbGlzdCcsXG4gICAgVEVOQU5UX0ZPUk06ICd0ZW5hbnQtZm9ybScsXG4gICAgVEVOQU5UX0NVU1RPTV9QUk9QRVJUSUVTOiAndGVuYW50LWN1c3RvbS1wcm9wZXJ0aWVzJyxcbiAgICBURU5BTlRfTElNSVRTOiAndGVuYW50LWxpbWl0cydcbiAgfSxcbiAgQUNUSU9OUzoge1xuICAgIFRFTkFOVF9DUkVBVElPTl9JTklUSUFMSVpFRDogJ3RlbmFudENyZWF0aW9uSW5pdGlhbGl6ZWQnLFxuICAgIFRFTkFOVF9DUkVBVElPTl9TVEFSVEVEX0ZJTExJTkc6ICd0ZW5hbnRDcmVhdGlvblN0YXJ0ZWRGaWxsaW5nJyxcbiAgICBURU5BTlRfQ1JFQVRJT05fU0FWRUQ6ICd0ZW5hbnRDcmVhdGlvblNhdmVkJyxcbiAgICBURU5BTlRfUFJPUEVSVElFU19PUEVORUQ6ICd0ZW5hbnRQcm9wZXJ0aWVzT3BlbmVkJyxcbiAgICBURU5BTlRfUFJPUEVSVElFU19TVEFSVEVEX0NIQU5HSU5HOiAndGVuYW50UHJvcGVydGllc1N0YXJ0ZWRDaGFuZ2luZycsXG4gICAgVEVOQU5UX1BST1BFUlRJRVNfU0FWRUQ6ICd0ZW5hbnRQcm9wZXJ0aWVzU2F2ZWQnLFxuICAgIFRFTkFOVF9DVVNUT01fUFJPUEVSVElFU19PUEVORUQ6ICd0ZW5hbnRDdXN0b21Qcm9wZXJ0aWVzT3BlbmVkJyxcbiAgICBURU5BTlRfQ1VTVE9NX1BST1BFUlRJRVNfU1RBUlRFRF9DSEFOR0lORzogJ3RlbmFudEN1c3RvbVByb3BlcnRpZXNTdGFydGVkQ2hhbmdpbmcnLFxuICAgIFRFTkFOVF9DVVNUT01fUFJPUEVSVElFU19TQVZFRDogJ3RlbmFudEN1c3RvbVByb3BlcnRpZXNTYXZlZCcsXG4gICAgVEVOQU5UX0xJTUlUU19PUEVORUQ6ICd0ZW5hbnRMaW1pdHNPcGVuZWQnLFxuICAgIFRFTkFOVF9MSU1JVFNfU1RBUlRFRF9DSEFOR0lORzogJ3RlbmFudExpbWl0c1N0YXJ0ZWRDaGFuZ2luZycsXG4gICAgVEVOQU5UX0xJTUlUU19TQVZFRDogJ3RlbmFudExpbWl0c1NhdmVkJ1xuICB9LFxuICBSRVNVTFRTOiB7XG4gICAgU1VDQ0VTUzogJ3N1Y2Nlc3MnLFxuICAgIEZBSUxVUkU6ICdmYWlsdXJlJ1xuICB9XG59IGFzIGNvbnN0O1xuZXhwb3J0IHR5cGUgVGVuYW50TWFuYWdlbWVudEFjdGlvblR5cGUgPVxuICAodHlwZW9mIFBST0RVQ1RfRVhQRVJJRU5DRV9URU5BTlRfTUFOQUdFTUVOVC5BQ1RJT05TKVtrZXlvZiB0eXBlb2YgUFJPRFVDVF9FWFBFUklFTkNFX1RFTkFOVF9NQU5BR0VNRU5ULkFDVElPTlNdO1xuZXhwb3J0IHR5cGUgVGVuYW50TWFuYWdlbWVudFJlc3VsdFR5cGUgPVxuICAodHlwZW9mIFBST0RVQ1RfRVhQRVJJRU5DRV9URU5BTlRfTUFOQUdFTUVOVC5SRVNVTFRTKVtrZXlvZiB0eXBlb2YgUFJPRFVDVF9FWFBFUklFTkNFX1RFTkFOVF9NQU5BR0VNRU5ULlJFU1VMVFNdO1xuIl19
|
|
@@ -1,15 +1,18 @@
|
|
|
1
1
|
import { NgModule } from '@angular/core';
|
|
2
2
|
import { RouterModule } from '@angular/router';
|
|
3
3
|
import { BsDatepickerModule } from 'ngx-bootstrap/datepicker';
|
|
4
|
-
import { CoreModule,
|
|
4
|
+
import { CoreModule, hookNavigator, hookRoute, gettext, ViewContext } from '@c8y/ngx-components';
|
|
5
5
|
import { TenantsNavigationFactory } from './tenants-navigation.factory';
|
|
6
6
|
import { TenantListComponent } from './tenant-list/tenant-list.component';
|
|
7
7
|
import { CreationTimeFilteringFormRendererComponent } from './tenant-list/creation-time.filtering-form-renderer.component';
|
|
8
8
|
import { StatusFilteringFormRendererComponent } from './tenant-list/status.filtering-form-renderer.component';
|
|
9
9
|
import { TENANTS_MODULE_CONFIG } from './tenants.model';
|
|
10
10
|
import { TenantListGuard } from './tenant-list/tenant-list.guard';
|
|
11
|
+
import { TenantFormComponent } from './tenant-form/tenant-form.component';
|
|
11
12
|
import { CustomPropertiesComponent } from './custom-properties/custom-properties.component';
|
|
12
13
|
import { TenantLimitsComponent } from './tenant-limits/tenant-limits.component';
|
|
14
|
+
import { SupportUserAccessComponent } from './support-user-access/support-user-access.component';
|
|
15
|
+
import { ExistingTenantGuard } from './existing-tenant.guard';
|
|
13
16
|
import * as i0 from "@angular/core";
|
|
14
17
|
export class TenantsModule {
|
|
15
18
|
static config(config = {}) {
|
|
@@ -26,24 +29,39 @@ export class TenantsModule {
|
|
|
26
29
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TenantsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
|
|
27
30
|
static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.13", ngImport: i0, type: TenantsModule, declarations: [TenantListComponent,
|
|
28
31
|
CreationTimeFilteringFormRendererComponent,
|
|
29
|
-
StatusFilteringFormRendererComponent
|
|
32
|
+
StatusFilteringFormRendererComponent,
|
|
33
|
+
TenantFormComponent], imports: [BsDatepickerModule, CoreModule, RouterModule, SupportUserAccessComponent] }); }
|
|
30
34
|
static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TenantsModule, providers: [
|
|
31
35
|
TenantListGuard,
|
|
36
|
+
ExistingTenantGuard,
|
|
32
37
|
hookNavigator(TenantsNavigationFactory),
|
|
33
|
-
hookRoute({
|
|
34
|
-
path: 'tenants',
|
|
35
|
-
icon: 'c8y-layers',
|
|
36
|
-
component: TenantListComponent,
|
|
37
|
-
canActivate: [TenantListGuard]
|
|
38
|
-
}),
|
|
39
38
|
hookRoute([
|
|
39
|
+
{
|
|
40
|
+
path: 'tenants',
|
|
41
|
+
icon: 'c8y-layers',
|
|
42
|
+
component: TenantListComponent,
|
|
43
|
+
canActivate: [TenantListGuard]
|
|
44
|
+
},
|
|
45
|
+
{
|
|
46
|
+
path: 'tenants/new',
|
|
47
|
+
component: TenantFormComponent
|
|
48
|
+
},
|
|
49
|
+
{
|
|
50
|
+
path: 'properties',
|
|
51
|
+
component: TenantFormComponent,
|
|
52
|
+
icon: 'new-property',
|
|
53
|
+
label: gettext('Properties'),
|
|
54
|
+
context: ViewContext.Tenant,
|
|
55
|
+
priority: 10000
|
|
56
|
+
},
|
|
40
57
|
{
|
|
41
58
|
path: 'custom_properties',
|
|
42
59
|
component: CustomPropertiesComponent,
|
|
43
60
|
icon: 'property-script',
|
|
44
61
|
label: gettext('Custom properties'),
|
|
45
62
|
context: ViewContext.Tenant,
|
|
46
|
-
priority: 4
|
|
63
|
+
priority: 4,
|
|
64
|
+
canActivate: [ExistingTenantGuard]
|
|
47
65
|
},
|
|
48
66
|
{
|
|
49
67
|
path: 'limits',
|
|
@@ -51,38 +69,54 @@ export class TenantsModule {
|
|
|
51
69
|
icon: 'sliders',
|
|
52
70
|
label: gettext('Limits'),
|
|
53
71
|
context: ViewContext.Tenant,
|
|
54
|
-
priority: 3
|
|
72
|
+
priority: 3,
|
|
73
|
+
canActivate: [ExistingTenantGuard]
|
|
55
74
|
}
|
|
56
75
|
])
|
|
57
|
-
], imports: [
|
|
76
|
+
], imports: [BsDatepickerModule, CoreModule, RouterModule, SupportUserAccessComponent] }); }
|
|
58
77
|
}
|
|
59
78
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TenantsModule, decorators: [{
|
|
60
79
|
type: NgModule,
|
|
61
80
|
args: [{
|
|
62
|
-
imports: [
|
|
81
|
+
imports: [BsDatepickerModule, CoreModule, RouterModule, SupportUserAccessComponent],
|
|
63
82
|
exports: [],
|
|
64
83
|
declarations: [
|
|
65
84
|
TenantListComponent,
|
|
66
85
|
CreationTimeFilteringFormRendererComponent,
|
|
67
|
-
StatusFilteringFormRendererComponent
|
|
86
|
+
StatusFilteringFormRendererComponent,
|
|
87
|
+
TenantFormComponent
|
|
68
88
|
],
|
|
69
89
|
providers: [
|
|
70
90
|
TenantListGuard,
|
|
91
|
+
ExistingTenantGuard,
|
|
71
92
|
hookNavigator(TenantsNavigationFactory),
|
|
72
|
-
hookRoute({
|
|
73
|
-
path: 'tenants',
|
|
74
|
-
icon: 'c8y-layers',
|
|
75
|
-
component: TenantListComponent,
|
|
76
|
-
canActivate: [TenantListGuard]
|
|
77
|
-
}),
|
|
78
93
|
hookRoute([
|
|
94
|
+
{
|
|
95
|
+
path: 'tenants',
|
|
96
|
+
icon: 'c8y-layers',
|
|
97
|
+
component: TenantListComponent,
|
|
98
|
+
canActivate: [TenantListGuard]
|
|
99
|
+
},
|
|
100
|
+
{
|
|
101
|
+
path: 'tenants/new',
|
|
102
|
+
component: TenantFormComponent
|
|
103
|
+
},
|
|
104
|
+
{
|
|
105
|
+
path: 'properties',
|
|
106
|
+
component: TenantFormComponent,
|
|
107
|
+
icon: 'new-property',
|
|
108
|
+
label: gettext('Properties'),
|
|
109
|
+
context: ViewContext.Tenant,
|
|
110
|
+
priority: 10000
|
|
111
|
+
},
|
|
79
112
|
{
|
|
80
113
|
path: 'custom_properties',
|
|
81
114
|
component: CustomPropertiesComponent,
|
|
82
115
|
icon: 'property-script',
|
|
83
116
|
label: gettext('Custom properties'),
|
|
84
117
|
context: ViewContext.Tenant,
|
|
85
|
-
priority: 4
|
|
118
|
+
priority: 4,
|
|
119
|
+
canActivate: [ExistingTenantGuard]
|
|
86
120
|
},
|
|
87
121
|
{
|
|
88
122
|
path: 'limits',
|
|
@@ -90,10 +124,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
90
124
|
icon: 'sliders',
|
|
91
125
|
label: gettext('Limits'),
|
|
92
126
|
context: ViewContext.Tenant,
|
|
93
|
-
priority: 3
|
|
127
|
+
priority: 3,
|
|
128
|
+
canActivate: [ExistingTenantGuard]
|
|
94
129
|
}
|
|
95
130
|
])
|
|
96
131
|
]
|
|
97
132
|
}]
|
|
98
133
|
}] });
|
|
99
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
134
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tenants.module.js","sourceRoot":"","sources":["../../../tenants/tenants.module.ts"],"names":[],"mappings":"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","sourcesContent":["import { ModuleWithProviders, NgModule } from '@angular/core';\nimport { RouterModule } from '@angular/router';\nimport { BsDatepickerModule } from 'ngx-bootstrap/datepicker';\nimport { CoreModule, hookNavigator, hookRoute, gettext, ViewContext } from '@c8y/ngx-components';\nimport { TenantsNavigationFactory } from './tenants-navigation.factory';\nimport { TenantListComponent } from './tenant-list/tenant-list.component';\nimport { CreationTimeFilteringFormRendererComponent } from './tenant-list/creation-time.filtering-form-renderer.component';\nimport { StatusFilteringFormRendererComponent } from './tenant-list/status.filtering-form-renderer.component';\nimport { TenantsModuleConfig, TENANTS_MODULE_CONFIG } from './tenants.model';\nimport { TenantListGuard } from './tenant-list/tenant-list.guard';\nimport { TenantFormComponent } from './tenant-form/tenant-form.component';\nimport { CustomPropertiesComponent } from './custom-properties/custom-properties.component';\nimport { TenantLimitsComponent } from './tenant-limits/tenant-limits.component';\nimport { SupportUserAccessComponent } from './support-user-access/support-user-access.component';\nimport { ExistingTenantGuard } from './existing-tenant.guard';\n\n@NgModule({\n  imports: [BsDatepickerModule, CoreModule, RouterModule, SupportUserAccessComponent],\n  exports: [],\n  declarations: [\n    TenantListComponent,\n    CreationTimeFilteringFormRendererComponent,\n    StatusFilteringFormRendererComponent,\n    TenantFormComponent\n  ],\n  providers: [\n    TenantListGuard,\n    ExistingTenantGuard,\n    hookNavigator(TenantsNavigationFactory),\n    hookRoute([\n      {\n        path: 'tenants',\n        icon: 'c8y-layers',\n        component: TenantListComponent,\n        canActivate: [TenantListGuard]\n      },\n      {\n        path: 'tenants/new',\n        component: TenantFormComponent\n      },\n      {\n        path: 'properties',\n        component: TenantFormComponent,\n        icon: 'new-property',\n        label: gettext('Properties'),\n        context: ViewContext.Tenant,\n        priority: 10000\n      },\n      {\n        path: 'custom_properties',\n        component: CustomPropertiesComponent,\n        icon: 'property-script',\n        label: gettext('Custom properties'),\n        context: ViewContext.Tenant,\n        priority: 4,\n        canActivate: [ExistingTenantGuard]\n      },\n      {\n        path: 'limits',\n        component: TenantLimitsComponent,\n        icon: 'sliders',\n        label: gettext('Limits'),\n        context: ViewContext.Tenant,\n        priority: 3,\n        canActivate: [ExistingTenantGuard]\n      }\n    ])\n  ]\n})\nexport class TenantsModule {\n  static config(config: TenantsModuleConfig = {}): ModuleWithProviders<TenantsModule> {\n    return {\n      ngModule: TenantsModule,\n      providers: [\n        {\n          provide: TENANTS_MODULE_CONFIG,\n          useValue: config\n        }\n      ]\n    };\n  }\n}\n"]}
|
|
@@ -1,15 +1,16 @@
|
|
|
1
1
|
import { Injectable } from '@angular/core';
|
|
2
|
-
import { ApplicationAvailability, ApplicationService, ApplicationType } from '@c8y/client';
|
|
2
|
+
import { ApplicationAvailability, ApplicationService, ApplicationType, FetchClient } from '@c8y/client';
|
|
3
3
|
import { AppStateService, ZipService } from '@c8y/ngx-components';
|
|
4
4
|
import { uniq } from 'lodash-es';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
6
|
import * as i1 from "@c8y/client";
|
|
7
7
|
import * as i2 from "@c8y/ngx-components";
|
|
8
8
|
export class TranslationStoreService {
|
|
9
|
-
constructor(appService, appState, zip) {
|
|
9
|
+
constructor(appService, appState, zip, fetchClient) {
|
|
10
10
|
this.appService = appService;
|
|
11
11
|
this.appState = appState;
|
|
12
12
|
this.zip = zip;
|
|
13
|
+
this.fetchClient = fetchClient;
|
|
13
14
|
this.translationAppName = 'User defined translations';
|
|
14
15
|
this.translationAppContextPath = 'user-defined-translations';
|
|
15
16
|
}
|
|
@@ -30,11 +31,12 @@ export class TranslationStoreService {
|
|
|
30
31
|
for (const locale of locales) {
|
|
31
32
|
try {
|
|
32
33
|
const translationsForLanguageAndApp = await this.getFileFromPath(`/apps/${contextPath}/${locale}.json`);
|
|
33
|
-
|
|
34
|
+
const translationsForLang = translationsForLanguageAndApp[locale];
|
|
35
|
+
for (const key of Object.keys(translationsForLang)) {
|
|
34
36
|
if (!translations[key]) {
|
|
35
37
|
translations[key] = {};
|
|
36
38
|
}
|
|
37
|
-
translations[key][locale] =
|
|
39
|
+
translations[key][locale] = translationsForLang[key];
|
|
38
40
|
}
|
|
39
41
|
}
|
|
40
42
|
catch (e) {
|
|
@@ -158,13 +160,14 @@ export class TranslationStoreService {
|
|
|
158
160
|
}
|
|
159
161
|
async getFileFromPath(path) {
|
|
160
162
|
const finalPath = `${path}?nocache=${Date.now()}`;
|
|
161
|
-
|
|
163
|
+
// using fetch client instead of fetch directly here to ensure loading indicator is shown during these requests
|
|
164
|
+
const response = await this.fetchClient.fetch(finalPath);
|
|
162
165
|
if (response.status !== 200) {
|
|
163
166
|
throw new Error(`Failed to fetch file from ${path}`);
|
|
164
167
|
}
|
|
165
168
|
return response.json();
|
|
166
169
|
}
|
|
167
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TranslationStoreService, deps: [{ token: i1.ApplicationService }, { token: i2.AppStateService }, { token: i2.ZipService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
170
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TranslationStoreService, deps: [{ token: i1.ApplicationService }, { token: i2.AppStateService }, { token: i2.ZipService }, { token: i1.FetchClient }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
168
171
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TranslationStoreService, providedIn: 'root' }); }
|
|
169
172
|
}
|
|
170
173
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TranslationStoreService, decorators: [{
|
|
@@ -172,5 +175,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
172
175
|
args: [{
|
|
173
176
|
providedIn: 'root'
|
|
174
177
|
}]
|
|
175
|
-
}], ctorParameters: () => [{ type: i1.ApplicationService }, { type: i2.AppStateService }, { type: i2.ZipService }] });
|
|
176
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"translation-store.service.js","sourceRoot":"","sources":["../../../../translation-editor/data/translation-store.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n  ApplicationAvailability,\n  ApplicationService,\n  ApplicationType,\n  IApplication\n} from '@c8y/client';\nimport { AppStateService, ZipService } from '@c8y/ngx-components';\nimport { uniq } from 'lodash-es';\n\nexport interface TranslationLocale {\n  label: string;\n  locale: string;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class TranslationStoreService {\n  protected readonly translationAppName = 'User defined translations';\n  protected readonly translationAppContextPath = 'user-defined-translations';\n\n  constructor(\n    private appService: ApplicationService,\n    private appState: AppStateService,\n    private zip: ZipService\n  ) {}\n\n  /**\n   * Retrieves a list of available translations for the given locales from all the hosted apps available on the tenant.\n   */\n  async getAvailableTranslations(locales: string[]) {\n    const { data: hostedApps } = await this.appService.list({\n      tenant: this.appState.currentTenant.value.name,\n      type: 'HOSTED',\n      pageSize: 2000\n    });\n\n    const uniqueContextPaths = uniq(\n      hostedApps\n        .filter(app => app.contextPath && app.manifest?.webSdkVersion)\n        .map(app => app.contextPath)\n    );\n\n    const translations: { [key: string]: { [locale: string]: string } } = {};\n    for (const contextPath of uniqueContextPaths) {\n      for (const locale of locales) {\n        try {\n          const translationsForLanguageAndApp = await this.getFileFromPath(\n            `/apps/${contextPath}/${locale}.json`\n          );\n          for (const key of Object.keys(translationsForLanguageAndApp)) {\n            if (!translations[key]) {\n              translations[key] = {};\n            }\n            translations[key][locale] = translationsForLanguageAndApp[key];\n          }\n        } catch (e) {\n          continue;\n        }\n      }\n    }\n\n    return translations;\n  }\n\n  /**\n   * Retrieves the translations for the given locale from the translation app.\n   */\n  async getTranslationsForLocale<T extends string>(locale: T): Promise<{ [key: string]: string }> {\n    try {\n      const translations = await this.getFileFromTranslationApp(`${locale}.json`);\n      if (translations[locale]) {\n        return translations[locale];\n      }\n    } catch (e) {\n      // do nothing.\n    }\n    return {};\n  }\n\n  /**\n   * Retrieves the translations for the given locales from the translation app.\n   */\n  async loadTranslationsForLocales(locales: string[]): Promise<{\n    [key: string]: {\n      [key: string]: string;\n    };\n  }> {\n    const translations: {\n      [key: string]: { [key: string]: string };\n    } = {};\n\n    const loadAndAddLocale = async (locale: string) => {\n      translations[locale] = await this.getTranslationsForLocale(locale);\n    };\n\n    await Promise.all(locales.map(loadAndAddLocale));\n\n    return translations;\n  }\n\n  /**\n   * Retrieves a combined list of translations for all the provided locales.\n   * The locales are combined into a single object per key and an array of these objects is returned.\n   */\n  async getCombinedListOfTranslationsForPerKey(\n    locales: string[]\n  ): Promise<{ key: string; [locale: string]: string }[]> {\n    const translations = await this.loadTranslationsForLocales(locales);\n    const translationMap: { [key: string]: { [locale: string]: string } } = {};\n\n    for (const locale of Object.keys(translations)) {\n      for (const key of Object.keys(translations[locale])) {\n        let translationsForKey = translationMap[key];\n        if (!translationsForKey) {\n          translationsForKey = {};\n          translationMap[key] = translationsForKey;\n        }\n\n        translationsForKey[locale] = translations[locale][key];\n      }\n    }\n\n    const keys = Object.keys(translationMap).sort((a, b) => a.localeCompare(b));\n\n    return keys.map(key => Object.assign({ key }, translationMap[key]));\n  }\n\n  /**\n   * Updates the files of the translation application with the provided translations.\n   */\n  async updateTranslations(\n    translations: { key: string; [locale: string]: string }[]\n  ): Promise<void> {\n    const filesToUpload = new Array<{ path: string; contents: File }>();\n    const translationsPerLocale: { [locale: string]: { [key: string]: string } } = {};\n    for (const entry of translations) {\n      const key = entry.key;\n      for (const locale of Object.keys(entry)) {\n        if (locale === 'key') {\n          continue;\n        }\n        if (!translationsPerLocale[locale]) {\n          translationsPerLocale[locale] = {};\n        }\n\n        if (entry[locale]) {\n          translationsPerLocale[locale][key] = entry[locale];\n        }\n      }\n    }\n\n    const langs = this.appState.state.langs as string[];\n\n    for (const locale of langs) {\n      const fileName = `${locale}.json`;\n      filesToUpload.push({\n        path: fileName,\n        contents: new File(\n          [JSON.stringify({ [locale]: translationsPerLocale[locale] || {} })],\n          fileName\n        )\n      });\n    }\n    const app = await this.getOrCreateTranslationApp();\n    await this.appService.binary(app).updateFiles(filesToUpload);\n  }\n\n  /**\n   * @returns The translation app for the current tenant. If it does not exist, it will be created.\n   */\n  async getOrCreateTranslationApp(): Promise<IApplication> {\n    const { data: apps } = await this.appService.listByName(this.translationAppName);\n    const ownApp = apps.find(\n      app =>\n        app.contextPath === this.translationAppContextPath &&\n        app.owner.tenant.id === this.appState.currentTenant.value.name\n    );\n    if (ownApp) {\n      return ownApp;\n    }\n\n    const { data: app } = await this.appService.create({\n      name: this.translationAppName,\n      contextPath: this.translationAppContextPath,\n      key: `${this.translationAppContextPath}-app-key`,\n      type: ApplicationType.HOSTED,\n      availability: ApplicationAvailability.MARKET,\n      config: {\n        icon: {\n          class: 'language1'\n        }\n      },\n      noAppSwitcher: true,\n      description: 'Providing user defined translations'\n    });\n    const zip = await this.zip.createZip([]);\n\n    const { data: binary } = await this.appService.binary(app).upload(zip, 'translations.zip');\n    const { data: updatedApp } = await this.appService.update({\n      id: app.id,\n      activeVersionId: binary.id as string\n    });\n    return updatedApp;\n  }\n\n  private async getFileFromTranslationApp(file: string) {\n    return this.getFileFromPath(`/apps/public/${this.translationAppContextPath}/${file}`);\n  }\n\n  private async getFileFromPath(path: string) {\n    const finalPath = `${path}?nocache=${Date.now()}`;\n    const response = await fetch(finalPath);\n\n    if (response.status !== 200) {\n      throw new Error(`Failed to fetch file from ${path}`);\n    }\n\n    return response.json();\n  }\n}\n"]}
|
|
178
|
+
}], ctorParameters: () => [{ type: i1.ApplicationService }, { type: i2.AppStateService }, { type: i2.ZipService }, { type: i1.FetchClient }] });
|
|
179
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"translation-store.service.js","sourceRoot":"","sources":["../../../../translation-editor/data/translation-store.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n  ApplicationAvailability,\n  ApplicationService,\n  ApplicationType,\n  FetchClient,\n  IApplication\n} from '@c8y/client';\nimport { AppStateService, ZipService } from '@c8y/ngx-components';\nimport { uniq } from 'lodash-es';\n\nexport interface TranslationLocale {\n  label: string;\n  locale: string;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class TranslationStoreService {\n  protected readonly translationAppName = 'User defined translations';\n  protected readonly translationAppContextPath = 'user-defined-translations';\n\n  constructor(\n    private appService: ApplicationService,\n    private appState: AppStateService,\n    private zip: ZipService,\n    private fetchClient: FetchClient\n  ) {}\n\n  /**\n   * Retrieves a list of available translations for the given locales from all the hosted apps available on the tenant.\n   */\n  async getAvailableTranslations(locales: string[]) {\n    const { data: hostedApps } = await this.appService.list({\n      tenant: this.appState.currentTenant.value.name,\n      type: 'HOSTED',\n      pageSize: 2000\n    });\n\n    const uniqueContextPaths = uniq(\n      hostedApps\n        .filter(app => app.contextPath && app.manifest?.webSdkVersion)\n        .map(app => app.contextPath)\n    );\n\n    const translations: { [key: string]: { [locale: string]: string } } = {};\n    for (const contextPath of uniqueContextPaths) {\n      for (const locale of locales) {\n        try {\n          const translationsForLanguageAndApp = await this.getFileFromPath(\n            `/apps/${contextPath}/${locale}.json`\n          );\n          const translationsForLang = translationsForLanguageAndApp[locale];\n          for (const key of Object.keys(translationsForLang)) {\n            if (!translations[key]) {\n              translations[key] = {};\n            }\n            translations[key][locale] = translationsForLang[key];\n          }\n        } catch (e) {\n          continue;\n        }\n      }\n    }\n\n    return translations;\n  }\n\n  /**\n   * Retrieves the translations for the given locale from the translation app.\n   */\n  async getTranslationsForLocale<T extends string>(locale: T): Promise<{ [key: string]: string }> {\n    try {\n      const translations = await this.getFileFromTranslationApp(`${locale}.json`);\n      if (translations[locale]) {\n        return translations[locale];\n      }\n    } catch (e) {\n      // do nothing.\n    }\n    return {};\n  }\n\n  /**\n   * Retrieves the translations for the given locales from the translation app.\n   */\n  async loadTranslationsForLocales(locales: string[]): Promise<{\n    [key: string]: {\n      [key: string]: string;\n    };\n  }> {\n    const translations: {\n      [key: string]: { [key: string]: string };\n    } = {};\n\n    const loadAndAddLocale = async (locale: string) => {\n      translations[locale] = await this.getTranslationsForLocale(locale);\n    };\n\n    await Promise.all(locales.map(loadAndAddLocale));\n\n    return translations;\n  }\n\n  /**\n   * Retrieves a combined list of translations for all the provided locales.\n   * The locales are combined into a single object per key and an array of these objects is returned.\n   */\n  async getCombinedListOfTranslationsForPerKey(\n    locales: string[]\n  ): Promise<{ key: string; [locale: string]: string }[]> {\n    const translations = await this.loadTranslationsForLocales(locales);\n    const translationMap: { [key: string]: { [locale: string]: string } } = {};\n\n    for (const locale of Object.keys(translations)) {\n      for (const key of Object.keys(translations[locale])) {\n        let translationsForKey = translationMap[key];\n        if (!translationsForKey) {\n          translationsForKey = {};\n          translationMap[key] = translationsForKey;\n        }\n\n        translationsForKey[locale] = translations[locale][key];\n      }\n    }\n\n    const keys = Object.keys(translationMap).sort((a, b) => a.localeCompare(b));\n\n    return keys.map(key => Object.assign({ key }, translationMap[key]));\n  }\n\n  /**\n   * Updates the files of the translation application with the provided translations.\n   */\n  async updateTranslations(\n    translations: { key: string; [locale: string]: string }[]\n  ): Promise<void> {\n    const filesToUpload = new Array<{ path: string; contents: File }>();\n    const translationsPerLocale: { [locale: string]: { [key: string]: string } } = {};\n    for (const entry of translations) {\n      const key = entry.key;\n      for (const locale of Object.keys(entry)) {\n        if (locale === 'key') {\n          continue;\n        }\n        if (!translationsPerLocale[locale]) {\n          translationsPerLocale[locale] = {};\n        }\n\n        if (entry[locale]) {\n          translationsPerLocale[locale][key] = entry[locale];\n        }\n      }\n    }\n\n    const langs = this.appState.state.langs as string[];\n\n    for (const locale of langs) {\n      const fileName = `${locale}.json`;\n      filesToUpload.push({\n        path: fileName,\n        contents: new File(\n          [JSON.stringify({ [locale]: translationsPerLocale[locale] || {} })],\n          fileName\n        )\n      });\n    }\n    const app = await this.getOrCreateTranslationApp();\n    await this.appService.binary(app).updateFiles(filesToUpload);\n  }\n\n  /**\n   * @returns The translation app for the current tenant. If it does not exist, it will be created.\n   */\n  async getOrCreateTranslationApp(): Promise<IApplication> {\n    const { data: apps } = await this.appService.listByName(this.translationAppName);\n    const ownApp = apps.find(\n      app =>\n        app.contextPath === this.translationAppContextPath &&\n        app.owner.tenant.id === this.appState.currentTenant.value.name\n    );\n    if (ownApp) {\n      return ownApp;\n    }\n\n    const { data: app } = await this.appService.create({\n      name: this.translationAppName,\n      contextPath: this.translationAppContextPath,\n      key: `${this.translationAppContextPath}-app-key`,\n      type: ApplicationType.HOSTED,\n      availability: ApplicationAvailability.MARKET,\n      config: {\n        icon: {\n          class: 'language1'\n        }\n      },\n      noAppSwitcher: true,\n      description: 'Providing user defined translations'\n    });\n    const zip = await this.zip.createZip([]);\n\n    const { data: binary } = await this.appService.binary(app).upload(zip, 'translations.zip');\n    const { data: updatedApp } = await this.appService.update({\n      id: app.id,\n      activeVersionId: binary.id as string\n    });\n    return updatedApp;\n  }\n\n  private async getFileFromTranslationApp(file: string) {\n    return this.getFileFromPath(`/apps/public/${this.translationAppContextPath}/${file}`);\n  }\n\n  private async getFileFromPath(path: string) {\n    const finalPath = `${path}?nocache=${Date.now()}`;\n\n    // using fetch client instead of fetch directly here to ensure loading indicator is shown during these requests\n    const response = await this.fetchClient.fetch(finalPath);\n\n    if (response.status !== 200) {\n      throw new Error(`Failed to fetch file from ${path}`);\n    }\n\n    return response.json();\n  }\n}\n"]}
|
|
@@ -1,10 +1,27 @@
|
|
|
1
|
-
import { hookNavigator, hookRoute } from '@c8y/ngx-components';
|
|
1
|
+
import { hookNavigator, hookRoute, hookTab } from '@c8y/ngx-components';
|
|
2
2
|
import { TranslationEditorNaviagtorFactoryService } from './translation-editor-naviagtor-factory.service';
|
|
3
|
+
import { translationEditorAdvancedPath, translationEditorBasePath, translationEditorBasicPath } from './translation-editor.constants';
|
|
4
|
+
import { TranslationEditorTabFactoryService } from './translation-editor-tab-factory.service';
|
|
3
5
|
export const tranlationEditorProviders = [
|
|
4
6
|
hookRoute({
|
|
5
|
-
path:
|
|
6
|
-
|
|
7
|
+
path: translationEditorBasePath,
|
|
8
|
+
children: [
|
|
9
|
+
{
|
|
10
|
+
path: '',
|
|
11
|
+
pathMatch: 'full',
|
|
12
|
+
redirectTo: translationEditorBasicPath
|
|
13
|
+
},
|
|
14
|
+
{
|
|
15
|
+
path: translationEditorBasicPath,
|
|
16
|
+
loadComponent: () => import('@c8y/ngx-components/translation-editor/lazy').then(m => m.TranslationEditorComponent)
|
|
17
|
+
},
|
|
18
|
+
{
|
|
19
|
+
path: translationEditorAdvancedPath,
|
|
20
|
+
loadComponent: () => import('@c8y/ngx-components/translation-editor/lazy').then(m => m.AdvancedTranslationEditorComponent)
|
|
21
|
+
}
|
|
22
|
+
]
|
|
7
23
|
}),
|
|
8
|
-
hookNavigator(TranslationEditorNaviagtorFactoryService)
|
|
24
|
+
hookNavigator(TranslationEditorNaviagtorFactoryService),
|
|
25
|
+
hookTab(TranslationEditorTabFactoryService)
|
|
9
26
|
];
|
|
10
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cmFuc2xhdGlvbi1lZGl0b3IvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGFBQWEsRUFBRSxTQUFTLEVBQUUsT0FBTyxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDeEUsT0FBTyxFQUFFLHdDQUF3QyxFQUFFLE1BQU0sZ0RBQWdELENBQUM7QUFDMUcsT0FBTyxFQUNMLDZCQUE2QixFQUM3Qix5QkFBeUIsRUFDekIsMEJBQTBCLEVBQzNCLE1BQU0sZ0NBQWdDLENBQUM7QUFDeEMsT0FBTyxFQUFFLGtDQUFrQyxFQUFFLE1BQU0sMENBQTBDLENBQUM7QUFFOUYsTUFBTSxDQUFDLE1BQU0seUJBQXlCLEdBQUc7SUFDdkMsU0FBUyxDQUFDO1FBQ1IsSUFBSSxFQUFFLHlCQUF5QjtRQUMvQixRQUFRLEVBQUU7WUFDUjtnQkFDRSxJQUFJLEVBQUUsRUFBRTtnQkFDUixTQUFTLEVBQUUsTUFBTTtnQkFDakIsVUFBVSxFQUFFLDBCQUEwQjthQUN2QztZQUNEO2dCQUNFLElBQUksRUFBRSwwQkFBMEI7Z0JBQ2hDLGFBQWEsRUFBRSxHQUFHLEVBQUUsQ0FDbEIsTUFBTSxDQUFDLDZDQUE2QyxDQUFDLENBQUMsSUFBSSxDQUN4RCxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQywwQkFBMEIsQ0FDbEM7YUFDSjtZQUNEO2dCQUNFLElBQUksRUFBRSw2QkFBNkI7Z0JBQ25DLGFBQWEsRUFBRSxHQUFHLEVBQUUsQ0FDbEIsTUFBTSxDQUFDLDZDQUE2QyxDQUFDLENBQUMsSUFBSSxDQUN4RCxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxrQ0FBa0MsQ0FDMUM7YUFDSjtTQUNGO0tBQ0YsQ0FBQztJQUNGLGFBQWEsQ0FBQyx3Q0FBd0MsQ0FBQztJQUN2RCxPQUFPLENBQUMsa0NBQWtDLENBQUM7Q0FDNUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGhvb2tOYXZpZ2F0b3IsIGhvb2tSb3V0ZSwgaG9va1RhYiB9IGZyb20gJ0BjOHkvbmd4LWNvbXBvbmVudHMnO1xuaW1wb3J0IHsgVHJhbnNsYXRpb25FZGl0b3JOYXZpYWd0b3JGYWN0b3J5U2VydmljZSB9IGZyb20gJy4vdHJhbnNsYXRpb24tZWRpdG9yLW5hdmlhZ3Rvci1mYWN0b3J5LnNlcnZpY2UnO1xuaW1wb3J0IHtcbiAgdHJhbnNsYXRpb25FZGl0b3JBZHZhbmNlZFBhdGgsXG4gIHRyYW5zbGF0aW9uRWRpdG9yQmFzZVBhdGgsXG4gIHRyYW5zbGF0aW9uRWRpdG9yQmFzaWNQYXRoXG59IGZyb20gJy4vdHJhbnNsYXRpb24tZWRpdG9yLmNvbnN0YW50cyc7XG5pbXBvcnQgeyBUcmFuc2xhdGlvbkVkaXRvclRhYkZhY3RvcnlTZXJ2aWNlIH0gZnJvbSAnLi90cmFuc2xhdGlvbi1lZGl0b3ItdGFiLWZhY3Rvcnkuc2VydmljZSc7XG5cbmV4cG9ydCBjb25zdCB0cmFubGF0aW9uRWRpdG9yUHJvdmlkZXJzID0gW1xuICBob29rUm91dGUoe1xuICAgIHBhdGg6IHRyYW5zbGF0aW9uRWRpdG9yQmFzZVBhdGgsXG4gICAgY2hpbGRyZW46IFtcbiAgICAgIHtcbiAgICAgICAgcGF0aDogJycsXG4gICAgICAgIHBhdGhNYXRjaDogJ2Z1bGwnLFxuICAgICAgICByZWRpcmVjdFRvOiB0cmFuc2xhdGlvbkVkaXRvckJhc2ljUGF0aFxuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgcGF0aDogdHJhbnNsYXRpb25FZGl0b3JCYXNpY1BhdGgsXG4gICAgICAgIGxvYWRDb21wb25lbnQ6ICgpID0+XG4gICAgICAgICAgaW1wb3J0KCdAYzh5L25neC1jb21wb25lbnRzL3RyYW5zbGF0aW9uLWVkaXRvci9sYXp5JykudGhlbihcbiAgICAgICAgICAgIG0gPT4gbS5UcmFuc2xhdGlvbkVkaXRvckNvbXBvbmVudFxuICAgICAgICAgIClcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIHBhdGg6IHRyYW5zbGF0aW9uRWRpdG9yQWR2YW5jZWRQYXRoLFxuICAgICAgICBsb2FkQ29tcG9uZW50OiAoKSA9PlxuICAgICAgICAgIGltcG9ydCgnQGM4eS9uZ3gtY29tcG9uZW50cy90cmFuc2xhdGlvbi1lZGl0b3IvbGF6eScpLnRoZW4oXG4gICAgICAgICAgICBtID0+IG0uQWR2YW5jZWRUcmFuc2xhdGlvbkVkaXRvckNvbXBvbmVudFxuICAgICAgICAgIClcbiAgICAgIH1cbiAgICBdXG4gIH0pLFxuICBob29rTmF2aWdhdG9yKFRyYW5zbGF0aW9uRWRpdG9yTmF2aWFndG9yRmFjdG9yeVNlcnZpY2UpLFxuICBob29rVGFiKFRyYW5zbGF0aW9uRWRpdG9yVGFiRmFjdG9yeVNlcnZpY2UpXG5dO1xuIl19
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
import { NgIf } from '@angular/common';
|
|
2
|
+
import { Component, signal, ViewChild, computed } from '@angular/core';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import { ActionBarItemComponent, AlertService, AppStateService, IconDirective, LoadingComponent, TitleComponent } from '@c8y/ngx-components';
|
|
5
|
+
import { EditorComponent, MonacoEditorMarkerValidatorDirective } from '@c8y/ngx-components/editor';
|
|
6
|
+
import { TranslationStoreService } from '@c8y/ngx-components/translation-editor/data';
|
|
7
|
+
import { gettext } from '@c8y/ngx-components/gettext';
|
|
8
|
+
import { TranslateService } from '@ngx-translate/core';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
import * as i1 from "@c8y/ngx-components";
|
|
11
|
+
import * as i2 from "@c8y/ngx-components/translation-editor/data";
|
|
12
|
+
import * as i3 from "@ngx-translate/core";
|
|
13
|
+
import * as i4 from "@angular/forms";
|
|
14
|
+
export class AdvancedTranslationEditorComponent {
|
|
15
|
+
constructor(appState, translationStore, alert, translation) {
|
|
16
|
+
this.appState = appState;
|
|
17
|
+
this.translationStore = translationStore;
|
|
18
|
+
this.alert = alert;
|
|
19
|
+
this.translation = translation;
|
|
20
|
+
this.valueString = signal('');
|
|
21
|
+
this.editorOptions = {
|
|
22
|
+
language: 'json'
|
|
23
|
+
};
|
|
24
|
+
this.isLoading = signal(false);
|
|
25
|
+
this.isValidJSON = computed(() => {
|
|
26
|
+
const data = this.valueString();
|
|
27
|
+
try {
|
|
28
|
+
JSON.parse(data);
|
|
29
|
+
return true;
|
|
30
|
+
}
|
|
31
|
+
catch (e) {
|
|
32
|
+
return false;
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
this.availableLangs = [];
|
|
36
|
+
this.availableLangs = this.appState.state.langs.sort((a, b) => a.localeCompare(b));
|
|
37
|
+
this.refresh();
|
|
38
|
+
}
|
|
39
|
+
onValueChange(data) {
|
|
40
|
+
this.valueString.set(data);
|
|
41
|
+
}
|
|
42
|
+
onEditorInit(_data) {
|
|
43
|
+
this.editorComponent.monaco.languages.json.jsonDefaults.setDiagnosticsOptions({
|
|
44
|
+
validate: true,
|
|
45
|
+
schemas: [{ schema: this.JSONSchema, fileMatch: ['*'], uri: 'translation' }],
|
|
46
|
+
enableSchemaRequest: false,
|
|
47
|
+
allowComments: false
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
async refresh() {
|
|
51
|
+
this.isLoading.set(true);
|
|
52
|
+
const translationProperties = this.availableLangs.reduceRight((acc, lang) => {
|
|
53
|
+
acc[lang] = {
|
|
54
|
+
type: 'string',
|
|
55
|
+
description: this.translation.instant(gettext('Translation for "{{ languageCode }}" language.'), { languageCode: lang })
|
|
56
|
+
};
|
|
57
|
+
return acc;
|
|
58
|
+
}, {
|
|
59
|
+
key: {
|
|
60
|
+
type: 'string',
|
|
61
|
+
description: this.translation.instant(gettext('The key to be translated.'))
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
this.JSONSchema = {
|
|
65
|
+
$schema: 'http://json-schema.org/draft-04/schema#',
|
|
66
|
+
type: 'array',
|
|
67
|
+
items: {
|
|
68
|
+
type: 'object',
|
|
69
|
+
properties: translationProperties,
|
|
70
|
+
required: ['key'],
|
|
71
|
+
minProperties: 2
|
|
72
|
+
}
|
|
73
|
+
};
|
|
74
|
+
const translations = await this.translationStore.getCombinedListOfTranslationsForPerKey(this.availableLangs);
|
|
75
|
+
this.valueString.set(JSON.stringify(translations, undefined, 2));
|
|
76
|
+
this.isLoading.set(false);
|
|
77
|
+
}
|
|
78
|
+
async importTranslationKeys() {
|
|
79
|
+
try {
|
|
80
|
+
this.alert.info(gettext('Importing translation from available Web SDK based applications. This may take a while...'));
|
|
81
|
+
// since we are only interested in the keys, we only need to retrieve the translations for one (well supported) language
|
|
82
|
+
// we assume that the keys are the same for all other languages, otherwise we would need to perform a lot more requests.
|
|
83
|
+
const data = await this.translationStore.getAvailableTranslations(['de']);
|
|
84
|
+
const valueString = this.valueString();
|
|
85
|
+
const values = JSON.parse(valueString);
|
|
86
|
+
const previousNumberOfEntries = values.length;
|
|
87
|
+
const retrievedKeys = Object.keys(data);
|
|
88
|
+
for (const key of retrievedKeys) {
|
|
89
|
+
const foundKey = values.find(value => value.key === key);
|
|
90
|
+
if (!foundKey) {
|
|
91
|
+
values.push({ key });
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
const newpreviousNumberOfEntries = values.length;
|
|
95
|
+
this.valueString.set(JSON.stringify(values, undefined, 2));
|
|
96
|
+
this.alert.success(this.translation.instant(gettext('Retrieved {{ numberOfRetrievedKeys }} keys from available applications. {{ numberOfNewKeys }} new translation keys were added.'), {
|
|
97
|
+
numberOfRetrievedKeys: retrievedKeys.length,
|
|
98
|
+
numberOfNewKeys: newpreviousNumberOfEntries - previousNumberOfEntries
|
|
99
|
+
}));
|
|
100
|
+
}
|
|
101
|
+
catch (e) {
|
|
102
|
+
this.alert.danger(gettext('Failed to import translation keys'));
|
|
103
|
+
this.alert.addServerFailure(e);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
async saveTranslations() {
|
|
107
|
+
try {
|
|
108
|
+
await this.translationStore.updateTranslations(JSON.parse(this.valueString()));
|
|
109
|
+
await this.refresh();
|
|
110
|
+
this.alert.success(gettext('Translations saved'));
|
|
111
|
+
}
|
|
112
|
+
catch (e) {
|
|
113
|
+
this.alert.danger(gettext('Failed to save translations'));
|
|
114
|
+
this.alert.addServerFailure(e);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AdvancedTranslationEditorComponent, deps: [{ token: i1.AppStateService }, { token: i2.TranslationStoreService }, { token: i1.AlertService }, { token: i3.TranslateService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
118
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: AdvancedTranslationEditorComponent, isStandalone: true, selector: "c8y-advanced-translation-editor", viewQueries: [{ propertyName: "editorComponent", first: true, predicate: EditorComponent, descendants: true }], ngImport: i0, template: "<c8y-title translate>Localization</c8y-title>\n\n<ng-container *ngIf=\"!isLoading(); else loading\">\n <c8y-action-bar-item placement=\"right\">\n <button\n class=\"btn btn-link\"\n [disabled]=\"!isValidJSON()\"\n (click)=\"saveTranslations()\"\n data-cy=\"c8y-translation-editor--save-and-apply\"\n >\n <i c8yIcon=\"save\"></i>\n <span translate>Save & apply</span>\n </button>\n </c8y-action-bar-item>\n\n <c8y-action-bar-item placement=\"right\">\n <button\n class=\"btn btn-link\"\n [disabled]=\"!isValidJSON()\"\n (click)=\"importTranslationKeys()\"\n data-cy=\"c8y-translation-editor--import-translation-keys\"\n >\n <i [c8yIcon]=\"'import'\"></i>\n <span translate>Import translation keys from Apps</span>\n </button>\n </c8y-action-bar-item>\n\n <div class=\"card card--fullpage content-fullpage\">\n <c8y-editor\n class=\"flex-grow\"\n [editorOptions]=\"editorOptions\"\n [ngModel]=\"valueString()\"\n (ngModelChange)=\"onValueChange($event)\"\n (editorInit)=\"onEditorInit($event)\"\n monacoEditorMarkerValidator\n ></c8y-editor>\n </div>\n</ng-container>\n\n<ng-template #loading>\n <c8y-loading></c8y-loading>\n</ng-template>", dependencies: [{ kind: "component", type: EditorComponent, selector: "c8y-editor", inputs: ["editorOptions"], outputs: ["editorInit"] }, { kind: "directive", type: MonacoEditorMarkerValidatorDirective, selector: "c8y-editor [monacoEditorMarkerValidator]" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: TitleComponent, selector: "c8y-title", inputs: ["pageTitleUpdate"] }, { kind: "component", type: LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ActionBarItemComponent, selector: "c8y-action-bar-item", inputs: ["placement", "priority", "itemClass", "injector", "groupId", "inGroupPriority"] }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }] }); }
|
|
119
|
+
}
|
|
120
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AdvancedTranslationEditorComponent, decorators: [{
|
|
121
|
+
type: Component,
|
|
122
|
+
args: [{ selector: 'c8y-advanced-translation-editor', standalone: true, imports: [
|
|
123
|
+
EditorComponent,
|
|
124
|
+
MonacoEditorMarkerValidatorDirective,
|
|
125
|
+
FormsModule,
|
|
126
|
+
TitleComponent,
|
|
127
|
+
LoadingComponent,
|
|
128
|
+
NgIf,
|
|
129
|
+
ActionBarItemComponent,
|
|
130
|
+
IconDirective
|
|
131
|
+
], template: "<c8y-title translate>Localization</c8y-title>\n\n<ng-container *ngIf=\"!isLoading(); else loading\">\n <c8y-action-bar-item placement=\"right\">\n <button\n class=\"btn btn-link\"\n [disabled]=\"!isValidJSON()\"\n (click)=\"saveTranslations()\"\n data-cy=\"c8y-translation-editor--save-and-apply\"\n >\n <i c8yIcon=\"save\"></i>\n <span translate>Save & apply</span>\n </button>\n </c8y-action-bar-item>\n\n <c8y-action-bar-item placement=\"right\">\n <button\n class=\"btn btn-link\"\n [disabled]=\"!isValidJSON()\"\n (click)=\"importTranslationKeys()\"\n data-cy=\"c8y-translation-editor--import-translation-keys\"\n >\n <i [c8yIcon]=\"'import'\"></i>\n <span translate>Import translation keys from Apps</span>\n </button>\n </c8y-action-bar-item>\n\n <div class=\"card card--fullpage content-fullpage\">\n <c8y-editor\n class=\"flex-grow\"\n [editorOptions]=\"editorOptions\"\n [ngModel]=\"valueString()\"\n (ngModelChange)=\"onValueChange($event)\"\n (editorInit)=\"onEditorInit($event)\"\n monacoEditorMarkerValidator\n ></c8y-editor>\n </div>\n</ng-container>\n\n<ng-template #loading>\n <c8y-loading></c8y-loading>\n</ng-template>" }]
|
|
132
|
+
}], ctorParameters: () => [{ type: i1.AppStateService }, { type: i2.TranslationStoreService }, { type: i1.AlertService }, { type: i3.TranslateService }], propDecorators: { editorComponent: [{
|
|
133
|
+
type: ViewChild,
|
|
134
|
+
args: [EditorComponent]
|
|
135
|
+
}] } });
|
|
136
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"advanced-translation-editor.component.js","sourceRoot":"","sources":["../../../../../translation-editor/lazy/advanced-translation-editor/advanced-translation-editor.component.ts","../../../../../translation-editor/lazy/advanced-translation-editor/advanced-translation-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgIf } from '@angular/common';\nimport { Component, signal, ViewChild, computed } from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport {\n  ActionBarItemComponent,\n  AlertService,\n  AppStateService,\n  IconDirective,\n  LoadingComponent,\n  TitleComponent\n} from '@c8y/ngx-components';\nimport { EditorComponent, MonacoEditorMarkerValidatorDirective } from '@c8y/ngx-components/editor';\nimport { TranslationStoreService } from '@c8y/ngx-components/translation-editor/data';\nimport { gettext } from '@c8y/ngx-components/gettext';\nimport type * as Monaco from 'monaco-editor';\nimport { TranslateService } from '@ngx-translate/core';\n\n@Component({\n  selector: 'c8y-advanced-translation-editor',\n  templateUrl: './advanced-translation-editor.component.html',\n  standalone: true,\n  imports: [\n    EditorComponent,\n    MonacoEditorMarkerValidatorDirective,\n    FormsModule,\n    TitleComponent,\n    LoadingComponent,\n    NgIf,\n    ActionBarItemComponent,\n    IconDirective\n  ]\n})\nexport class AdvancedTranslationEditorComponent {\n  valueString = signal('');\n  editorOptions: EditorComponent['editorOptions'] = {\n    language: 'json'\n  };\n  isLoading = signal(false);\n  isValidJSON = computed(() => {\n    const data = this.valueString();\n    try {\n      JSON.parse(data);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  });\n  JSONSchema: any;\n  @ViewChild(EditorComponent) editorComponent!: EditorComponent;\n  availableLangs: string[] = [];\n\n  constructor(\n    private appState: AppStateService,\n    private translationStore: TranslationStoreService,\n    private alert: AlertService,\n    private translation: TranslateService\n  ) {\n    this.availableLangs = (this.appState.state.langs as string[]).sort((a, b) =>\n      a.localeCompare(b)\n    );\n    this.refresh();\n  }\n\n  onValueChange(data: string) {\n    this.valueString.set(data);\n  }\n\n  onEditorInit(_data: Monaco.editor.IStandaloneCodeEditor) {\n    this.editorComponent.monaco.languages.json.jsonDefaults.setDiagnosticsOptions({\n      validate: true,\n      schemas: [{ schema: this.JSONSchema, fileMatch: ['*'], uri: 'translation' }],\n      enableSchemaRequest: false,\n      allowComments: false\n    });\n  }\n\n  async refresh() {\n    this.isLoading.set(true);\n\n    const translationProperties: { key: { type: 'string' }; [key: string]: { type: 'string' } } =\n      this.availableLangs.reduceRight(\n        (acc, lang) => {\n          acc[lang] = {\n            type: 'string',\n            description: this.translation.instant(\n              gettext('Translation for \"{{ languageCode }}\" language.'),\n              { languageCode: lang }\n            )\n          };\n          return acc;\n        },\n        {\n          key: {\n            type: 'string',\n            description: this.translation.instant(gettext('The key to be translated.'))\n          }\n        }\n      );\n    this.JSONSchema = {\n      $schema: 'http://json-schema.org/draft-04/schema#',\n\n      type: 'array',\n      items: {\n        type: 'object',\n        properties: translationProperties,\n        required: ['key'],\n        minProperties: 2\n      }\n    };\n\n    const translations = await this.translationStore.getCombinedListOfTranslationsForPerKey(\n      this.availableLangs\n    );\n\n    this.valueString.set(JSON.stringify(translations, undefined, 2));\n\n    this.isLoading.set(false);\n  }\n\n  async importTranslationKeys() {\n    try {\n      this.alert.info(\n        gettext(\n          'Importing translation from available Web SDK based applications. This may take a while...'\n        )\n      );\n\n      // since we are only interested in the keys, we only need to retrieve the translations for one (well supported) language\n      // we assume that the keys are the same for all other languages, otherwise we would need to perform a lot more requests.\n      const data = await this.translationStore.getAvailableTranslations(['de']);\n      const valueString = this.valueString();\n      const values: Array<{ key: string }> = JSON.parse(valueString);\n      const previousNumberOfEntries = values.length;\n      const retrievedKeys = Object.keys(data);\n      for (const key of retrievedKeys) {\n        const foundKey = values.find(value => value.key === key);\n        if (!foundKey) {\n          values.push({ key });\n        }\n      }\n      const newpreviousNumberOfEntries = values.length;\n      this.valueString.set(JSON.stringify(values, undefined, 2));\n      this.alert.success(\n        this.translation.instant(\n          gettext(\n            'Retrieved {{ numberOfRetrievedKeys }} keys from available applications. {{ numberOfNewKeys }} new translation keys were added.'\n          ),\n          {\n            numberOfRetrievedKeys: retrievedKeys.length,\n            numberOfNewKeys: newpreviousNumberOfEntries - previousNumberOfEntries\n          }\n        )\n      );\n    } catch (e) {\n      this.alert.danger(gettext('Failed to import translation keys'));\n      this.alert.addServerFailure(e);\n    }\n  }\n\n  async saveTranslations() {\n    try {\n      await this.translationStore.updateTranslations(JSON.parse(this.valueString()));\n      await this.refresh();\n      this.alert.success(gettext('Translations saved'));\n    } catch (e) {\n      this.alert.danger(gettext('Failed to save translations'));\n      this.alert.addServerFailure(e);\n    }\n  }\n}\n","<c8y-title translate>Localization</c8y-title>\n\n<ng-container *ngIf=\"!isLoading(); else loading\">\n  <c8y-action-bar-item placement=\"right\">\n    <button\n      class=\"btn btn-link\"\n      [disabled]=\"!isValidJSON()\"\n      (click)=\"saveTranslations()\"\n      data-cy=\"c8y-translation-editor--save-and-apply\"\n    >\n      <i c8yIcon=\"save\"></i>\n      <span translate>Save & apply</span>\n    </button>\n  </c8y-action-bar-item>\n\n  <c8y-action-bar-item placement=\"right\">\n    <button\n      class=\"btn btn-link\"\n      [disabled]=\"!isValidJSON()\"\n      (click)=\"importTranslationKeys()\"\n      data-cy=\"c8y-translation-editor--import-translation-keys\"\n    >\n      <i [c8yIcon]=\"'import'\"></i>\n      <span translate>Import translation keys from Apps</span>\n    </button>\n  </c8y-action-bar-item>\n\n  <div class=\"card card--fullpage content-fullpage\">\n    <c8y-editor\n      class=\"flex-grow\"\n      [editorOptions]=\"editorOptions\"\n      [ngModel]=\"valueString()\"\n      (ngModelChange)=\"onValueChange($event)\"\n      (editorInit)=\"onEditorInit($event)\"\n      monacoEditorMarkerValidator\n    ></c8y-editor>\n  </div>\n</ng-container>\n\n<ng-template #loading>\n  <c8y-loading></c8y-loading>\n</ng-template>"]}
|
|
@@ -1,2 +1,3 @@
|
|
|
1
1
|
export * from './translation-editor/translation-editor.component';
|
|
2
|
-
|
|
2
|
+
export * from './advanced-translation-editor/advanced-translation-editor.component';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi90cmFuc2xhdGlvbi1lZGl0b3IvbGF6eS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG1EQUFtRCxDQUFDO0FBQ2xFLGNBQWMscUVBQXFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3RyYW5zbGF0aW9uLWVkaXRvci90cmFuc2xhdGlvbi1lZGl0b3IuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vYWR2YW5jZWQtdHJhbnNsYXRpb24tZWRpdG9yL2FkdmFuY2VkLXRyYW5zbGF0aW9uLWVkaXRvci5jb21wb25lbnQnO1xuIl19
|