2017-10-29 04:13:06 +03:00
|
|
|
/*
|
|
|
|
Copyright 2017 Travis Ralston
|
2020-07-29 01:31:13 +03:00
|
|
|
Copyright 2019, 2020 The Matrix.org Foundation C.I.C.
|
2017-10-29 04:13:06 +03: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.
|
|
|
|
*/
|
|
|
|
|
2019-12-21 00:13:46 +03:00
|
|
|
import {MatrixClientPeg} from '../../MatrixClientPeg';
|
2019-02-23 02:33:20 +03:00
|
|
|
import MatrixClientBackedSettingsHandler from "./MatrixClientBackedSettingsHandler";
|
2020-06-22 23:14:43 +03:00
|
|
|
import {objectClone, objectKeyChanges} from "../../utils/objects";
|
2020-07-28 20:53:43 +03:00
|
|
|
import {SettingLevel} from "../SettingLevel";
|
2020-07-29 01:31:13 +03:00
|
|
|
import { WatchManager } from "../WatchManager";
|
|
|
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
|
|
|
import { MatrixEvent } from "matrix-js-sdk/src/models/event";
|
2017-10-29 04:13:06 +03:00
|
|
|
|
2019-10-29 23:12:05 +03:00
|
|
|
const BREADCRUMBS_LEGACY_EVENT_TYPE = "im.vector.riot.breadcrumb_rooms";
|
|
|
|
const BREADCRUMBS_EVENT_TYPE = "im.vector.setting.breadcrumbs";
|
|
|
|
const BREADCRUMBS_EVENT_TYPES = [BREADCRUMBS_LEGACY_EVENT_TYPE, BREADCRUMBS_EVENT_TYPE];
|
2020-07-16 06:15:32 +03:00
|
|
|
const RECENT_EMOJI_EVENT_TYPE = "io.element.recent_emoji";
|
2019-10-29 23:20:53 +03:00
|
|
|
const INTEG_PROVISIONING_EVENT_TYPE = "im.vector.setting.integration_provisioning";
|
|
|
|
|
2017-10-29 04:13:06 +03:00
|
|
|
/**
|
|
|
|
* Gets and sets settings at the "account" level for the current user.
|
|
|
|
* This handler does not make use of the roomId parameter.
|
|
|
|
*/
|
2019-02-23 02:33:20 +03:00
|
|
|
export default class AccountSettingsHandler extends MatrixClientBackedSettingsHandler {
|
2020-07-29 01:31:13 +03:00
|
|
|
constructor(private watchers: WatchManager) {
|
2019-02-23 02:33:20 +03:00
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
public initMatrixClient(oldClient: MatrixClient, newClient: MatrixClient) {
|
2019-02-23 02:33:20 +03:00
|
|
|
if (oldClient) {
|
2020-07-29 01:31:13 +03:00
|
|
|
oldClient.removeListener("accountData", this.onAccountData);
|
2019-02-23 02:33:20 +03:00
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
newClient.on("accountData", this.onAccountData);
|
2019-02-23 02:33:20 +03:00
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
private onAccountData = (event: MatrixEvent, prevEvent: MatrixEvent) => {
|
2019-02-23 02:33:20 +03:00
|
|
|
if (event.getType() === "org.matrix.preview_urls") {
|
|
|
|
let val = event.getContent()['disable'];
|
|
|
|
if (typeof(val) !== "boolean") {
|
|
|
|
val = null;
|
|
|
|
} else {
|
|
|
|
val = !val;
|
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
this.watchers.notifyUpdate("urlPreviewsEnabled", null, SettingLevel.ACCOUNT, val);
|
2019-02-23 02:33:20 +03:00
|
|
|
} else if (event.getType() === "im.vector.web.settings") {
|
2020-06-22 19:18:38 +03:00
|
|
|
// Figure out what changed and fire those updates
|
|
|
|
const prevContent = prevEvent ? prevEvent.getContent() : {};
|
2020-08-05 11:13:01 +03:00
|
|
|
const changedSettings = objectKeyChanges<Record<string, any>>(prevContent, event.getContent());
|
2020-06-22 19:18:38 +03:00
|
|
|
for (const settingName of changedSettings) {
|
2019-02-26 22:43:10 +03:00
|
|
|
const val = event.getContent()[settingName];
|
2020-07-29 01:31:13 +03:00
|
|
|
this.watchers.notifyUpdate(settingName, null, SettingLevel.ACCOUNT, val);
|
2019-02-23 02:33:20 +03:00
|
|
|
}
|
2019-10-29 23:12:05 +03:00
|
|
|
} else if (BREADCRUMBS_EVENT_TYPES.includes(event.getType())) {
|
2020-07-29 01:31:13 +03:00
|
|
|
this.notifyBreadcrumbsUpdate(event);
|
2019-10-29 23:20:53 +03:00
|
|
|
} else if (event.getType() === INTEG_PROVISIONING_EVENT_TYPE) {
|
2019-10-29 23:23:59 +03:00
|
|
|
const val = event.getContent()['enabled'];
|
2020-07-29 01:31:13 +03:00
|
|
|
this.watchers.notifyUpdate("integrationProvisioning", null, SettingLevel.ACCOUNT, val);
|
2020-07-16 06:15:32 +03:00
|
|
|
} else if (event.getType() === RECENT_EMOJI_EVENT_TYPE) {
|
|
|
|
const val = event.getContent()['enabled'];
|
2020-07-29 01:31:13 +03:00
|
|
|
this.watchers.notifyUpdate("recent_emoji", null, SettingLevel.ACCOUNT, val);
|
2019-02-23 02:33:20 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
public getValue(settingName: string, roomId: string): any {
|
2017-10-30 06:48:29 +03:00
|
|
|
// Special case URL previews
|
|
|
|
if (settingName === "urlPreviewsEnabled") {
|
2020-07-29 01:31:13 +03:00
|
|
|
const content = this.getSettings("org.matrix.preview_urls") || {};
|
2017-11-16 05:22:30 +03:00
|
|
|
|
|
|
|
// Check to make sure that we actually got a boolean
|
|
|
|
if (typeof(content['disable']) !== "boolean") return null;
|
2017-10-30 06:48:29 +03:00
|
|
|
return !content['disable'];
|
|
|
|
}
|
|
|
|
|
2019-04-05 00:06:03 +03:00
|
|
|
// Special case for breadcrumbs
|
|
|
|
if (settingName === "breadcrumb_rooms") {
|
2020-07-29 01:31:13 +03:00
|
|
|
let content = this.getSettings(BREADCRUMBS_EVENT_TYPE);
|
2019-10-29 23:12:05 +03:00
|
|
|
if (!content || !content['recent_rooms']) {
|
2020-07-29 01:31:13 +03:00
|
|
|
content = this.getSettings(BREADCRUMBS_LEGACY_EVENT_TYPE);
|
2019-10-29 23:12:05 +03:00
|
|
|
|
|
|
|
// This is a bit of a hack, but it makes things slightly easier
|
|
|
|
if (content) content['recent_rooms'] = content['rooms'];
|
|
|
|
}
|
|
|
|
|
|
|
|
return content && content['recent_rooms'] ? content['recent_rooms'] : [];
|
2019-04-05 00:06:03 +03:00
|
|
|
}
|
|
|
|
|
2020-07-16 06:15:32 +03:00
|
|
|
// Special case recent emoji
|
|
|
|
if (settingName === "recent_emoji") {
|
2020-07-29 01:31:13 +03:00
|
|
|
const content = this.getSettings(RECENT_EMOJI_EVENT_TYPE);
|
2020-07-16 06:15:32 +03:00
|
|
|
return content ? content["recent_emoji"] : null;
|
|
|
|
}
|
|
|
|
|
2019-10-29 23:20:53 +03:00
|
|
|
// Special case integration manager provisioning
|
2019-10-29 23:26:35 +03:00
|
|
|
if (settingName === "integrationProvisioning") {
|
2020-07-29 01:31:13 +03:00
|
|
|
const content = this.getSettings(INTEG_PROVISIONING_EVENT_TYPE);
|
2019-10-29 23:20:53 +03:00
|
|
|
return content ? content['enabled'] : null;
|
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
const settings = this.getSettings() || {};
|
2018-01-17 21:17:26 +03:00
|
|
|
let preferredValue = settings[settingName];
|
2017-11-13 22:58:10 +03:00
|
|
|
|
|
|
|
if (preferredValue === null || preferredValue === undefined) {
|
|
|
|
// Honour the old setting on read only
|
|
|
|
if (settingName === "hideAvatarChanges" || settingName === "hideDisplaynameChanges") {
|
2018-01-17 21:17:26 +03:00
|
|
|
preferredValue = settings["hideAvatarDisplaynameChanges"];
|
2017-11-13 22:58:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return preferredValue;
|
2017-10-29 04:13:06 +03:00
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
public setValue(settingName: string, roomId: string, newValue: any): Promise<void> {
|
2017-10-30 06:48:29 +03:00
|
|
|
// Special case URL previews
|
|
|
|
if (settingName === "urlPreviewsEnabled") {
|
2020-07-29 01:31:13 +03:00
|
|
|
const content = this.getSettings("org.matrix.preview_urls") || {};
|
2017-10-30 06:48:29 +03:00
|
|
|
content['disable'] = !newValue;
|
|
|
|
return MatrixClientPeg.get().setAccountData("org.matrix.preview_urls", content);
|
|
|
|
}
|
|
|
|
|
2019-04-05 00:06:03 +03:00
|
|
|
// Special case for breadcrumbs
|
|
|
|
if (settingName === "breadcrumb_rooms") {
|
2019-10-29 23:12:05 +03:00
|
|
|
// We read the value first just to make sure we preserve whatever random keys might be present.
|
2020-07-29 01:31:13 +03:00
|
|
|
let content = this.getSettings(BREADCRUMBS_EVENT_TYPE);
|
2019-10-29 23:12:05 +03:00
|
|
|
if (!content || !content['recent_rooms']) {
|
2020-07-29 01:31:13 +03:00
|
|
|
content = this.getSettings(BREADCRUMBS_LEGACY_EVENT_TYPE);
|
2019-10-29 23:12:05 +03:00
|
|
|
}
|
2019-11-19 00:17:31 +03:00
|
|
|
if (!content) content = {}; // If we still don't have content, make some
|
2019-10-29 23:12:05 +03:00
|
|
|
|
|
|
|
content['recent_rooms'] = newValue;
|
2019-04-05 00:06:03 +03:00
|
|
|
return MatrixClientPeg.get().setAccountData(BREADCRUMBS_EVENT_TYPE, content);
|
|
|
|
}
|
|
|
|
|
2020-07-16 06:15:32 +03:00
|
|
|
// Special case recent emoji
|
|
|
|
if (settingName === "recent_emoji") {
|
2020-07-29 01:31:13 +03:00
|
|
|
const content = this.getSettings(RECENT_EMOJI_EVENT_TYPE) || {};
|
2020-07-16 06:15:32 +03:00
|
|
|
content["recent_emoji"] = newValue;
|
|
|
|
return MatrixClientPeg.get().setAccountData(RECENT_EMOJI_EVENT_TYPE, content);
|
|
|
|
}
|
|
|
|
|
2019-10-29 23:20:53 +03:00
|
|
|
// Special case integration manager provisioning
|
2019-10-29 23:26:35 +03:00
|
|
|
if (settingName === "integrationProvisioning") {
|
2020-07-29 01:31:13 +03:00
|
|
|
const content = this.getSettings(INTEG_PROVISIONING_EVENT_TYPE) || {};
|
2019-10-29 23:20:53 +03:00
|
|
|
content['enabled'] = newValue;
|
|
|
|
return MatrixClientPeg.get().setAccountData(INTEG_PROVISIONING_EVENT_TYPE, content);
|
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
const content = this.getSettings() || {};
|
2017-10-29 04:45:48 +03:00
|
|
|
content[settingName] = newValue;
|
|
|
|
return MatrixClientPeg.get().setAccountData("im.vector.web.settings", content);
|
2017-10-29 04:13:06 +03:00
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
public canSetValue(settingName: string, roomId: string): boolean {
|
2017-10-29 04:13:06 +03:00
|
|
|
return true; // It's their account, so they should be able to
|
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
public isSupported(): boolean {
|
2017-11-04 08:19:45 +03:00
|
|
|
const cli = MatrixClientPeg.get();
|
2021-01-14 18:58:26 +03:00
|
|
|
return cli !== undefined && cli !== null && !cli.isGuest();
|
2017-10-29 04:13:06 +03:00
|
|
|
}
|
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
private getSettings(eventType = "im.vector.web.settings"): any { // TODO: [TS] Types on return
|
2017-11-15 14:15:49 +03:00
|
|
|
const cli = MatrixClientPeg.get();
|
2018-01-17 21:17:26 +03:00
|
|
|
if (!cli) return null;
|
|
|
|
|
2017-11-15 14:15:49 +03:00
|
|
|
const event = cli.getAccountData(eventType);
|
2018-01-17 21:17:26 +03:00
|
|
|
if (!event || !event.getContent()) return null;
|
2020-06-22 23:14:43 +03:00
|
|
|
return objectClone(event.getContent()); // clone to prevent mutation
|
2017-10-29 04:13:06 +03:00
|
|
|
}
|
2019-10-29 23:12:05 +03:00
|
|
|
|
2020-07-29 01:31:13 +03:00
|
|
|
private notifyBreadcrumbsUpdate(event: MatrixEvent) {
|
2019-10-29 23:12:05 +03:00
|
|
|
let val = [];
|
|
|
|
if (event.getType() === BREADCRUMBS_LEGACY_EVENT_TYPE) {
|
|
|
|
// This seems fishy - try and get the event for the new rooms
|
2020-07-29 01:31:13 +03:00
|
|
|
const newType = this.getSettings(BREADCRUMBS_EVENT_TYPE);
|
2019-10-29 23:12:05 +03:00
|
|
|
if (newType) val = newType['recent_rooms'];
|
2019-11-19 00:25:04 +03:00
|
|
|
else val = event.getContent()['rooms'];
|
2019-10-29 23:12:05 +03:00
|
|
|
} else if (event.getType() === BREADCRUMBS_EVENT_TYPE) {
|
|
|
|
val = event.getContent()['recent_rooms'];
|
|
|
|
} else {
|
|
|
|
return; // for sanity, not because we expect to be here.
|
|
|
|
}
|
2020-07-29 01:31:13 +03:00
|
|
|
this.watchers.notifyUpdate("breadcrumb_rooms", null, SettingLevel.ACCOUNT, val || []);
|
2019-10-29 23:12:05 +03:00
|
|
|
}
|
2017-10-29 04:53:12 +03:00
|
|
|
}
|