@sapui5/sap.fe.core 1.124.7 → 1.124.9
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/package.json +1 -1
- package/src/sap/fe/core/.library +1 -1
- package/src/sap/fe/core/ActionRuntime.js +7 -3
- package/src/sap/fe/core/ActionRuntime.ts +6 -2
- package/src/sap/fe/core/AppComponent.js +11 -4
- package/src/sap/fe/core/AppComponent.ts +12 -3
- package/src/sap/fe/core/controllerextensions/EditFlow.js +6 -2
- package/src/sap/fe/core/controllerextensions/EditFlow.ts +5 -1
- package/src/sap/fe/core/controllerextensions/PageReady.js +19 -18
- package/src/sap/fe/core/controllerextensions/PageReady.ts +20 -17
- package/src/sap/fe/core/controllerextensions/SideEffects.js +55 -7
- package/src/sap/fe/core/controllerextensions/SideEffects.ts +55 -2
- package/src/sap/fe/core/controls/ComponentLoader.js +2 -2
- package/src/sap/fe/core/controls/ComponentLoader.ts +1 -1
- package/src/sap/fe/core/converters/controls/Common/Table.js +11 -7
- package/src/sap/fe/core/converters/controls/Common/Table.ts +8 -6
- package/src/sap/fe/core/fpm/manifest.json +1 -1
- package/src/sap/fe/core/helpers/AppStartupHelper.js +32 -2
- package/src/sap/fe/core/helpers/AppStartupHelper.ts +32 -1
- package/src/sap/fe/core/library.js +1 -1
- package/src/sap/fe/core/manifestMerger/ChangePageConfiguration.js +8 -17
- package/src/sap/fe/core/manifestMerger/ChangePageConfiguration.ts +14 -16
- package/src/sap/fe/core/templating/UIFormatters.js +18 -2
- package/src/sap/fe/core/templating/UIFormatters.ts +15 -0
- package/src/sap/fe/core/type/Email.js +2 -2
- package/src/sap/fe/core/type/Email.ts +1 -1
|
@@ -3396,22 +3396,24 @@ export function addComputedColumns(
|
|
|
3396
3396
|
// so we have to disable it here to ensure, that the field building block
|
|
3397
3397
|
// does not render it into the ID column
|
|
3398
3398
|
// The additional column is only added for tables on a LR and in case tehe entity is draft enabled!
|
|
3399
|
-
const
|
|
3400
|
-
(column) => column.formatOptions?.hasDraftIndicator !== undefined && column.formatOptions?.hasDraftIndicator === true
|
|
3401
|
-
);
|
|
3399
|
+
const columnsWithDraftIndicator: TableColumn[] = tableColumns.filter((column) => column.formatOptions?.hasDraftIndicator === true);
|
|
3402
3400
|
if (
|
|
3403
|
-
|
|
3401
|
+
columnsWithDraftIndicator.length &&
|
|
3404
3402
|
converterContext.getTemplateType() === TemplateType.ListReport &&
|
|
3405
3403
|
(ModelHelper.isDraftNode(converterContext.getEntitySet()) || ModelHelper.isDraftRoot(converterContext.getEntitySet()))
|
|
3406
3404
|
) {
|
|
3407
3405
|
// Switch off the draft indicator in the found column
|
|
3408
|
-
columnWithDraftIndicator
|
|
3406
|
+
columnsWithDraftIndicator.forEach((columnWithDraftIndicator: TableColumn) => {
|
|
3407
|
+
if (columnWithDraftIndicator?.formatOptions) {
|
|
3408
|
+
columnWithDraftIndicator.formatOptions.hasDraftIndicator = false;
|
|
3409
|
+
}
|
|
3410
|
+
});
|
|
3409
3411
|
|
|
3410
3412
|
// Insert a separate draft indicator column
|
|
3411
3413
|
const draftIndicatorColumn: ComputedTableColumn = {
|
|
3412
3414
|
key: "draftStatus",
|
|
3413
3415
|
name: "draftStatus",
|
|
3414
|
-
propertyKey:
|
|
3416
|
+
propertyKey: columnsWithDraftIndicator[0].name,
|
|
3415
3417
|
isDraftIndicator: true,
|
|
3416
3418
|
availability: "Default",
|
|
3417
3419
|
label: "{sap.fe.i18n>C_DRAFT_STATUS_COLUMN_LABEL_TOOLTIP}",
|
|
@@ -64,6 +64,34 @@ sap.ui.define(["sap/ui/model/Filter", "sap/ui/model/FilterOperator", "../convert
|
|
|
64
64
|
}
|
|
65
65
|
return new Filter(aFilters, true);
|
|
66
66
|
},
|
|
67
|
+
/**
|
|
68
|
+
* Makes sure key values have the correct format.
|
|
69
|
+
* This is to handle special cases, for example, where Boolean values are 'X' or '', or guid values have the following format: guid'00000000-0000-0000-0000-000000000000'.".
|
|
70
|
+
* @param keys
|
|
71
|
+
* @param entityType
|
|
72
|
+
*/
|
|
73
|
+
_sanitizeKeys: function (keys, entityType) {
|
|
74
|
+
keys.forEach(key => {
|
|
75
|
+
const propertyType = entityType[key.name].$Type;
|
|
76
|
+
switch (propertyType) {
|
|
77
|
+
case "Edm.Boolean":
|
|
78
|
+
if (key.value !== "true" && key.value !== "false") {
|
|
79
|
+
// If the value is neither "true" nor "false", it's considered as false if it contains only whitespaces (e.g. to handle 'X'/'' for true/false values)
|
|
80
|
+
key.value = key.value.trim().length ? "true" : "false";
|
|
81
|
+
}
|
|
82
|
+
break;
|
|
83
|
+
case "Edm.Guid":
|
|
84
|
+
{
|
|
85
|
+
const match = /[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}/.exec(key.value);
|
|
86
|
+
key.value = match ? match[0] : key.value;
|
|
87
|
+
break;
|
|
88
|
+
}
|
|
89
|
+
default:
|
|
90
|
+
// Do nothing
|
|
91
|
+
}
|
|
92
|
+
});
|
|
93
|
+
},
|
|
94
|
+
|
|
67
95
|
/**
|
|
68
96
|
* Loads all contexts for a list of page infos.
|
|
69
97
|
*
|
|
@@ -75,14 +103,16 @@ sap.ui.define(["sap/ui/model/Filter", "sap/ui/model/FilterOperator", "../convert
|
|
|
75
103
|
_requestObjectsFromParameters: async function (aStartupPages, oModel, alwaysCheckExistence) {
|
|
76
104
|
// Load the respective objects for all object pages found in aExternallyNavigablePages
|
|
77
105
|
const aContextPromises = aStartupPages.map(async pageInfo => {
|
|
106
|
+
const entityType = oModel.getMetaModel().getObject(`${pageInfo.contextPath}/`);
|
|
78
107
|
if (pageInfo.technicalKeys && !alwaysCheckExistence) {
|
|
108
|
+
this._sanitizeKeys(pageInfo.technicalKeys, entityType);
|
|
79
109
|
// Create a context with the proper path
|
|
80
|
-
const entityType = oModel.getMetaModel().getObject(`${pageInfo.contextPath}/`);
|
|
81
110
|
const path = `${pageInfo.contextPath}(${SemanticKeyHelper.getPathContent(pageInfo.technicalKeys, entityType)})`;
|
|
82
111
|
const context = oModel.bindContext(path).getBoundContext();
|
|
83
112
|
return Promise.resolve([context]);
|
|
84
113
|
} else {
|
|
85
114
|
const aKeys = pageInfo.semanticKeys || pageInfo.technicalKeys || [];
|
|
115
|
+
this._sanitizeKeys(aKeys, entityType);
|
|
86
116
|
const oFilter = this._createFilterFromKeys(aKeys, pageInfo.draftMode, oModel.getMetaModel());
|
|
87
117
|
|
|
88
118
|
// only request a minimum of fields to boost backend performance since this is only used to check if an object exists
|
|
@@ -376,4 +406,4 @@ sap.ui.define(["sap/ui/model/Filter", "sap/ui/model/FilterOperator", "../convert
|
|
|
376
406
|
};
|
|
377
407
|
return AppStartupHelper;
|
|
378
408
|
}, false);
|
|
379
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["AppStartupHelper","_getKeysFromStartupParams","aKeyNames","oStartupParameters","bAllFound","aKeys","map","name","length","value","undefined","_createFilterFromKeys","bDraftMode","oMetaModel","bFilterCaseSensitive","ModelHelper","isFilteringCaseSensitive","bFilterOnActiveEntity","aFilters","key","Filter","path","operator","FilterOperator","EQ","value1","caseSensitive","oDraftFilter","filters","and","push","_requestObjectsFromParameters","aStartupPages","oModel","alwaysCheckExistence","aContextPromises","pageInfo","technicalKeys","entityType","getMetaModel","getObject","contextPath","SemanticKeyHelper","getPathContent","context","bindContext","getBoundContext","Promise","resolve","semanticKeys","oFilter","draftMode","oListBind","bindList","$select","join","requestContexts","all","_getReachablePageInfoFromRoute","oRoute","oManifestRouting","sPattern","pattern","replace","endsWith","sTargetName","Array","isArray","target","oTarget","targets","aPatternSegments","split","pageLevel","options","settings","allowDeepLinking","sContextPath","entitySet","metaContext","createBindingContext","objectPath","getInvolvedDataModelObjects","targetEntitySet","oEntityType","aSemanticKeyNames","aSemantickKeys","semKey","$PropertyPath","aTechnicalKeys","_getReachablePages","aRoutes","routes","mPagesByLevel","forEach","oPageInfo","aReachablePages","level","_getStartupPagesFromStartupParams","result","current","findRecursive","aCurrentLevelPages","lastPage","nextPage","indexOf","pop","slice","_getDeepLinkObject","aContexts","hash","oContext","getPath","getDeepLinkStartupHash","requestObject","startupPages","allContexts","contextsForStartup","foundContexts","getCreateStartupHash","oRouter","then","oEntitySetAnnotations","sMetaPath","bCreatable","bNewActionOperationAvailable","oInsertRestrictions","Insertable","getDefaultCreateHash","sDefaultCreateHash","preferredMode","sHash","includes","substring","getRouteInfoByHash","Error","verifyEditAnnotations","metaModel","entitySetAnnotations","updateHidden","operationAvailable"],"sourceRoot":".","sources":["AppStartupHelper.ts"],"sourcesContent":["import type { PropertyPath } from \"@sap-ux/vocabularies-types\";\nimport type { StartupParameters } from \"sap/fe/core/AppComponent\";\nimport type { RoutingConfiguration, RoutingRoute } from \"sap/ui/core/Manifest\";\nimport type Router from \"sap/ui/core/routing/Router\";\nimport Filter from \"sap/ui/model/Filter\";\nimport FilterOperator from \"sap/ui/model/FilterOperator\";\nimport type Context from \"sap/ui/model/odata/v4/Context\";\nimport type ODataMetaModel from \"sap/ui/model/odata/v4/ODataMetaModel\";\nimport type ODataModel from \"sap/ui/model/odata/v4/ODataModel\";\nimport type { ExpandPathType, MetaModelEntitySetAnnotation } from \"types/metamodel_types\";\nimport { getInvolvedDataModelObjects } from \"../converters/MetaModelConverter\";\nimport ModelHelper from \"./ModelHelper\";\nimport SemanticKeyHelper from \"./SemanticKeyHelper\";\n\ntype ValuedKey = {\n\tname: string;\n\tvalue: string;\n};\n\ntype PageInfo = {\n\tpattern: string;\n\tcontextPath: string;\n\tdraftMode: boolean;\n\ttechnicalKeys: ValuedKey[] | undefined;\n\tsemanticKeys: ValuedKey[] | undefined;\n\ttarget: string;\n\tpageLevel: number;\n};\n\nconst AppStartupHelper = {\n\t/**\n\t * Retrieves a set of key values from startup parameters.\n\t *\n\t * @param aKeyNames The array of key names\n\t * @param oStartupParameters The startup parameters\n\t * @returns An array of pairs \\{name, value\\} if all key values could be found in the startup parameters, undefined otherwise\n\t */\n\t_getKeysFromStartupParams: function (aKeyNames: string[], oStartupParameters: StartupParameters | null): ValuedKey[] | undefined {\n\t\tlet bAllFound = true;\n\t\tconst aKeys = aKeyNames.map((name) => {\n\t\t\tif (oStartupParameters?.[name] && oStartupParameters[name].length === 1) {\n\t\t\t\treturn { name, value: oStartupParameters[name][0] as string };\n\t\t\t} else {\n\t\t\t\t// A unique key value couldn't be found in the startup parameters\n\t\t\t\tbAllFound = false;\n\t\t\t\treturn { name, value: \"\" };\n\t\t\t}\n\t\t});\n\n\t\treturn bAllFound ? aKeys : undefined;\n\t},\n\n\t/**\n\t * Creates a filter from a list of key values.\n\t *\n\t * @param aKeys Array of semantic keys or technical keys (with values)\n\t * @param bDraftMode True if the entity supports draft mode\n\t * @param oMetaModel The metamodel\n\t * @returns The filter\n\t */\n\t_createFilterFromKeys: function (aKeys: ValuedKey[], bDraftMode: Boolean, oMetaModel: ODataMetaModel): Filter {\n\t\tconst bFilterCaseSensitive = ModelHelper.isFilteringCaseSensitive(oMetaModel);\n\n\t\tlet bFilterOnActiveEntity = false;\n\t\tconst aFilters = aKeys.map((key) => {\n\t\t\tif (key.name === \"IsActiveEntity\") {\n\t\t\t\tbFilterOnActiveEntity = true;\n\t\t\t}\n\t\t\treturn new Filter({\n\t\t\t\tpath: key.name,\n\t\t\t\toperator: FilterOperator.EQ,\n\t\t\t\tvalue1: key.value,\n\t\t\t\tcaseSensitive: bFilterCaseSensitive\n\t\t\t});\n\t\t});\n\t\tif (bDraftMode && !bFilterOnActiveEntity) {\n\t\t\tconst oDraftFilter = new Filter({\n\t\t\t\tfilters: [new Filter(\"IsActiveEntity\", \"EQ\", false), new Filter(\"SiblingEntity/IsActiveEntity\", \"EQ\", null)],\n\t\t\t\tand: false\n\t\t\t});\n\t\t\taFilters.push(oDraftFilter);\n\t\t}\n\n\t\treturn new Filter(aFilters, true);\n\t},\n\n\t/**\n\t * Loads all contexts for a list of page infos.\n\t *\n\t * @param aStartupPages The list of page infos\n\t * @param oModel The model used to load the contexts\n\t * @param alwaysCheckExistence If true, always check if the object exists, even if semantic keys are present\n\t * @returns A Promise for all contexts\n\t */\n\t_requestObjectsFromParameters: async function (\n\t\taStartupPages: PageInfo[],\n\t\toModel: ODataModel,\n\t\talwaysCheckExistence: boolean\n\t): Promise<Context[][]> {\n\t\t// Load the respective objects for all object pages found in aExternallyNavigablePages\n\t\tconst aContextPromises = aStartupPages.map(async (pageInfo) => {\n\t\t\tif (pageInfo.technicalKeys && !alwaysCheckExistence) {\n\t\t\t\t// Create a context with the proper path\n\t\t\t\tconst entityType = oModel.getMetaModel().getObject(`${pageInfo.contextPath}/`);\n\t\t\t\tconst path = `${pageInfo.contextPath}(${SemanticKeyHelper.getPathContent(pageInfo.technicalKeys, entityType)})`;\n\t\t\t\tconst context = oModel.bindContext(path).getBoundContext();\n\t\t\t\treturn Promise.resolve([context]);\n\t\t\t} else {\n\t\t\t\tconst aKeys = pageInfo.semanticKeys || pageInfo.technicalKeys || [];\n\t\t\t\tconst oFilter = this._createFilterFromKeys(aKeys, pageInfo.draftMode, oModel.getMetaModel());\n\n\t\t\t\t// only request a minimum of fields to boost backend performance since this is only used to check if an object exists\n\t\t\t\tconst oListBind = oModel.bindList(pageInfo.contextPath, undefined, undefined, oFilter, {\n\t\t\t\t\t$select: aKeys\n\t\t\t\t\t\t.map((key) => {\n\t\t\t\t\t\t\treturn key.name;\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.join(\",\")\n\t\t\t\t});\n\t\t\t\treturn oListBind.requestContexts(0, 2);\n\t\t\t}\n\t\t});\n\n\t\treturn Promise.all(aContextPromises);\n\t},\n\n\t/**\n\t * Creates a PageInfo from a route if it's reachable from the startup parameters.\n\t *\n\t * @param oRoute The route\n\t * @param oManifestRouting The app manifest routing section\n\t * @param oStartupParameters The startup parameters\n\t * @param oMetaModel The app metamodel\n\t * @returns A page info if the page is reachable, undefined otherwise\n\t */\n\t_getReachablePageInfoFromRoute: function (\n\t\toRoute: RoutingRoute,\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toMetaModel: ODataMetaModel\n\t): PageInfo | undefined {\n\t\t// Remove trailing ':?query:' and '/'\n\t\tlet sPattern: string = oRoute.pattern!.replace(\":?query:\", \"\");\n\t\tsPattern = sPattern.replace(/\\/$/, \"\");\n\n\t\tif (!sPattern || !sPattern.endsWith(\")\")) {\n\t\t\t// Ignore level-0 routes (ListReport) or routes corresponding to a 1-1 relation (no keys in the URL in this case)\n\t\t\treturn undefined;\n\t\t}\n\n\t\tsPattern = sPattern.replace(/\\(\\{[^}]*\\}\\)/g, \"(#)\"); // Replace keys with #\n\n\t\t// Get the rightmost target for this route\n\t\tconst sTargetName: string = Array.isArray(oRoute.target) ? oRoute.target[oRoute.target.length - 1] : oRoute.target;\n\t\tconst oTarget = oManifestRouting?.targets?.[sTargetName];\n\n\t\tconst aPatternSegments = sPattern.split(\"/\");\n\t\tconst pageLevel = aPatternSegments.length - 1;\n\n\t\tif (pageLevel !== 0 && oTarget?.options?.settings?.allowDeepLinking !== true) {\n\t\t\t// The first level of object page allows deep linking by default.\n\t\t\t// Otherwise, the target must allow deep linking explicitely in the manifest\n\t\t\treturn undefined;\n\t\t}\n\n\t\tlet sContextPath = \"\";\n\t\tif (oTarget?.options?.settings?.entitySet) {\n\t\t\tsContextPath = `/${oTarget.options.settings.entitySet}`;\n\t\t} else if (oTarget?.options?.settings?.contextPath) {\n\t\t\tconst metaContext = oMetaModel.createBindingContext(oTarget.options.settings.contextPath)!;\n\t\t\tconst objectPath = getInvolvedDataModelObjects(metaContext);\n\t\t\tif (objectPath.targetEntitySet) {\n\t\t\t\tsContextPath = `/${objectPath.targetEntitySet.name}`;\n\t\t\t}\n\t\t}\n\n\t\tif (!sContextPath) {\n\t\t\treturn undefined;\n\t\t}\n\t\tconst oEntityType = sContextPath && oMetaModel.getObject(`/$EntityContainer${sContextPath}/`);\n\n\t\tif (!oEntityType) {\n\t\t\treturn undefined;\n\t\t}\n\n\t\t// Get the semantic key values for the entity\n\t\tconst aSemanticKeyNames: ExpandPathType<PropertyPath>[] = oMetaModel.getObject(\n\t\t\t`/$EntityContainer${sContextPath}/@com.sap.vocabularies.Common.v1.SemanticKey`\n\t\t);\n\n\t\tconst aSemantickKeys = aSemanticKeyNames\n\t\t\t? this._getKeysFromStartupParams(\n\t\t\t\t\taSemanticKeyNames.map((semKey) => {\n\t\t\t\t\t\treturn semKey.$PropertyPath;\n\t\t\t\t\t}),\n\t\t\t\t\toStartupParameters\n\t\t\t  )\n\t\t\t: undefined;\n\n\t\t// Get the technical keys only if we couldn't find the semantic key values, and on first level OP\n\t\tconst aTechnicalKeys =\n\t\t\t!aSemantickKeys && pageLevel === 0 ? this._getKeysFromStartupParams(oEntityType[\"$Key\"], oStartupParameters) : undefined;\n\n\t\tif (aSemantickKeys === undefined && aTechnicalKeys === undefined) {\n\t\t\t// We couldn't find the semantic/technical keys in the startup parameters\n\t\t\treturn undefined;\n\t\t}\n\n\t\t// The startup parameters contain values for all semantic keys (or technical keys) --> we can store the page info in the corresponding level\n\t\tconst draftMode =\n\t\t\toMetaModel.getObject(`/$EntityContainer${sContextPath}@com.sap.vocabularies.Common.v1.DraftRoot`) ||\n\t\t\toMetaModel.getObject(`/$EntityContainer${sContextPath}@com.sap.vocabularies.Common.v1.DraftNode`)\n\t\t\t\t? true\n\t\t\t\t: false;\n\n\t\treturn {\n\t\t\tpattern: sPattern,\n\t\t\tcontextPath: sContextPath,\n\t\t\tdraftMode,\n\t\t\ttechnicalKeys: aTechnicalKeys,\n\t\t\tsemanticKeys: aSemantickKeys,\n\t\t\ttarget: sTargetName,\n\t\t\tpageLevel\n\t\t};\n\t},\n\n\t/**\n\t * Returns the list of all pages that allow deeplink and that can be reached using the startup parameters.\n\t *\n\t * @param oManifestRouting The routing information from the app manifest\n\t * @param oStartupParameters The startup parameters\n\t * @param oMetaModel The metamodel\n\t * @returns The reachable pages\n\t */\n\t_getReachablePages: function (\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toMetaModel: ODataMetaModel\n\t): PageInfo[][] {\n\t\tconst aRoutes: RoutingRoute[] = oManifestRouting?.routes ?? [];\n\t\tconst mPagesByLevel: Record<number, PageInfo[]> = {};\n\n\t\taRoutes.forEach((oRoute) => {\n\t\t\tconst oPageInfo = this._getReachablePageInfoFromRoute(oRoute, oManifestRouting, oStartupParameters, oMetaModel);\n\n\t\t\tif (oPageInfo) {\n\t\t\t\tif (!mPagesByLevel[oPageInfo.pageLevel]) {\n\t\t\t\t\tmPagesByLevel[oPageInfo.pageLevel] = [];\n\t\t\t\t}\n\t\t\t\tmPagesByLevel[oPageInfo.pageLevel].push(oPageInfo);\n\t\t\t}\n\t\t});\n\n\t\t// A page is reachable only if all its parents are also reachable\n\t\t// So if we couldn't find any pages for a given level, all pages with a higher level won't be reachable anyway\n\t\tconst aReachablePages: PageInfo[][] = [];\n\t\tlet level = 0;\n\t\twhile (mPagesByLevel[level]) {\n\t\t\taReachablePages.push(mPagesByLevel[level]);\n\t\t\tlevel++;\n\t\t}\n\n\t\treturn aReachablePages;\n\t},\n\n\t/**\n\t * Get the list of startup pages.\n\t *\n\t * @param oManifestRouting The routing information from the app manifest\n\t * @param oStartupParameters The startup parameters\n\t * @param oMetaModel The metamodel\n\t * @returns An array of startup page infos\n\t */\n\t_getStartupPagesFromStartupParams: function (\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toMetaModel: ODataMetaModel\n\t): PageInfo[] {\n\t\t// Find all pages that can be reached with the startup parameters\n\t\tconst aReachablePages = this._getReachablePages(oManifestRouting, oStartupParameters, oMetaModel);\n\n\t\tif (aReachablePages.length === 0) {\n\t\t\treturn [];\n\t\t}\n\n\t\t// Find the longest sequence of pages that can be reached (recursively)\n\t\tlet result: PageInfo[] = [];\n\t\tconst current: PageInfo[] = [];\n\n\t\tfunction findRecursive(level: number): void {\n\t\t\tconst aCurrentLevelPages = aReachablePages[level];\n\t\t\tconst lastPage = current.length ? current[current.length - 1] : undefined;\n\n\t\t\tif (aCurrentLevelPages) {\n\t\t\t\taCurrentLevelPages.forEach(function (nextPage) {\n\t\t\t\t\tif (!lastPage || nextPage.pattern.indexOf(lastPage.pattern) === 0) {\n\t\t\t\t\t\t// We only consider pages that can be reached from the page at the previous level,\n\t\t\t\t\t\t// --> their pattern must be the pattern of the previous page with another segment appended\n\t\t\t\t\t\tcurrent.push(nextPage);\n\t\t\t\t\t\tfindRecursive(level + 1);\n\t\t\t\t\t\tcurrent.pop();\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t}\n\t\t\tif (current.length > result.length) {\n\t\t\t\tresult = current.slice(); // We have found a sequence longer than our previous best --> store it as the new longest\n\t\t\t}\n\t\t}\n\n\t\tfindRecursive(0);\n\n\t\treturn result;\n\t},\n\n\t/**\n\t * Creates the startup object from the list of pages and contexts.\n\t *\n\t * @param aStartupPages The pages\n\t * @param aContexts The contexts\n\t * @returns An object containing either a hash or a context to navigate to, or an empty object if no deep link was found\n\t */\n\t_getDeepLinkObject: function (aStartupPages: PageInfo[], aContexts: Context[]): { hash?: string; context?: Context } {\n\t\tif (aContexts.length === 1) {\n\t\t\treturn { context: aContexts[0] };\n\t\t} else if (aContexts.length > 1) {\n\t\t\t// Navigation to a deeper level --> use the pattern of the deepest object page\n\t\t\t// and replace the parameters by the ID from the contexts\n\t\t\tlet hash = aStartupPages[aStartupPages.length - 1].pattern;\n\t\t\taContexts.forEach(function (oContext) {\n\t\t\t\thash = hash.replace(\"(#)\", `(${oContext.getPath().split(\"(\")[1]}`);\n\t\t\t});\n\n\t\t\treturn { hash };\n\t\t} else {\n\t\t\treturn {};\n\t\t}\n\t},\n\n\t/**\n\t * Calculates startup parameters for a deeplink case, from startup parameters and routing infoirmation.\n\t *\n\t * @param oManifestRouting The routing information from the app manifest\n\t * @param oStartupParameters The startup parameters\n\t * @param oModel The OData model\n\t * @param alwaysCheckExistence\n\t * @returns An object containing either a hash or a context to navigate to, or an empty object if no deep link was found\n\t */\n\tgetDeepLinkStartupHash: async function (\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toModel: ODataModel,\n\t\talwaysCheckExistence: boolean\n\t): Promise<{ hash?: string; context?: Context }> {\n\t\tawait oModel.getMetaModel().requestObject(\"/$EntityContainer/\");\n\n\t\t// Check if semantic keys are present in url parameters for every object page at each level\n\t\tconst startupPages = this._getStartupPagesFromStartupParams(oManifestRouting, oStartupParameters, oModel.getMetaModel());\n\n\t\t// Load the contexts\n\t\tconst allContexts = await this._requestObjectsFromParameters(startupPages, oModel, alwaysCheckExistence);\n\t\tif (allContexts.length) {\n\t\t\t// Make sure we only get 1 context per promise, and flatten the array\n\t\t\tconst contextsForStartup: Context[] = [];\n\t\t\tallContexts.forEach(function (foundContexts) {\n\t\t\t\tif (foundContexts.length === 1) {\n\t\t\t\t\tcontextsForStartup.push(foundContexts[0]);\n\t\t\t\t}\n\t\t\t});\n\n\t\t\treturn contextsForStartup.length === allContexts.length ? this._getDeepLinkObject(startupPages, contextsForStartup) : {};\n\t\t} else {\n\t\t\treturn {};\n\t\t}\n\t},\n\n\t/**\n\t * Calculates the new hash based on the startup parameters.\n\t *\n\t * @param oStartupParameters The startup parameter values (map parameter name -> array of values)\n\t * @param sContextPath The context path for the startup of the app (generally the path to the main entity set)\n\t * @param oRouter The router instance\n\t * @param oMetaModel The meta model\n\t * @returns A promise containing the hash to navigate to, or an empty string if there's no need to navigate\n\t */\n\tgetCreateStartupHash: async function (\n\t\toStartupParameters: StartupParameters,\n\t\tsContextPath: string,\n\t\toRouter: Router,\n\t\toMetaModel: ODataMetaModel\n\t): Promise<string> {\n\t\treturn oMetaModel.requestObject(`${sContextPath}@`).then((oEntitySetAnnotations: MetaModelEntitySetAnnotation) => {\n\t\t\tlet sMetaPath = \"\";\n\t\t\tlet bCreatable = true;\n\n\t\t\tif (\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Common.v1.DraftRoot\"] &&\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Common.v1.DraftRoot\"][\"NewAction\"]\n\t\t\t) {\n\t\t\t\tsMetaPath = `${sContextPath}@com.sap.vocabularies.Common.v1.DraftRoot/NewAction@Org.OData.Core.V1.OperationAvailable`;\n\t\t\t} else if (\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Session.v1.StickySessionSupported\"] &&\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Session.v1.StickySessionSupported\"][\"NewAction\"]\n\t\t\t) {\n\t\t\t\tsMetaPath = `${sContextPath}@com.sap.vocabularies.Session.v1.StickySessionSupported/NewAction@Org.OData.Core.V1.OperationAvailable`;\n\t\t\t}\n\n\t\t\tif (sMetaPath) {\n\t\t\t\tconst bNewActionOperationAvailable = oMetaModel.getObject(sMetaPath);\n\t\t\t\tif (bNewActionOperationAvailable === false) {\n\t\t\t\t\tbCreatable = false;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconst oInsertRestrictions = oEntitySetAnnotations[\"@Org.OData.Capabilities.V1.InsertRestrictions\"];\n\t\t\t\tif (oInsertRestrictions && oInsertRestrictions.Insertable === false) {\n\t\t\t\t\tbCreatable = false;\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (bCreatable) {\n\t\t\t\treturn this.getDefaultCreateHash(oStartupParameters, sContextPath, oRouter);\n\t\t\t} else {\n\t\t\t\treturn \"\";\n\t\t\t}\n\t\t});\n\t},\n\n\t/**\n\t * Calculates the hash to create a new object.\n\t *\n\t * @param oStartupParameters The startup parameter values (map parameter name -> array of values)\n\t * @param sContextPath The context path of the entity set to be used for the creation\n\t * @param oRouter The router instance\n\t * @returns The hash\n\t */\n\tgetDefaultCreateHash: function (\n\t\toStartupParameters: StartupParameters | null | undefined,\n\t\tsContextPath: string,\n\t\toRouter: Router\n\t): string {\n\t\tlet sDefaultCreateHash = oStartupParameters && oStartupParameters.preferredMode ? oStartupParameters.preferredMode[0] : \"create\";\n\t\tlet sHash = \"\";\n\n\t\tsDefaultCreateHash =\n\t\t\tsDefaultCreateHash.includes(\":\") && sDefaultCreateHash.length > sDefaultCreateHash.indexOf(\":\") + 1\n\t\t\t\t? sDefaultCreateHash.substring(0, sDefaultCreateHash.indexOf(\":\"))\n\t\t\t\t: \"create\";\n\t\tsHash = `${sContextPath.substring(1)}(...)?i-action=${sDefaultCreateHash}`;\n\t\tif (oRouter.getRouteInfoByHash(sHash)) {\n\t\t\treturn sHash;\n\t\t} else {\n\t\t\tthrow new Error(`No route match for creating a new ${sContextPath.substring(1)}`);\n\t\t}\n\t},\n\n\t/**\n\t * Verifies whether the entity in the given context path is editable by looking at the following annotations:\n\t *\n\t * - @Core.OperationAvailable of the edit action\n\t * - @UI.UpdateHidden\n\t *\n\t * Only static values (true/false) are considered here as we don't have the actual binding context to evaluate dynamic bindings.\n\t *\n\t * @param contextPath\n\t * @param metaModel\n\t * @returns Whether the entity is editable or not\n\t */\n\tverifyEditAnnotations: async function (contextPath: string, metaModel: ODataMetaModel): Promise<boolean> {\n\t\tconst entitySetAnnotations: MetaModelEntitySetAnnotation = await metaModel.requestObject(`${contextPath}@`);\n\t\tconst updateHidden = entitySetAnnotations?.[\"@com.sap.vocabularies.UI.v1.UpdateHidden\"];\n\t\tlet operationAvailable = true;\n\n\t\tif (entitySetAnnotations?.[\"@com.sap.vocabularies.Common.v1.DraftRoot\"]?.[\"EditAction\"]) {\n\t\t\toperationAvailable = metaModel.getObject(\n\t\t\t\t`${contextPath}@com.sap.vocabularies.Common.v1.DraftRoot/EditAction@Org.OData.Core.V1.OperationAvailable`\n\t\t\t);\n\t\t}\n\n\t\treturn operationAvailable !== false && updateHidden !== true;\n\t}\n};\n\nexport default AppStartupHelper;\n"],"mappings":"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"}
|
|
409
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["AppStartupHelper","_getKeysFromStartupParams","aKeyNames","oStartupParameters","bAllFound","aKeys","map","name","length","value","undefined","_createFilterFromKeys","bDraftMode","oMetaModel","bFilterCaseSensitive","ModelHelper","isFilteringCaseSensitive","bFilterOnActiveEntity","aFilters","key","Filter","path","operator","FilterOperator","EQ","value1","caseSensitive","oDraftFilter","filters","and","push","_sanitizeKeys","keys","entityType","forEach","propertyType","$Type","trim","match","exec","_requestObjectsFromParameters","aStartupPages","oModel","alwaysCheckExistence","aContextPromises","pageInfo","getMetaModel","getObject","contextPath","technicalKeys","SemanticKeyHelper","getPathContent","context","bindContext","getBoundContext","Promise","resolve","semanticKeys","oFilter","draftMode","oListBind","bindList","$select","join","requestContexts","all","_getReachablePageInfoFromRoute","oRoute","oManifestRouting","sPattern","pattern","replace","endsWith","sTargetName","Array","isArray","target","oTarget","targets","aPatternSegments","split","pageLevel","options","settings","allowDeepLinking","sContextPath","entitySet","metaContext","createBindingContext","objectPath","getInvolvedDataModelObjects","targetEntitySet","oEntityType","aSemanticKeyNames","aSemantickKeys","semKey","$PropertyPath","aTechnicalKeys","_getReachablePages","aRoutes","routes","mPagesByLevel","oPageInfo","aReachablePages","level","_getStartupPagesFromStartupParams","result","current","findRecursive","aCurrentLevelPages","lastPage","nextPage","indexOf","pop","slice","_getDeepLinkObject","aContexts","hash","oContext","getPath","getDeepLinkStartupHash","requestObject","startupPages","allContexts","contextsForStartup","foundContexts","getCreateStartupHash","oRouter","then","oEntitySetAnnotations","sMetaPath","bCreatable","bNewActionOperationAvailable","oInsertRestrictions","Insertable","getDefaultCreateHash","sDefaultCreateHash","preferredMode","sHash","includes","substring","getRouteInfoByHash","Error","verifyEditAnnotations","metaModel","entitySetAnnotations","updateHidden","operationAvailable"],"sourceRoot":".","sources":["AppStartupHelper.ts"],"sourcesContent":["import type { PropertyPath } from \"@sap-ux/vocabularies-types\";\nimport type { StartupParameters } from \"sap/fe/core/AppComponent\";\nimport type { RoutingConfiguration, RoutingRoute } from \"sap/ui/core/Manifest\";\nimport type Router from \"sap/ui/core/routing/Router\";\nimport Filter from \"sap/ui/model/Filter\";\nimport FilterOperator from \"sap/ui/model/FilterOperator\";\nimport type Context from \"sap/ui/model/odata/v4/Context\";\nimport type ODataMetaModel from \"sap/ui/model/odata/v4/ODataMetaModel\";\nimport type ODataModel from \"sap/ui/model/odata/v4/ODataModel\";\nimport type { ExpandPathType, MetaModelEntitySetAnnotation } from \"types/metamodel_types\";\nimport { getInvolvedDataModelObjects } from \"../converters/MetaModelConverter\";\nimport ModelHelper from \"./ModelHelper\";\nimport SemanticKeyHelper from \"./SemanticKeyHelper\";\n\ntype ValuedKey = {\n\tname: string;\n\tvalue: string;\n};\n\ntype PageInfo = {\n\tpattern: string;\n\tcontextPath: string;\n\tdraftMode: boolean;\n\ttechnicalKeys: ValuedKey[] | undefined;\n\tsemanticKeys: ValuedKey[] | undefined;\n\ttarget: string;\n\tpageLevel: number;\n};\n\nconst AppStartupHelper = {\n\t/**\n\t * Retrieves a set of key values from startup parameters.\n\t *\n\t * @param aKeyNames The array of key names\n\t * @param oStartupParameters The startup parameters\n\t * @returns An array of pairs \\{name, value\\} if all key values could be found in the startup parameters, undefined otherwise\n\t */\n\t_getKeysFromStartupParams: function (aKeyNames: string[], oStartupParameters: StartupParameters | null): ValuedKey[] | undefined {\n\t\tlet bAllFound = true;\n\t\tconst aKeys = aKeyNames.map((name) => {\n\t\t\tif (oStartupParameters?.[name] && oStartupParameters[name].length === 1) {\n\t\t\t\treturn { name, value: oStartupParameters[name][0] as string };\n\t\t\t} else {\n\t\t\t\t// A unique key value couldn't be found in the startup parameters\n\t\t\t\tbAllFound = false;\n\t\t\t\treturn { name, value: \"\" };\n\t\t\t}\n\t\t});\n\n\t\treturn bAllFound ? aKeys : undefined;\n\t},\n\n\t/**\n\t * Creates a filter from a list of key values.\n\t *\n\t * @param aKeys Array of semantic keys or technical keys (with values)\n\t * @param bDraftMode True if the entity supports draft mode\n\t * @param oMetaModel The metamodel\n\t * @returns The filter\n\t */\n\t_createFilterFromKeys: function (aKeys: ValuedKey[], bDraftMode: Boolean, oMetaModel: ODataMetaModel): Filter {\n\t\tconst bFilterCaseSensitive = ModelHelper.isFilteringCaseSensitive(oMetaModel);\n\n\t\tlet bFilterOnActiveEntity = false;\n\t\tconst aFilters = aKeys.map((key) => {\n\t\t\tif (key.name === \"IsActiveEntity\") {\n\t\t\t\tbFilterOnActiveEntity = true;\n\t\t\t}\n\t\t\treturn new Filter({\n\t\t\t\tpath: key.name,\n\t\t\t\toperator: FilterOperator.EQ,\n\t\t\t\tvalue1: key.value,\n\t\t\t\tcaseSensitive: bFilterCaseSensitive\n\t\t\t});\n\t\t});\n\t\tif (bDraftMode && !bFilterOnActiveEntity) {\n\t\t\tconst oDraftFilter = new Filter({\n\t\t\t\tfilters: [new Filter(\"IsActiveEntity\", \"EQ\", false), new Filter(\"SiblingEntity/IsActiveEntity\", \"EQ\", null)],\n\t\t\t\tand: false\n\t\t\t});\n\t\t\taFilters.push(oDraftFilter);\n\t\t}\n\n\t\treturn new Filter(aFilters, true);\n\t},\n\n\t/**\n\t * Makes sure key values have the correct format.\n\t * This is to handle special cases, for example, where Boolean values are 'X' or '', or guid values have the following format: guid'00000000-0000-0000-0000-000000000000'.\".\n\t * @param keys\n\t * @param entityType\n\t */\n\t_sanitizeKeys: function (keys: ValuedKey[], entityType: Record<string, { $Type: string }>): void {\n\t\tkeys.forEach((key) => {\n\t\t\tconst propertyType = entityType[key.name].$Type;\n\t\t\tswitch (propertyType) {\n\t\t\t\tcase \"Edm.Boolean\":\n\t\t\t\t\tif (key.value !== \"true\" && key.value !== \"false\") {\n\t\t\t\t\t\t// If the value is neither \"true\" nor \"false\", it's considered as false if it contains only whitespaces (e.g. to handle 'X'/'' for true/false values)\n\t\t\t\t\t\tkey.value = key.value.trim().length ? \"true\" : \"false\";\n\t\t\t\t\t}\n\t\t\t\t\tbreak;\n\n\t\t\t\tcase \"Edm.Guid\": {\n\t\t\t\t\tconst match = /[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}/.exec(key.value);\n\t\t\t\t\tkey.value = match ? match[0] : key.value;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tdefault:\n\t\t\t\t// Do nothing\n\t\t\t}\n\t\t});\n\t},\n\n\t/**\n\t * Loads all contexts for a list of page infos.\n\t *\n\t * @param aStartupPages The list of page infos\n\t * @param oModel The model used to load the contexts\n\t * @param alwaysCheckExistence If true, always check if the object exists, even if semantic keys are present\n\t * @returns A Promise for all contexts\n\t */\n\t_requestObjectsFromParameters: async function (\n\t\taStartupPages: PageInfo[],\n\t\toModel: ODataModel,\n\t\talwaysCheckExistence: boolean\n\t): Promise<Context[][]> {\n\t\t// Load the respective objects for all object pages found in aExternallyNavigablePages\n\t\tconst aContextPromises = aStartupPages.map(async (pageInfo) => {\n\t\t\tconst entityType = oModel.getMetaModel().getObject(`${pageInfo.contextPath}/`);\n\t\t\tif (pageInfo.technicalKeys && !alwaysCheckExistence) {\n\t\t\t\tthis._sanitizeKeys(pageInfo.technicalKeys, entityType);\n\t\t\t\t// Create a context with the proper path\n\t\t\t\tconst path = `${pageInfo.contextPath}(${SemanticKeyHelper.getPathContent(pageInfo.technicalKeys, entityType)})`;\n\t\t\t\tconst context = oModel.bindContext(path).getBoundContext();\n\t\t\t\treturn Promise.resolve([context]);\n\t\t\t} else {\n\t\t\t\tconst aKeys = pageInfo.semanticKeys || pageInfo.technicalKeys || [];\n\t\t\t\tthis._sanitizeKeys(aKeys, entityType);\n\t\t\t\tconst oFilter = this._createFilterFromKeys(aKeys, pageInfo.draftMode, oModel.getMetaModel());\n\n\t\t\t\t// only request a minimum of fields to boost backend performance since this is only used to check if an object exists\n\t\t\t\tconst oListBind = oModel.bindList(pageInfo.contextPath, undefined, undefined, oFilter, {\n\t\t\t\t\t$select: aKeys\n\t\t\t\t\t\t.map((key) => {\n\t\t\t\t\t\t\treturn key.name;\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.join(\",\")\n\t\t\t\t});\n\t\t\t\treturn oListBind.requestContexts(0, 2);\n\t\t\t}\n\t\t});\n\n\t\treturn Promise.all(aContextPromises);\n\t},\n\n\t/**\n\t * Creates a PageInfo from a route if it's reachable from the startup parameters.\n\t *\n\t * @param oRoute The route\n\t * @param oManifestRouting The app manifest routing section\n\t * @param oStartupParameters The startup parameters\n\t * @param oMetaModel The app metamodel\n\t * @returns A page info if the page is reachable, undefined otherwise\n\t */\n\t_getReachablePageInfoFromRoute: function (\n\t\toRoute: RoutingRoute,\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toMetaModel: ODataMetaModel\n\t): PageInfo | undefined {\n\t\t// Remove trailing ':?query:' and '/'\n\t\tlet sPattern: string = oRoute.pattern!.replace(\":?query:\", \"\");\n\t\tsPattern = sPattern.replace(/\\/$/, \"\");\n\n\t\tif (!sPattern || !sPattern.endsWith(\")\")) {\n\t\t\t// Ignore level-0 routes (ListReport) or routes corresponding to a 1-1 relation (no keys in the URL in this case)\n\t\t\treturn undefined;\n\t\t}\n\n\t\tsPattern = sPattern.replace(/\\(\\{[^}]*\\}\\)/g, \"(#)\"); // Replace keys with #\n\n\t\t// Get the rightmost target for this route\n\t\tconst sTargetName: string = Array.isArray(oRoute.target) ? oRoute.target[oRoute.target.length - 1] : oRoute.target;\n\t\tconst oTarget = oManifestRouting?.targets?.[sTargetName];\n\n\t\tconst aPatternSegments = sPattern.split(\"/\");\n\t\tconst pageLevel = aPatternSegments.length - 1;\n\n\t\tif (pageLevel !== 0 && oTarget?.options?.settings?.allowDeepLinking !== true) {\n\t\t\t// The first level of object page allows deep linking by default.\n\t\t\t// Otherwise, the target must allow deep linking explicitely in the manifest\n\t\t\treturn undefined;\n\t\t}\n\n\t\tlet sContextPath = \"\";\n\t\tif (oTarget?.options?.settings?.entitySet) {\n\t\t\tsContextPath = `/${oTarget.options.settings.entitySet}`;\n\t\t} else if (oTarget?.options?.settings?.contextPath) {\n\t\t\tconst metaContext = oMetaModel.createBindingContext(oTarget.options.settings.contextPath)!;\n\t\t\tconst objectPath = getInvolvedDataModelObjects(metaContext);\n\t\t\tif (objectPath.targetEntitySet) {\n\t\t\t\tsContextPath = `/${objectPath.targetEntitySet.name}`;\n\t\t\t}\n\t\t}\n\n\t\tif (!sContextPath) {\n\t\t\treturn undefined;\n\t\t}\n\t\tconst oEntityType = sContextPath && oMetaModel.getObject(`/$EntityContainer${sContextPath}/`);\n\n\t\tif (!oEntityType) {\n\t\t\treturn undefined;\n\t\t}\n\n\t\t// Get the semantic key values for the entity\n\t\tconst aSemanticKeyNames: ExpandPathType<PropertyPath>[] = oMetaModel.getObject(\n\t\t\t`/$EntityContainer${sContextPath}/@com.sap.vocabularies.Common.v1.SemanticKey`\n\t\t);\n\n\t\tconst aSemantickKeys = aSemanticKeyNames\n\t\t\t? this._getKeysFromStartupParams(\n\t\t\t\t\taSemanticKeyNames.map((semKey) => {\n\t\t\t\t\t\treturn semKey.$PropertyPath;\n\t\t\t\t\t}),\n\t\t\t\t\toStartupParameters\n\t\t\t  )\n\t\t\t: undefined;\n\n\t\t// Get the technical keys only if we couldn't find the semantic key values, and on first level OP\n\t\tconst aTechnicalKeys =\n\t\t\t!aSemantickKeys && pageLevel === 0 ? this._getKeysFromStartupParams(oEntityType[\"$Key\"], oStartupParameters) : undefined;\n\n\t\tif (aSemantickKeys === undefined && aTechnicalKeys === undefined) {\n\t\t\t// We couldn't find the semantic/technical keys in the startup parameters\n\t\t\treturn undefined;\n\t\t}\n\n\t\t// The startup parameters contain values for all semantic keys (or technical keys) --> we can store the page info in the corresponding level\n\t\tconst draftMode =\n\t\t\toMetaModel.getObject(`/$EntityContainer${sContextPath}@com.sap.vocabularies.Common.v1.DraftRoot`) ||\n\t\t\toMetaModel.getObject(`/$EntityContainer${sContextPath}@com.sap.vocabularies.Common.v1.DraftNode`)\n\t\t\t\t? true\n\t\t\t\t: false;\n\n\t\treturn {\n\t\t\tpattern: sPattern,\n\t\t\tcontextPath: sContextPath,\n\t\t\tdraftMode,\n\t\t\ttechnicalKeys: aTechnicalKeys,\n\t\t\tsemanticKeys: aSemantickKeys,\n\t\t\ttarget: sTargetName,\n\t\t\tpageLevel\n\t\t};\n\t},\n\n\t/**\n\t * Returns the list of all pages that allow deeplink and that can be reached using the startup parameters.\n\t *\n\t * @param oManifestRouting The routing information from the app manifest\n\t * @param oStartupParameters The startup parameters\n\t * @param oMetaModel The metamodel\n\t * @returns The reachable pages\n\t */\n\t_getReachablePages: function (\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toMetaModel: ODataMetaModel\n\t): PageInfo[][] {\n\t\tconst aRoutes: RoutingRoute[] = oManifestRouting?.routes ?? [];\n\t\tconst mPagesByLevel: Record<number, PageInfo[]> = {};\n\n\t\taRoutes.forEach((oRoute) => {\n\t\t\tconst oPageInfo = this._getReachablePageInfoFromRoute(oRoute, oManifestRouting, oStartupParameters, oMetaModel);\n\n\t\t\tif (oPageInfo) {\n\t\t\t\tif (!mPagesByLevel[oPageInfo.pageLevel]) {\n\t\t\t\t\tmPagesByLevel[oPageInfo.pageLevel] = [];\n\t\t\t\t}\n\t\t\t\tmPagesByLevel[oPageInfo.pageLevel].push(oPageInfo);\n\t\t\t}\n\t\t});\n\n\t\t// A page is reachable only if all its parents are also reachable\n\t\t// So if we couldn't find any pages for a given level, all pages with a higher level won't be reachable anyway\n\t\tconst aReachablePages: PageInfo[][] = [];\n\t\tlet level = 0;\n\t\twhile (mPagesByLevel[level]) {\n\t\t\taReachablePages.push(mPagesByLevel[level]);\n\t\t\tlevel++;\n\t\t}\n\n\t\treturn aReachablePages;\n\t},\n\n\t/**\n\t * Get the list of startup pages.\n\t *\n\t * @param oManifestRouting The routing information from the app manifest\n\t * @param oStartupParameters The startup parameters\n\t * @param oMetaModel The metamodel\n\t * @returns An array of startup page infos\n\t */\n\t_getStartupPagesFromStartupParams: function (\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toMetaModel: ODataMetaModel\n\t): PageInfo[] {\n\t\t// Find all pages that can be reached with the startup parameters\n\t\tconst aReachablePages = this._getReachablePages(oManifestRouting, oStartupParameters, oMetaModel);\n\n\t\tif (aReachablePages.length === 0) {\n\t\t\treturn [];\n\t\t}\n\n\t\t// Find the longest sequence of pages that can be reached (recursively)\n\t\tlet result: PageInfo[] = [];\n\t\tconst current: PageInfo[] = [];\n\n\t\tfunction findRecursive(level: number): void {\n\t\t\tconst aCurrentLevelPages = aReachablePages[level];\n\t\t\tconst lastPage = current.length ? current[current.length - 1] : undefined;\n\n\t\t\tif (aCurrentLevelPages) {\n\t\t\t\taCurrentLevelPages.forEach(function (nextPage) {\n\t\t\t\t\tif (!lastPage || nextPage.pattern.indexOf(lastPage.pattern) === 0) {\n\t\t\t\t\t\t// We only consider pages that can be reached from the page at the previous level,\n\t\t\t\t\t\t// --> their pattern must be the pattern of the previous page with another segment appended\n\t\t\t\t\t\tcurrent.push(nextPage);\n\t\t\t\t\t\tfindRecursive(level + 1);\n\t\t\t\t\t\tcurrent.pop();\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t}\n\t\t\tif (current.length > result.length) {\n\t\t\t\tresult = current.slice(); // We have found a sequence longer than our previous best --> store it as the new longest\n\t\t\t}\n\t\t}\n\n\t\tfindRecursive(0);\n\n\t\treturn result;\n\t},\n\n\t/**\n\t * Creates the startup object from the list of pages and contexts.\n\t *\n\t * @param aStartupPages The pages\n\t * @param aContexts The contexts\n\t * @returns An object containing either a hash or a context to navigate to, or an empty object if no deep link was found\n\t */\n\t_getDeepLinkObject: function (aStartupPages: PageInfo[], aContexts: Context[]): { hash?: string; context?: Context } {\n\t\tif (aContexts.length === 1) {\n\t\t\treturn { context: aContexts[0] };\n\t\t} else if (aContexts.length > 1) {\n\t\t\t// Navigation to a deeper level --> use the pattern of the deepest object page\n\t\t\t// and replace the parameters by the ID from the contexts\n\t\t\tlet hash = aStartupPages[aStartupPages.length - 1].pattern;\n\t\t\taContexts.forEach(function (oContext) {\n\t\t\t\thash = hash.replace(\"(#)\", `(${oContext.getPath().split(\"(\")[1]}`);\n\t\t\t});\n\n\t\t\treturn { hash };\n\t\t} else {\n\t\t\treturn {};\n\t\t}\n\t},\n\n\t/**\n\t * Calculates startup parameters for a deeplink case, from startup parameters and routing infoirmation.\n\t *\n\t * @param oManifestRouting The routing information from the app manifest\n\t * @param oStartupParameters The startup parameters\n\t * @param oModel The OData model\n\t * @param alwaysCheckExistence\n\t * @returns An object containing either a hash or a context to navigate to, or an empty object if no deep link was found\n\t */\n\tgetDeepLinkStartupHash: async function (\n\t\toManifestRouting: RoutingConfiguration | undefined,\n\t\toStartupParameters: StartupParameters | null,\n\t\toModel: ODataModel,\n\t\talwaysCheckExistence: boolean\n\t): Promise<{ hash?: string; context?: Context }> {\n\t\tawait oModel.getMetaModel().requestObject(\"/$EntityContainer/\");\n\n\t\t// Check if semantic keys are present in url parameters for every object page at each level\n\t\tconst startupPages = this._getStartupPagesFromStartupParams(oManifestRouting, oStartupParameters, oModel.getMetaModel());\n\n\t\t// Load the contexts\n\t\tconst allContexts = await this._requestObjectsFromParameters(startupPages, oModel, alwaysCheckExistence);\n\t\tif (allContexts.length) {\n\t\t\t// Make sure we only get 1 context per promise, and flatten the array\n\t\t\tconst contextsForStartup: Context[] = [];\n\t\t\tallContexts.forEach(function (foundContexts) {\n\t\t\t\tif (foundContexts.length === 1) {\n\t\t\t\t\tcontextsForStartup.push(foundContexts[0]);\n\t\t\t\t}\n\t\t\t});\n\n\t\t\treturn contextsForStartup.length === allContexts.length ? this._getDeepLinkObject(startupPages, contextsForStartup) : {};\n\t\t} else {\n\t\t\treturn {};\n\t\t}\n\t},\n\n\t/**\n\t * Calculates the new hash based on the startup parameters.\n\t *\n\t * @param oStartupParameters The startup parameter values (map parameter name -> array of values)\n\t * @param sContextPath The context path for the startup of the app (generally the path to the main entity set)\n\t * @param oRouter The router instance\n\t * @param oMetaModel The meta model\n\t * @returns A promise containing the hash to navigate to, or an empty string if there's no need to navigate\n\t */\n\tgetCreateStartupHash: async function (\n\t\toStartupParameters: StartupParameters,\n\t\tsContextPath: string,\n\t\toRouter: Router,\n\t\toMetaModel: ODataMetaModel\n\t): Promise<string> {\n\t\treturn oMetaModel.requestObject(`${sContextPath}@`).then((oEntitySetAnnotations: MetaModelEntitySetAnnotation) => {\n\t\t\tlet sMetaPath = \"\";\n\t\t\tlet bCreatable = true;\n\n\t\t\tif (\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Common.v1.DraftRoot\"] &&\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Common.v1.DraftRoot\"][\"NewAction\"]\n\t\t\t) {\n\t\t\t\tsMetaPath = `${sContextPath}@com.sap.vocabularies.Common.v1.DraftRoot/NewAction@Org.OData.Core.V1.OperationAvailable`;\n\t\t\t} else if (\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Session.v1.StickySessionSupported\"] &&\n\t\t\t\toEntitySetAnnotations[\"@com.sap.vocabularies.Session.v1.StickySessionSupported\"][\"NewAction\"]\n\t\t\t) {\n\t\t\t\tsMetaPath = `${sContextPath}@com.sap.vocabularies.Session.v1.StickySessionSupported/NewAction@Org.OData.Core.V1.OperationAvailable`;\n\t\t\t}\n\n\t\t\tif (sMetaPath) {\n\t\t\t\tconst bNewActionOperationAvailable = oMetaModel.getObject(sMetaPath);\n\t\t\t\tif (bNewActionOperationAvailable === false) {\n\t\t\t\t\tbCreatable = false;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconst oInsertRestrictions = oEntitySetAnnotations[\"@Org.OData.Capabilities.V1.InsertRestrictions\"];\n\t\t\t\tif (oInsertRestrictions && oInsertRestrictions.Insertable === false) {\n\t\t\t\t\tbCreatable = false;\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (bCreatable) {\n\t\t\t\treturn this.getDefaultCreateHash(oStartupParameters, sContextPath, oRouter);\n\t\t\t} else {\n\t\t\t\treturn \"\";\n\t\t\t}\n\t\t});\n\t},\n\n\t/**\n\t * Calculates the hash to create a new object.\n\t *\n\t * @param oStartupParameters The startup parameter values (map parameter name -> array of values)\n\t * @param sContextPath The context path of the entity set to be used for the creation\n\t * @param oRouter The router instance\n\t * @returns The hash\n\t */\n\tgetDefaultCreateHash: function (\n\t\toStartupParameters: StartupParameters | null | undefined,\n\t\tsContextPath: string,\n\t\toRouter: Router\n\t): string {\n\t\tlet sDefaultCreateHash = oStartupParameters && oStartupParameters.preferredMode ? oStartupParameters.preferredMode[0] : \"create\";\n\t\tlet sHash = \"\";\n\n\t\tsDefaultCreateHash =\n\t\t\tsDefaultCreateHash.includes(\":\") && sDefaultCreateHash.length > sDefaultCreateHash.indexOf(\":\") + 1\n\t\t\t\t? sDefaultCreateHash.substring(0, sDefaultCreateHash.indexOf(\":\"))\n\t\t\t\t: \"create\";\n\t\tsHash = `${sContextPath.substring(1)}(...)?i-action=${sDefaultCreateHash}`;\n\t\tif (oRouter.getRouteInfoByHash(sHash)) {\n\t\t\treturn sHash;\n\t\t} else {\n\t\t\tthrow new Error(`No route match for creating a new ${sContextPath.substring(1)}`);\n\t\t}\n\t},\n\n\t/**\n\t * Verifies whether the entity in the given context path is editable by looking at the following annotations:\n\t *\n\t * - @Core.OperationAvailable of the edit action\n\t * - @UI.UpdateHidden\n\t *\n\t * Only static values (true/false) are considered here as we don't have the actual binding context to evaluate dynamic bindings.\n\t *\n\t * @param contextPath\n\t * @param metaModel\n\t * @returns Whether the entity is editable or not\n\t */\n\tverifyEditAnnotations: async function (contextPath: string, metaModel: ODataMetaModel): Promise<boolean> {\n\t\tconst entitySetAnnotations: MetaModelEntitySetAnnotation = await metaModel.requestObject(`${contextPath}@`);\n\t\tconst updateHidden = entitySetAnnotations?.[\"@com.sap.vocabularies.UI.v1.UpdateHidden\"];\n\t\tlet operationAvailable = true;\n\n\t\tif (entitySetAnnotations?.[\"@com.sap.vocabularies.Common.v1.DraftRoot\"]?.[\"EditAction\"]) {\n\t\t\toperationAvailable = metaModel.getObject(\n\t\t\t\t`${contextPath}@com.sap.vocabularies.Common.v1.DraftRoot/EditAction@Org.OData.Core.V1.OperationAvailable`\n\t\t\t);\n\t\t}\n\n\t\treturn operationAvailable !== false && updateHidden !== true;\n\t}\n};\n\nexport default AppStartupHelper;\n"],"mappings":";AAAA;AAAA;AAAA;;;;;EA6BA,MAAMA,gBAAgB,GAAG;IACxB;AACD;AACA;AACA;AACA;AACA;AACA;IACCC,yBAAyB,EAAE,UAAUC,SAAmB,EAAEC,kBAA4C,EAA2B;MAChI,IAAIC,SAAS,GAAG,IAAI;MACpB,MAAMC,KAAK,GAAGH,SAAS,CAACI,GAAG,CAAEC,IAAI,IAAK;QACrC,IAAIJ,kBAAkB,aAAlBA,kBAAkB,eAAlBA,kBAAkB,CAAGI,IAAI,CAAC,IAAIJ,kBAAkB,CAACI,IAAI,CAAC,CAACC,MAAM,KAAK,CAAC,EAAE;UACxE,OAAO;YAAED,IAAI;YAAEE,KAAK,EAAEN,kBAAkB,CAACI,IAAI,CAAC,CAAC,CAAC;UAAY,CAAC;QAC9D,CAAC,MAAM;UACN;UACAH,SAAS,GAAG,KAAK;UACjB,OAAO;YAAEG,IAAI;YAAEE,KAAK,EAAE;UAAG,CAAC;QAC3B;MACD,CAAC,CAAC;MAEF,OAAOL,SAAS,GAAGC,KAAK,GAAGK,SAAS;IACrC,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;IACCC,qBAAqB,EAAE,UAAUN,KAAkB,EAAEO,UAAmB,EAAEC,UAA0B,EAAU;MAC7G,MAAMC,oBAAoB,GAAGC,WAAW,CAACC,wBAAwB,CAACH,UAAU,CAAC;MAE7E,IAAII,qBAAqB,GAAG,KAAK;MACjC,MAAMC,QAAQ,GAAGb,KAAK,CAACC,GAAG,CAAEa,GAAG,IAAK;QACnC,IAAIA,GAAG,CAACZ,IAAI,KAAK,gBAAgB,EAAE;UAClCU,qBAAqB,GAAG,IAAI;QAC7B;QACA,OAAO,IAAIG,MAAM,CAAC;UACjBC,IAAI,EAAEF,GAAG,CAACZ,IAAI;UACde,QAAQ,EAAEC,cAAc,CAACC,EAAE;UAC3BC,MAAM,EAAEN,GAAG,CAACV,KAAK;UACjBiB,aAAa,EAAEZ;QAChB,CAAC,CAAC;MACH,CAAC,CAAC;MACF,IAAIF,UAAU,IAAI,CAACK,qBAAqB,EAAE;QACzC,MAAMU,YAAY,GAAG,IAAIP,MAAM,CAAC;UAC/BQ,OAAO,EAAE,CAAC,IAAIR,MAAM,CAAC,gBAAgB,EAAE,IAAI,EAAE,KAAK,CAAC,EAAE,IAAIA,MAAM,CAAC,8BAA8B,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;UAC5GS,GAAG,EAAE;QACN,CAAC,CAAC;QACFX,QAAQ,CAACY,IAAI,CAACH,YAAY,CAAC;MAC5B;MAEA,OAAO,IAAIP,MAAM,CAACF,QAAQ,EAAE,IAAI,CAAC;IAClC,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;IACCa,aAAa,EAAE,UAAUC,IAAiB,EAAEC,UAA6C,EAAQ;MAChGD,IAAI,CAACE,OAAO,CAAEf,GAAG,IAAK;QACrB,MAAMgB,YAAY,GAAGF,UAAU,CAACd,GAAG,CAACZ,IAAI,CAAC,CAAC6B,KAAK;QAC/C,QAAQD,YAAY;UACnB,KAAK,aAAa;YACjB,IAAIhB,GAAG,CAACV,KAAK,KAAK,MAAM,IAAIU,GAAG,CAACV,KAAK,KAAK,OAAO,EAAE;cAClD;cACAU,GAAG,CAACV,KAAK,GAAGU,GAAG,CAACV,KAAK,CAAC4B,IAAI,EAAE,CAAC7B,MAAM,GAAG,MAAM,GAAG,OAAO;YACvD;YACA;UAED,KAAK,UAAU;YAAE;cAChB,MAAM8B,KAAK,GAAG,6EAA6E,CAACC,IAAI,CAACpB,GAAG,CAACV,KAAK,CAAC;cAC3GU,GAAG,CAACV,KAAK,GAAG6B,KAAK,GAAGA,KAAK,CAAC,CAAC,CAAC,GAAGnB,GAAG,CAACV,KAAK;cACxC;YACD;UAEA;UACA;QAAA;MAEF,CAAC,CAAC;IACH,CAAC;;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;IACC+B,6BAA6B,EAAE,gBAC9BC,aAAyB,EACzBC,MAAkB,EAClBC,oBAA6B,EACN;MACvB;MACA,MAAMC,gBAAgB,GAAGH,aAAa,CAACnC,GAAG,CAAC,MAAOuC,QAAQ,IAAK;QAC9D,MAAMZ,UAAU,GAAGS,MAAM,CAACI,YAAY,EAAE,CAACC,SAAS,CAAE,GAAEF,QAAQ,CAACG,WAAY,GAAE,CAAC;QAC9E,IAAIH,QAAQ,CAACI,aAAa,IAAI,CAACN,oBAAoB,EAAE;UACpD,IAAI,CAACZ,aAAa,CAACc,QAAQ,CAACI,aAAa,EAAEhB,UAAU,CAAC;UACtD;UACA,MAAMZ,IAAI,GAAI,GAAEwB,QAAQ,CAACG,WAAY,IAAGE,iBAAiB,CAACC,cAAc,CAACN,QAAQ,CAACI,aAAa,EAAEhB,UAAU,CAAE,GAAE;UAC/G,MAAMmB,OAAO,GAAGV,MAAM,CAACW,WAAW,CAAChC,IAAI,CAAC,CAACiC,eAAe,EAAE;UAC1D,OAAOC,OAAO,CAACC,OAAO,CAAC,CAACJ,OAAO,CAAC,CAAC;QAClC,CAAC,MAAM;UACN,MAAM/C,KAAK,GAAGwC,QAAQ,CAACY,YAAY,IAAIZ,QAAQ,CAACI,aAAa,IAAI,EAAE;UACnE,IAAI,CAAClB,aAAa,CAAC1B,KAAK,EAAE4B,UAAU,CAAC;UACrC,MAAMyB,OAAO,GAAG,IAAI,CAAC/C,qBAAqB,CAACN,KAAK,EAAEwC,QAAQ,CAACc,SAAS,EAAEjB,MAAM,CAACI,YAAY,EAAE,CAAC;;UAE5F;UACA,MAAMc,SAAS,GAAGlB,MAAM,CAACmB,QAAQ,CAAChB,QAAQ,CAACG,WAAW,EAAEtC,SAAS,EAAEA,SAAS,EAAEgD,OAAO,EAAE;YACtFI,OAAO,EAAEzD,KAAK,CACZC,GAAG,CAAEa,GAAG,IAAK;cACb,OAAOA,GAAG,CAACZ,IAAI;YAChB,CAAC,CAAC,CACDwD,IAAI,CAAC,GAAG;UACX,CAAC,CAAC;UACF,OAAOH,SAAS,CAACI,eAAe,CAAC,CAAC,EAAE,CAAC,CAAC;QACvC;MACD,CAAC,CAAC;MAEF,OAAOT,OAAO,CAACU,GAAG,CAACrB,gBAAgB,CAAC;IACrC,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;IACCsB,8BAA8B,EAAE,UAC/BC,MAAoB,EACpBC,gBAAkD,EAClDjE,kBAA4C,EAC5CU,UAA0B,EACH;MAAA;MACvB;MACA,IAAIwD,QAAgB,GAAGF,MAAM,CAACG,OAAO,CAAEC,OAAO,CAAC,UAAU,EAAE,EAAE,CAAC;MAC9DF,QAAQ,GAAGA,QAAQ,CAACE,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC;MAEtC,IAAI,CAACF,QAAQ,IAAI,CAACA,QAAQ,CAACG,QAAQ,CAAC,GAAG,CAAC,EAAE;QACzC;QACA,OAAO9D,SAAS;MACjB;MAEA2D,QAAQ,GAAGA,QAAQ,CAACE,OAAO,CAAC,gBAAgB,EAAE,KAAK,CAAC,CAAC,CAAC;;MAEtD;MACA,MAAME,WAAmB,GAAGC,KAAK,CAACC,OAAO,CAACR,MAAM,CAACS,MAAM,CAAC,GAAGT,MAAM,CAACS,MAAM,CAACT,MAAM,CAACS,MAAM,CAACpE,MAAM,GAAG,CAAC,CAAC,GAAG2D,MAAM,CAACS,MAAM;MAClH,MAAMC,OAAO,GAAGT,gBAAgB,aAAhBA,gBAAgB,gDAAhBA,gBAAgB,CAAEU,OAAO,0DAAzB,sBAA4BL,WAAW,CAAC;MAExD,MAAMM,gBAAgB,GAAGV,QAAQ,CAACW,KAAK,CAAC,GAAG,CAAC;MAC5C,MAAMC,SAAS,GAAGF,gBAAgB,CAACvE,MAAM,GAAG,CAAC;MAE7C,IAAIyE,SAAS,KAAK,CAAC,IAAI,CAAAJ,OAAO,aAAPA,OAAO,2CAAPA,OAAO,CAAEK,OAAO,8EAAhB,iBAAkBC,QAAQ,0DAA1B,sBAA4BC,gBAAgB,MAAK,IAAI,EAAE;QAC7E;QACA;QACA,OAAO1E,SAAS;MACjB;MAEA,IAAI2E,YAAY,GAAG,EAAE;MACrB,IAAIR,OAAO,aAAPA,OAAO,oCAAPA,OAAO,CAAEK,OAAO,uEAAhB,kBAAkBC,QAAQ,kDAA1B,sBAA4BG,SAAS,EAAE;QAC1CD,YAAY,GAAI,IAAGR,OAAO,CAACK,OAAO,CAACC,QAAQ,CAACG,SAAU,EAAC;MACxD,CAAC,MAAM,IAAIT,OAAO,aAAPA,OAAO,oCAAPA,OAAO,CAAEK,OAAO,uEAAhB,kBAAkBC,QAAQ,kDAA1B,sBAA4BnC,WAAW,EAAE;QACnD,MAAMuC,WAAW,GAAG1E,UAAU,CAAC2E,oBAAoB,CAACX,OAAO,CAACK,OAAO,CAACC,QAAQ,CAACnC,WAAW,CAAE;QAC1F,MAAMyC,UAAU,GAAGC,2BAA2B,CAACH,WAAW,CAAC;QAC3D,IAAIE,UAAU,CAACE,eAAe,EAAE;UAC/BN,YAAY,GAAI,IAAGI,UAAU,CAACE,eAAe,CAACpF,IAAK,EAAC;QACrD;MACD;MAEA,IAAI,CAAC8E,YAAY,EAAE;QAClB,OAAO3E,SAAS;MACjB;MACA,MAAMkF,WAAW,GAAGP,YAAY,IAAIxE,UAAU,CAACkC,SAAS,CAAE,oBAAmBsC,YAAa,GAAE,CAAC;MAE7F,IAAI,CAACO,WAAW,EAAE;QACjB,OAAOlF,SAAS;MACjB;;MAEA;MACA,MAAMmF,iBAAiD,GAAGhF,UAAU,CAACkC,SAAS,CAC5E,oBAAmBsC,YAAa,8CAA6C,CAC9E;MAED,MAAMS,cAAc,GAAGD,iBAAiB,GACrC,IAAI,CAAC5F,yBAAyB,CAC9B4F,iBAAiB,CAACvF,GAAG,CAAEyF,MAAM,IAAK;QACjC,OAAOA,MAAM,CAACC,aAAa;MAC5B,CAAC,CAAC,EACF7F,kBAAkB,CACjB,GACDO,SAAS;;MAEZ;MACA,MAAMuF,cAAc,GACnB,CAACH,cAAc,IAAIb,SAAS,KAAK,CAAC,GAAG,IAAI,CAAChF,yBAAyB,CAAC2F,WAAW,CAAC,MAAM,CAAC,EAAEzF,kBAAkB,CAAC,GAAGO,SAAS;MAEzH,IAAIoF,cAAc,KAAKpF,SAAS,IAAIuF,cAAc,KAAKvF,SAAS,EAAE;QACjE;QACA,OAAOA,SAAS;MACjB;;MAEA;MACA,MAAMiD,SAAS,GACd9C,UAAU,CAACkC,SAAS,CAAE,oBAAmBsC,YAAa,2CAA0C,CAAC,IACjGxE,UAAU,CAACkC,SAAS,CAAE,oBAAmBsC,YAAa,2CAA0C,CAAC,GAC9F,IAAI,GACJ,KAAK;MAET,OAAO;QACNf,OAAO,EAAED,QAAQ;QACjBrB,WAAW,EAAEqC,YAAY;QACzB1B,SAAS;QACTV,aAAa,EAAEgD,cAAc;QAC7BxC,YAAY,EAAEqC,cAAc;QAC5BlB,MAAM,EAAEH,WAAW;QACnBQ;MACD,CAAC;IACF,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;IACCiB,kBAAkB,EAAE,UACnB9B,gBAAkD,EAClDjE,kBAA4C,EAC5CU,UAA0B,EACX;MACf,MAAMsF,OAAuB,GAAG,CAAA/B,gBAAgB,aAAhBA,gBAAgB,uBAAhBA,gBAAgB,CAAEgC,MAAM,KAAI,EAAE;MAC9D,MAAMC,aAAyC,GAAG,CAAC,CAAC;MAEpDF,OAAO,CAACjE,OAAO,CAAEiC,MAAM,IAAK;QAC3B,MAAMmC,SAAS,GAAG,IAAI,CAACpC,8BAA8B,CAACC,MAAM,EAAEC,gBAAgB,EAAEjE,kBAAkB,EAAEU,UAAU,CAAC;QAE/G,IAAIyF,SAAS,EAAE;UACd,IAAI,CAACD,aAAa,CAACC,SAAS,CAACrB,SAAS,CAAC,EAAE;YACxCoB,aAAa,CAACC,SAAS,CAACrB,SAAS,CAAC,GAAG,EAAE;UACxC;UACAoB,aAAa,CAACC,SAAS,CAACrB,SAAS,CAAC,CAACnD,IAAI,CAACwE,SAAS,CAAC;QACnD;MACD,CAAC,CAAC;;MAEF;MACA;MACA,MAAMC,eAA6B,GAAG,EAAE;MACxC,IAAIC,KAAK,GAAG,CAAC;MACb,OAAOH,aAAa,CAACG,KAAK,CAAC,EAAE;QAC5BD,eAAe,CAACzE,IAAI,CAACuE,aAAa,CAACG,KAAK,CAAC,CAAC;QAC1CA,KAAK,EAAE;MACR;MAEA,OAAOD,eAAe;IACvB,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;IACCE,iCAAiC,EAAE,UAClCrC,gBAAkD,EAClDjE,kBAA4C,EAC5CU,UAA0B,EACb;MACb;MACA,MAAM0F,eAAe,GAAG,IAAI,CAACL,kBAAkB,CAAC9B,gBAAgB,EAAEjE,kBAAkB,EAAEU,UAAU,CAAC;MAEjG,IAAI0F,eAAe,CAAC/F,MAAM,KAAK,CAAC,EAAE;QACjC,OAAO,EAAE;MACV;;MAEA;MACA,IAAIkG,MAAkB,GAAG,EAAE;MAC3B,MAAMC,OAAmB,GAAG,EAAE;MAE9B,SAASC,aAAa,CAACJ,KAAa,EAAQ;QAC3C,MAAMK,kBAAkB,GAAGN,eAAe,CAACC,KAAK,CAAC;QACjD,MAAMM,QAAQ,GAAGH,OAAO,CAACnG,MAAM,GAAGmG,OAAO,CAACA,OAAO,CAACnG,MAAM,GAAG,CAAC,CAAC,GAAGE,SAAS;QAEzE,IAAImG,kBAAkB,EAAE;UACvBA,kBAAkB,CAAC3E,OAAO,CAAC,UAAU6E,QAAQ,EAAE;YAC9C,IAAI,CAACD,QAAQ,IAAIC,QAAQ,CAACzC,OAAO,CAAC0C,OAAO,CAACF,QAAQ,CAACxC,OAAO,CAAC,KAAK,CAAC,EAAE;cAClE;cACA;cACAqC,OAAO,CAAC7E,IAAI,CAACiF,QAAQ,CAAC;cACtBH,aAAa,CAACJ,KAAK,GAAG,CAAC,CAAC;cACxBG,OAAO,CAACM,GAAG,EAAE;YACd;UACD,CAAC,CAAC;QACH;QACA,IAAIN,OAAO,CAACnG,MAAM,GAAGkG,MAAM,CAAClG,MAAM,EAAE;UACnCkG,MAAM,GAAGC,OAAO,CAACO,KAAK,EAAE,CAAC,CAAC;QAC3B;MACD;;MAEAN,aAAa,CAAC,CAAC,CAAC;MAEhB,OAAOF,MAAM;IACd,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;IACCS,kBAAkB,EAAE,UAAU1E,aAAyB,EAAE2E,SAAoB,EAAwC;MACpH,IAAIA,SAAS,CAAC5G,MAAM,KAAK,CAAC,EAAE;QAC3B,OAAO;UAAE4C,OAAO,EAAEgE,SAAS,CAAC,CAAC;QAAE,CAAC;MACjC,CAAC,MAAM,IAAIA,SAAS,CAAC5G,MAAM,GAAG,CAAC,EAAE;QAChC;QACA;QACA,IAAI6G,IAAI,GAAG5E,aAAa,CAACA,aAAa,CAACjC,MAAM,GAAG,CAAC,CAAC,CAAC8D,OAAO;QAC1D8C,SAAS,CAAClF,OAAO,CAAC,UAAUoF,QAAQ,EAAE;UACrCD,IAAI,GAAGA,IAAI,CAAC9C,OAAO,CAAC,KAAK,EAAG,IAAG+C,QAAQ,CAACC,OAAO,EAAE,CAACvC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAE,EAAC,CAAC;QACnE,CAAC,CAAC;QAEF,OAAO;UAAEqC;QAAK,CAAC;MAChB,CAAC,MAAM;QACN,OAAO,CAAC,CAAC;MACV;IACD,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;IACCG,sBAAsB,EAAE,gBACvBpD,gBAAkD,EAClDjE,kBAA4C,EAC5CuC,MAAkB,EAClBC,oBAA6B,EACmB;MAChD,MAAMD,MAAM,CAACI,YAAY,EAAE,CAAC2E,aAAa,CAAC,oBAAoB,CAAC;;MAE/D;MACA,MAAMC,YAAY,GAAG,IAAI,CAACjB,iCAAiC,CAACrC,gBAAgB,EAAEjE,kBAAkB,EAAEuC,MAAM,CAACI,YAAY,EAAE,CAAC;;MAExH;MACA,MAAM6E,WAAW,GAAG,MAAM,IAAI,CAACnF,6BAA6B,CAACkF,YAAY,EAAEhF,MAAM,EAAEC,oBAAoB,CAAC;MACxG,IAAIgF,WAAW,CAACnH,MAAM,EAAE;QACvB;QACA,MAAMoH,kBAA6B,GAAG,EAAE;QACxCD,WAAW,CAACzF,OAAO,CAAC,UAAU2F,aAAa,EAAE;UAC5C,IAAIA,aAAa,CAACrH,MAAM,KAAK,CAAC,EAAE;YAC/BoH,kBAAkB,CAAC9F,IAAI,CAAC+F,aAAa,CAAC,CAAC,CAAC,CAAC;UAC1C;QACD,CAAC,CAAC;QAEF,OAAOD,kBAAkB,CAACpH,MAAM,KAAKmH,WAAW,CAACnH,MAAM,GAAG,IAAI,CAAC2G,kBAAkB,CAACO,YAAY,EAAEE,kBAAkB,CAAC,GAAG,CAAC,CAAC;MACzH,CAAC,MAAM;QACN,OAAO,CAAC,CAAC;MACV;IACD,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;IACCE,oBAAoB,EAAE,gBACrB3H,kBAAqC,EACrCkF,YAAoB,EACpB0C,OAAe,EACflH,UAA0B,EACR;MAClB,OAAOA,UAAU,CAAC4G,aAAa,CAAE,GAAEpC,YAAa,GAAE,CAAC,CAAC2C,IAAI,CAAEC,qBAAmD,IAAK;QACjH,IAAIC,SAAS,GAAG,EAAE;QAClB,IAAIC,UAAU,GAAG,IAAI;QAErB,IACCF,qBAAqB,CAAC,2CAA2C,CAAC,IAClEA,qBAAqB,CAAC,2CAA2C,CAAC,CAAC,WAAW,CAAC,EAC9E;UACDC,SAAS,GAAI,GAAE7C,YAAa,0FAAyF;QACtH,CAAC,MAAM,IACN4C,qBAAqB,CAAC,yDAAyD,CAAC,IAChFA,qBAAqB,CAAC,yDAAyD,CAAC,CAAC,WAAW,CAAC,EAC5F;UACDC,SAAS,GAAI,GAAE7C,YAAa,wGAAuG;QACpI;QAEA,IAAI6C,SAAS,EAAE;UACd,MAAME,4BAA4B,GAAGvH,UAAU,CAACkC,SAAS,CAACmF,SAAS,CAAC;UACpE,IAAIE,4BAA4B,KAAK,KAAK,EAAE;YAC3CD,UAAU,GAAG,KAAK;UACnB;QACD,CAAC,MAAM;UACN,MAAME,mBAAmB,GAAGJ,qBAAqB,CAAC,+CAA+C,CAAC;UAClG,IAAII,mBAAmB,IAAIA,mBAAmB,CAACC,UAAU,KAAK,KAAK,EAAE;YACpEH,UAAU,GAAG,KAAK;UACnB;QACD;QACA,IAAIA,UAAU,EAAE;UACf,OAAO,IAAI,CAACI,oBAAoB,CAACpI,kBAAkB,EAAEkF,YAAY,EAAE0C,OAAO,CAAC;QAC5E,CAAC,MAAM;UACN,OAAO,EAAE;QACV;MACD,CAAC,CAAC;IACH,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;IACCQ,oBAAoB,EAAE,UACrBpI,kBAAwD,EACxDkF,YAAoB,EACpB0C,OAAe,EACN;MACT,IAAIS,kBAAkB,GAAGrI,kBAAkB,IAAIA,kBAAkB,CAACsI,aAAa,GAAGtI,kBAAkB,CAACsI,aAAa,CAAC,CAAC,CAAC,GAAG,QAAQ;MAChI,IAAIC,KAAK,GAAG,EAAE;MAEdF,kBAAkB,GACjBA,kBAAkB,CAACG,QAAQ,CAAC,GAAG,CAAC,IAAIH,kBAAkB,CAAChI,MAAM,GAAGgI,kBAAkB,CAACxB,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,GAChGwB,kBAAkB,CAACI,SAAS,CAAC,CAAC,EAAEJ,kBAAkB,CAACxB,OAAO,CAAC,GAAG,CAAC,CAAC,GAChE,QAAQ;MACZ0B,KAAK,GAAI,GAAErD,YAAY,CAACuD,SAAS,CAAC,CAAC,CAAE,kBAAiBJ,kBAAmB,EAAC;MAC1E,IAAIT,OAAO,CAACc,kBAAkB,CAACH,KAAK,CAAC,EAAE;QACtC,OAAOA,KAAK;MACb,CAAC,MAAM;QACN,MAAM,IAAII,KAAK,CAAE,qCAAoCzD,YAAY,CAACuD,SAAS,CAAC,CAAC,CAAE,EAAC,CAAC;MAClF;IACD,CAAC;IAED;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;IACCG,qBAAqB,EAAE,gBAAgB/F,WAAmB,EAAEgG,SAAyB,EAAoB;MAAA;MACxG,MAAMC,oBAAkD,GAAG,MAAMD,SAAS,CAACvB,aAAa,CAAE,GAAEzE,WAAY,GAAE,CAAC;MAC3G,MAAMkG,YAAY,GAAGD,oBAAoB,aAApBA,oBAAoB,uBAApBA,oBAAoB,CAAG,0CAA0C,CAAC;MACvF,IAAIE,kBAAkB,GAAG,IAAI;MAE7B,IAAIF,oBAAoB,aAApBA,oBAAoB,wCAApBA,oBAAoB,CAAG,2CAA2C,CAAC,kDAAnE,sBAAsE,YAAY,CAAC,EAAE;QACxFE,kBAAkB,GAAGH,SAAS,CAACjG,SAAS,CACtC,GAAEC,WAAY,2FAA0F,CACzG;MACF;MAEA,OAAOmG,kBAAkB,KAAK,KAAK,IAAID,YAAY,KAAK,IAAI;IAC7D;EACD,CAAC;EAAC,OAEalJ,gBAAgB;AAAA"}
|
|
@@ -84,6 +84,35 @@ const AppStartupHelper = {
|
|
|
84
84
|
return new Filter(aFilters, true);
|
|
85
85
|
},
|
|
86
86
|
|
|
87
|
+
/**
|
|
88
|
+
* Makes sure key values have the correct format.
|
|
89
|
+
* This is to handle special cases, for example, where Boolean values are 'X' or '', or guid values have the following format: guid'00000000-0000-0000-0000-000000000000'.".
|
|
90
|
+
* @param keys
|
|
91
|
+
* @param entityType
|
|
92
|
+
*/
|
|
93
|
+
_sanitizeKeys: function (keys: ValuedKey[], entityType: Record<string, { $Type: string }>): void {
|
|
94
|
+
keys.forEach((key) => {
|
|
95
|
+
const propertyType = entityType[key.name].$Type;
|
|
96
|
+
switch (propertyType) {
|
|
97
|
+
case "Edm.Boolean":
|
|
98
|
+
if (key.value !== "true" && key.value !== "false") {
|
|
99
|
+
// If the value is neither "true" nor "false", it's considered as false if it contains only whitespaces (e.g. to handle 'X'/'' for true/false values)
|
|
100
|
+
key.value = key.value.trim().length ? "true" : "false";
|
|
101
|
+
}
|
|
102
|
+
break;
|
|
103
|
+
|
|
104
|
+
case "Edm.Guid": {
|
|
105
|
+
const match = /[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}/.exec(key.value);
|
|
106
|
+
key.value = match ? match[0] : key.value;
|
|
107
|
+
break;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
default:
|
|
111
|
+
// Do nothing
|
|
112
|
+
}
|
|
113
|
+
});
|
|
114
|
+
},
|
|
115
|
+
|
|
87
116
|
/**
|
|
88
117
|
* Loads all contexts for a list of page infos.
|
|
89
118
|
*
|
|
@@ -99,14 +128,16 @@ const AppStartupHelper = {
|
|
|
99
128
|
): Promise<Context[][]> {
|
|
100
129
|
// Load the respective objects for all object pages found in aExternallyNavigablePages
|
|
101
130
|
const aContextPromises = aStartupPages.map(async (pageInfo) => {
|
|
131
|
+
const entityType = oModel.getMetaModel().getObject(`${pageInfo.contextPath}/`);
|
|
102
132
|
if (pageInfo.technicalKeys && !alwaysCheckExistence) {
|
|
133
|
+
this._sanitizeKeys(pageInfo.technicalKeys, entityType);
|
|
103
134
|
// Create a context with the proper path
|
|
104
|
-
const entityType = oModel.getMetaModel().getObject(`${pageInfo.contextPath}/`);
|
|
105
135
|
const path = `${pageInfo.contextPath}(${SemanticKeyHelper.getPathContent(pageInfo.technicalKeys, entityType)})`;
|
|
106
136
|
const context = oModel.bindContext(path).getBoundContext();
|
|
107
137
|
return Promise.resolve([context]);
|
|
108
138
|
} else {
|
|
109
139
|
const aKeys = pageInfo.semanticKeys || pageInfo.technicalKeys || [];
|
|
140
|
+
this._sanitizeKeys(aKeys, entityType);
|
|
110
141
|
const oFilter = this._createFilterFromKeys(aKeys, pageInfo.draftMode, oModel.getMetaModel());
|
|
111
142
|
|
|
112
143
|
// only request a minimum of fields to boost backend performance since this is only used to check if an object exists
|
|
@@ -48,7 +48,7 @@ sap.ui.define(["sap/base/Log", "sap/fe/core/formatters/CriticalityFormatter", "s
|
|
|
48
48
|
controls: [],
|
|
49
49
|
elements: [],
|
|
50
50
|
// eslint-disable-next-line no-template-curly-in-string
|
|
51
|
-
version: "1.124.
|
|
51
|
+
version: "1.124.9",
|
|
52
52
|
noLibraryCSS: true,
|
|
53
53
|
extensions: {
|
|
54
54
|
//Configuration used for rule loading of Support Assistant
|
|
@@ -6,7 +6,6 @@ sap.ui.define(["sap/base/Log", "sap/base/util/ObjectPath"], function (Log, Objec
|
|
|
6
6
|
"use strict";
|
|
7
7
|
|
|
8
8
|
var _exports = {};
|
|
9
|
-
let pageConfigurationChanges = {};
|
|
10
9
|
/**
|
|
11
10
|
* Apply change method.
|
|
12
11
|
*
|
|
@@ -40,18 +39,19 @@ sap.ui.define(["sap/base/Log", "sap/base/util/ObjectPath"], function (Log, Objec
|
|
|
40
39
|
* @param pageId The ID of the page for which the configuration is to be changed.
|
|
41
40
|
* @param path The path in the page settings for which the configuration is to be changed.
|
|
42
41
|
* @param value The new value of the configuration. This could be a plain value like a string, or a Boolean, or a structured object.
|
|
43
|
-
* @param lateChange Indicates that the change was done after application startup (
|
|
42
|
+
* @param lateChange Indicates that the change was done after application startup (for example when using the feature toggle).
|
|
43
|
+
* @param appComponent The appComponent, in case the change is done after application startup.
|
|
44
44
|
* @returns The changed or unchanged manifest.
|
|
45
45
|
*/
|
|
46
46
|
_exports.applyChange = applyChange;
|
|
47
|
-
function changeConfiguration(manifest, pageId, path, value, lateChange) {
|
|
47
|
+
function changeConfiguration(manifest, pageId, path, value, lateChange, appComponent) {
|
|
48
48
|
const pageSettings = getPageSettings(manifest, pageId);
|
|
49
49
|
if (pageSettings) {
|
|
50
50
|
const propertyPath = retrievePropertyPath(path);
|
|
51
51
|
ObjectPath.set(propertyPath, value, pageSettings);
|
|
52
|
-
if (lateChange) {
|
|
53
|
-
pageConfigurationChanges[pageId] = pageConfigurationChanges[pageId] || [];
|
|
54
|
-
pageConfigurationChanges[pageId].push(path);
|
|
52
|
+
if (lateChange && appComponent) {
|
|
53
|
+
appComponent.pageConfigurationChanges[pageId] = appComponent.pageConfigurationChanges[pageId] || [];
|
|
54
|
+
appComponent.pageConfigurationChanges[pageId].push(path);
|
|
55
55
|
}
|
|
56
56
|
} else {
|
|
57
57
|
Log.error(`No Fiori elements page with ID ${pageId} found in routing targets.`);
|
|
@@ -118,7 +118,7 @@ sap.ui.define(["sap/base/Log", "sap/base/util/ObjectPath"], function (Log, Objec
|
|
|
118
118
|
*/
|
|
119
119
|
function applyPageConfigurationChanges(manifest, viewData, appComponent, pageId) {
|
|
120
120
|
viewData = viewData ?? {};
|
|
121
|
-
const pageChanges = pageConfigurationChanges[pageId] || [];
|
|
121
|
+
const pageChanges = appComponent.pageConfigurationChanges[pageId] || [];
|
|
122
122
|
for (const path of pageChanges) {
|
|
123
123
|
const propertyPath = retrievePropertyPath(path);
|
|
124
124
|
const manifestValue = ObjectPath.get(propertyPath, manifest);
|
|
@@ -126,16 +126,7 @@ sap.ui.define(["sap/base/Log", "sap/base/util/ObjectPath"], function (Log, Objec
|
|
|
126
126
|
}
|
|
127
127
|
return viewData;
|
|
128
128
|
}
|
|
129
|
-
|
|
130
|
-
/**
|
|
131
|
-
* Cleans all registered page configuration changes.
|
|
132
|
-
*
|
|
133
|
-
*/
|
|
134
129
|
_exports.applyPageConfigurationChanges = applyPageConfigurationChanges;
|
|
135
|
-
function cleanPageConfigurationChanges() {
|
|
136
|
-
pageConfigurationChanges = {};
|
|
137
|
-
}
|
|
138
|
-
_exports.cleanPageConfigurationChanges = cleanPageConfigurationChanges;
|
|
139
130
|
return _exports;
|
|
140
131
|
}, false);
|
|
141
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["pageConfigurationChanges","applyChange","manifest","change","changeContent","getContent","pageId","page","propertyChange","entityPropertyChange","operation","propertyPath","propertyValue","undefined","startsWith","Log","error","changeConfiguration","path","value","lateChange","pageSettings","getPageSettings","retrievePropertyPath","ObjectPath","set","push","split","annotationPath","i","length","includes","concat","slice","targets","routing","p","id","name","options","settings","applyPageConfigurationChanges","viewData","appComponent","pageChanges","manifestValue","get","cleanPageConfigurationChanges"],"sourceRoot":".","sources":["ChangePageConfiguration.ts"],"sourcesContent":["import Log from \"sap/base/Log\";\nimport ObjectPath from \"sap/base/util/ObjectPath\";\nimport type AppComponent from \"sap/fe/core/AppComponent\";\nimport type { ViewData } from \"sap/fe/core/services/TemplatedViewServiceFactory\";\nimport type { ManifestContent } from \"sap/ui/core/Manifest\";\n\nlet pageConfigurationChanges: Record<string, string[]> = {};\n\nexport type Change = {\n\tgetContent(): ChangeContent;\n};\n\ntype ChangeContent = {\n\tpage: string; // ID of the page to be changed\n\tentityPropertyChange: EntityPropertyChange;\n};\n\ntype EntityPropertyChange = {\n\tpropertyPath: string; // path to the property to be changed\n\toperation: string; // only UPSERT supported\n\tpropertyValue: string | Object; //what to be changed\n};\n\n/**\n * Apply change method.\n *\n * This method is being called by the FLEX framework in case a manifest change with the change type\n * 'appdescr_fe_changePageConfiguration' was created for the current application. This method is not meant to be\n * called by anyone else but the FLEX framework.\n *\n * @param manifest The original manifest.\n * @param change The change content.\n * @returns The changed or unchanged manifest.\n */\nexport function applyChange(manifest: ManifestContent, change: Change): object {\n\tconst changeContent = change.getContent();\n\tconst pageId = changeContent?.page;\n\tconst propertyChange = changeContent?.entityPropertyChange;\n\n\t// return unmodified manifest in case change not valid\n\tif (\n\t\tpropertyChange?.operation !== \"UPSERT\" ||\n\t\t!propertyChange?.propertyPath ||\n\t\tpropertyChange?.propertyValue === undefined ||\n\t\tpropertyChange?.propertyPath.startsWith(\"/\")\n\t) {\n\t\tLog.error(\"Change content is not a valid\");\n\t\treturn manifest;\n\t}\n\n\treturn changeConfiguration(manifest, pageId, propertyChange.propertyPath, propertyChange.propertyValue);\n}\n\n/**\n * Changes the page configuration of SAP Fiori elements.\n *\n * This method enables you to change the page configuration of SAP Fiori elements.\n *\n * @param manifest The original manifest.\n * @param pageId The ID of the page for which the configuration is to be changed.\n * @param path The path in the page settings for which the configuration is to be changed.\n * @param value The new value of the configuration. This could be a plain value like a string, or a Boolean, or a structured object.\n * @param lateChange Indicates that the change was done after application startup (e.g. feature toggle).\n * @returns The changed or unchanged manifest.\n */\nexport function changeConfiguration(manifest: ManifestContent, pageId: string, path: string, value: unknown, lateChange?: boolean): object {\n\tconst pageSettings = getPageSettings(manifest, pageId);\n\n\tif (pageSettings) {\n\t\tconst propertyPath = retrievePropertyPath(path);\n\t\tObjectPath.set(propertyPath, value, pageSettings);\n\t\tif (lateChange) {\n\t\t\tpageConfigurationChanges[pageId] = pageConfigurationChanges[pageId] || [];\n\t\t\tpageConfigurationChanges[pageId].push(path);\n\t\t}\n\t} else {\n\t\tLog.error(`No Fiori elements page with ID ${pageId} found in routing targets.`);\n\t}\n\n\treturn manifest;\n}\n\n/**\n * Retrieves an array with the property path parts and consider the controlConfiguration specially.\n *\n * @param path The given property path\n * @returns An array with the property path parts.\n */\nfunction retrievePropertyPath(path: string): string[] {\n\tlet propertyPath = path.split(\"/\");\n\tif (propertyPath[0] === \"controlConfiguration\") {\n\t\tlet annotationPath = \"\";\n\t\t// the annotation path in the control configuration has to stay together. For now rely on the fact the @ is in the last part\n\t\tfor (let i = 1; i < propertyPath.length; i++) {\n\t\t\tannotationPath += (i > 1 ? \"/\" : \"\") + propertyPath[i];\n\t\t\tif (annotationPath.includes(\"@\")) {\n\t\t\t\tpropertyPath = [\"controlConfiguration\", annotationPath].concat(propertyPath.slice(i + 1));\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t}\n\treturn propertyPath;\n}\n\n/**\n * Search the page settings in the manifest for a given page ID.\n *\n * @param manifest The manifest where the search is carried out to find the page settings.\n * @param pageId The ID of the page.\n * @returns The page settings for the page ID or undefined if not found.\n */\nfunction getPageSettings(manifest: ManifestContent, pageId: string): object | undefined {\n\tlet pageSettings;\n\tconst targets = manifest[\"sap.ui5\"]?.routing?.targets ?? {};\n\tfor (const p in targets) {\n\t\tif (targets[p].id === pageId && targets[p].name.startsWith(\"sap.fe.templates.\")) {\n\t\t\tpageSettings = targets[p].options?.settings ?? {};\n\t\t\tbreak;\n\t\t}\n\t}\n\treturn pageSettings;\n}\n\n/**\n * Applies page configuration changes to view data object.\n *\n * UI5 routing clones the manifest settings during the app init, even before the router was initialized.\n * As we allow changing the manifest in the async initializeFeatureToggle hook, the view data might not fit the current\n * manifest settings, therefore (re)applying the registered page configuration changes to the view data object.\n *\n * @param manifest The current page manifest settings.\n * @param viewData The current viewData settings.\n * @param appComponent The app component instance.\n * @param pageId The ID of the page.\n * @returns The updated viewData settings.\n */\nexport function applyPageConfigurationChanges(manifest: object, viewData: ViewData, appComponent: AppComponent, pageId: string): ViewData {\n\tviewData = viewData ?? {};\n\tconst pageChanges: string[] = pageConfigurationChanges[pageId] || [];\n\tfor (const path of pageChanges) {\n\t\tconst propertyPath = retrievePropertyPath(path);\n\t\tconst manifestValue = ObjectPath.get(propertyPath, manifest);\n\t\tObjectPath.set(propertyPath, manifestValue, viewData);\n\t}\n\treturn viewData;\n}\n\n/**\n * Cleans all registered page configuration changes.\n *\n */\nexport function cleanPageConfigurationChanges(): void {\n\tpageConfigurationChanges = {};\n}\n"],"mappings":"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"}
|
|
132
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["applyChange","manifest","change","changeContent","getContent","pageId","page","propertyChange","entityPropertyChange","operation","propertyPath","propertyValue","undefined","startsWith","Log","error","changeConfiguration","path","value","lateChange","appComponent","pageSettings","getPageSettings","retrievePropertyPath","ObjectPath","set","pageConfigurationChanges","push","split","annotationPath","i","length","includes","concat","slice","targets","routing","p","id","name","options","settings","applyPageConfigurationChanges","viewData","pageChanges","manifestValue","get"],"sourceRoot":".","sources":["ChangePageConfiguration.ts"],"sourcesContent":["import Log from \"sap/base/Log\";\nimport ObjectPath from \"sap/base/util/ObjectPath\";\nimport type AppComponent from \"sap/fe/core/AppComponent\";\nimport type { ViewData } from \"sap/fe/core/services/TemplatedViewServiceFactory\";\nimport type { ManifestContent } from \"sap/ui/core/Manifest\";\n\nexport type Change = {\n\tgetContent(): ChangeContent;\n};\n\ntype ChangeContent = {\n\tpage: string; // ID of the page to be changed\n\tentityPropertyChange: EntityPropertyChange;\n};\n\ntype EntityPropertyChange = {\n\tpropertyPath: string; // path to the property to be changed\n\toperation: string; // only UPSERT supported\n\tpropertyValue: string | Object; //what to be changed\n};\n\n/**\n * Apply change method.\n *\n * This method is being called by the FLEX framework in case a manifest change with the change type\n * 'appdescr_fe_changePageConfiguration' was created for the current application. This method is not meant to be\n * called by anyone else but the FLEX framework.\n *\n * @param manifest The original manifest.\n * @param change The change content.\n * @returns The changed or unchanged manifest.\n */\nexport function applyChange(manifest: ManifestContent, change: Change): object {\n\tconst changeContent = change.getContent();\n\tconst pageId = changeContent?.page;\n\tconst propertyChange = changeContent?.entityPropertyChange;\n\n\t// return unmodified manifest in case change not valid\n\tif (\n\t\tpropertyChange?.operation !== \"UPSERT\" ||\n\t\t!propertyChange?.propertyPath ||\n\t\tpropertyChange?.propertyValue === undefined ||\n\t\tpropertyChange?.propertyPath.startsWith(\"/\")\n\t) {\n\t\tLog.error(\"Change content is not a valid\");\n\t\treturn manifest;\n\t}\n\n\treturn changeConfiguration(manifest, pageId, propertyChange.propertyPath, propertyChange.propertyValue);\n}\n\n/**\n * Changes the page configuration of SAP Fiori elements.\n *\n * This method enables you to change the page configuration of SAP Fiori elements.\n *\n * @param manifest The original manifest.\n * @param pageId The ID of the page for which the configuration is to be changed.\n * @param path The path in the page settings for which the configuration is to be changed.\n * @param value The new value of the configuration. This could be a plain value like a string, or a Boolean, or a structured object.\n * @param lateChange Indicates that the change was done after application startup (for example when using the feature toggle).\n * @param appComponent The appComponent, in case the change is done after application startup.\n * @returns The changed or unchanged manifest.\n */\nexport function changeConfiguration(\n\tmanifest: ManifestContent,\n\tpageId: string,\n\tpath: string,\n\tvalue: unknown,\n\tlateChange?: boolean,\n\tappComponent?: AppComponent\n): object {\n\tconst pageSettings = getPageSettings(manifest, pageId);\n\n\tif (pageSettings) {\n\t\tconst propertyPath = retrievePropertyPath(path);\n\t\tObjectPath.set(propertyPath, value, pageSettings);\n\t\tif (lateChange && appComponent) {\n\t\t\tappComponent.pageConfigurationChanges[pageId] = appComponent.pageConfigurationChanges[pageId] || [];\n\t\t\tappComponent.pageConfigurationChanges[pageId].push(path);\n\t\t}\n\t} else {\n\t\tLog.error(`No Fiori elements page with ID ${pageId} found in routing targets.`);\n\t}\n\n\treturn manifest;\n}\n\n/**\n * Retrieves an array with the property path parts and consider the controlConfiguration specially.\n *\n * @param path The given property path\n * @returns An array with the property path parts.\n */\nfunction retrievePropertyPath(path: string): string[] {\n\tlet propertyPath = path.split(\"/\");\n\tif (propertyPath[0] === \"controlConfiguration\") {\n\t\tlet annotationPath = \"\";\n\t\t// the annotation path in the control configuration has to stay together. For now rely on the fact the @ is in the last part\n\t\tfor (let i = 1; i < propertyPath.length; i++) {\n\t\t\tannotationPath += (i > 1 ? \"/\" : \"\") + propertyPath[i];\n\t\t\tif (annotationPath.includes(\"@\")) {\n\t\t\t\tpropertyPath = [\"controlConfiguration\", annotationPath].concat(propertyPath.slice(i + 1));\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t}\n\treturn propertyPath;\n}\n\n/**\n * Search the page settings in the manifest for a given page ID.\n *\n * @param manifest The manifest where the search is carried out to find the page settings.\n * @param pageId The ID of the page.\n * @returns The page settings for the page ID or undefined if not found.\n */\nfunction getPageSettings(manifest: ManifestContent, pageId: string): object | undefined {\n\tlet pageSettings;\n\tconst targets = manifest[\"sap.ui5\"]?.routing?.targets ?? {};\n\tfor (const p in targets) {\n\t\tif (targets[p].id === pageId && targets[p].name.startsWith(\"sap.fe.templates.\")) {\n\t\t\tpageSettings = targets[p].options?.settings ?? {};\n\t\t\tbreak;\n\t\t}\n\t}\n\treturn pageSettings;\n}\n\n/**\n * Applies page configuration changes to view data object.\n *\n * UI5 routing clones the manifest settings during the app init, even before the router was initialized.\n * As we allow changing the manifest in the async initializeFeatureToggle hook, the view data might not fit the current\n * manifest settings, therefore (re)applying the registered page configuration changes to the view data object.\n *\n * @param manifest The current page manifest settings.\n * @param viewData The current viewData settings.\n * @param appComponent The app component instance.\n * @param pageId The ID of the page.\n * @returns The updated viewData settings.\n */\nexport function applyPageConfigurationChanges(manifest: object, viewData: ViewData, appComponent: AppComponent, pageId: string): ViewData {\n\tviewData = viewData ?? {};\n\tconst pageChanges: string[] = appComponent.pageConfigurationChanges[pageId] || [];\n\tfor (const path of pageChanges) {\n\t\tconst propertyPath = retrievePropertyPath(path);\n\t\tconst manifestValue = ObjectPath.get(propertyPath, manifest);\n\t\tObjectPath.set(propertyPath, manifestValue, viewData);\n\t}\n\treturn viewData;\n}\n"],"mappings":"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"}
|
|
@@ -4,8 +4,6 @@ import type AppComponent from "sap/fe/core/AppComponent";
|
|
|
4
4
|
import type { ViewData } from "sap/fe/core/services/TemplatedViewServiceFactory";
|
|
5
5
|
import type { ManifestContent } from "sap/ui/core/Manifest";
|
|
6
6
|
|
|
7
|
-
let pageConfigurationChanges: Record<string, string[]> = {};
|
|
8
|
-
|
|
9
7
|
export type Change = {
|
|
10
8
|
getContent(): ChangeContent;
|
|
11
9
|
};
|
|
@@ -60,18 +58,26 @@ export function applyChange(manifest: ManifestContent, change: Change): object {
|
|
|
60
58
|
* @param pageId The ID of the page for which the configuration is to be changed.
|
|
61
59
|
* @param path The path in the page settings for which the configuration is to be changed.
|
|
62
60
|
* @param value The new value of the configuration. This could be a plain value like a string, or a Boolean, or a structured object.
|
|
63
|
-
* @param lateChange Indicates that the change was done after application startup (
|
|
61
|
+
* @param lateChange Indicates that the change was done after application startup (for example when using the feature toggle).
|
|
62
|
+
* @param appComponent The appComponent, in case the change is done after application startup.
|
|
64
63
|
* @returns The changed or unchanged manifest.
|
|
65
64
|
*/
|
|
66
|
-
export function changeConfiguration(
|
|
65
|
+
export function changeConfiguration(
|
|
66
|
+
manifest: ManifestContent,
|
|
67
|
+
pageId: string,
|
|
68
|
+
path: string,
|
|
69
|
+
value: unknown,
|
|
70
|
+
lateChange?: boolean,
|
|
71
|
+
appComponent?: AppComponent
|
|
72
|
+
): object {
|
|
67
73
|
const pageSettings = getPageSettings(manifest, pageId);
|
|
68
74
|
|
|
69
75
|
if (pageSettings) {
|
|
70
76
|
const propertyPath = retrievePropertyPath(path);
|
|
71
77
|
ObjectPath.set(propertyPath, value, pageSettings);
|
|
72
|
-
if (lateChange) {
|
|
73
|
-
pageConfigurationChanges[pageId] = pageConfigurationChanges[pageId] || [];
|
|
74
|
-
pageConfigurationChanges[pageId].push(path);
|
|
78
|
+
if (lateChange && appComponent) {
|
|
79
|
+
appComponent.pageConfigurationChanges[pageId] = appComponent.pageConfigurationChanges[pageId] || [];
|
|
80
|
+
appComponent.pageConfigurationChanges[pageId].push(path);
|
|
75
81
|
}
|
|
76
82
|
} else {
|
|
77
83
|
Log.error(`No Fiori elements page with ID ${pageId} found in routing targets.`);
|
|
@@ -136,7 +142,7 @@ function getPageSettings(manifest: ManifestContent, pageId: string): object | un
|
|
|
136
142
|
*/
|
|
137
143
|
export function applyPageConfigurationChanges(manifest: object, viewData: ViewData, appComponent: AppComponent, pageId: string): ViewData {
|
|
138
144
|
viewData = viewData ?? {};
|
|
139
|
-
const pageChanges: string[] = pageConfigurationChanges[pageId] || [];
|
|
145
|
+
const pageChanges: string[] = appComponent.pageConfigurationChanges[pageId] || [];
|
|
140
146
|
for (const path of pageChanges) {
|
|
141
147
|
const propertyPath = retrievePropertyPath(path);
|
|
142
148
|
const manifestValue = ObjectPath.get(propertyPath, manifest);
|
|
@@ -144,11 +150,3 @@ export function applyPageConfigurationChanges(manifest: object, viewData: ViewDa
|
|
|
144
150
|
}
|
|
145
151
|
return viewData;
|
|
146
152
|
}
|
|
147
|
-
|
|
148
|
-
/**
|
|
149
|
-
* Cleans all registered page configuration changes.
|
|
150
|
-
*
|
|
151
|
-
*/
|
|
152
|
-
export function cleanPageConfigurationChanges(): void {
|
|
153
|
-
pageConfigurationChanges = {};
|
|
154
|
-
}
|