@yuuvis/client-core 2.0.0-beta.0 → 2.0.0-beta.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/fesm2022/yuuvis-client-core.mjs +773 -645
- package/fesm2022/yuuvis-client-core.mjs.map +1 -1
- package/index.d.ts +7 -4
- package/lib/model/object-flavor.interface.d.ts +4 -1
- package/lib/provide.client.core.d.ts +58 -0
- package/lib/service/backend/backend.service.d.ts +0 -1
- package/lib/service/config/config.service.d.ts +0 -8
- package/lib/service/core-init/core-init.service.d.ts +1 -7
- package/lib/service/core-init/translate-json-loader.d.ts +1 -1
- package/lib/service/dms/dms.service.d.ts +2 -1
- package/lib/service/object-config/object-config.interface.d.ts +1 -1
- package/lib/service/object-form/object-form-translate.service.d.ts +15 -0
- package/lib/service/{system → object-form}/object-form.interface.d.ts +2 -2
- package/lib/service/object-form/object-form.model.d.ts +45 -0
- package/lib/service/system/system.interface.d.ts +1 -1
- package/lib/service/system/system.service.d.ts +1 -1
- package/lib/service/toast/toast.interface.d.ts +8 -0
- package/lib/service/toast/toast.service.d.ts +17 -0
- package/lib/service/toast/toast.styles.d.ts +1 -0
- package/lib/service/upload/upload.interface.d.ts +1 -1
- package/lib/service/user/user.providers.d.ts +4 -0
- package/lib/service/user/user.service.d.ts +2 -4
- package/package.json +7 -10
- package/esm2022/index.mjs +0 -69
- package/esm2022/lib/client-core.module.mjs +0 -117
- package/esm2022/lib/client-core.shared.module.mjs +0 -18
- package/esm2022/lib/common/pipes/filesize.pipe.mjs +0 -37
- package/esm2022/lib/common/pipes/index.mjs +0 -6
- package/esm2022/lib/common/pipes/keys.pipe.mjs +0 -17
- package/esm2022/lib/common/pipes/locale-date.pipe.mjs +0 -45
- package/esm2022/lib/common/pipes/locale-number.pipe.mjs +0 -115
- package/esm2022/lib/common/pipes/safe-html.pipe.mjs +0 -44
- package/esm2022/lib/common/services/native-notifications.interface.mjs +0 -2
- package/esm2022/lib/common/services/native-notifications.mjs +0 -40
- package/esm2022/lib/model/dms-object.interface.mjs +0 -2
- package/esm2022/lib/model/dms-object.model.mjs +0 -79
- package/esm2022/lib/model/object-flavor.interface.mjs +0 -2
- package/esm2022/lib/model/range-value.interface.mjs +0 -2
- package/esm2022/lib/model/yuv-error.model.mjs +0 -37
- package/esm2022/lib/model/yuv-user.model.mjs +0 -38
- package/esm2022/lib/service/audit/audit.interface.mjs +0 -2
- package/esm2022/lib/service/audit/audit.service.mjs +0 -135
- package/esm2022/lib/service/auth/auth.interceptor.mjs +0 -50
- package/esm2022/lib/service/auth/auth.interface.mjs +0 -7
- package/esm2022/lib/service/auth/auth.service.mjs +0 -130
- package/esm2022/lib/service/auth/oidc.service.mjs +0 -81
- package/esm2022/lib/service/backend/api.enum.mjs +0 -11
- package/esm2022/lib/service/backend/backend.interface.mjs +0 -2
- package/esm2022/lib/service/backend/backend.service.mjs +0 -206
- package/esm2022/lib/service/bpm/bpm.interface.mjs +0 -8
- package/esm2022/lib/service/bpm/bpm.service.mjs +0 -51
- package/esm2022/lib/service/cache/app-cache.service.mjs +0 -51
- package/esm2022/lib/service/catalog/catalog.interface.mjs +0 -2
- package/esm2022/lib/service/catalog/catalog.service.mjs +0 -13
- package/esm2022/lib/service/clipboard/clipboard.interface.mjs +0 -2
- package/esm2022/lib/service/clipboard/clipboard.service.mjs +0 -99
- package/esm2022/lib/service/config/config.interface.mjs +0 -6
- package/esm2022/lib/service/config/config.service.mjs +0 -115
- package/esm2022/lib/service/config/core-config.mjs +0 -20
- package/esm2022/lib/service/config/core-config.tokens.mjs +0 -9
- package/esm2022/lib/service/connection/connection.service.mjs +0 -36
- package/esm2022/lib/service/connection/offline.interceptor.mjs +0 -28
- package/esm2022/lib/service/core-init/core-init.service.mjs +0 -29
- package/esm2022/lib/service/core-init/missing-translation-handler.mjs +0 -10
- package/esm2022/lib/service/core-init/translate-json-loader.mjs +0 -117
- package/esm2022/lib/service/device/device.interface.mjs +0 -6
- package/esm2022/lib/service/device/device.service.mjs +0 -144
- package/esm2022/lib/service/dms/dms.service.interface.mjs +0 -2
- package/esm2022/lib/service/dms/dms.service.mjs +0 -440
- package/esm2022/lib/service/event/event.interface.mjs +0 -2
- package/esm2022/lib/service/event/event.service.mjs +0 -38
- package/esm2022/lib/service/event/events.mjs +0 -14
- package/esm2022/lib/service/idm/idm.interface.mjs +0 -2
- package/esm2022/lib/service/idm/idm.service.mjs +0 -34
- package/esm2022/lib/service/logger/logger-console.service.mjs +0 -73
- package/esm2022/lib/service/logger/logger.interface.mjs +0 -2
- package/esm2022/lib/service/logger/logger.mjs +0 -27
- package/esm2022/lib/service/notification/notification.service.mjs +0 -131
- package/esm2022/lib/service/object-config/object-config.interface.mjs +0 -2
- package/esm2022/lib/service/object-config/object-config.service.mjs +0 -233
- package/esm2022/lib/service/pending-changes/pending-changes-component.interface.mjs +0 -5
- package/esm2022/lib/service/pending-changes/pending-changes-guard.service.mjs +0 -25
- package/esm2022/lib/service/pending-changes/pending-changes.service.mjs +0 -123
- package/esm2022/lib/service/prediction/prediction.interface.mjs +0 -2
- package/esm2022/lib/service/prediction/prediction.service.mjs +0 -60
- package/esm2022/lib/service/retention/retention.interface.mjs +0 -2
- package/esm2022/lib/service/retention/retention.service.mjs +0 -65
- package/esm2022/lib/service/search/search.service.interface.mjs +0 -39
- package/esm2022/lib/service/search/search.service.mjs +0 -261
- package/esm2022/lib/service/search/search.utils.mjs +0 -142
- package/esm2022/lib/service/session-storage/session-storage.service.mjs +0 -50
- package/esm2022/lib/service/system/object-form.interface.mjs +0 -2
- package/esm2022/lib/service/system/system.enum.mjs +0 -167
- package/esm2022/lib/service/system/system.interface.mjs +0 -2
- package/esm2022/lib/service/system/system.service.mjs +0 -597
- package/esm2022/lib/service/upload/upload.interface.mjs +0 -2
- package/esm2022/lib/service/upload/upload.service.mjs +0 -228
- package/esm2022/lib/service/user/user-storage.service.mjs +0 -38
- package/esm2022/lib/service/user/user.service.mjs +0 -211
- package/esm2022/lib/util/utils.helper.enum.mjs +0 -15
- package/esm2022/lib/util/utils.mjs +0 -373
- package/esm2022/yuuvis-client-core.mjs +0 -5
- package/lib/client-core.module.d.ts +0 -20
- package/lib/client-core.shared.module.d.ts +0 -10
- package/lib/service/auth/oidc.service.d.ts +0 -16
|
@@ -1,597 +0,0 @@
|
|
|
1
|
-
import { inject, Injectable } from '@angular/core';
|
|
2
|
-
import { forkJoin, of, ReplaySubject } from 'rxjs';
|
|
3
|
-
import { catchError, map, switchMap, tap } from 'rxjs/operators';
|
|
4
|
-
import { Utils } from '../../util/utils';
|
|
5
|
-
import { ApiBase } from '../backend/api.enum';
|
|
6
|
-
import { BackendService } from '../backend/backend.service';
|
|
7
|
-
import { AppCacheService } from '../cache/app-cache.service';
|
|
8
|
-
import { Logger } from '../logger/logger';
|
|
9
|
-
import { BaseObjectTypeField, Classification, ContentStreamAllowed, InternalFieldType, ObjectTypeClassification, SystemType } from './system.enum';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
/**
|
|
12
|
-
* Providing system definitions.
|
|
13
|
-
*/
|
|
14
|
-
export class SystemService {
|
|
15
|
-
constructor() {
|
|
16
|
-
this.#backend = inject(BackendService);
|
|
17
|
-
this.#appCache = inject(AppCacheService);
|
|
18
|
-
this.#logger = inject(Logger);
|
|
19
|
-
this.#STORAGE_KEY = 'yuv.core.system.definition';
|
|
20
|
-
this.#STORAGE_KEY_AUTH_DATA = 'yuv.core.auth.data';
|
|
21
|
-
// cached icons to avoid backend calls (session cache)
|
|
22
|
-
this.#iconCache = {};
|
|
23
|
-
this.#resolvedClassificationsCache = {};
|
|
24
|
-
this.#systemSource = new ReplaySubject();
|
|
25
|
-
this.system$ = this.#systemSource.asObservable();
|
|
26
|
-
// cache for resolved visible tags because they are used in lists and therefore should not be re-evaluated all the time
|
|
27
|
-
this.#visibleTagsCache = {};
|
|
28
|
-
}
|
|
29
|
-
#backend;
|
|
30
|
-
#appCache;
|
|
31
|
-
#logger;
|
|
32
|
-
#STORAGE_KEY;
|
|
33
|
-
#STORAGE_KEY_AUTH_DATA;
|
|
34
|
-
// cached icons to avoid backend calls (session cache)
|
|
35
|
-
#iconCache;
|
|
36
|
-
#resolvedClassificationsCache;
|
|
37
|
-
#systemSource;
|
|
38
|
-
// cache for resolved visible tags because they are used in lists and therefore should not be re-evaluated all the time
|
|
39
|
-
#visibleTagsCache;
|
|
40
|
-
#permissions;
|
|
41
|
-
/**
|
|
42
|
-
* Get all object types
|
|
43
|
-
* @param withLabels Whether or not to also add the types labels
|
|
44
|
-
*/
|
|
45
|
-
getObjectTypes(withLabels, situation) {
|
|
46
|
-
// Filter by user permissions based on situation
|
|
47
|
-
const objectTypes = this.#filterByPermissions([...this.system.objectTypes, ...this.system.secondaryObjectTypes.map((sot) => this.#sotToGenericType(sot))], situation);
|
|
48
|
-
return withLabels ? objectTypes.map((t) => ({ ...t, label: this.getLocalizedResource(`${t.id}_label`) })) : objectTypes;
|
|
49
|
-
}
|
|
50
|
-
#sotToGenericType(sot) {
|
|
51
|
-
return {
|
|
52
|
-
...sot,
|
|
53
|
-
isFolder: false,
|
|
54
|
-
creatable: true,
|
|
55
|
-
secondaryObjectTypes: [],
|
|
56
|
-
isSot: true
|
|
57
|
-
};
|
|
58
|
-
}
|
|
59
|
-
/**
|
|
60
|
-
* Get all secondary object types
|
|
61
|
-
* @param withLabels Whether or not to also add the types labels
|
|
62
|
-
*/
|
|
63
|
-
getSecondaryObjectTypes(withLabels, situation) {
|
|
64
|
-
const sots = this.#filterByPermissions(this.system.secondaryObjectTypes.map((sot) => this.#sotToGenericType(sot)), situation);
|
|
65
|
-
return ((withLabels ? sots.map((t) => ({ ...t, label: this.getLocalizedResource(`${t.id}_label`) })) : sots)
|
|
66
|
-
// ignore
|
|
67
|
-
.filter((t) => t.id !== t.baseId && !t.id.startsWith('system:') && t.id !== 'appClientsystem:leadingType'));
|
|
68
|
-
}
|
|
69
|
-
#filterByPermissions(types, situation) {
|
|
70
|
-
if (!situation)
|
|
71
|
-
return types;
|
|
72
|
-
const allowedTypes = situation === 'search' ? this.#permissions.searchableObjectTypes : this.#permissions.createableObjectTypes;
|
|
73
|
-
return types.filter((t) => allowedTypes.includes(t.id));
|
|
74
|
-
}
|
|
75
|
-
/**
|
|
76
|
-
* Get a particular object type
|
|
77
|
-
* @param objectTypeId ID of the object type
|
|
78
|
-
* @param withLabel Whether or not to also add the types label
|
|
79
|
-
*/
|
|
80
|
-
getObjectType(objectTypeId, withLabel) {
|
|
81
|
-
let objectType = objectTypeId === SystemType.OBJECT ? this.getBaseType() : this.system.objectTypes.find((ot) => ot.id === objectTypeId);
|
|
82
|
-
if (objectType && withLabel) {
|
|
83
|
-
objectType.label = this.getLocalizedResource(`${objectType.id}_label`) || objectTypeId;
|
|
84
|
-
}
|
|
85
|
-
if (!objectType) {
|
|
86
|
-
// no 'real' object type found. Try to find a matching SOT and treat it like a real type
|
|
87
|
-
// by filling up the missing properties
|
|
88
|
-
const sot = this.getSecondaryObjectType(objectTypeId, withLabel);
|
|
89
|
-
if (sot) {
|
|
90
|
-
objectType = this.#sotToGenericType(sot);
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
return objectType;
|
|
94
|
-
}
|
|
95
|
-
/**
|
|
96
|
-
* Get a particular secondary object type
|
|
97
|
-
* @param objectTypeId ID of the object type
|
|
98
|
-
* @param withLabel Whether or not to also add the types label
|
|
99
|
-
*/
|
|
100
|
-
getSecondaryObjectType(objectTypeId, withLabel) {
|
|
101
|
-
const objectType = this.system.secondaryObjectTypes.find((ot) => ot.id === objectTypeId);
|
|
102
|
-
if (objectType && withLabel) {
|
|
103
|
-
objectType.label = this.getLocalizedResource(`${objectType.id}_label`) || objectType.id;
|
|
104
|
-
}
|
|
105
|
-
return objectType;
|
|
106
|
-
}
|
|
107
|
-
/**
|
|
108
|
-
* Get the base document type all documents belong to
|
|
109
|
-
* @param withLabel Whether or not to also add the types label
|
|
110
|
-
*/
|
|
111
|
-
getBaseDocumentType(withLabel) {
|
|
112
|
-
return this.getObjectType(SystemType.DOCUMENT, withLabel);
|
|
113
|
-
}
|
|
114
|
-
/**
|
|
115
|
-
* Get the base folder type all folders belong to
|
|
116
|
-
* @param withLabel Whether or not to also add the types label
|
|
117
|
-
*/
|
|
118
|
-
getBaseFolderType(withLabel) {
|
|
119
|
-
return this.getObjectType(SystemType.FOLDER, withLabel);
|
|
120
|
-
}
|
|
121
|
-
/**
|
|
122
|
-
* Get the base object type all dms objects belong to
|
|
123
|
-
*/
|
|
124
|
-
getBaseType() {
|
|
125
|
-
const sysFolder = this.getBaseFolderType();
|
|
126
|
-
const sysDocument = this.getBaseDocumentType();
|
|
127
|
-
// base type contains only fields that are shared by base document and base folder ...
|
|
128
|
-
const folderTypeFieldIDs = sysFolder.fields.map((f) => f.id);
|
|
129
|
-
const baseTypeFields = sysDocument.fields.filter((f) => folderTypeFieldIDs.includes(f.id));
|
|
130
|
-
return {
|
|
131
|
-
id: SystemType.OBJECT,
|
|
132
|
-
creatable: false,
|
|
133
|
-
isFolder: false,
|
|
134
|
-
secondaryObjectTypes: [],
|
|
135
|
-
fields: baseTypeFields
|
|
136
|
-
// rawFields: baseTypeFields
|
|
137
|
-
};
|
|
138
|
-
}
|
|
139
|
-
/**
|
|
140
|
-
* Get the resolved object type with all fields ( including fields from related secondary types )
|
|
141
|
-
*/
|
|
142
|
-
getResolvedType(objectTypeId) {
|
|
143
|
-
const abstractTypes = Object.values(SystemType);
|
|
144
|
-
if (!objectTypeId || abstractTypes.includes(objectTypeId)) {
|
|
145
|
-
const baseType = this.getBaseType();
|
|
146
|
-
return { id: baseType.id, fields: baseType.fields };
|
|
147
|
-
}
|
|
148
|
-
const ot = this.getObjectType(objectTypeId);
|
|
149
|
-
if (!ot) {
|
|
150
|
-
const sot = this.getSecondaryObjectType(objectTypeId) || { id: objectTypeId, fields: [] };
|
|
151
|
-
const baseType = this.getBaseType();
|
|
152
|
-
return {
|
|
153
|
-
id: sot.id,
|
|
154
|
-
fields: [...sot.fields, ...baseType.fields]
|
|
155
|
-
};
|
|
156
|
-
}
|
|
157
|
-
return {
|
|
158
|
-
id: ot.id,
|
|
159
|
-
fields: ot.fields
|
|
160
|
-
};
|
|
161
|
-
}
|
|
162
|
-
/**
|
|
163
|
-
* Get the resolved object tags
|
|
164
|
-
*/
|
|
165
|
-
getResolvedTags(objectTypeId) {
|
|
166
|
-
const vTags = this.getVisibleTags(objectTypeId);
|
|
167
|
-
return Object.keys(vTags).map((k) => ({
|
|
168
|
-
id: objectTypeId,
|
|
169
|
-
tagName: k,
|
|
170
|
-
tagValues: vTags[k],
|
|
171
|
-
fields: this.getBaseType().fields.filter((f) => f.id === BaseObjectTypeField.TAGS)
|
|
172
|
-
}));
|
|
173
|
-
}
|
|
174
|
-
/**
|
|
175
|
-
* Get a list of classifications for a given object type including the
|
|
176
|
-
* classifications of its static secondary object types
|
|
177
|
-
* @param objectTypeId ID of the object type
|
|
178
|
-
*/
|
|
179
|
-
getResolvedClassifications(objectTypeId) {
|
|
180
|
-
return this.#resolvedClassificationsCache[objectTypeId] || this.#resolveClassifications(objectTypeId);
|
|
181
|
-
}
|
|
182
|
-
#resolveClassifications(objectTypeId) {
|
|
183
|
-
let classifications = [];
|
|
184
|
-
const ot = this.getObjectType(objectTypeId);
|
|
185
|
-
if (ot) {
|
|
186
|
-
classifications = ot.classification || [];
|
|
187
|
-
const staticSOTs = ot.secondaryObjectTypes ? ot.secondaryObjectTypes.filter((sot) => sot.static).map((sot) => sot.id) : [];
|
|
188
|
-
staticSOTs.forEach((id) => {
|
|
189
|
-
const sot = this.getSecondaryObjectType(id);
|
|
190
|
-
classifications = sot?.classification
|
|
191
|
-
? [
|
|
192
|
-
...classifications,
|
|
193
|
-
...sot.classification.filter((c) => {
|
|
194
|
-
// also filter classifications that should not be inherited
|
|
195
|
-
return c !== ObjectTypeClassification.CREATE_FALSE && c !== ObjectTypeClassification.SEARCH_FALSE;
|
|
196
|
-
})
|
|
197
|
-
]
|
|
198
|
-
: classifications;
|
|
199
|
-
});
|
|
200
|
-
this.#resolvedClassificationsCache[objectTypeId] = classifications;
|
|
201
|
-
}
|
|
202
|
-
return classifications;
|
|
203
|
-
}
|
|
204
|
-
/**
|
|
205
|
-
* Visible tags are defined by a classification on the object type (e.g. 'tag[tenkolibri:process,1,2,3]').
|
|
206
|
-
*
|
|
207
|
-
* The example will only return tags with the name 'tenkolibri:process'
|
|
208
|
-
* and values of either 1, 2 or 3. All other tags will be ignored.
|
|
209
|
-
*
|
|
210
|
-
* @param objectTypeId ID of the object type to get the visible tags for
|
|
211
|
-
* @returns object where the property name is the name of the tag and its value are the visible values
|
|
212
|
-
* for that tag (if values is emoty all values are allowed)
|
|
213
|
-
*/
|
|
214
|
-
getVisibleTags(objectTypeId) {
|
|
215
|
-
return this.#visibleTagsCache[objectTypeId] || this.fetchVisibleTags(objectTypeId);
|
|
216
|
-
}
|
|
217
|
-
fetchVisibleTags(objectTypeId) {
|
|
218
|
-
const ot = this.getObjectType(objectTypeId) || this.getSecondaryObjectType(objectTypeId);
|
|
219
|
-
const tagClassifications = this.getResolvedClassifications(objectTypeId).filter((t) => t.startsWith('tag['));
|
|
220
|
-
const parentType = ot && ot.id;
|
|
221
|
-
const to = {};
|
|
222
|
-
(tagClassifications || []).forEach((tag) => {
|
|
223
|
-
const m = tag.match(/\[(.*)\]/i)[1].split(',');
|
|
224
|
-
const tagName = m.splice(0, 1)[0];
|
|
225
|
-
const tagValues = m.map((v) => parseInt(v.trim()));
|
|
226
|
-
to[tagName] = tagValues;
|
|
227
|
-
});
|
|
228
|
-
this.#visibleTagsCache[objectTypeId] = parentType ? { ...this.getVisibleTags(parentType), ...to } : to;
|
|
229
|
-
return this.#visibleTagsCache[objectTypeId];
|
|
230
|
-
}
|
|
231
|
-
filterVisibleTags(objectTypeId, tagsValue) {
|
|
232
|
-
if (!tagsValue)
|
|
233
|
-
return [];
|
|
234
|
-
const vTags = this.getVisibleTags(objectTypeId);
|
|
235
|
-
// Tag value looks like this: [tagName: string, state: number, date: Date, traceId: string]
|
|
236
|
-
return tagsValue.filter((v) => !!vTags[v[0]] && vTags[v[0]].includes(v[1]));
|
|
237
|
-
}
|
|
238
|
-
/**
|
|
239
|
-
* Get the icon for an object type. This will return an SVG as a string.
|
|
240
|
-
* @param objectTypeId ID of the object type
|
|
241
|
-
* @param fallback ID of a fallback icon that should be used if the given object type has no icon yet
|
|
242
|
-
*/
|
|
243
|
-
getObjectTypeIcon(objectTypeId, fallback) {
|
|
244
|
-
if (this.#iconCache[objectTypeId] && this.#iconCache[objectTypeId].icon) {
|
|
245
|
-
return of(this.#iconCache[objectTypeId].icon);
|
|
246
|
-
}
|
|
247
|
-
else {
|
|
248
|
-
const iconUri = this.getObjectTypeIconUri(objectTypeId, fallback);
|
|
249
|
-
return this.#backend.get(iconUri).pipe(tap((icon) => (this.#iconCache[objectTypeId] = { uri: iconUri, icon })));
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
/**
|
|
253
|
-
* Get the URI of an object type icon.
|
|
254
|
-
* @param objectTypeId ID of the object type
|
|
255
|
-
* @param fallback ID of a fallback icon that should be used if the given object type has no icon yet
|
|
256
|
-
*/
|
|
257
|
-
getObjectTypeIconUri(objectTypeId, fallback) {
|
|
258
|
-
if (this.#iconCache[objectTypeId]) {
|
|
259
|
-
return this.#iconCache[objectTypeId].uri;
|
|
260
|
-
}
|
|
261
|
-
else {
|
|
262
|
-
const ci = this.#getIconFromClassification(objectTypeId);
|
|
263
|
-
const fb = this.getFallbackIcon(objectTypeId, fallback);
|
|
264
|
-
const uri = `/resources/icons/${encodeURIComponent(ci || objectTypeId)}${fb ? `?fallback=${encodeURIComponent(fb)}` : ''}`;
|
|
265
|
-
this.#iconCache[objectTypeId] = { uri: `${this.#backend.getApiBase(ApiBase.apiWeb)}${uri}` };
|
|
266
|
-
return this.#iconCache[objectTypeId].uri;
|
|
267
|
-
}
|
|
268
|
-
}
|
|
269
|
-
getFallbackIcon(objectTypeId, fallback) {
|
|
270
|
-
const ot = this.getObjectType(objectTypeId);
|
|
271
|
-
if (ot && !fallback) {
|
|
272
|
-
// add default fallbacks for system:document and system:folder if now other fallback has been provided
|
|
273
|
-
fallback = ot.isFolder ? 'system:folder' : 'system:document';
|
|
274
|
-
// if (this.isFloatingObjectType(ot)) {
|
|
275
|
-
// // types that do not have no object type assigned to them (primary FSOTs)
|
|
276
|
-
// fallback = 'system:dlm';
|
|
277
|
-
// }
|
|
278
|
-
}
|
|
279
|
-
return fallback;
|
|
280
|
-
}
|
|
281
|
-
#getIconFromClassification(objectTypeId) {
|
|
282
|
-
const ce = this.getClassifications(this.getResolvedClassifications(objectTypeId));
|
|
283
|
-
return ce.has(ObjectTypeClassification.OBJECT_TYPE_ICON) ? ce.get(ObjectTypeClassification.OBJECT_TYPE_ICON).options[0] : null;
|
|
284
|
-
}
|
|
285
|
-
getLocalizedResource(key) {
|
|
286
|
-
return this.system.i18n[key];
|
|
287
|
-
}
|
|
288
|
-
getLocalizedLabel(id) {
|
|
289
|
-
return this.getLocalizedResource(`${id}_label`);
|
|
290
|
-
}
|
|
291
|
-
getLocalizedDescription(id) {
|
|
292
|
-
return this.getLocalizedResource(`${id}_description`);
|
|
293
|
-
}
|
|
294
|
-
/**
|
|
295
|
-
* Determine whether or not the given object type field is a system field
|
|
296
|
-
* @param field Object type field to be checked
|
|
297
|
-
*/
|
|
298
|
-
isSystemProperty(field) {
|
|
299
|
-
return field.id.startsWith('system:');
|
|
300
|
-
}
|
|
301
|
-
/**
|
|
302
|
-
* Fetches the backends system definition and updates system$ Observable.
|
|
303
|
-
* Subscribe to the system$ observable instead of calling this function, otherwise you'll trigger fetching the
|
|
304
|
-
* system definition every time.
|
|
305
|
-
*
|
|
306
|
-
* @param user The user to load the system definition for
|
|
307
|
-
*/
|
|
308
|
-
getSystemDefinition(authData) {
|
|
309
|
-
// TODO: Supposed to return 304 if nothing changes
|
|
310
|
-
return this.#fetchSystemDefinition(authData);
|
|
311
|
-
// TODO: remove when 304 is there???
|
|
312
|
-
// // try to fetch system definition from cache first
|
|
313
|
-
// return this.appCache.getItem(this.STORAGE_KEY).pipe(
|
|
314
|
-
// switchMap(res => {
|
|
315
|
-
// if (res) {
|
|
316
|
-
// // check if the system definition from the cache is up to date
|
|
317
|
-
// this.system = res;
|
|
318
|
-
// this.systemSource.next(this.system);
|
|
319
|
-
// return of(true);
|
|
320
|
-
// } else {
|
|
321
|
-
// // nothing cached so far
|
|
322
|
-
// return this.fetchSystemDefinition();
|
|
323
|
-
// }
|
|
324
|
-
// })
|
|
325
|
-
// );
|
|
326
|
-
}
|
|
327
|
-
/**
|
|
328
|
-
* Actually fetch the system definition from the backend.
|
|
329
|
-
* @param user User to fetch definition for
|
|
330
|
-
*/
|
|
331
|
-
#fetchSystemDefinition(authData) {
|
|
332
|
-
return (authData ? of(authData) : this.#appCache.getItem(this.#STORAGE_KEY_AUTH_DATA)).pipe(switchMap((data) => {
|
|
333
|
-
this.updateAuthData(data).subscribe();
|
|
334
|
-
const fetchTasks = [this.#backend.get('/dms/schema/native.json', ApiBase.core), this.#fetchLocalizations()];
|
|
335
|
-
return forkJoin(fetchTasks);
|
|
336
|
-
}), catchError((error) => {
|
|
337
|
-
this.#logger.error('Error fetching recent version of system definition from server.', error);
|
|
338
|
-
this.#systemSource.error('Error fetching recent version of system definition from server.');
|
|
339
|
-
return of(null);
|
|
340
|
-
}), map((data) => {
|
|
341
|
-
if (data?.length) {
|
|
342
|
-
this.setSchema(data[0], data[1]);
|
|
343
|
-
}
|
|
344
|
-
return !!data;
|
|
345
|
-
}));
|
|
346
|
-
}
|
|
347
|
-
setPermissions(p) {
|
|
348
|
-
this.#permissions = p;
|
|
349
|
-
}
|
|
350
|
-
/**
|
|
351
|
-
* Create the schema from the servers schema response
|
|
352
|
-
* @param schemaResponse Response from the backend
|
|
353
|
-
*/
|
|
354
|
-
setSchema(schemaResponse, localizedResource = {}) {
|
|
355
|
-
// prepare a quick access object for the fields
|
|
356
|
-
const propertiesQA = {};
|
|
357
|
-
const orgTypeFields = [BaseObjectTypeField.MODIFIED_BY, BaseObjectTypeField.CREATED_BY];
|
|
358
|
-
schemaResponse.propertyDefinition.forEach((p) => {
|
|
359
|
-
p.classifications = p.classification;
|
|
360
|
-
// TODO: Remove once schema supports organization classification for base params
|
|
361
|
-
// map certain fields to organization type (fake it until you make it ;-)
|
|
362
|
-
if (orgTypeFields.includes(p.id)) {
|
|
363
|
-
p.classifications = [Classification.STRING_ORGANIZATION];
|
|
364
|
-
}
|
|
365
|
-
propertiesQA[p.id] = p;
|
|
366
|
-
});
|
|
367
|
-
// prepare a quick access object for object types (including secondary objects)
|
|
368
|
-
const objectTypesQA = {};
|
|
369
|
-
schemaResponse.typeFolderDefinition.forEach((ot) => {
|
|
370
|
-
objectTypesQA[ot.id] = ot;
|
|
371
|
-
});
|
|
372
|
-
schemaResponse.typeDocumentDefinition.forEach((ot) => {
|
|
373
|
-
objectTypesQA[ot.id] = ot;
|
|
374
|
-
});
|
|
375
|
-
schemaResponse.typeSecondaryDefinition.forEach((sot) => {
|
|
376
|
-
objectTypesQA[sot.id] = sot;
|
|
377
|
-
});
|
|
378
|
-
const objectTypes = [
|
|
379
|
-
// folder types
|
|
380
|
-
...schemaResponse.typeFolderDefinition.map((fd) => ({
|
|
381
|
-
id: fd.id,
|
|
382
|
-
description: fd.description,
|
|
383
|
-
classification: fd.classification,
|
|
384
|
-
baseId: fd.baseId,
|
|
385
|
-
creatable: this.#isCreatable(fd.id),
|
|
386
|
-
contentStreamAllowed: ContentStreamAllowed.NOT_ALLOWED,
|
|
387
|
-
isFolder: true,
|
|
388
|
-
secondaryObjectTypes: fd.secondaryObjectTypeId ? fd.secondaryObjectTypeId.map((t) => ({ id: t.value, static: t.static })) : [],
|
|
389
|
-
fields: this.#resolveObjectTypeFields(fd, propertiesQA, objectTypesQA)
|
|
390
|
-
// rawFields: this.resolveObjectTypeFields(fd, propertiesQA, objectTypesQA, true),
|
|
391
|
-
})),
|
|
392
|
-
// document types
|
|
393
|
-
...schemaResponse.typeDocumentDefinition.map((dd) => ({
|
|
394
|
-
id: dd.id,
|
|
395
|
-
description: dd.description,
|
|
396
|
-
classification: dd.classification,
|
|
397
|
-
baseId: dd.baseId,
|
|
398
|
-
creatable: this.#isCreatable(dd.id),
|
|
399
|
-
contentStreamAllowed: dd.contentStreamAllowed,
|
|
400
|
-
isFolder: false,
|
|
401
|
-
secondaryObjectTypes: dd.secondaryObjectTypeId ? dd.secondaryObjectTypeId.map((t) => ({ id: t.value, static: t.static })) : [],
|
|
402
|
-
fields: this.#resolveObjectTypeFields(dd, propertiesQA, objectTypesQA)
|
|
403
|
-
// rawFields: this.resolveObjectTypeFields(dd, propertiesQA, objectTypesQA, true),
|
|
404
|
-
}))
|
|
405
|
-
];
|
|
406
|
-
const secondaryObjectTypes = schemaResponse.typeSecondaryDefinition.map((std) => ({
|
|
407
|
-
id: std.id,
|
|
408
|
-
description: std.description,
|
|
409
|
-
classification: std.classification,
|
|
410
|
-
contentStreamAllowed: std.contentStreamAllowed,
|
|
411
|
-
baseId: std.baseId,
|
|
412
|
-
// TODO: Could a SOT be a folder too?
|
|
413
|
-
isFolder: false,
|
|
414
|
-
fields: this.#resolveObjectTypeFields(std, propertiesQA, objectTypesQA)
|
|
415
|
-
}));
|
|
416
|
-
this.system = {
|
|
417
|
-
version: schemaResponse.version,
|
|
418
|
-
lastModificationDate: schemaResponse.lastModificationDate,
|
|
419
|
-
objectTypes,
|
|
420
|
-
secondaryObjectTypes,
|
|
421
|
-
i18n: localizedResource,
|
|
422
|
-
allFields: propertiesQA
|
|
423
|
-
};
|
|
424
|
-
this.#appCache.setItem(this.#STORAGE_KEY, this.system).subscribe();
|
|
425
|
-
this.#systemSource.next(this.system);
|
|
426
|
-
}
|
|
427
|
-
/**
|
|
428
|
-
* Resolve all the fields for an object type. This also includes secondary object types and the fields inherited from
|
|
429
|
-
* the base type (... and of course the base type (and its secondary object types) of the base type and so on)
|
|
430
|
-
* @param schemaTypeDefinition object type definition from the native schema
|
|
431
|
-
* @param propertiesQA Quick access object of all properties
|
|
432
|
-
* @param objectTypesQA Quick access object of all object types
|
|
433
|
-
* @param raw If set to 'true' only the properties of the object type itself will be returned (without SOTs)
|
|
434
|
-
*/
|
|
435
|
-
#resolveObjectTypeFields(schemaTypeDefinition, propertiesQA, objectTypesQA) {
|
|
436
|
-
const objectTypeFieldIDs = schemaTypeDefinition.propertyReference.map((pr) => pr.value);
|
|
437
|
-
if (schemaTypeDefinition.secondaryObjectTypeId) {
|
|
438
|
-
schemaTypeDefinition.secondaryObjectTypeId
|
|
439
|
-
.filter((sot) => sot.static)
|
|
440
|
-
.map((sot) => sot.value)
|
|
441
|
-
.forEach((sotID) => objectTypesQA[sotID].propertyReference.forEach((pr) => objectTypeFieldIDs.push(pr.value)));
|
|
442
|
-
}
|
|
443
|
-
let fields = objectTypeFieldIDs.map((id) => ({
|
|
444
|
-
...propertiesQA[id],
|
|
445
|
-
_internalType: this.getInternalFormElementType(propertiesQA[id].propertyType, propertiesQA[id].classifications)
|
|
446
|
-
}));
|
|
447
|
-
// also resolve properties of the base type
|
|
448
|
-
if (schemaTypeDefinition.baseId !== schemaTypeDefinition.id && !!objectTypesQA[schemaTypeDefinition.baseId]) {
|
|
449
|
-
fields = fields.concat(this.#resolveObjectTypeFields(objectTypesQA[schemaTypeDefinition.baseId], propertiesQA, objectTypesQA));
|
|
450
|
-
}
|
|
451
|
-
return fields;
|
|
452
|
-
}
|
|
453
|
-
#isCreatable(objectTypeId) {
|
|
454
|
-
return ![SystemType.FOLDER, SystemType.DOCUMENT].includes(objectTypeId);
|
|
455
|
-
}
|
|
456
|
-
/**
|
|
457
|
-
* Fetch a collection of form models.
|
|
458
|
-
* @param objectTypeIDs Object type IDs to fetch form model for
|
|
459
|
-
* @param situation Form situation
|
|
460
|
-
* @returns Object where the object type id is key and the form model is the value
|
|
461
|
-
*/
|
|
462
|
-
getObjectTypeForms(objectTypeIDs, situation) {
|
|
463
|
-
return forkJoin(objectTypeIDs.map((o) => this.getObjectTypeForm(o, situation).pipe(catchError((e) => of(null)), map((res) => ({
|
|
464
|
-
id: o,
|
|
465
|
-
formModel: res
|
|
466
|
-
}))))).pipe(map((res) => {
|
|
467
|
-
const resMap = {};
|
|
468
|
-
res.filter((r) => this.#formHasElements(r.formModel)).forEach((r) => (resMap[r.id] = r.formModel));
|
|
469
|
-
return resMap;
|
|
470
|
-
}));
|
|
471
|
-
}
|
|
472
|
-
/**
|
|
473
|
-
* Get the form model of an object type.
|
|
474
|
-
*
|
|
475
|
-
* @param objectTypeId ID of the object type to fetch the form for
|
|
476
|
-
* @param situation The form situation to be fetched
|
|
477
|
-
* @returns Form model
|
|
478
|
-
*/
|
|
479
|
-
getObjectTypeForm(objectTypeId, situation) {
|
|
480
|
-
return this.#backend.get(Utils.buildUri(`/dms/forms/${objectTypeId}`, { situation }));
|
|
481
|
-
}
|
|
482
|
-
/**
|
|
483
|
-
* Check whether or not the model has at least one form element. Recursive.
|
|
484
|
-
* @param element Form element to check child elements for
|
|
485
|
-
*/
|
|
486
|
-
#formHasElements(element) {
|
|
487
|
-
let hasElement = false;
|
|
488
|
-
element.elements?.forEach((e) => {
|
|
489
|
-
if (!['o2mGroup', 'o2mGroupStack'].includes(e.type)) {
|
|
490
|
-
hasElement = true;
|
|
491
|
-
}
|
|
492
|
-
else if (!hasElement) {
|
|
493
|
-
hasElement = this.#formHasElements(e);
|
|
494
|
-
}
|
|
495
|
-
});
|
|
496
|
-
return hasElement;
|
|
497
|
-
}
|
|
498
|
-
/**
|
|
499
|
-
* Generates an internal type for a given object type field.
|
|
500
|
-
* Adding this to a form element or object type field enables us to render forms
|
|
501
|
-
* based on object type fields in a more performant way. Otherwise we would
|
|
502
|
-
* have to evaluate the conditions for every form element on every digest cycle.
|
|
503
|
-
* @param type propertyType of the ObjectTypeField
|
|
504
|
-
* @param classifications classifications of the ObjectTypeField
|
|
505
|
-
*/
|
|
506
|
-
getInternalFormElementType(type, classifications) {
|
|
507
|
-
const _classifications = this.getClassifications(classifications || []);
|
|
508
|
-
if (type === 'string' && _classifications.has(Classification.STRING_REFERENCE)) {
|
|
509
|
-
return InternalFieldType.STRING_REFERENCE;
|
|
510
|
-
}
|
|
511
|
-
else if (type === 'string' && _classifications.has(Classification.STRING_ORGANIZATION)) {
|
|
512
|
-
return InternalFieldType.STRING_ORGANIZATION;
|
|
513
|
-
}
|
|
514
|
-
else if (type === 'string' && _classifications.has(Classification.STRING_ORGANIZATION_SET)) {
|
|
515
|
-
return InternalFieldType.STRING_ORGANIZATION_SET;
|
|
516
|
-
}
|
|
517
|
-
else if (type === 'string' && _classifications.has(Classification.STRING_CATALOG)) {
|
|
518
|
-
return InternalFieldType.STRING_CATALOG;
|
|
519
|
-
}
|
|
520
|
-
else if (type === 'boolean' && _classifications.has(Classification.BOOLEAN_SWITCH)) {
|
|
521
|
-
return InternalFieldType.BOOLEAN_SWITCH;
|
|
522
|
-
}
|
|
523
|
-
else if (type === 'string' &&
|
|
524
|
-
(_classifications.has(Classification.STRING_CATALOG_DYNAMIC) || _classifications.has(Classification.STRING_CATALOG_CUSTOM))) {
|
|
525
|
-
return InternalFieldType.STRING_DYNAMIC_CATALOG;
|
|
526
|
-
}
|
|
527
|
-
else {
|
|
528
|
-
// if there are no matching conditions just return the original type
|
|
529
|
-
return type;
|
|
530
|
-
}
|
|
531
|
-
}
|
|
532
|
-
getObjectTypeField(id) {
|
|
533
|
-
const f = this.system?.allFields[id];
|
|
534
|
-
return f ? { ...f, _internalType: this.getInternalFormElementType(f.propertyType, f.classifications) } : undefined;
|
|
535
|
-
}
|
|
536
|
-
/**
|
|
537
|
-
* Extract classifications from object type fields classification
|
|
538
|
-
* string. This string may contain more than one classification entry.
|
|
539
|
-
*
|
|
540
|
-
* Classification is a comma separated string that may contain additional
|
|
541
|
-
* properties related to on classification entry. Example:
|
|
542
|
-
*
|
|
543
|
-
* `id:reference[system:folder], email`
|
|
544
|
-
*
|
|
545
|
-
* @param classifications Object type fields classification property (schema)
|
|
546
|
-
*/
|
|
547
|
-
getClassifications(classifications) {
|
|
548
|
-
const res = new Map();
|
|
549
|
-
if (classifications) {
|
|
550
|
-
classifications.forEach((c) => {
|
|
551
|
-
const matches = c.match(/^([^\[]*)(\[(.*)\])?$/);
|
|
552
|
-
if (matches && matches.length) {
|
|
553
|
-
res.set(matches[1], {
|
|
554
|
-
classification: matches[1],
|
|
555
|
-
options: matches[3] ? matches[3].split(',').map((o) => o.trim()) : []
|
|
556
|
-
});
|
|
557
|
-
}
|
|
558
|
-
});
|
|
559
|
-
}
|
|
560
|
-
return res;
|
|
561
|
-
}
|
|
562
|
-
toFormElement(field) {
|
|
563
|
-
return { ...field, label: this.getLocalizedLabel(field.id), name: field.id, type: field.propertyType };
|
|
564
|
-
}
|
|
565
|
-
updateAuthData(data) {
|
|
566
|
-
this.authData = { ...this.authData, ...data };
|
|
567
|
-
this.#backend.setHeader('Accept-Language', this.authData.language);
|
|
568
|
-
return this.#appCache.setItem(this.#STORAGE_KEY_AUTH_DATA, this.authData);
|
|
569
|
-
}
|
|
570
|
-
updateLocalizations(iso) {
|
|
571
|
-
return this.updateAuthData({ language: iso }).pipe(switchMap(() => this.#fetchLocalizations()), tap((res) => {
|
|
572
|
-
this.system.i18n = res;
|
|
573
|
-
this.#appCache.setItem(this.#STORAGE_KEY, this.system).subscribe();
|
|
574
|
-
this.#systemSource.next(this.system);
|
|
575
|
-
}));
|
|
576
|
-
}
|
|
577
|
-
#fetchLocalizations() {
|
|
578
|
-
return this.#backend.get('/resources/text');
|
|
579
|
-
}
|
|
580
|
-
fetchResources(id) {
|
|
581
|
-
return this.#backend
|
|
582
|
-
.batch([
|
|
583
|
-
{ uri: `/system/resources/${id}`, base: ApiBase.core },
|
|
584
|
-
{ uri: `/admin/resources/${id}`, base: ApiBase.core }
|
|
585
|
-
])
|
|
586
|
-
.pipe(map(([global, tenant]) => ({ global, tenant })));
|
|
587
|
-
}
|
|
588
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SystemService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
589
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SystemService, providedIn: 'root' }); }
|
|
590
|
-
}
|
|
591
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SystemService, decorators: [{
|
|
592
|
-
type: Injectable,
|
|
593
|
-
args: [{
|
|
594
|
-
providedIn: 'root'
|
|
595
|
-
}]
|
|
596
|
-
}] });
|
|
597
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system.service.js","sourceRoot":"","sources":["../../../../../../../../libs/yuuvis/client-core/src/lib/service/system/system.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { forkJoin, Observable, of, ReplaySubject } from 'rxjs';\nimport { catchError, map, switchMap, tap } from 'rxjs/operators';\nimport { Utils } from '../../util/utils';\nimport { ApiBase } from '../backend/api.enum';\nimport { BackendService } from '../backend/backend.service';\nimport { AppCacheService } from '../cache/app-cache.service';\nimport { Logger } from '../logger/logger';\nimport { AuthData } from './../auth/auth.service';\nimport { ObjectTypeFieldInternalType, ObjectTypeFieldType } from './object-form.interface';\nimport { BaseObjectTypeField, Classification, ContentStreamAllowed, InternalFieldType, ObjectTypeClassification, SystemType } from './system.enum';\nimport {\n  ClassificationEntry,\n  GenericObjectType,\n  Localization,\n  ObjectType,\n  ObjectTypeField,\n  ObjectTypePermissions,\n  SchemaResponse,\n  SchemaResponseFieldDefinition,\n  SchemaResponseTypeDefinition,\n  SecondaryObjectType,\n  SystemDefinition\n} from './system.interface';\n\n/**\n * Providing system definitions.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class SystemService {\n  #backend = inject(BackendService);\n  #appCache = inject(AppCacheService);\n  #logger = inject(Logger);\n\n  #STORAGE_KEY = 'yuv.core.system.definition';\n  #STORAGE_KEY_AUTH_DATA = 'yuv.core.auth.data';\n  // cached icons to avoid backend calls (session cache)\n  #iconCache: {\n    [objectTypeId: string]: {\n      uri: string;\n      icon?: string;\n    };\n  } = {};\n  #resolvedClassificationsCache: { [objectTypeId: string]: any } = {};\n\n  system?: SystemDefinition;\n  #systemSource = new ReplaySubject<SystemDefinition>();\n  system$: Observable<SystemDefinition> = this.#systemSource.asObservable();\n\n  // cache for resolved visible tags because they are used in lists and therefore should not be re-evaluated all the time\n  #visibleTagsCache: { [objectId: string]: { [tagName: string]: any[] } } = {};\n  #permissions?: ObjectTypePermissions;\n  authData?: AuthData;\n\n  /**\n   * Get all object types\n   * @param withLabels Whether or not to also add the types labels\n   */\n  getObjectTypes(withLabels?: boolean, situation?: 'search' | 'create'): GenericObjectType[] {\n    // Filter by user permissions based on situation\n    const objectTypes = this.#filterByPermissions(\n      [...this.system!.objectTypes, ...this.system!.secondaryObjectTypes.map((sot) => this.#sotToGenericType(sot))],\n      situation\n    ) as GenericObjectType[];\n    return withLabels ? objectTypes.map((t) => ({ ...t, label: this.getLocalizedResource(`${t.id}_label`) })) : objectTypes;\n  }\n\n  #sotToGenericType(sot: SecondaryObjectType): GenericObjectType {\n    return {\n      ...sot,\n      isFolder: false,\n      creatable: true,\n      secondaryObjectTypes: [],\n      isSot: true\n    };\n  }\n\n  /**\n   * Get all secondary object types\n   * @param withLabels Whether or not to also add the types labels\n   */\n  getSecondaryObjectTypes(withLabels?: boolean, situation?: 'search' | 'create'): SecondaryObjectType[] {\n    const sots = this.#filterByPermissions(\n      this.system!.secondaryObjectTypes.map((sot) => this.#sotToGenericType(sot)),\n      situation\n    ) as SecondaryObjectType[];\n\n    return (\n      (withLabels ? sots.map((t) => ({ ...t, label: this.getLocalizedResource(`${t.id}_label`) })) : sots)\n        // ignore\n        .filter((t) => t.id !== t.baseId && !t.id.startsWith('system:') && t.id !== 'appClientsystem:leadingType')\n    );\n  }\n\n  #filterByPermissions(types: GenericObjectType[], situation?: 'search' | 'create'): GenericObjectType[] {\n    if (!situation) return types;\n    const allowedTypes = situation === 'search' ? this.#permissions!.searchableObjectTypes : this.#permissions!.createableObjectTypes;\n    return types.filter((t) => allowedTypes.includes(t.id));\n  }\n\n  /**\n   * Get a particular object type\n   * @param objectTypeId ID of the object type\n   * @param withLabel Whether or not to also add the types label\n   */\n  getObjectType(objectTypeId: string, withLabel?: boolean): GenericObjectType | undefined {\n    let objectType: GenericObjectType | undefined =\n      objectTypeId === SystemType.OBJECT ? this.getBaseType() : this.system!.objectTypes.find((ot) => ot.id === objectTypeId);\n\n    if (objectType && withLabel) {\n      objectType.label = this.getLocalizedResource(`${objectType.id}_label`) || objectTypeId;\n    }\n\n    if (!objectType) {\n      // no 'real' object type found. Try to find a matching SOT and treat it like a real type\n      // by filling up the missing properties\n      const sot = this.getSecondaryObjectType(objectTypeId, withLabel);\n      if (sot) {\n        objectType = this.#sotToGenericType(sot);\n      }\n    }\n    return objectType;\n  }\n\n  /**\n   * Get a particular secondary object type\n   * @param objectTypeId ID of the object type\n   * @param withLabel Whether or not to also add the types label\n   */\n  getSecondaryObjectType(objectTypeId: string, withLabel?: boolean): SecondaryObjectType | undefined {\n    const objectType: SecondaryObjectType | undefined = this.system!.secondaryObjectTypes.find((ot) => ot.id === objectTypeId);\n    if (objectType && withLabel) {\n      objectType.label = this.getLocalizedResource(`${objectType.id}_label`) || objectType.id;\n    }\n    return objectType;\n  }\n\n  /**\n   * Get the base document type all documents belong to\n   * @param withLabel Whether or not to also add the types label\n   */\n  getBaseDocumentType(withLabel?: boolean): ObjectType {\n    return this.getObjectType(SystemType.DOCUMENT, withLabel)!;\n  }\n\n  /**\n   * Get the base folder type all folders belong to\n   * @param withLabel Whether or not to also add the types label\n   */\n  getBaseFolderType(withLabel?: boolean): ObjectType {\n    return this.getObjectType(SystemType.FOLDER, withLabel)!;\n  }\n\n  /**\n   * Get the base object type all dms objects belong to\n   */\n  getBaseType(): ObjectType {\n    const sysFolder = this.getBaseFolderType();\n    const sysDocument = this.getBaseDocumentType();\n\n    // base type contains only fields that are shared by base document and base folder ...\n    const folderTypeFieldIDs = sysFolder.fields.map((f) => f.id);\n    const baseTypeFields: ObjectTypeField[] = sysDocument.fields.filter((f) => folderTypeFieldIDs.includes(f.id));\n\n    return {\n      id: SystemType.OBJECT,\n      creatable: false,\n      isFolder: false,\n      secondaryObjectTypes: [],\n      fields: baseTypeFields\n      // rawFields: baseTypeFields\n    };\n  }\n\n  /**\n   * Get the resolved object type with all fields ( including fields from related secondary types )\n   */\n  getResolvedType(objectTypeId?: string): { id: string; fields: ObjectTypeField[] } {\n    const abstractTypes = Object.values(SystemType);\n    if (!objectTypeId || abstractTypes.includes(objectTypeId)) {\n      const baseType = this.getBaseType();\n      return { id: baseType.id, fields: baseType.fields };\n    }\n\n    const ot = this.getObjectType(objectTypeId);\n    if (!ot) {\n      const sot = this.getSecondaryObjectType(objectTypeId) || { id: objectTypeId, fields: [] };\n      const baseType = this.getBaseType();\n      return {\n        id: sot.id,\n        fields: [...sot.fields, ...baseType.fields]\n      };\n    }\n\n    return {\n      id: ot.id,\n      fields: ot.fields\n    };\n  }\n\n  /**\n   * Get the resolved object tags\n   */\n  getResolvedTags(objectTypeId: string): { id: string; tagName: string; tagValues: any; fields: ObjectTypeField[] }[] {\n    const vTags = this.getVisibleTags(objectTypeId);\n    return Object.keys(vTags).map((k) => ({\n      id: objectTypeId,\n      tagName: k,\n      tagValues: vTags[k],\n      fields: this.getBaseType().fields.filter((f) => f.id === BaseObjectTypeField.TAGS)\n    }));\n  }\n\n  /**\n   * Get a list of classifications for a given object type including the\n   * classifications of its static secondary object types\n   * @param objectTypeId ID of the object type\n   */\n  getResolvedClassifications(objectTypeId: string): string[] {\n    return this.#resolvedClassificationsCache[objectTypeId] || this.#resolveClassifications(objectTypeId);\n  }\n\n  #resolveClassifications(objectTypeId: string): string[] {\n    let classifications: string[] = [];\n    const ot = this.getObjectType(objectTypeId);\n    if (ot) {\n      classifications = ot.classification || [];\n      const staticSOTs: string[] = ot.secondaryObjectTypes ? ot.secondaryObjectTypes.filter((sot) => sot.static).map((sot) => sot.id) : [];\n      staticSOTs.forEach((id) => {\n        const sot = this.getSecondaryObjectType(id);\n        classifications = sot?.classification\n          ? [\n              ...classifications,\n              ...sot.classification.filter((c) => {\n                // also filter classifications that should not be inherited\n                return c !== ObjectTypeClassification.CREATE_FALSE && c !== ObjectTypeClassification.SEARCH_FALSE;\n              })\n            ]\n          : classifications;\n      });\n      this.#resolvedClassificationsCache[objectTypeId] = classifications;\n    }\n    return classifications;\n  }\n\n  /**\n   * Visible tags are defined by a classification on the object type (e.g. 'tag[tenkolibri:process,1,2,3]').\n   *\n   * The example will only return tags with the name 'tenkolibri:process'\n   * and values of either 1, 2 or 3. All other tags will be ignored.\n   *\n   * @param objectTypeId ID of the object type to get the visible tags for\n   * @returns object where the property name is the name of the tag and its value are the visible values\n   * for that tag (if values is emoty all values are allowed)\n   */\n  getVisibleTags(objectTypeId: string): { [tagName: string]: any[] } {\n    return this.#visibleTagsCache[objectTypeId] || this.fetchVisibleTags(objectTypeId);\n  }\n\n  private fetchVisibleTags(objectTypeId: string): { [tagName: string]: any[] } {\n    const ot = this.getObjectType(objectTypeId) || this.getSecondaryObjectType(objectTypeId);\n    const tagClassifications = this.getResolvedClassifications(objectTypeId).filter((t) => t.startsWith('tag['));\n    const parentType = ot && (ot as ObjectType).id;\n\n    const to: { [tagName: string]: any[] } = {};\n    (tagClassifications || []).forEach((tag) => {\n      const m = tag.match(/\\[(.*)\\]/i)![1].split(',');\n      const tagName = m.splice(0, 1)[0];\n      const tagValues = m.map((v) => parseInt(v.trim()));\n      to[tagName] = tagValues;\n    });\n\n    this.#visibleTagsCache[objectTypeId] = parentType ? { ...this.getVisibleTags(parentType), ...to } : to;\n    return this.#visibleTagsCache[objectTypeId];\n  }\n\n  filterVisibleTags(objectTypeId: string, tagsValue: Array<Array<any>>): Array<Array<any>> {\n    if (!tagsValue) return [];\n    const vTags: { [tagName: string]: any[] } = this.getVisibleTags(objectTypeId);\n    // Tag value looks like this: [tagName: string, state: number, date: Date, traceId: string]\n    return tagsValue.filter((v: any[]) => !!vTags[v[0]] && vTags[v[0]].includes(v[1]));\n  }\n\n  /**\n   * Get the icon for an object type. This will return an SVG as a string.\n   * @param objectTypeId ID of the object type\n   * @param fallback ID of a fallback icon that should be used if the given object type has no icon yet\n   */\n  getObjectTypeIcon(objectTypeId: string, fallback?: string): Observable<string> {\n    if (this.#iconCache[objectTypeId] && this.#iconCache[objectTypeId].icon) {\n      return of(this.#iconCache[objectTypeId].icon!);\n    } else {\n      const iconUri = this.getObjectTypeIconUri(objectTypeId, fallback);\n      return this.#backend.get(iconUri).pipe(tap((icon) => (this.#iconCache[objectTypeId] = { uri: iconUri, icon })));\n    }\n  }\n\n  /**\n   * Get the URI of an object type icon.\n   * @param objectTypeId ID of the object type\n   * @param fallback ID of a fallback icon that should be used if the given object type has no icon yet\n   */\n  getObjectTypeIconUri(objectTypeId: string, fallback?: string): string {\n    if (this.#iconCache[objectTypeId]) {\n      return this.#iconCache[objectTypeId].uri;\n    } else {\n      const ci = this.#getIconFromClassification(objectTypeId);\n      const fb = this.getFallbackIcon(objectTypeId, fallback);\n      const uri = `/resources/icons/${encodeURIComponent(ci || objectTypeId)}${fb ? `?fallback=${encodeURIComponent(fb)}` : ''}`;\n      this.#iconCache[objectTypeId] = { uri: `${this.#backend.getApiBase(ApiBase.apiWeb)}${uri}` };\n      return this.#iconCache[objectTypeId].uri;\n    }\n  }\n\n  private getFallbackIcon(objectTypeId: string, fallback?: string): string {\n    const ot = this.getObjectType(objectTypeId);\n    if (ot && !fallback) {\n      // add default fallbacks for system:document and system:folder if now other fallback has been provided\n      fallback = ot.isFolder ? 'system:folder' : 'system:document';\n      // if (this.isFloatingObjectType(ot)) {\n      //   // types that do not have no object type assigned to them (primary FSOTs)\n      //   fallback = 'system:dlm';\n      // }\n    }\n    return fallback!;\n  }\n\n  #getIconFromClassification(objectTypeId: string) {\n    const ce: Map<string, ClassificationEntry> = this.getClassifications(this.getResolvedClassifications(objectTypeId));\n    return ce.has(ObjectTypeClassification.OBJECT_TYPE_ICON) ? ce.get(ObjectTypeClassification.OBJECT_TYPE_ICON)!.options[0] : null;\n  }\n\n  getLocalizedResource(key: string): string {\n    return this.system!.i18n[key];\n  }\n\n  getLocalizedLabel(id: string) {\n    return this.getLocalizedResource(`${id}_label`);\n  }\n\n  getLocalizedDescription(id: string) {\n    return this.getLocalizedResource(`${id}_description`);\n  }\n\n  /**\n   * Determine whether or not the given object type field is a system field\n   * @param field Object type field to be checked\n   */\n  isSystemProperty(field: ObjectTypeField): boolean {\n    return field.id.startsWith('system:');\n  }\n\n  /**\n   * Fetches the backends system definition and updates system$ Observable.\n   * Subscribe to the system$ observable instead of calling this function, otherwise you'll trigger fetching the\n   * system definition every time.\n   *\n   * @param user The user to load the system definition for\n   */\n  getSystemDefinition(authData?: AuthData): Observable<boolean> {\n    // TODO: Supposed to return 304 if nothing changes\n    return this.#fetchSystemDefinition(authData);\n\n    // TODO: remove when 304 is there???\n    // // try to fetch system definition from cache first\n    // return this.appCache.getItem(this.STORAGE_KEY).pipe(\n    //   switchMap(res => {\n    //     if (res) {\n    //       // check if the system definition from the cache is up to date\n    //       this.system = res;\n    //       this.systemSource.next(this.system);\n    //       return of(true);\n    //     } else {\n    //       // nothing cached so far\n    //       return this.fetchSystemDefinition();\n    //     }\n    //   })\n    // );\n  }\n\n  /**\n   * Actually fetch the system definition from the backend.\n   * @param user User to fetch definition for\n   */\n  #fetchSystemDefinition(authData?: AuthData): Observable<boolean> {\n    return (authData ? of(authData) : this.#appCache.getItem(this.#STORAGE_KEY_AUTH_DATA)).pipe(\n      switchMap((data: AuthData) => {\n        this.updateAuthData(data).subscribe();\n        const fetchTasks = [this.#backend.get('/dms/schema/native.json', ApiBase.core), this.#fetchLocalizations()];\n        return forkJoin(fetchTasks);\n      }),\n      catchError((error) => {\n        this.#logger.error('Error fetching recent version of system definition from server.', error);\n        this.#systemSource.error('Error fetching recent version of system definition from server.');\n        return of(null);\n      }),\n      map((data) => {\n        if (data?.length) {\n          this.setSchema(data[0], data[1]);\n        }\n        return !!data;\n      })\n    );\n  }\n\n  setPermissions(p: ObjectTypePermissions) {\n    this.#permissions = p;\n  }\n\n  /**\n   * Create the schema from the servers schema response\n   * @param schemaResponse Response from the backend\n   */\n  setSchema(schemaResponse: SchemaResponse, localizedResource: Localization = {}) {\n    // prepare a quick access object for the fields\n    const propertiesQA: Record<string, SchemaResponseFieldDefinition> = {};\n    const orgTypeFields = [BaseObjectTypeField.MODIFIED_BY, BaseObjectTypeField.CREATED_BY];\n    schemaResponse.propertyDefinition.forEach((p: any) => {\n      p.classifications = p.classification;\n      // TODO: Remove once schema supports organization classification for base params\n      // map certain fields to organization type (fake it until you make it ;-)\n      if (orgTypeFields.includes(p.id)) {\n        p.classifications = [Classification.STRING_ORGANIZATION];\n      }\n      propertiesQA[p.id] = p;\n    });\n    // prepare a quick access object for object types (including secondary objects)\n    const objectTypesQA: { [id: string]: SchemaResponseTypeDefinition } = {};\n    schemaResponse.typeFolderDefinition.forEach((ot: any) => {\n      objectTypesQA[ot.id] = ot;\n    });\n    schemaResponse.typeDocumentDefinition.forEach((ot: any) => {\n      objectTypesQA[ot.id] = ot;\n    });\n    schemaResponse.typeSecondaryDefinition.forEach((sot: any) => {\n      objectTypesQA[sot.id] = sot;\n    });\n\n    const objectTypes: ObjectType[] = [\n      // folder types\n      ...schemaResponse.typeFolderDefinition.map((fd) => ({\n        id: fd.id,\n        description: fd.description,\n        classification: fd.classification,\n        baseId: fd.baseId,\n        creatable: this.#isCreatable(fd.id),\n        contentStreamAllowed: ContentStreamAllowed.NOT_ALLOWED,\n        isFolder: true,\n        secondaryObjectTypes: fd.secondaryObjectTypeId ? fd.secondaryObjectTypeId.map((t) => ({ id: t.value, static: t.static })) : [],\n        fields: this.#resolveObjectTypeFields(fd, propertiesQA, objectTypesQA)\n        // rawFields: this.resolveObjectTypeFields(fd, propertiesQA, objectTypesQA, true),\n      })),\n      // document types\n      ...schemaResponse.typeDocumentDefinition.map((dd) => ({\n        id: dd.id,\n        description: dd.description,\n        classification: dd.classification,\n        baseId: dd.baseId,\n        creatable: this.#isCreatable(dd.id),\n        contentStreamAllowed: dd.contentStreamAllowed,\n        isFolder: false,\n        secondaryObjectTypes: dd.secondaryObjectTypeId ? dd.secondaryObjectTypeId.map((t) => ({ id: t.value, static: t.static })) : [],\n        fields: this.#resolveObjectTypeFields(dd, propertiesQA, objectTypesQA)\n        // rawFields: this.resolveObjectTypeFields(dd, propertiesQA, objectTypesQA, true),\n      }))\n    ];\n\n    const secondaryObjectTypes: SecondaryObjectType[] = schemaResponse.typeSecondaryDefinition.map((std) => ({\n      id: std.id,\n      description: std.description,\n      classification: std.classification,\n      contentStreamAllowed: std.contentStreamAllowed,\n      baseId: std.baseId,\n      // TODO: Could a SOT be a folder too?\n      isFolder: false,\n      fields: this.#resolveObjectTypeFields(std, propertiesQA, objectTypesQA)\n    }));\n\n    this.system = {\n      version: schemaResponse.version,\n      lastModificationDate: schemaResponse.lastModificationDate,\n      objectTypes,\n      secondaryObjectTypes,\n      i18n: localizedResource,\n      allFields: propertiesQA\n    };\n    this.#appCache.setItem(this.#STORAGE_KEY, this.system).subscribe();\n    this.#systemSource.next(this.system);\n  }\n\n  /**\n   * Resolve all the fields for an object type. This also includes secondary object types and the fields inherited from\n   * the base type (... and of course the base type (and its secondary object types) of the base type and so on)\n   * @param schemaTypeDefinition object type definition from the native schema\n   * @param propertiesQA Quick access object of all properties\n   * @param objectTypesQA Quick access object of all object types\n   * @param raw If set to 'true' only the properties of the object type itself will be returned (without SOTs)\n   */\n  #resolveObjectTypeFields(\n    schemaTypeDefinition: SchemaResponseTypeDefinition,\n    propertiesQA: { [id: string]: SchemaResponseFieldDefinition },\n    objectTypesQA: { [id: string]: SchemaResponseTypeDefinition }\n  ): ObjectTypeField[] {\n    const objectTypeFieldIDs = schemaTypeDefinition.propertyReference.map((pr) => pr.value);\n    if (schemaTypeDefinition.secondaryObjectTypeId) {\n      schemaTypeDefinition.secondaryObjectTypeId\n        .filter((sot) => sot.static)\n        .map((sot) => sot.value)\n        .forEach((sotID) => objectTypesQA[sotID].propertyReference.forEach((pr: { value: string }) => objectTypeFieldIDs.push(pr.value)));\n    }\n\n    let fields: ObjectTypeField[] = objectTypeFieldIDs.map((id) => ({\n      ...propertiesQA[id],\n      _internalType: this.getInternalFormElementType(propertiesQA[id].propertyType, propertiesQA[id].classifications)\n    }));\n\n    // also resolve properties of the base type\n    if (schemaTypeDefinition.baseId !== schemaTypeDefinition.id && !!objectTypesQA[schemaTypeDefinition.baseId]) {\n      fields = fields.concat(this.#resolveObjectTypeFields(objectTypesQA[schemaTypeDefinition.baseId], propertiesQA, objectTypesQA));\n    }\n    return fields;\n  }\n\n  #isCreatable(objectTypeId: string) {\n    return ![SystemType.FOLDER, SystemType.DOCUMENT].includes(objectTypeId);\n  }\n\n  /**\n   * Fetch a collection of form models.\n   * @param objectTypeIDs Object type IDs to fetch form model for\n   * @param situation Form situation\n   * @returns Object where the object type id is key and the form model is the value\n   */\n  getObjectTypeForms(objectTypeIDs: string[], situation: string): Observable<Record<string, any>> {\n    return forkJoin(\n      objectTypeIDs.map((o) =>\n        this.getObjectTypeForm(o, situation).pipe(\n          catchError((e) => of(null)),\n          map((res) => ({\n            id: o,\n            formModel: res\n          }))\n        )\n      )\n    ).pipe(\n      map((res) => {\n        const resMap: Record<string, any> = {};\n        res.filter((r) => this.#formHasElements(r.formModel)).forEach((r: any) => (resMap[r.id] = r.formModel));\n        return resMap;\n      })\n    );\n  }\n\n  /**\n   * Get the form model of an object type.\n   *\n   * @param objectTypeId ID of the object type to fetch the form for\n   * @param situation The form situation to be fetched\n   * @returns Form model\n   */\n  getObjectTypeForm(objectTypeId: string, situation: string): Observable<any> {\n    return this.#backend.get(Utils.buildUri(`/dms/forms/${objectTypeId}`, { situation }));\n  }\n\n  /**\n   * Check whether or not the model has at least one form element. Recursive.\n   * @param element Form element to check child elements for\n   */\n  #formHasElements(element: any): boolean {\n    let hasElement = false;\n    element.elements?.forEach((e: any) => {\n      if (!['o2mGroup', 'o2mGroupStack'].includes(e.type)) {\n        hasElement = true;\n      } else if (!hasElement) {\n        hasElement = this.#formHasElements(e);\n      }\n    });\n    return hasElement;\n  }\n\n  /**\n   * Generates an internal type for a given object type field.\n   * Adding this to a form element or object type field enables us to render forms\n   * based on object type fields in a more performant way. Otherwise we would\n   * have to evaluate the conditions for every form element on every digest cycle.\n   * @param type propertyType of the ObjectTypeField\n   * @param classifications classifications of the ObjectTypeField\n   */\n  getInternalFormElementType(type: string, classifications?: string[]): ObjectTypeFieldInternalType {\n    const _classifications = this.getClassifications(classifications || []);\n\n    if (type === 'string' && _classifications.has(Classification.STRING_REFERENCE)) {\n      return InternalFieldType.STRING_REFERENCE;\n    } else if (type === 'string' && _classifications.has(Classification.STRING_ORGANIZATION)) {\n      return InternalFieldType.STRING_ORGANIZATION;\n    } else if (type === 'string' && _classifications.has(Classification.STRING_ORGANIZATION_SET)) {\n      return InternalFieldType.STRING_ORGANIZATION_SET;\n    } else if (type === 'string' && _classifications.has(Classification.STRING_CATALOG)) {\n      return InternalFieldType.STRING_CATALOG;\n    } else if (type === 'boolean' && _classifications.has(Classification.BOOLEAN_SWITCH)) {\n      return InternalFieldType.BOOLEAN_SWITCH;\n    } else if (\n      type === 'string' &&\n      (_classifications.has(Classification.STRING_CATALOG_DYNAMIC) || _classifications.has(Classification.STRING_CATALOG_CUSTOM))\n    ) {\n      return InternalFieldType.STRING_DYNAMIC_CATALOG;\n    } else {\n      // if there are no matching conditions just return the original type\n      return type as ObjectTypeFieldType;\n    }\n  }\n\n  getObjectTypeField(id: string): ObjectTypeField | undefined {\n    const f = this.system?.allFields[id];\n    return f ? { ...f, _internalType: this.getInternalFormElementType(f.propertyType, f.classifications) } : undefined;\n  }\n\n  /**\n   * Extract classifications from object type fields classification\n   * string. This string may contain more than one classification entry.\n   *\n   * Classification is a comma separated string that may contain additional\n   * properties related to on classification entry. Example:\n   *\n   * `id:reference[system:folder], email`\n   *\n   * @param classifications Object type fields classification property (schema)\n   */\n  getClassifications(classifications: string[]): Map<string, ClassificationEntry> {\n    const res = new Map<string, ClassificationEntry>();\n    if (classifications) {\n      classifications.forEach((c) => {\n        const matches: RegExpMatchArray | null = c.match(/^([^\\[]*)(\\[(.*)\\])?$/);\n        if (matches && matches.length) {\n          res.set(matches[1], {\n            classification: matches[1],\n            options: matches[3] ? matches[3].split(',').map((o) => o.trim()) : []\n          });\n        }\n      });\n    }\n    return res;\n  }\n\n  toFormElement(field: ObjectTypeField): any {\n    return { ...field, label: this.getLocalizedLabel(field.id), name: field.id, type: field.propertyType };\n  }\n\n  updateAuthData(data: Partial<AuthData>) {\n    this.authData = { ...this.authData!, ...data };\n    this.#backend.setHeader('Accept-Language', this.authData.language);\n    return this.#appCache.setItem(this.#STORAGE_KEY_AUTH_DATA, this.authData);\n  }\n\n  updateLocalizations(iso: string): Observable<any> {\n    return this.updateAuthData({ language: iso }).pipe(\n      switchMap(() => this.#fetchLocalizations()),\n      tap((res) => {\n        this.system!.i18n = res;\n        this.#appCache.setItem(this.#STORAGE_KEY, this.system).subscribe();\n        this.#systemSource.next(this.system!);\n      })\n    );\n  }\n\n  #fetchLocalizations(): Observable<Localization> {\n    return this.#backend.get('/resources/text');\n  }\n\n  fetchResources(id: string): Observable<{ global: any; tenant: any }> {\n    return this.#backend\n      .batch([\n        { uri: `/system/resources/${id}`, base: ApiBase.core },\n        { uri: `/admin/resources/${id}`, base: ApiBase.core }\n      ])\n      .pipe(map(([global, tenant]) => ({ global, tenant })));\n  }\n}\n"]}
|