2017-11-29 19:11:03 +01:00
|
|
|
/*
|
2017-11-30 11:20:29 +01:00
|
|
|
Copyright 2017 New Vector Ltd
|
2017-11-29 19:11:03 +01:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2017-12-01 15:44:14 +01:00
|
|
|
/*
|
2018-02-23 16:11:28 +01:00
|
|
|
* See - https://docs.google.com/document/d/1uPF7XWY_dXTKVKV7jZQ2KmsI19wn9-kFRgQ1tFQP7wQ/edit?usp=sharing for
|
|
|
|
* spec. details / documentation.
|
2017-12-01 15:44:14 +01:00
|
|
|
*/
|
|
|
|
|
2018-03-28 13:22:06 +02:00
|
|
|
import FromWidgetPostMessageApi from './FromWidgetPostMessageApi';
|
|
|
|
import ToWidgetPostMessageApi from './ToWidgetPostMessageApi';
|
2017-12-05 01:08:17 +01:00
|
|
|
|
2018-03-28 13:22:06 +02:00
|
|
|
if (!global.mxFromWidgetMessaging) {
|
|
|
|
global.mxFromWidgetMessaging = new FromWidgetPostMessageApi();
|
|
|
|
global.mxFromWidgetMessaging.start();
|
2017-12-04 18:54:00 +01:00
|
|
|
}
|
2018-03-28 13:22:06 +02:00
|
|
|
if (!global.mxToWidgetMessaging) {
|
|
|
|
global.mxToWidgetMessaging = new ToWidgetPostMessageApi();
|
|
|
|
global.mxToWidgetMessaging.start();
|
2017-12-04 18:54:00 +01:00
|
|
|
}
|
2017-11-30 12:30:30 +01:00
|
|
|
|
2018-03-28 13:22:06 +02:00
|
|
|
const OUTBOUND_API_NAME = 'toWidget';
|
2018-01-18 14:16:06 +01:00
|
|
|
|
2018-03-28 13:22:06 +02:00
|
|
|
export default class WidgetMessaging {
|
|
|
|
constructor(widgetId, widgetUrl, target) {
|
|
|
|
this.widgetId = widgetId;
|
|
|
|
this.widgetUrl = widgetUrl;
|
|
|
|
this.target = target;
|
|
|
|
this.fromWidget = global.mxFromWidgetMessaging;
|
|
|
|
this.toWidget = global.mxToWidgetMessaging;
|
|
|
|
this.start();
|
2017-12-01 17:17:18 +01:00
|
|
|
}
|
|
|
|
|
2018-03-28 13:22:06 +02:00
|
|
|
messageToWidget(action) {
|
2018-05-12 21:49:43 +02:00
|
|
|
action.widgetId = this.widgetId; // Required to be sent for all outbound requests
|
|
|
|
|
2018-03-28 13:22:06 +02:00
|
|
|
return this.toWidget.exec(action, this.target).then((data) => {
|
|
|
|
// Check for errors and reject if found
|
2017-12-15 16:24:22 +01:00
|
|
|
if (data.response === undefined) { // null is valid
|
|
|
|
throw new Error("Missing 'response' field");
|
|
|
|
}
|
|
|
|
if (data.response && data.response.error) {
|
|
|
|
const err = data.response.error;
|
|
|
|
const msg = String(err.message ? err.message : "An error was returned");
|
|
|
|
if (err._error) {
|
|
|
|
console.error(err._error);
|
|
|
|
}
|
|
|
|
// Potential XSS attack if 'msg' is not appropriately sanitized,
|
|
|
|
// as it is untrusted input by our parent window (which we assume is Riot).
|
|
|
|
// We can't aggressively sanitize [A-z0-9] since it might be a translation.
|
|
|
|
throw new Error(msg);
|
|
|
|
}
|
2018-03-28 13:22:06 +02:00
|
|
|
// Return the response field for the request
|
2017-12-15 16:24:22 +01:00
|
|
|
return data.response;
|
|
|
|
});
|
2017-12-01 17:17:18 +01:00
|
|
|
}
|
2017-12-15 16:24:22 +01:00
|
|
|
|
2017-12-15 17:39:04 +01:00
|
|
|
/**
|
|
|
|
* Request a screenshot from a widget
|
2018-02-25 23:21:30 +01:00
|
|
|
* @return {Promise} To be resolved with screenshot data when it has been generated
|
2017-12-15 17:39:04 +01:00
|
|
|
*/
|
|
|
|
getScreenshot() {
|
2018-03-28 13:22:06 +02:00
|
|
|
console.warn('Requesting screenshot for', this.widgetId);
|
|
|
|
return this.messageToWidget({
|
2018-02-23 16:11:28 +01:00
|
|
|
api: OUTBOUND_API_NAME,
|
2017-12-15 22:36:02 +01:00
|
|
|
action: "screenshot",
|
2018-03-28 13:22:06 +02:00
|
|
|
})
|
|
|
|
.catch((error) => new Error("Failed to get screenshot: " + error.message))
|
|
|
|
.then((response) => response.screenshot);
|
2017-12-15 17:39:04 +01:00
|
|
|
}
|
2017-12-16 10:16:24 +01:00
|
|
|
|
2018-02-25 23:21:30 +01:00
|
|
|
/**
|
|
|
|
* Request capabilities required by the widget
|
|
|
|
* @return {Promise} To be resolved with an array of requested widget capabilities
|
|
|
|
*/
|
2017-12-16 10:16:24 +01:00
|
|
|
getCapabilities() {
|
2018-03-28 13:22:06 +02:00
|
|
|
console.warn('Requesting capabilities for', this.widgetId);
|
|
|
|
return this.messageToWidget({
|
2018-02-23 16:11:28 +01:00
|
|
|
api: OUTBOUND_API_NAME,
|
2017-12-16 10:16:24 +01:00
|
|
|
action: "capabilities",
|
2018-03-28 13:22:06 +02:00
|
|
|
}).then((response) => {
|
|
|
|
console.warn('Got capabilities for', this.widgetId, response.capabilities);
|
|
|
|
return response.capabilities;
|
|
|
|
});
|
2017-12-16 10:16:24 +01:00
|
|
|
}
|
2017-12-01 15:56:27 +01:00
|
|
|
|
2018-05-11 17:22:54 +02:00
|
|
|
sendVisibility(visible) {
|
|
|
|
return this.messageToWidget({
|
|
|
|
api: OUTBOUND_API_NAME,
|
|
|
|
action: "visibility",
|
|
|
|
visible,
|
|
|
|
})
|
|
|
|
.catch((error) => {
|
|
|
|
console.error("Failed to send visibility: ", error);
|
|
|
|
});
|
|
|
|
}
|
2018-03-28 13:22:06 +02:00
|
|
|
|
|
|
|
start() {
|
|
|
|
this.fromWidget.addEndpoint(this.widgetId, this.widgetUrl);
|
|
|
|
}
|
|
|
|
|
|
|
|
stop() {
|
|
|
|
this.fromWidget.removeEndpoint(this.widgetId, this.widgetUrl);
|
2017-11-30 12:30:30 +01:00
|
|
|
}
|
|
|
|
}
|