@sapui5/sap.fe.test 1.101.1 → 1.102.0
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/test/.library +1 -1
- package/src/sap/fe/test/CollaborationClient.js +77 -0
- package/src/sap/fe/test/CollaborationClient.ts +105 -0
- package/src/sap/fe/test/FeMocks.js +1 -1
- package/src/sap/fe/test/FeMocks.ts +6 -2
- package/src/sap/fe/test/JestTemplatingHelper.js +19 -1
- package/src/sap/fe/test/JestTemplatingHelper.ts +13 -0
- package/src/sap/fe/test/ListReport.js +7 -15
- package/src/sap/fe/test/LocationUtil.js +12 -16
- package/src/sap/fe/test/ObjectPage.js +38 -6
- package/src/sap/fe/test/Shell.js +94 -34
- package/src/sap/fe/test/TemplatePage.js +6 -0
- package/src/sap/fe/test/UI5MockHelper.js +98 -18
- package/src/sap/fe/test/UI5MockHelper.ts +79 -14
- package/src/sap/fe/test/api/CollaborationAPI.js +215 -0
- package/src/sap/fe/test/api/CollaborationAPI.ts +240 -0
- package/src/sap/fe/test/api/HeaderAssertions.js +76 -41
- package/src/sap/fe/test/api/TableAPI.js +30 -34
- package/src/sap/fe/test/api/TableActions.js +1 -1
- package/src/sap/fe/test/api/TableAssertions.js +51 -46
- package/src/sap/fe/test/builder/MacroFieldBuilder.js +161 -156
- package/src/sap/fe/test/builder/MdcTableBuilder.js +89 -12
- package/src/sap/fe/test/library.js +1 -1
|
@@ -6,13 +6,17 @@ import Control from "sap/ui/core/Control";
|
|
|
6
6
|
import UI5Element from "sap/ui/core/Element";
|
|
7
7
|
import Controller from "sap/ui/core/mvc/Controller";
|
|
8
8
|
import View from "sap/ui/core/mvc/View";
|
|
9
|
+
import CompositeBinding from "sap/ui/model/CompositeBinding";
|
|
10
|
+
import JSONModel from "sap/ui/model/json/JSONModel";
|
|
9
11
|
import Context from "sap/ui/model/odata/v4/Context";
|
|
10
|
-
import ODataListBinding from "sap/ui/model/odata/v4/ODataListBinding";
|
|
11
12
|
import ODataContextBinding from "sap/ui/model/odata/v4/ODataContextBinding";
|
|
13
|
+
import ODataListBinding from "sap/ui/model/odata/v4/ODataListBinding";
|
|
12
14
|
import ODataMetaModel from "sap/ui/model/odata/v4/ODataMetaModel";
|
|
13
15
|
import ODataModel from "sap/ui/model/odata/v4/ODataModel";
|
|
16
|
+
import ODataPropertyBinding from "sap/ui/model/odata/v4/ODataPropertyBinding";
|
|
14
17
|
|
|
15
18
|
export class MockContext implements Partial<Context> {
|
|
19
|
+
private isKeptAlive = false;
|
|
16
20
|
public constructor(private oValues?: any, private oBinding?: any) {}
|
|
17
21
|
|
|
18
22
|
public isA(sClassName: string): boolean {
|
|
@@ -52,9 +56,14 @@ export class MockContext implements Partial<Context> {
|
|
|
52
56
|
return this.oBinding;
|
|
53
57
|
});
|
|
54
58
|
public getModel = jest.fn(() => {
|
|
55
|
-
return
|
|
59
|
+
return this.oBinding?.getModel() as any as ODataModel;
|
|
56
60
|
});
|
|
57
61
|
public isTransient = jest.fn();
|
|
62
|
+
public setKeepAlive = jest.fn((bool: boolean) => {
|
|
63
|
+
this.isKeptAlive = bool;
|
|
64
|
+
});
|
|
65
|
+
public isKeepAlive = jest.fn(() => this.isKeptAlive);
|
|
66
|
+
public requestSideEffects = jest.fn();
|
|
58
67
|
}
|
|
59
68
|
|
|
60
69
|
export class MockUI5Element implements Partial<UI5Element> {
|
|
@@ -70,12 +79,13 @@ export class MockUI5Element implements Partial<UI5Element> {
|
|
|
70
79
|
|
|
71
80
|
export class MockControl extends MockUI5Element implements Partial<Control> {
|
|
72
81
|
public getBindingContext: jest.Mock<any, any> = jest.fn();
|
|
82
|
+
public getModel: jest.Mock<any, any> = jest.fn();
|
|
73
83
|
}
|
|
74
84
|
|
|
75
85
|
export class MockEvent implements Partial<Event> {
|
|
76
86
|
public constructor(public params: { [key: string]: any } = {}, public source: MockUI5Element = new MockUI5Element()) {}
|
|
77
87
|
|
|
78
|
-
public getParameter = jest.fn(name => this.params[name]);
|
|
88
|
+
public getParameter = jest.fn((name) => this.params[name]);
|
|
79
89
|
|
|
80
90
|
public getSource = jest.fn(() => this.source as any);
|
|
81
91
|
}
|
|
@@ -89,7 +99,7 @@ export class MockListBinding implements Partial<ODataListBinding> {
|
|
|
89
99
|
public constructor(aContexts?: any[], private mockModel?: MockModel) {
|
|
90
100
|
aContexts = aContexts || [];
|
|
91
101
|
|
|
92
|
-
this.aMockContexts = aContexts.map(context => {
|
|
102
|
+
this.aMockContexts = aContexts.map((context) => {
|
|
93
103
|
return new MockContext(context, this);
|
|
94
104
|
});
|
|
95
105
|
}
|
|
@@ -109,19 +119,73 @@ export class MockListBinding implements Partial<ODataListBinding> {
|
|
|
109
119
|
|
|
110
120
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
111
121
|
public requestContexts = jest.fn((...args) => {
|
|
112
|
-
return Promise.resolve(
|
|
122
|
+
return Promise.resolve(this.aMockContexts as any as Context[]);
|
|
113
123
|
});
|
|
114
124
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
115
125
|
public getCurrentContexts = jest.fn((...args) => {
|
|
116
|
-
return
|
|
126
|
+
return this.aMockContexts as any as Context[];
|
|
127
|
+
});
|
|
128
|
+
public getModel = jest.fn(() => {
|
|
129
|
+
return this.mockModel as any as ODataModel;
|
|
117
130
|
});
|
|
131
|
+
public getUpdateGroupId = jest.fn(() => "auto");
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
export class MockPropertyBinding implements Partial<ODataPropertyBinding> {
|
|
135
|
+
private mockModel!: JSONModel;
|
|
136
|
+
private mockPath!: string;
|
|
137
|
+
|
|
138
|
+
public constructor(mockModel: JSONModel, mockPath: string) {
|
|
139
|
+
this.setModel(mockModel);
|
|
140
|
+
this.setPath(mockPath);
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
public isA(sClassName: string): boolean {
|
|
144
|
+
return sClassName === "sap/ui/model/odata/v4/ODataPropertyBinding";
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
public setModel(oModel: JSONModel) {
|
|
148
|
+
this.mockModel = oModel;
|
|
149
|
+
}
|
|
118
150
|
public getModel = jest.fn(() => {
|
|
119
|
-
return
|
|
151
|
+
return this.mockModel as any as ODataModel;
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
public setPath(path: string | undefined) {
|
|
155
|
+
this.mockPath = path ? path : "";
|
|
156
|
+
}
|
|
157
|
+
public getPath = jest.fn(() => {
|
|
158
|
+
return this.mockPath;
|
|
159
|
+
});
|
|
160
|
+
|
|
161
|
+
public getValue = jest.fn(() => {
|
|
162
|
+
return this?.mockModel?.getProperty("/" + this.mockPath);
|
|
163
|
+
});
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
export class MockCompositeBinding implements Partial<CompositeBinding> {
|
|
167
|
+
private aBindings: MockPropertyBinding[];
|
|
168
|
+
|
|
169
|
+
public constructor(aBindings: MockPropertyBinding[]) {
|
|
170
|
+
this.aBindings = aBindings;
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
public isA(sClassName: string): boolean {
|
|
174
|
+
return sClassName === "sap.ui.model.CompositeBinding";
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
public getBindings = jest.fn(() => {
|
|
178
|
+
return this.aBindings;
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
public getValue = jest.fn(() => {
|
|
182
|
+
return this.aBindings.map((binding) => binding.getValue());
|
|
120
183
|
});
|
|
121
184
|
}
|
|
122
185
|
|
|
123
186
|
export class MockContextBinding implements Partial<ODataContextBinding> {
|
|
124
187
|
private oMockContext: MockContext;
|
|
188
|
+
private isKeptAlive: boolean = false;
|
|
125
189
|
|
|
126
190
|
public constructor(oContext?: any, private mockModel?: MockModel) {
|
|
127
191
|
this.oMockContext = new MockContext(oContext || {}, this);
|
|
@@ -139,11 +203,11 @@ export class MockContextBinding implements Partial<ODataContextBinding> {
|
|
|
139
203
|
|
|
140
204
|
// Mocked API
|
|
141
205
|
public getBoundContext = jest.fn(() => {
|
|
142
|
-
return
|
|
206
|
+
return this.oMockContext as any as Context;
|
|
143
207
|
});
|
|
144
208
|
public attachEventOnce = jest.fn();
|
|
145
209
|
public getModel = jest.fn(() => {
|
|
146
|
-
return
|
|
210
|
+
return this.mockModel as any as ODataModel;
|
|
147
211
|
});
|
|
148
212
|
}
|
|
149
213
|
|
|
@@ -161,14 +225,14 @@ export class MockMetaModel implements Partial<ODataMetaModel> {
|
|
|
161
225
|
// Mocked API
|
|
162
226
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
163
227
|
public getMetaContext = jest.fn((sPath: string) => {
|
|
164
|
-
return
|
|
228
|
+
return new MockContext({ $path: sPath }) as any as Context;
|
|
165
229
|
});
|
|
166
230
|
public getObject = jest.fn((sPath: string) => {
|
|
167
231
|
return this.oMetaContext.getProperty(sPath);
|
|
168
232
|
});
|
|
169
233
|
public requestObject = jest.fn();
|
|
170
234
|
public createBindingContext = jest.fn((sPath: string) => {
|
|
171
|
-
return
|
|
235
|
+
return new MockContext({ $path: sPath }) as any as Context;
|
|
172
236
|
});
|
|
173
237
|
public getMetaPath = jest.fn((sPath: string) => {
|
|
174
238
|
return sPath;
|
|
@@ -207,19 +271,20 @@ export class MockModel implements Partial<ODataModel> {
|
|
|
207
271
|
// Mocked API
|
|
208
272
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
209
273
|
public bindList = jest.fn((...args) => {
|
|
210
|
-
return
|
|
274
|
+
return this.mockListBinding as any as ODataListBinding;
|
|
211
275
|
});
|
|
212
276
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
213
277
|
public bindContext = jest.fn((...args) => {
|
|
214
|
-
return
|
|
278
|
+
return this.mockContextBinding as any as ODataContextBinding;
|
|
215
279
|
});
|
|
216
280
|
public getMetaModel = jest.fn(() => {
|
|
217
|
-
return
|
|
281
|
+
return this.oMetaModel as any as ODataMetaModel;
|
|
218
282
|
});
|
|
219
283
|
|
|
220
284
|
public getProperty = jest.fn();
|
|
221
285
|
public setProperty = jest.fn();
|
|
222
286
|
public requestObject = jest.fn();
|
|
287
|
+
public getKeepAliveContext = jest.fn();
|
|
223
288
|
}
|
|
224
289
|
|
|
225
290
|
export class MockView implements Partial<View> {
|
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
/*!
|
|
2
|
+
* SAP UI development toolkit for HTML5 (SAPUI5)
|
|
3
|
+
* (c) Copyright 2009-2021 SAP SE. All rights reserved
|
|
4
|
+
*/
|
|
5
|
+
sap.ui.define(["sap/base/Log", "sap/fe/core/controllerextensions/collaboration/ActivityBase", "sap/fe/core/controllerextensions/collaboration/CollaborationCommon", "sap/fe/core/controllerextensions/editFlow/draft", "sap/ui/model/json/JSONModel"], function (Log, ActivityBase, CollaborationCommon, draft, JSONModel) {
|
|
6
|
+
"use strict";
|
|
7
|
+
|
|
8
|
+
var Activity = CollaborationCommon.Activity;
|
|
9
|
+
var initializeCollaboration = ActivityBase.initializeCollaboration;
|
|
10
|
+
var endCollaboration = ActivityBase.endCollaboration;
|
|
11
|
+
var broadcastCollaborationMessage = ActivityBase.broadcastCollaborationMessage;
|
|
12
|
+
|
|
13
|
+
var CollaborationAPI = {
|
|
14
|
+
_lastReceivedMessage: undefined,
|
|
15
|
+
_rootPath: "",
|
|
16
|
+
_oModel: undefined,
|
|
17
|
+
_lockedPropertyPath: "",
|
|
18
|
+
_internalModel: undefined,
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Open an existing collaborative draft with a new user, and creates a 'ghost client' for this user.
|
|
22
|
+
*
|
|
23
|
+
* @param oContext The context of the collaborative draft
|
|
24
|
+
* @param userID The ID of the user
|
|
25
|
+
* @param userName The name of the user
|
|
26
|
+
*/
|
|
27
|
+
enterDraft: function (oContext, userID, userName) {
|
|
28
|
+
var webSocketBaseURL = oContext.getModel().getMetaModel().getObject("/@com.sap.vocabularies.Common.v1.WebSocketBaseURL");
|
|
29
|
+
|
|
30
|
+
if (!webSocketBaseURL) {
|
|
31
|
+
Log.error("Cannot find WebSocketBaseURL annotation");
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
var sDraftUUID = oContext.getProperty("DraftAdministrativeData/DraftUUID");
|
|
36
|
+
this._internalModel = new JSONModel({});
|
|
37
|
+
initializeCollaboration({
|
|
38
|
+
id: userID,
|
|
39
|
+
name: userName,
|
|
40
|
+
initialName: userName
|
|
41
|
+
}, webSocketBaseURL, sDraftUUID, this._internalModel, this._onMessageReceived.bind(this), true);
|
|
42
|
+
this._rootPath = oContext.getPath();
|
|
43
|
+
this._oModel = oContext.getModel();
|
|
44
|
+
},
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Checks if the ghost client has revieved a given message.
|
|
48
|
+
*
|
|
49
|
+
* @param message The message content to be looked for
|
|
50
|
+
* @returns True if the last recieved message matches the content
|
|
51
|
+
*/
|
|
52
|
+
checkReceived: function (message) {
|
|
53
|
+
if (!this._lastReceivedMessage) {
|
|
54
|
+
return false;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
var found = (!message.userID || message.userID === this._lastReceivedMessage.userID) && (!message.userAction || message.userAction === this._lastReceivedMessage.userAction) && (!message.clientContent || message.clientContent === this._lastReceivedMessage.clientContent);
|
|
58
|
+
this._lastReceivedMessage = undefined; // reset history to avoid finding the same message twice
|
|
59
|
+
|
|
60
|
+
return found;
|
|
61
|
+
},
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Closes the ghost client and removes the user from the collaborative draft.
|
|
65
|
+
*/
|
|
66
|
+
leaveDraft: function () {
|
|
67
|
+
if (this._internalModel) {
|
|
68
|
+
endCollaboration(this._internalModel);
|
|
69
|
+
|
|
70
|
+
this._internalModel.destroy();
|
|
71
|
+
|
|
72
|
+
this._internalModel = undefined;
|
|
73
|
+
}
|
|
74
|
+
},
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Simulates that the user starts typing in an input (live change).
|
|
78
|
+
*
|
|
79
|
+
* @param sPropertyPath The path of the property being modified
|
|
80
|
+
*/
|
|
81
|
+
startLiveChange: function (sPropertyPath) {
|
|
82
|
+
if (this._internalModel) {
|
|
83
|
+
if (this._lockedPropertyPath) {
|
|
84
|
+
// Unlock previous property path
|
|
85
|
+
this.undoChange();
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
this._lockedPropertyPath = sPropertyPath;
|
|
89
|
+
broadcastCollaborationMessage(Activity.LiveChange, this._rootPath + "/" + sPropertyPath, this._internalModel);
|
|
90
|
+
}
|
|
91
|
+
},
|
|
92
|
+
|
|
93
|
+
/**
|
|
94
|
+
* Simulates that the user has modified a property.
|
|
95
|
+
*
|
|
96
|
+
* @param sPropertyPath The path of the property being modified
|
|
97
|
+
* @param value The new value of the property being modified
|
|
98
|
+
*/
|
|
99
|
+
updatePropertyValue: function (sPropertyPath, value) {
|
|
100
|
+
var _this = this;
|
|
101
|
+
|
|
102
|
+
if (this._internalModel) {
|
|
103
|
+
if (this._lockedPropertyPath !== sPropertyPath) {
|
|
104
|
+
this.startLiveChange(sPropertyPath);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
var oContextBinding = this._oModel.bindContext(this._rootPath, undefined, {
|
|
108
|
+
$$patchWithoutSideEffects: true,
|
|
109
|
+
$$groupId: "$auto",
|
|
110
|
+
$$updateGroupId: "$auto"
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
var oPropertyBinding = this._oModel.bindProperty(sPropertyPath, oContextBinding.getBoundContext());
|
|
114
|
+
|
|
115
|
+
oPropertyBinding.requestValue().then(function () {
|
|
116
|
+
oPropertyBinding.setValue(value);
|
|
117
|
+
oContextBinding.attachEventOnce("patchCompleted", function () {
|
|
118
|
+
broadcastCollaborationMessage(Activity.Change, _this._rootPath + "/" + sPropertyPath, _this._internalModel);
|
|
119
|
+
_this._lockedPropertyPath = "";
|
|
120
|
+
});
|
|
121
|
+
}).catch(function (err) {
|
|
122
|
+
Log.error(err);
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
},
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Simulates that the user did an 'undo' (to be called after startLiveChange).
|
|
129
|
+
*/
|
|
130
|
+
undoChange: function () {
|
|
131
|
+
if (this._lockedPropertyPath) {
|
|
132
|
+
broadcastCollaborationMessage(Activity.Undo, this._rootPath + "/" + this._lockedPropertyPath, this._internalModel);
|
|
133
|
+
this._lockedPropertyPath = "";
|
|
134
|
+
}
|
|
135
|
+
},
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Simulates that the user has discarded the draft.
|
|
139
|
+
*/
|
|
140
|
+
discardDraft: function () {
|
|
141
|
+
var _this2 = this;
|
|
142
|
+
|
|
143
|
+
if (this._internalModel) {
|
|
144
|
+
var draftContext = this._getDraftContext();
|
|
145
|
+
|
|
146
|
+
draftContext.requestProperty("IsActiveEntity").then(function () {
|
|
147
|
+
draft.deleteDraft(draftContext);
|
|
148
|
+
}).then(function () {
|
|
149
|
+
broadcastCollaborationMessage(Activity.Discard, _this2._rootPath.replace("IsActiveEntity=false", "IsActiveEntity=true"), _this2._internalModel);
|
|
150
|
+
|
|
151
|
+
_this2._internalModel.destroy();
|
|
152
|
+
|
|
153
|
+
_this2._internalModel = undefined;
|
|
154
|
+
}).catch(function (err) {
|
|
155
|
+
Log.error(err);
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
},
|
|
159
|
+
|
|
160
|
+
/**
|
|
161
|
+
* Simulates that the user has deleted the draft.
|
|
162
|
+
*/
|
|
163
|
+
deleteDraft: function () {
|
|
164
|
+
var _this3 = this;
|
|
165
|
+
|
|
166
|
+
if (this._internalModel) {
|
|
167
|
+
var draftContext = this._getDraftContext();
|
|
168
|
+
|
|
169
|
+
var activeContext;
|
|
170
|
+
draftContext.requestProperty("IsActiveEntity").then(function () {
|
|
171
|
+
return draftContext.getModel().bindContext(_this3._rootPath + "/SiblingEntity").getBoundContext();
|
|
172
|
+
}).then(function (context) {
|
|
173
|
+
activeContext = context;
|
|
174
|
+
return context.requestCanonicalPath();
|
|
175
|
+
}).then(function () {
|
|
176
|
+
return draft.deleteDraft(draftContext);
|
|
177
|
+
}).then(function () {
|
|
178
|
+
activeContext.delete();
|
|
179
|
+
broadcastCollaborationMessage(Activity.Delete, _this3._rootPath, _this3._internalModel);
|
|
180
|
+
|
|
181
|
+
_this3._internalModel.destroy();
|
|
182
|
+
|
|
183
|
+
_this3._internalModel = undefined;
|
|
184
|
+
}).catch(function (err) {
|
|
185
|
+
Log.error(err);
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
},
|
|
189
|
+
// /////////////////////////////
|
|
190
|
+
// Private methods
|
|
191
|
+
_getDraftContext: function () {
|
|
192
|
+
return this._oModel.bindContext(this._rootPath, undefined, {
|
|
193
|
+
$$patchWithoutSideEffects: true,
|
|
194
|
+
$$groupId: "$auto",
|
|
195
|
+
$$updateGroupId: "$auto"
|
|
196
|
+
}).getBoundContext();
|
|
197
|
+
},
|
|
198
|
+
|
|
199
|
+
/**
|
|
200
|
+
* Callback of the ghost client when receiving a message on the web socket.
|
|
201
|
+
*
|
|
202
|
+
* @param oMessage The message
|
|
203
|
+
*/
|
|
204
|
+
_onMessageReceived: function (oMessage) {
|
|
205
|
+
oMessage.userAction = oMessage.userAction || oMessage.clientAction;
|
|
206
|
+
this._lastReceivedMessage = oMessage;
|
|
207
|
+
|
|
208
|
+
if (oMessage.userAction === Activity.Join) {
|
|
209
|
+
broadcastCollaborationMessage(Activity.JoinEcho, this._lockedPropertyPath ? this._rootPath + "/" + this._lockedPropertyPath : undefined, this._internalModel);
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
};
|
|
213
|
+
return CollaborationAPI;
|
|
214
|
+
}, false);
|
|
215
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["CollaborationAPI.ts"],"names":["CollaborationAPI","_lastReceivedMessage","undefined","_rootPath","_oModel","_lockedPropertyPath","_internalModel","enterDraft","oContext","userID","userName","webSocketBaseURL","getModel","getMetaModel","getObject","Log","error","sDraftUUID","getProperty","JSONModel","initializeCollaboration","id","name","initialName","_onMessageReceived","bind","getPath","checkReceived","message","found","userAction","clientContent","leaveDraft","endCollaboration","destroy","startLiveChange","sPropertyPath","undoChange","broadcastCollaborationMessage","Activity","LiveChange","updatePropertyValue","value","oContextBinding","bindContext","$$patchWithoutSideEffects","$$groupId","$$updateGroupId","oPropertyBinding","bindProperty","getBoundContext","requestValue","then","setValue","attachEventOnce","Change","catch","err","Undo","discardDraft","draftContext","_getDraftContext","requestProperty","draft","deleteDraft","Discard","replace","activeContext","context","requestCanonicalPath","delete","Delete","oMessage","clientAction","Join","JoinEcho"],"mappings":";AAAA;AACA;AACA;;;;;;;;;AAUA,MAAMA,gBAAgB,GAAG;AACxBC,IAAAA,oBAAoB,EAAEC,SADE;AAExBC,IAAAA,SAAS,EAAE,EAFa;AAGxBC,IAAAA,OAAO,EAAEF,SAHe;AAIxBG,IAAAA,mBAAmB,EAAE,EAJG;AAKxBC,IAAAA,cAAc,EAAEJ,SALQ;;AAOxB;AACD;AACA;AACA;AACA;AACA;AACA;AACCK,IAAAA,UAAU,EAAE,UAAUC,QAAV,EAA6BC,MAA7B,EAA6CC,QAA7C,EAA+D;AAC1E,UAAMC,gBAAwB,GAAGH,QAAQ,CAACI,QAAT,GAAoBC,YAApB,GAAmCC,SAAnC,CAA6C,mDAA7C,CAAjC;;AAEA,UAAI,CAACH,gBAAL,EAAuB;AACtBI,QAAAA,GAAG,CAACC,KAAJ,CAAU,yCAAV;AACA;AACA;;AAED,UAAMC,UAAkB,GAAGT,QAAQ,CAACU,WAAT,CAAqB,mCAArB,CAA3B;AACA,WAAKZ,cAAL,GAAsB,IAAIa,SAAJ,CAAc,EAAd,CAAtB;AAEAC,MAAAA,uBAAuB,CACtB;AACCC,QAAAA,EAAE,EAAEZ,MADL;AAECa,QAAAA,IAAI,EAAEZ,QAFP;AAGCa,QAAAA,WAAW,EAAEb;AAHd,OADsB,EAMtBC,gBANsB,EAOtBM,UAPsB,EAQtB,KAAKX,cARiB,EAStB,KAAKkB,kBAAL,CAAwBC,IAAxB,CAA6B,IAA7B,CATsB,EAUtB,IAVsB,CAAvB;AAaA,WAAKtB,SAAL,GAAiBK,QAAQ,CAACkB,OAAT,EAAjB;AACA,WAAKtB,OAAL,GAAeI,QAAQ,CAACI,QAAT,EAAf;AACA,KAxCuB;;AA0CxB;AACD;AACA;AACA;AACA;AACA;AACCe,IAAAA,aAAa,EAAE,UAAUC,OAAV,EAA8C;AAC5D,UAAI,CAAC,KAAK3B,oBAAV,EAAgC;AAC/B,eAAO,KAAP;AACA;;AAED,UAAM4B,KAAK,GACV,CAAC,CAACD,OAAO,CAACnB,MAAT,IAAmBmB,OAAO,CAACnB,MAAR,KAAmB,KAAKR,oBAAL,CAA0BQ,MAAjE,MACC,CAACmB,OAAO,CAACE,UAAT,IAAuBF,OAAO,CAACE,UAAR,KAAuB,KAAK7B,oBAAL,CAA0B6B,UADzE,MAEC,CAACF,OAAO,CAACG,aAAT,IAA0BH,OAAO,CAACG,aAAR,KAA0B,KAAK9B,oBAAL,CAA0B8B,aAF/E,CADD;AAKA,WAAK9B,oBAAL,GAA4BC,SAA5B,CAV4D,CAUrB;;AAEvC,aAAO2B,KAAP;AACA,KA7DuB;;AA+DxB;AACD;AACA;AACCG,IAAAA,UAAU,EAAE,YAAY;AACvB,UAAI,KAAK1B,cAAT,EAAyB;AACxB2B,QAAAA,gBAAgB,CAAC,KAAK3B,cAAN,CAAhB;;AACA,aAAKA,cAAL,CAAoB4B,OAApB;;AACA,aAAK5B,cAAL,GAAsBJ,SAAtB;AACA;AACD,KAxEuB;;AA0ExB;AACD;AACA;AACA;AACA;AACCiC,IAAAA,eAAe,EAAE,UAAUC,aAAV,EAAiC;AACjD,UAAI,KAAK9B,cAAT,EAAyB;AACxB,YAAI,KAAKD,mBAAT,EAA8B;AAC7B;AACA,eAAKgC,UAAL;AACA;;AACD,aAAKhC,mBAAL,GAA2B+B,aAA3B;AACAE,QAAAA,6BAA6B,CAACC,QAAQ,CAACC,UAAV,EAAsB,KAAKrC,SAAL,GAAiB,GAAjB,GAAuBiC,aAA7C,EAA4D,KAAK9B,cAAjE,CAA7B;AACA;AACD,KAxFuB;;AA0FxB;AACD;AACA;AACA;AACA;AACA;AACCmC,IAAAA,mBAAmB,EAAE,UAAUL,aAAV,EAAiCM,KAAjC,EAA6C;AAAA;;AACjE,UAAI,KAAKpC,cAAT,EAAyB;AACxB,YAAI,KAAKD,mBAAL,KAA6B+B,aAAjC,EAAgD;AAC/C,eAAKD,eAAL,CAAqBC,aAArB;AACA;;AAED,YAAMO,eAAe,GAAG,KAAKvC,OAAL,CAAcwC,WAAd,CAA0B,KAAKzC,SAA/B,EAA0CD,SAA1C,EAAqD;AAC5E2C,UAAAA,yBAAyB,EAAE,IADiD;AAE5EC,UAAAA,SAAS,EAAE,OAFiE;AAG5EC,UAAAA,eAAe,EAAE;AAH2D,SAArD,CAAxB;;AAMA,YAAMC,gBAAgB,GAAG,KAAK5C,OAAL,CAAc6C,YAAd,CAA2Bb,aAA3B,EAA0CO,eAAe,CAACO,eAAhB,EAA1C,CAAzB;;AAEAF,QAAAA,gBAAgB,CACdG,YADF,GAEEC,IAFF,CAEO,YAAM;AACXJ,UAAAA,gBAAgB,CAACK,QAAjB,CAA0BX,KAA1B;AACAC,UAAAA,eAAe,CAACW,eAAhB,CAAgC,gBAAhC,EAAkD,YAAM;AACvDhB,YAAAA,6BAA6B,CAACC,QAAQ,CAACgB,MAAV,EAAkB,KAAI,CAACpD,SAAL,GAAiB,GAAjB,GAAuBiC,aAAzC,EAAwD,KAAI,CAAC9B,cAA7D,CAA7B;AACA,YAAA,KAAI,CAACD,mBAAL,GAA2B,EAA3B;AACA,WAHD;AAIA,SARF,EASEmD,KATF,CASQ,UAAUC,GAAV,EAAe;AACrB1C,UAAAA,GAAG,CAACC,KAAJ,CAAUyC,GAAV;AACA,SAXF;AAYA;AACD,KA3HuB;;AA6HxB;AACD;AACA;AACCpB,IAAAA,UAAU,EAAE,YAAY;AACvB,UAAI,KAAKhC,mBAAT,EAA8B;AAC7BiC,QAAAA,6BAA6B,CAACC,QAAQ,CAACmB,IAAV,EAAgB,KAAKvD,SAAL,GAAiB,GAAjB,GAAuB,KAAKE,mBAA5C,EAAiE,KAAKC,cAAtE,CAA7B;AACA,aAAKD,mBAAL,GAA2B,EAA3B;AACA;AACD,KArIuB;;AAuIxB;AACD;AACA;AACCsD,IAAAA,YAAY,EAAE,YAAY;AAAA;;AACzB,UAAI,KAAKrD,cAAT,EAAyB;AACxB,YAAMsD,YAAY,GAAG,KAAKC,gBAAL,EAArB;;AAEAD,QAAAA,YAAY,CACVE,eADF,CACkB,gBADlB,EAEEV,IAFF,CAEO,YAAM;AACXW,UAAAA,KAAK,CAACC,WAAN,CAAkBJ,YAAlB;AACA,SAJF,EAKER,IALF,CAKO,YAAM;AACXd,UAAAA,6BAA6B,CAC5BC,QAAQ,CAAC0B,OADmB,EAE5B,MAAI,CAAC9D,SAAL,CAAe+D,OAAf,CAAuB,sBAAvB,EAA+C,qBAA/C,CAF4B,EAG5B,MAAI,CAAC5D,cAHuB,CAA7B;;AAKA,UAAA,MAAI,CAACA,cAAL,CAAqB4B,OAArB;;AACA,UAAA,MAAI,CAAC5B,cAAL,GAAsBJ,SAAtB;AACA,SAbF,EAcEsD,KAdF,CAcQ,UAAUC,GAAV,EAAoB;AAC1B1C,UAAAA,GAAG,CAACC,KAAJ,CAAUyC,GAAV;AACA,SAhBF;AAiBA;AACD,KAhKuB;;AAkKxB;AACD;AACA;AACCO,IAAAA,WAAW,EAAE,YAAY;AAAA;;AACxB,UAAI,KAAK1D,cAAT,EAAyB;AACxB,YAAMsD,YAAY,GAAG,KAAKC,gBAAL,EAArB;;AACA,YAAIM,aAAJ;AAEAP,QAAAA,YAAY,CACVE,eADF,CACkB,gBADlB,EAEEV,IAFF,CAEO,YAAM;AACX,iBAAOQ,YAAY,CACjBhD,QADK,GAELgC,WAFK,CAEO,MAAI,CAACzC,SAAL,GAAiB,gBAFxB,EAGL+C,eAHK,EAAP;AAIA,SAPF,EAQEE,IARF,CAQO,UAACgB,OAAD,EAAkB;AACvBD,UAAAA,aAAa,GAAGC,OAAhB;AACA,iBAAOA,OAAO,CAACC,oBAAR,EAAP;AACA,SAXF,EAYEjB,IAZF,CAYO,YAAM;AACX,iBAAOW,KAAK,CAACC,WAAN,CAAkBJ,YAAlB,CAAP;AACA,SAdF,EAeER,IAfF,CAeO,YAAM;AACVe,UAAAA,aAAD,CAAuBG,MAAvB;AACAhC,UAAAA,6BAA6B,CAACC,QAAQ,CAACgC,MAAV,EAAkB,MAAI,CAACpE,SAAvB,EAAkC,MAAI,CAACG,cAAvC,CAA7B;;AACA,UAAA,MAAI,CAACA,cAAL,CAAqB4B,OAArB;;AACA,UAAA,MAAI,CAAC5B,cAAL,GAAsBJ,SAAtB;AACA,SApBF,EAqBEsD,KArBF,CAqBQ,UAAUC,GAAV,EAAoB;AAC1B1C,UAAAA,GAAG,CAACC,KAAJ,CAAUyC,GAAV;AACA,SAvBF;AAwBA;AACD,KAnMuB;AAqMxB;AACA;AAEAI,IAAAA,gBAAgB,EAAE,YAAiB;AAClC,aAAO,KAAKzD,OAAL,CAAcwC,WAAd,CAA0B,KAAKzC,SAA/B,EAA0CD,SAA1C,EAAqD;AAC3D2C,QAAAA,yBAAyB,EAAE,IADgC;AAE3DC,QAAAA,SAAS,EAAE,OAFgD;AAG3DC,QAAAA,eAAe,EAAE;AAH0C,OAArD,EAIJG,eAJI,EAAP;AAKA,KA9MuB;;AAgNxB;AACD;AACA;AACA;AACA;AACC1B,IAAAA,kBAAkB,EAAE,UAAUgD,QAAV,EAA6B;AAChDA,MAAAA,QAAQ,CAAC1C,UAAT,GAAsB0C,QAAQ,CAAC1C,UAAT,IAAuB0C,QAAQ,CAACC,YAAtD;AACA,WAAKxE,oBAAL,GAA4BuE,QAA5B;;AAEA,UAAIA,QAAQ,CAAC1C,UAAT,KAAwBS,QAAQ,CAACmC,IAArC,EAA2C;AAC1CpC,QAAAA,6BAA6B,CAC5BC,QAAQ,CAACoC,QADmB,EAE5B,KAAKtE,mBAAL,GAA2B,KAAKF,SAAL,GAAiB,GAAjB,GAAuB,KAAKE,mBAAvD,GAA6EH,SAFjD,EAG5B,KAAKI,cAHuB,CAA7B;AAKA;AACD;AAhOuB,GAAzB;SAmOeN,gB","sourceRoot":".","sourcesContent":["import Log from \"sap/base/Log\";\nimport {\n\tbroadcastCollaborationMessage,\n\tendCollaboration,\n\tinitializeCollaboration\n} from \"sap/fe/core/controllerextensions/collaboration/ActivityBase\";\nimport { Activity, Message } from \"sap/fe/core/controllerextensions/collaboration/CollaborationCommon\";\nimport draft from \"sap/fe/core/controllerextensions/editFlow/draft\";\nimport Context from \"sap/ui/model/Context\";\nimport JSONModel from \"sap/ui/model/json/JSONModel\";\nimport ODataModel from \"sap/ui/model/odata/v4/ODataModel\";\n\nconst CollaborationAPI = {\n\t_lastReceivedMessage: undefined as Message | undefined,\n\t_rootPath: \"\",\n\t_oModel: undefined as ODataModel | undefined,\n\t_lockedPropertyPath: \"\",\n\t_internalModel: undefined as JSONModel | undefined,\n\n\t/**\n\t * Open an existing collaborative draft with a new user, and creates a 'ghost client' for this user.\n\t *\n\t * @param oContext The context of the collaborative draft\n\t * @param userID The ID of the user\n\t * @param userName The name of the user\n\t */\n\tenterDraft: function (oContext: Context, userID: string, userName: string) {\n\t\tconst webSocketBaseURL: string = oContext.getModel().getMetaModel().getObject(\"/@com.sap.vocabularies.Common.v1.WebSocketBaseURL\");\n\n\t\tif (!webSocketBaseURL) {\n\t\t\tLog.error(\"Cannot find WebSocketBaseURL annotation\");\n\t\t\treturn;\n\t\t}\n\n\t\tconst sDraftUUID: string = oContext.getProperty(\"DraftAdministrativeData/DraftUUID\");\n\t\tthis._internalModel = new JSONModel({});\n\n\t\tinitializeCollaboration(\n\t\t\t{\n\t\t\t\tid: userID,\n\t\t\t\tname: userName,\n\t\t\t\tinitialName: userName\n\t\t\t},\n\t\t\twebSocketBaseURL,\n\t\t\tsDraftUUID,\n\t\t\tthis._internalModel,\n\t\t\tthis._onMessageReceived.bind(this),\n\t\t\ttrue\n\t\t);\n\n\t\tthis._rootPath = oContext.getPath();\n\t\tthis._oModel = oContext.getModel() as ODataModel;\n\t},\n\n\t/**\n\t * Checks if the ghost client has revieved a given message.\n\t *\n\t * @param message The message content to be looked for\n\t * @returns True if the last recieved message matches the content\n\t */\n\tcheckReceived: function (message: Partial<Message>): boolean {\n\t\tif (!this._lastReceivedMessage) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst found =\n\t\t\t(!message.userID || message.userID === this._lastReceivedMessage.userID) &&\n\t\t\t(!message.userAction || message.userAction === this._lastReceivedMessage.userAction) &&\n\t\t\t(!message.clientContent || message.clientContent === this._lastReceivedMessage.clientContent);\n\n\t\tthis._lastReceivedMessage = undefined; // reset history to avoid finding the same message twice\n\n\t\treturn found;\n\t},\n\n\t/**\n\t * Closes the ghost client and removes the user from the collaborative draft.\n\t */\n\tleaveDraft: function () {\n\t\tif (this._internalModel) {\n\t\t\tendCollaboration(this._internalModel);\n\t\t\tthis._internalModel.destroy();\n\t\t\tthis._internalModel = undefined;\n\t\t}\n\t},\n\n\t/**\n\t * Simulates that the user starts typing in an input (live change).\n\t *\n\t * @param sPropertyPath The path of the property being modified\n\t */\n\tstartLiveChange: function (sPropertyPath: string) {\n\t\tif (this._internalModel) {\n\t\t\tif (this._lockedPropertyPath) {\n\t\t\t\t// Unlock previous property path\n\t\t\t\tthis.undoChange();\n\t\t\t}\n\t\t\tthis._lockedPropertyPath = sPropertyPath;\n\t\t\tbroadcastCollaborationMessage(Activity.LiveChange, this._rootPath + \"/\" + sPropertyPath, this._internalModel);\n\t\t}\n\t},\n\n\t/**\n\t * Simulates that the user has modified a property.\n\t *\n\t * @param sPropertyPath The path of the property being modified\n\t * @param value The new value of the property being modified\n\t */\n\tupdatePropertyValue: function (sPropertyPath: string, value: any) {\n\t\tif (this._internalModel) {\n\t\t\tif (this._lockedPropertyPath !== sPropertyPath) {\n\t\t\t\tthis.startLiveChange(sPropertyPath);\n\t\t\t}\n\n\t\t\tconst oContextBinding = this._oModel!.bindContext(this._rootPath, undefined, {\n\t\t\t\t$$patchWithoutSideEffects: true,\n\t\t\t\t$$groupId: \"$auto\",\n\t\t\t\t$$updateGroupId: \"$auto\"\n\t\t\t});\n\n\t\t\tconst oPropertyBinding = this._oModel!.bindProperty(sPropertyPath, oContextBinding.getBoundContext());\n\n\t\t\toPropertyBinding\n\t\t\t\t.requestValue()\n\t\t\t\t.then(() => {\n\t\t\t\t\toPropertyBinding.setValue(value);\n\t\t\t\t\toContextBinding.attachEventOnce(\"patchCompleted\", () => {\n\t\t\t\t\t\tbroadcastCollaborationMessage(Activity.Change, this._rootPath + \"/\" + sPropertyPath, this._internalModel!);\n\t\t\t\t\t\tthis._lockedPropertyPath = \"\";\n\t\t\t\t\t});\n\t\t\t\t})\n\t\t\t\t.catch(function (err) {\n\t\t\t\t\tLog.error(err);\n\t\t\t\t});\n\t\t}\n\t},\n\n\t/**\n\t * Simulates that the user did an 'undo' (to be called after startLiveChange).\n\t */\n\tundoChange: function () {\n\t\tif (this._lockedPropertyPath) {\n\t\t\tbroadcastCollaborationMessage(Activity.Undo, this._rootPath + \"/\" + this._lockedPropertyPath, this._internalModel!);\n\t\t\tthis._lockedPropertyPath = \"\";\n\t\t}\n\t},\n\n\t/**\n\t * Simulates that the user has discarded the draft.\n\t */\n\tdiscardDraft: function () {\n\t\tif (this._internalModel) {\n\t\t\tconst draftContext = this._getDraftContext();\n\n\t\t\tdraftContext\n\t\t\t\t.requestProperty(\"IsActiveEntity\")\n\t\t\t\t.then(() => {\n\t\t\t\t\tdraft.deleteDraft(draftContext);\n\t\t\t\t})\n\t\t\t\t.then(() => {\n\t\t\t\t\tbroadcastCollaborationMessage(\n\t\t\t\t\t\tActivity.Discard,\n\t\t\t\t\t\tthis._rootPath.replace(\"IsActiveEntity=false\", \"IsActiveEntity=true\"),\n\t\t\t\t\t\tthis._internalModel!\n\t\t\t\t\t);\n\t\t\t\t\tthis._internalModel!.destroy();\n\t\t\t\t\tthis._internalModel = undefined;\n\t\t\t\t})\n\t\t\t\t.catch(function (err: any) {\n\t\t\t\t\tLog.error(err);\n\t\t\t\t});\n\t\t}\n\t},\n\n\t/**\n\t * Simulates that the user has deleted the draft.\n\t */\n\tdeleteDraft: function () {\n\t\tif (this._internalModel) {\n\t\t\tconst draftContext = this._getDraftContext();\n\t\t\tlet activeContext: Context;\n\n\t\t\tdraftContext\n\t\t\t\t.requestProperty(\"IsActiveEntity\")\n\t\t\t\t.then(() => {\n\t\t\t\t\treturn draftContext\n\t\t\t\t\t\t.getModel()\n\t\t\t\t\t\t.bindContext(this._rootPath + \"/SiblingEntity\")\n\t\t\t\t\t\t.getBoundContext();\n\t\t\t\t})\n\t\t\t\t.then((context: any) => {\n\t\t\t\t\tactiveContext = context;\n\t\t\t\t\treturn context.requestCanonicalPath();\n\t\t\t\t})\n\t\t\t\t.then(() => {\n\t\t\t\t\treturn draft.deleteDraft(draftContext);\n\t\t\t\t})\n\t\t\t\t.then(() => {\n\t\t\t\t\t(activeContext as any).delete();\n\t\t\t\t\tbroadcastCollaborationMessage(Activity.Delete, this._rootPath, this._internalModel!);\n\t\t\t\t\tthis._internalModel!.destroy();\n\t\t\t\t\tthis._internalModel = undefined;\n\t\t\t\t})\n\t\t\t\t.catch(function (err: any) {\n\t\t\t\t\tLog.error(err);\n\t\t\t\t});\n\t\t}\n\t},\n\n\t// /////////////////////////////\n\t// Private methods\n\n\t_getDraftContext: function (): any {\n\t\treturn this._oModel!.bindContext(this._rootPath, undefined, {\n\t\t\t$$patchWithoutSideEffects: true,\n\t\t\t$$groupId: \"$auto\",\n\t\t\t$$updateGroupId: \"$auto\"\n\t\t}).getBoundContext();\n\t},\n\n\t/**\n\t * Callback of the ghost client when receiving a message on the web socket.\n\t *\n\t * @param oMessage The message\n\t */\n\t_onMessageReceived: function (oMessage: Message) {\n\t\toMessage.userAction = oMessage.userAction || oMessage.clientAction;\n\t\tthis._lastReceivedMessage = oMessage;\n\n\t\tif (oMessage.userAction === Activity.Join) {\n\t\t\tbroadcastCollaborationMessage(\n\t\t\t\tActivity.JoinEcho,\n\t\t\t\tthis._lockedPropertyPath ? this._rootPath + \"/\" + this._lockedPropertyPath : undefined,\n\t\t\t\tthis._internalModel!\n\t\t\t);\n\t\t}\n\t}\n};\n\nexport default CollaborationAPI;\n"]}
|
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
import Log from "sap/base/Log";
|
|
2
|
+
import {
|
|
3
|
+
broadcastCollaborationMessage,
|
|
4
|
+
endCollaboration,
|
|
5
|
+
initializeCollaboration
|
|
6
|
+
} from "sap/fe/core/controllerextensions/collaboration/ActivityBase";
|
|
7
|
+
import { Activity, Message } from "sap/fe/core/controllerextensions/collaboration/CollaborationCommon";
|
|
8
|
+
import draft from "sap/fe/core/controllerextensions/editFlow/draft";
|
|
9
|
+
import Context from "sap/ui/model/Context";
|
|
10
|
+
import JSONModel from "sap/ui/model/json/JSONModel";
|
|
11
|
+
import ODataModel from "sap/ui/model/odata/v4/ODataModel";
|
|
12
|
+
|
|
13
|
+
const CollaborationAPI = {
|
|
14
|
+
_lastReceivedMessage: undefined as Message | undefined,
|
|
15
|
+
_rootPath: "",
|
|
16
|
+
_oModel: undefined as ODataModel | undefined,
|
|
17
|
+
_lockedPropertyPath: "",
|
|
18
|
+
_internalModel: undefined as JSONModel | undefined,
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Open an existing collaborative draft with a new user, and creates a 'ghost client' for this user.
|
|
22
|
+
*
|
|
23
|
+
* @param oContext The context of the collaborative draft
|
|
24
|
+
* @param userID The ID of the user
|
|
25
|
+
* @param userName The name of the user
|
|
26
|
+
*/
|
|
27
|
+
enterDraft: function (oContext: Context, userID: string, userName: string) {
|
|
28
|
+
const webSocketBaseURL: string = oContext.getModel().getMetaModel().getObject("/@com.sap.vocabularies.Common.v1.WebSocketBaseURL");
|
|
29
|
+
|
|
30
|
+
if (!webSocketBaseURL) {
|
|
31
|
+
Log.error("Cannot find WebSocketBaseURL annotation");
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
const sDraftUUID: string = oContext.getProperty("DraftAdministrativeData/DraftUUID");
|
|
36
|
+
this._internalModel = new JSONModel({});
|
|
37
|
+
|
|
38
|
+
initializeCollaboration(
|
|
39
|
+
{
|
|
40
|
+
id: userID,
|
|
41
|
+
name: userName,
|
|
42
|
+
initialName: userName
|
|
43
|
+
},
|
|
44
|
+
webSocketBaseURL,
|
|
45
|
+
sDraftUUID,
|
|
46
|
+
this._internalModel,
|
|
47
|
+
this._onMessageReceived.bind(this),
|
|
48
|
+
true
|
|
49
|
+
);
|
|
50
|
+
|
|
51
|
+
this._rootPath = oContext.getPath();
|
|
52
|
+
this._oModel = oContext.getModel() as ODataModel;
|
|
53
|
+
},
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Checks if the ghost client has revieved a given message.
|
|
57
|
+
*
|
|
58
|
+
* @param message The message content to be looked for
|
|
59
|
+
* @returns True if the last recieved message matches the content
|
|
60
|
+
*/
|
|
61
|
+
checkReceived: function (message: Partial<Message>): boolean {
|
|
62
|
+
if (!this._lastReceivedMessage) {
|
|
63
|
+
return false;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
const found =
|
|
67
|
+
(!message.userID || message.userID === this._lastReceivedMessage.userID) &&
|
|
68
|
+
(!message.userAction || message.userAction === this._lastReceivedMessage.userAction) &&
|
|
69
|
+
(!message.clientContent || message.clientContent === this._lastReceivedMessage.clientContent);
|
|
70
|
+
|
|
71
|
+
this._lastReceivedMessage = undefined; // reset history to avoid finding the same message twice
|
|
72
|
+
|
|
73
|
+
return found;
|
|
74
|
+
},
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Closes the ghost client and removes the user from the collaborative draft.
|
|
78
|
+
*/
|
|
79
|
+
leaveDraft: function () {
|
|
80
|
+
if (this._internalModel) {
|
|
81
|
+
endCollaboration(this._internalModel);
|
|
82
|
+
this._internalModel.destroy();
|
|
83
|
+
this._internalModel = undefined;
|
|
84
|
+
}
|
|
85
|
+
},
|
|
86
|
+
|
|
87
|
+
/**
|
|
88
|
+
* Simulates that the user starts typing in an input (live change).
|
|
89
|
+
*
|
|
90
|
+
* @param sPropertyPath The path of the property being modified
|
|
91
|
+
*/
|
|
92
|
+
startLiveChange: function (sPropertyPath: string) {
|
|
93
|
+
if (this._internalModel) {
|
|
94
|
+
if (this._lockedPropertyPath) {
|
|
95
|
+
// Unlock previous property path
|
|
96
|
+
this.undoChange();
|
|
97
|
+
}
|
|
98
|
+
this._lockedPropertyPath = sPropertyPath;
|
|
99
|
+
broadcastCollaborationMessage(Activity.LiveChange, this._rootPath + "/" + sPropertyPath, this._internalModel);
|
|
100
|
+
}
|
|
101
|
+
},
|
|
102
|
+
|
|
103
|
+
/**
|
|
104
|
+
* Simulates that the user has modified a property.
|
|
105
|
+
*
|
|
106
|
+
* @param sPropertyPath The path of the property being modified
|
|
107
|
+
* @param value The new value of the property being modified
|
|
108
|
+
*/
|
|
109
|
+
updatePropertyValue: function (sPropertyPath: string, value: any) {
|
|
110
|
+
if (this._internalModel) {
|
|
111
|
+
if (this._lockedPropertyPath !== sPropertyPath) {
|
|
112
|
+
this.startLiveChange(sPropertyPath);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
const oContextBinding = this._oModel!.bindContext(this._rootPath, undefined, {
|
|
116
|
+
$$patchWithoutSideEffects: true,
|
|
117
|
+
$$groupId: "$auto",
|
|
118
|
+
$$updateGroupId: "$auto"
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
const oPropertyBinding = this._oModel!.bindProperty(sPropertyPath, oContextBinding.getBoundContext());
|
|
122
|
+
|
|
123
|
+
oPropertyBinding
|
|
124
|
+
.requestValue()
|
|
125
|
+
.then(() => {
|
|
126
|
+
oPropertyBinding.setValue(value);
|
|
127
|
+
oContextBinding.attachEventOnce("patchCompleted", () => {
|
|
128
|
+
broadcastCollaborationMessage(Activity.Change, this._rootPath + "/" + sPropertyPath, this._internalModel!);
|
|
129
|
+
this._lockedPropertyPath = "";
|
|
130
|
+
});
|
|
131
|
+
})
|
|
132
|
+
.catch(function (err) {
|
|
133
|
+
Log.error(err);
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
},
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Simulates that the user did an 'undo' (to be called after startLiveChange).
|
|
140
|
+
*/
|
|
141
|
+
undoChange: function () {
|
|
142
|
+
if (this._lockedPropertyPath) {
|
|
143
|
+
broadcastCollaborationMessage(Activity.Undo, this._rootPath + "/" + this._lockedPropertyPath, this._internalModel!);
|
|
144
|
+
this._lockedPropertyPath = "";
|
|
145
|
+
}
|
|
146
|
+
},
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* Simulates that the user has discarded the draft.
|
|
150
|
+
*/
|
|
151
|
+
discardDraft: function () {
|
|
152
|
+
if (this._internalModel) {
|
|
153
|
+
const draftContext = this._getDraftContext();
|
|
154
|
+
|
|
155
|
+
draftContext
|
|
156
|
+
.requestProperty("IsActiveEntity")
|
|
157
|
+
.then(() => {
|
|
158
|
+
draft.deleteDraft(draftContext);
|
|
159
|
+
})
|
|
160
|
+
.then(() => {
|
|
161
|
+
broadcastCollaborationMessage(
|
|
162
|
+
Activity.Discard,
|
|
163
|
+
this._rootPath.replace("IsActiveEntity=false", "IsActiveEntity=true"),
|
|
164
|
+
this._internalModel!
|
|
165
|
+
);
|
|
166
|
+
this._internalModel!.destroy();
|
|
167
|
+
this._internalModel = undefined;
|
|
168
|
+
})
|
|
169
|
+
.catch(function (err: any) {
|
|
170
|
+
Log.error(err);
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
},
|
|
174
|
+
|
|
175
|
+
/**
|
|
176
|
+
* Simulates that the user has deleted the draft.
|
|
177
|
+
*/
|
|
178
|
+
deleteDraft: function () {
|
|
179
|
+
if (this._internalModel) {
|
|
180
|
+
const draftContext = this._getDraftContext();
|
|
181
|
+
let activeContext: Context;
|
|
182
|
+
|
|
183
|
+
draftContext
|
|
184
|
+
.requestProperty("IsActiveEntity")
|
|
185
|
+
.then(() => {
|
|
186
|
+
return draftContext
|
|
187
|
+
.getModel()
|
|
188
|
+
.bindContext(this._rootPath + "/SiblingEntity")
|
|
189
|
+
.getBoundContext();
|
|
190
|
+
})
|
|
191
|
+
.then((context: any) => {
|
|
192
|
+
activeContext = context;
|
|
193
|
+
return context.requestCanonicalPath();
|
|
194
|
+
})
|
|
195
|
+
.then(() => {
|
|
196
|
+
return draft.deleteDraft(draftContext);
|
|
197
|
+
})
|
|
198
|
+
.then(() => {
|
|
199
|
+
(activeContext as any).delete();
|
|
200
|
+
broadcastCollaborationMessage(Activity.Delete, this._rootPath, this._internalModel!);
|
|
201
|
+
this._internalModel!.destroy();
|
|
202
|
+
this._internalModel = undefined;
|
|
203
|
+
})
|
|
204
|
+
.catch(function (err: any) {
|
|
205
|
+
Log.error(err);
|
|
206
|
+
});
|
|
207
|
+
}
|
|
208
|
+
},
|
|
209
|
+
|
|
210
|
+
// /////////////////////////////
|
|
211
|
+
// Private methods
|
|
212
|
+
|
|
213
|
+
_getDraftContext: function (): any {
|
|
214
|
+
return this._oModel!.bindContext(this._rootPath, undefined, {
|
|
215
|
+
$$patchWithoutSideEffects: true,
|
|
216
|
+
$$groupId: "$auto",
|
|
217
|
+
$$updateGroupId: "$auto"
|
|
218
|
+
}).getBoundContext();
|
|
219
|
+
},
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* Callback of the ghost client when receiving a message on the web socket.
|
|
223
|
+
*
|
|
224
|
+
* @param oMessage The message
|
|
225
|
+
*/
|
|
226
|
+
_onMessageReceived: function (oMessage: Message) {
|
|
227
|
+
oMessage.userAction = oMessage.userAction || oMessage.clientAction;
|
|
228
|
+
this._lastReceivedMessage = oMessage;
|
|
229
|
+
|
|
230
|
+
if (oMessage.userAction === Activity.Join) {
|
|
231
|
+
broadcastCollaborationMessage(
|
|
232
|
+
Activity.JoinEcho,
|
|
233
|
+
this._lockedPropertyPath ? this._rootPath + "/" + this._lockedPropertyPath : undefined,
|
|
234
|
+
this._internalModel!
|
|
235
|
+
);
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
};
|
|
239
|
+
|
|
240
|
+
export default CollaborationAPI;
|