2016-11-02 14:49:28 -04:00
|
|
|
// @flow
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copyright 2016 Aviral Dasgupta
|
|
|
|
Copyright 2016 OpenMarket Ltd
|
|
|
|
|
|
|
|
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-06-11 14:19:17 -04:00
|
|
|
import VectorBasePlatform, {updateCheckStatusEnum} from './VectorBasePlatform';
|
2016-11-02 14:49:28 -04:00
|
|
|
import dis from 'matrix-react-sdk/lib/dispatcher';
|
2017-05-31 12:53:32 -04:00
|
|
|
import { _t } from 'matrix-react-sdk/lib/languageHandler';
|
2017-07-12 19:48:31 -04:00
|
|
|
import Promise from 'bluebird';
|
2017-09-25 11:19:04 -04:00
|
|
|
import {remote, ipcRenderer, desktopCapturer} from 'electron';
|
2018-04-12 20:35:55 -04:00
|
|
|
import rageshake from 'matrix-react-sdk/lib/rageshake/rageshake';
|
2016-11-03 11:45:12 -04:00
|
|
|
|
2017-04-23 04:59:00 -04:00
|
|
|
remote.autoUpdater.on('update-downloaded', onUpdateDownloaded);
|
2016-11-03 11:45:12 -04:00
|
|
|
|
2017-06-22 13:25:19 -04:00
|
|
|
// try to flush the rageshake logs to indexeddb before quit.
|
|
|
|
ipcRenderer.on('before-quit', function () {
|
|
|
|
console.log('riot-desktop closing');
|
|
|
|
rageshake.flush();
|
|
|
|
});
|
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
function onUpdateDownloaded(ev: Event, releaseNotes: string, ver: string, date: Date, updateURL: string) {
|
2016-11-02 14:49:28 -04:00
|
|
|
dis.dispatch({
|
|
|
|
action: 'new_version',
|
2017-04-23 04:59:00 -04:00
|
|
|
currentVersion: remote.app.getVersion(),
|
2016-11-02 14:49:28 -04:00
|
|
|
newVersion: ver,
|
|
|
|
releaseNotes: releaseNotes,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
function platformFriendlyName(): string {
|
2016-11-24 11:46:15 -05:00
|
|
|
console.log(window.process);
|
|
|
|
switch (window.process.platform) {
|
|
|
|
case 'darwin':
|
|
|
|
return 'macOS';
|
|
|
|
case 'freebsd':
|
|
|
|
return 'FreeBSD';
|
|
|
|
case 'openbsd':
|
|
|
|
return 'OpenBSD';
|
|
|
|
case 'sunos':
|
|
|
|
return 'SunOS';
|
|
|
|
case 'win32':
|
|
|
|
return 'Windows';
|
|
|
|
default:
|
|
|
|
// Sorry, Linux users: you get lumped into here,
|
|
|
|
// but only because Linux's capitalisation is
|
|
|
|
// normal. We do care about you.
|
2016-12-06 17:59:17 -05:00
|
|
|
return window.process.platform[0].toUpperCase() + window.process.platform.slice(1);
|
2016-11-24 11:46:15 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-22 13:30:45 -04:00
|
|
|
function _onAction(payload: Object) {
|
|
|
|
// Whitelist payload actions, no point sending most across
|
|
|
|
if (['call_state'].includes(payload.action)) {
|
|
|
|
ipcRenderer.send('app_onAction', payload);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-11 14:19:17 -04:00
|
|
|
function getUpdateCheckStatus(status) {
|
|
|
|
if (status === true) {
|
|
|
|
return { status: updateCheckStatusEnum.DOWNLOADING };
|
|
|
|
} else if (status === false) {
|
|
|
|
return { status: updateCheckStatusEnum.NOTAVAILABLE };
|
|
|
|
} else {
|
|
|
|
return {
|
|
|
|
status: updateCheckStatusEnum.ERROR,
|
|
|
|
detail: status,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-02 14:49:28 -04:00
|
|
|
export default class ElectronPlatform extends VectorBasePlatform {
|
2017-05-22 13:30:45 -04:00
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
dis.register(_onAction);
|
2017-06-03 10:12:46 -04:00
|
|
|
this.updatable = Boolean(remote.autoUpdater.getFeedURL());
|
2017-06-11 14:19:17 -04:00
|
|
|
|
|
|
|
/*
|
|
|
|
IPC Call `check_updates` returns:
|
|
|
|
true if there is an update available
|
|
|
|
false if there is not
|
|
|
|
or the error if one is encountered
|
|
|
|
*/
|
|
|
|
ipcRenderer.on('check_updates', (event, status) => {
|
|
|
|
if (!this.showUpdateCheck) return;
|
|
|
|
dis.dispatch({
|
|
|
|
action: 'check_updates',
|
|
|
|
value: getUpdateCheckStatus(status),
|
|
|
|
});
|
|
|
|
this.showUpdateCheck = false;
|
|
|
|
});
|
|
|
|
|
|
|
|
this.startUpdateCheck = this.startUpdateCheck.bind(this);
|
|
|
|
this.stopUpdateCheck = this.stopUpdateCheck.bind(this);
|
2017-05-22 13:30:45 -04:00
|
|
|
}
|
|
|
|
|
2017-05-29 15:03:21 -04:00
|
|
|
getHumanReadableName(): string {
|
2017-05-31 09:51:08 -04:00
|
|
|
return 'Electron Platform'; // no translation required: only used for analytics
|
2017-05-29 14:51:28 -04:00
|
|
|
}
|
|
|
|
|
2016-11-02 14:49:28 -04:00
|
|
|
setNotificationCount(count: number) {
|
2017-01-26 07:58:29 -05:00
|
|
|
if (this.notificationCount === count) return;
|
2016-11-02 14:49:28 -04:00
|
|
|
super.setNotificationCount(count);
|
2017-05-17 05:39:43 -04:00
|
|
|
|
|
|
|
ipcRenderer.send('setBadgeCount', count);
|
2016-11-02 14:49:28 -04:00
|
|
|
}
|
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
supportsNotifications(): boolean {
|
2016-11-02 14:49:28 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
maySendNotifications(): boolean {
|
2016-11-02 14:49:28 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-12-06 08:28:59 -05:00
|
|
|
displayNotification(title: string, msg: string, avatarUrl: string, room: Object): Notification {
|
2017-04-03 15:30:05 -04:00
|
|
|
// GNOME notification spec parses HTML tags for styling...
|
|
|
|
// Electron Docs state all supported linux notification systems follow this markup spec
|
|
|
|
// https://github.com/electron/electron/blob/master/docs/tutorial/desktop-environment-integration.md#linux
|
|
|
|
// maybe we should pass basic styling (italics, bold, underline) through from MD
|
2017-04-22 11:05:08 -04:00
|
|
|
// we only have to strip out < and > as the spec doesn't include anything about things like &
|
|
|
|
// so we shouldn't assume that all implementations will treat those properly. Very basic tag parsing is done.
|
2017-04-03 15:30:05 -04:00
|
|
|
if (window.process.platform === 'linux') {
|
2017-04-23 04:59:00 -04:00
|
|
|
msg = msg.replace(/</g, '<').replace(/>/g, '>');
|
2017-04-03 15:30:05 -04:00
|
|
|
}
|
|
|
|
|
2016-11-02 14:49:28 -04:00
|
|
|
// Notifications in Electron use the HTML5 notification API
|
|
|
|
const notification = new global.Notification(
|
|
|
|
title,
|
|
|
|
{
|
|
|
|
body: msg,
|
|
|
|
icon: avatarUrl,
|
2017-04-23 04:59:00 -04:00
|
|
|
tag: 'vector',
|
2016-11-02 14:49:28 -04:00
|
|
|
silent: true, // we play our own sounds
|
2017-05-02 16:29:19 -04:00
|
|
|
},
|
2016-11-02 14:49:28 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
notification.onclick = function() {
|
|
|
|
dis.dispatch({
|
|
|
|
action: 'view_room',
|
2017-04-23 04:56:43 -04:00
|
|
|
room_id: room.roomId,
|
2016-11-02 14:49:28 -04:00
|
|
|
});
|
|
|
|
global.focus();
|
2017-04-23 04:59:00 -04:00
|
|
|
const win = remote.getCurrentWindow();
|
2017-04-23 04:56:43 -04:00
|
|
|
|
|
|
|
if (win.isMinimized()) win.restore();
|
|
|
|
else if (!win.isVisible()) win.show();
|
|
|
|
else win.focus();
|
2016-11-02 14:49:28 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
return notification;
|
|
|
|
}
|
|
|
|
|
2017-05-31 19:00:00 -04:00
|
|
|
loudNotification(ev: Event, room: Object) {
|
|
|
|
ipcRenderer.send('loudNotification');
|
|
|
|
}
|
|
|
|
|
2016-11-02 14:49:28 -04:00
|
|
|
clearNotification(notif: Notification) {
|
2018-10-18 13:31:06 -04:00
|
|
|
// This crashes on windows under certain circumstances: can't find any
|
2018-10-18 13:35:01 -04:00
|
|
|
// workaround other than not closing notifs.
|
2018-10-18 13:31:06 -04:00
|
|
|
// https://github.com/electron/electron/issues/15251
|
|
|
|
// https://github.com/vector-im/riot-web/issues/7512
|
|
|
|
if (window.process.platform === 'win32') return;
|
2016-11-02 14:49:28 -04:00
|
|
|
notif.close();
|
|
|
|
}
|
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
getAppVersion(): Promise<string> {
|
2017-07-12 19:32:07 -04:00
|
|
|
return Promise.resolve(remote.app.getVersion());
|
2016-11-08 05:47:01 -05:00
|
|
|
}
|
|
|
|
|
2017-06-11 14:19:17 -04:00
|
|
|
startUpdateCheck() {
|
|
|
|
if (this.showUpdateCheck) return;
|
|
|
|
super.startUpdateCheck();
|
2017-06-03 10:12:46 -04:00
|
|
|
|
2017-06-11 14:19:17 -04:00
|
|
|
ipcRenderer.send('check_updates');
|
2016-11-02 14:49:28 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
installUpdate() {
|
|
|
|
// IPC to the main process to install the update, since quitAndInstall
|
|
|
|
// doesn't fire the before-quit event so the main process needs to know
|
|
|
|
// it should exit.
|
2017-06-11 14:19:17 -04:00
|
|
|
ipcRenderer.send('install_update');
|
2016-11-02 14:49:28 -04:00
|
|
|
}
|
2016-11-24 11:46:15 -05:00
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
getDefaultDeviceDisplayName(): string {
|
2017-05-31 09:51:08 -04:00
|
|
|
return _t('Riot Desktop on %(platformName)s', { platformName: platformFriendlyName() });
|
2016-11-24 11:46:15 -05:00
|
|
|
}
|
2017-01-10 13:39:21 -05:00
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
screenCaptureErrorString(): ?string {
|
2017-01-10 13:39:21 -05:00
|
|
|
return null;
|
|
|
|
}
|
2017-01-19 08:25:56 -05:00
|
|
|
|
2017-05-24 10:52:08 -04:00
|
|
|
isElectron(): boolean { return true; }
|
|
|
|
|
2017-05-02 16:29:19 -04:00
|
|
|
requestNotificationPermission(): Promise<string> {
|
2017-07-12 19:32:07 -04:00
|
|
|
return Promise.resolve('granted');
|
2017-01-19 08:25:56 -05:00
|
|
|
}
|
2017-04-10 12:40:09 -04:00
|
|
|
|
|
|
|
reload() {
|
2017-04-23 04:59:00 -04:00
|
|
|
remote.getCurrentWebContents().reload();
|
2017-04-10 12:40:09 -04:00
|
|
|
}
|
2017-09-04 03:33:08 -04:00
|
|
|
|
|
|
|
/* BEGIN copied and slightly-modified code
|
|
|
|
* setupScreenSharingForIframe function from:
|
|
|
|
* https://github.com/jitsi/jitsi-meet-electron-utils
|
|
|
|
* Copied directly here to avoid the need for a native electron module for
|
|
|
|
* 'just a bit of JavaScript'
|
|
|
|
* NOTE: Apache v2.0 licensed
|
|
|
|
*/
|
|
|
|
setupScreenSharingForIframe(iframe: Object) {
|
|
|
|
iframe.contentWindow.JitsiMeetElectron = {
|
|
|
|
/**
|
|
|
|
* Get sources available for screensharing. The callback is invoked
|
|
|
|
* with an array of DesktopCapturerSources.
|
|
|
|
*
|
|
|
|
* @param {Function} callback - The success callback.
|
|
|
|
* @param {Function} errorCallback - The callback for errors.
|
|
|
|
* @param {Object} options - Configuration for getting sources.
|
|
|
|
* @param {Array} options.types - Specify the desktop source types
|
|
|
|
* to get, with valid sources being "window" and "screen".
|
|
|
|
* @param {Object} options.thumbnailSize - Specify how big the
|
|
|
|
* preview images for the sources should be. The valid keys are
|
|
|
|
* height and width, e.g. { height: number, width: number}. By
|
|
|
|
* default electron will return images with height and width of
|
|
|
|
* 150px.
|
|
|
|
*/
|
|
|
|
obtainDesktopStreams(callback, errorCallback, options = {}) {
|
2017-09-25 11:19:04 -04:00
|
|
|
desktopCapturer.getSources(options,
|
2017-09-04 03:33:08 -04:00
|
|
|
(error, sources) => {
|
|
|
|
if (error) {
|
|
|
|
errorCallback(error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
callback(sources);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
/* END of copied and slightly-modified code */
|
2016-11-02 14:49:28 -04:00
|
|
|
}
|