mirror of
https://github.com/vector-im/element-web.git
synced 2024-11-16 13:14:58 +08:00
Merge pull request #2928 from Half-Shot/hs/custom-notif-sounds
Custom notification sounds for rooms
This commit is contained in:
commit
635b1ff612
@ -71,3 +71,26 @@ limitations under the License.
|
|||||||
.mx_UserNotifSettings_notifTable .mx_Spinner {
|
.mx_UserNotifSettings_notifTable .mx_Spinner {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.mx_NotificationSound_soundUpload {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx_NotificationSound_browse {
|
||||||
|
color: $accent-color;
|
||||||
|
border: 1px solid $accent-color;
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx_NotificationSound_save {
|
||||||
|
margin-left: 5px;
|
||||||
|
color: white;
|
||||||
|
background-color: $accent-color;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx_NotificationSound_resetSound {
|
||||||
|
margin-top: 5px;
|
||||||
|
color: white;
|
||||||
|
border: $warning-color;
|
||||||
|
background-color: $warning-color;
|
||||||
|
}
|
@ -100,10 +100,55 @@ const Notifier = {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
_playAudioNotification: function(ev, room) {
|
getSoundForRoom: async function(roomId) {
|
||||||
const e = document.getElementById("messageAudio");
|
// We do no caching here because the SDK caches setting
|
||||||
if (e) {
|
// and the browser will cache the sound.
|
||||||
e.play();
|
const content = SettingsStore.getValue("notificationSound", roomId);
|
||||||
|
if (!content) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!content.url) {
|
||||||
|
console.warn(`${roomId} has custom notification sound event, but no url key`);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!content.url.startsWith("mxc://")) {
|
||||||
|
console.warn(`${roomId} has custom notification sound event, but url is not a mxc url`);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ideally in here we could use MSC1310 to detect the type of file, and reject it.
|
||||||
|
|
||||||
|
return {
|
||||||
|
url: MatrixClientPeg.get().mxcUrlToHttp(content.url),
|
||||||
|
name: content.name,
|
||||||
|
type: content.type,
|
||||||
|
size: content.size,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
_playAudioNotification: async function(ev, room) {
|
||||||
|
const sound = await this.getSoundForRoom(room.roomId);
|
||||||
|
console.log(`Got sound ${sound && sound.name || "default"} for ${room.roomId}`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const selector = document.querySelector(sound ? `audio[src='${sound.url}']` : "#messageAudio");
|
||||||
|
let audioElement = selector;
|
||||||
|
if (!selector) {
|
||||||
|
if (!sound) {
|
||||||
|
console.error("No audio element or sound to play for notification");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
audioElement = new Audio(sound.url);
|
||||||
|
if (sound.type) {
|
||||||
|
audioElement.type = sound.type;
|
||||||
|
}
|
||||||
|
document.body.appendChild(audioElement);
|
||||||
|
}
|
||||||
|
audioElement.play();
|
||||||
|
} catch (ex) {
|
||||||
|
console.warn("Caught error when trying to fetch room notification sound:", ex);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -23,8 +23,10 @@ import AdvancedRoomSettingsTab from "../settings/tabs/room/AdvancedRoomSettingsT
|
|||||||
import RolesRoomSettingsTab from "../settings/tabs/room/RolesRoomSettingsTab";
|
import RolesRoomSettingsTab from "../settings/tabs/room/RolesRoomSettingsTab";
|
||||||
import GeneralRoomSettingsTab from "../settings/tabs/room/GeneralRoomSettingsTab";
|
import GeneralRoomSettingsTab from "../settings/tabs/room/GeneralRoomSettingsTab";
|
||||||
import SecurityRoomSettingsTab from "../settings/tabs/room/SecurityRoomSettingsTab";
|
import SecurityRoomSettingsTab from "../settings/tabs/room/SecurityRoomSettingsTab";
|
||||||
|
import NotificationSettingsTab from "../settings/tabs/room/NotificationSettingsTab";
|
||||||
import sdk from "../../../index";
|
import sdk from "../../../index";
|
||||||
import MatrixClientPeg from "../../../MatrixClientPeg";
|
import MatrixClientPeg from "../../../MatrixClientPeg";
|
||||||
|
import SettingsStore from '../../../settings/SettingsStore';
|
||||||
import dis from "../../../dispatcher";
|
import dis from "../../../dispatcher";
|
||||||
|
|
||||||
export default class RoomSettingsDialog extends React.Component {
|
export default class RoomSettingsDialog extends React.Component {
|
||||||
@ -67,6 +69,11 @@ export default class RoomSettingsDialog extends React.Component {
|
|||||||
"mx_RoomSettingsDialog_rolesIcon",
|
"mx_RoomSettingsDialog_rolesIcon",
|
||||||
<RolesRoomSettingsTab roomId={this.props.roomId} />,
|
<RolesRoomSettingsTab roomId={this.props.roomId} />,
|
||||||
));
|
));
|
||||||
|
tabs.push(new Tab(
|
||||||
|
_td("Notifications"),
|
||||||
|
"mx_RoomSettingsDialog_rolesIcon",
|
||||||
|
<NotificationSettingsTab roomId={this.props.roomId} />,
|
||||||
|
));
|
||||||
tabs.push(new Tab(
|
tabs.push(new Tab(
|
||||||
_td("Advanced"),
|
_td("Advanced"),
|
||||||
"mx_RoomSettingsDialog_warningIcon",
|
"mx_RoomSettingsDialog_warningIcon",
|
||||||
|
@ -0,0 +1,178 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2019 New Vector 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import {_t} from "../../../../../languageHandler";
|
||||||
|
import MatrixClientPeg from "../../../../../MatrixClientPeg";
|
||||||
|
import AccessibleButton from "../../../elements/AccessibleButton";
|
||||||
|
import Notifier from "../../../../../Notifier";
|
||||||
|
import SettingsStore from '../../../../../settings/SettingsStore';
|
||||||
|
import { SettingLevel } from '../../../../../settings/SettingsStore';
|
||||||
|
|
||||||
|
export default class NotificationsSettingsTab extends React.Component {
|
||||||
|
static propTypes = {
|
||||||
|
roomId: PropTypes.string.isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
currentSound: "default",
|
||||||
|
uploadedFile: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillMount() {
|
||||||
|
Notifier.getSoundForRoom(this.props.roomId).then((soundData) => {
|
||||||
|
if (!soundData) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.setState({currentSound: soundData.name || soundData.url});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async _triggerUploader(e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
this.refs.soundUpload.click();
|
||||||
|
}
|
||||||
|
|
||||||
|
async _onSoundUploadChanged(e) {
|
||||||
|
if (!e.target.files || !e.target.files.length) {
|
||||||
|
this.setState({
|
||||||
|
uploadedFile: null,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const file = e.target.files[0];
|
||||||
|
this.setState({
|
||||||
|
uploadedFile: file,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async _onClickSaveSound(e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this._saveSound();
|
||||||
|
} catch (ex) {
|
||||||
|
console.error(
|
||||||
|
`Unable to save notification sound for ${this.props.roomId}`,
|
||||||
|
);
|
||||||
|
console.error(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async _saveSound() {
|
||||||
|
if (!this.state.uploadedFile) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let type = this.state.uploadedFile.type;
|
||||||
|
if (type === "video/ogg") {
|
||||||
|
// XXX: I've observed browsers allowing users to pick a audio/ogg files,
|
||||||
|
// and then calling it a video/ogg. This is a lame hack, but man browsers
|
||||||
|
// suck at detecting mimetypes.
|
||||||
|
type = "audio/ogg";
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = await MatrixClientPeg.get().uploadContent(
|
||||||
|
this.state.uploadedFile, {
|
||||||
|
type,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
await SettingsStore.setValue(
|
||||||
|
"notificationSound",
|
||||||
|
this.props.roomId,
|
||||||
|
SettingLevel.ROOM_ACCOUNT,
|
||||||
|
{
|
||||||
|
name: this.state.uploadedFile.name,
|
||||||
|
type: type,
|
||||||
|
size: this.state.uploadedFile.size,
|
||||||
|
url,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
uploadedFile: null,
|
||||||
|
currentSound: this.state.uploadedFile.name,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_clearSound(e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
SettingsStore.setValue(
|
||||||
|
"notificationSound",
|
||||||
|
this.props.roomId,
|
||||||
|
SettingLevel.ROOM_ACCOUNT,
|
||||||
|
null,
|
||||||
|
);
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
currentSound: "default",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
let currentUploadedFile = null;
|
||||||
|
if (this.state.uploadedFile) {
|
||||||
|
currentUploadedFile = (
|
||||||
|
<div>
|
||||||
|
<span>{_t("Uploaded sound")}: <code>{this.state.uploadedFile.name}</code></span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="mx_SettingsTab">
|
||||||
|
<div className="mx_SettingsTab_heading">{_t("Notifications")}</div>
|
||||||
|
<div className='mx_SettingsTab_section mx_SettingsTab_subsectionText'>
|
||||||
|
<span className='mx_SettingsTab_subheading'>{_t("Sounds")}</span>
|
||||||
|
<div>
|
||||||
|
<span>{_t("Notification sound")}: <code>{this.state.currentSound}</code></span><br />
|
||||||
|
<AccessibleButton className="mx_NotificationSound_resetSound" disabled={this.state.currentSound == "default"} onClick={this._clearSound.bind(this)} kind="primary">
|
||||||
|
{_t("Reset")}
|
||||||
|
</AccessibleButton>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h3>{_t("Set a new custom sound")}</h3>
|
||||||
|
<form autoComplete={false} noValidate={true}>
|
||||||
|
<input ref="soundUpload" className="mx_NotificationSound_soundUpload" type="file" onChange={this._onSoundUploadChanged.bind(this)} accept="audio/*" />
|
||||||
|
</form>
|
||||||
|
|
||||||
|
{currentUploadedFile}
|
||||||
|
|
||||||
|
<AccessibleButton className="mx_NotificationSound_browse" onClick={this._triggerUploader.bind(this)} kind="primary">
|
||||||
|
{_t("Browse")}
|
||||||
|
</AccessibleButton>
|
||||||
|
|
||||||
|
<AccessibleButton className="mx_NotificationSound_save" disabled={this.state.uploadedFile == null} onClick={this._onClickSaveSound.bind(this)} kind="primary">
|
||||||
|
{_t("Save")}
|
||||||
|
</AccessibleButton>
|
||||||
|
<br />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -304,6 +304,7 @@
|
|||||||
"Custom user status messages": "Custom user status messages",
|
"Custom user status messages": "Custom user status messages",
|
||||||
"Group & filter rooms by custom tags (refresh to apply changes)": "Group & filter rooms by custom tags (refresh to apply changes)",
|
"Group & filter rooms by custom tags (refresh to apply changes)": "Group & filter rooms by custom tags (refresh to apply changes)",
|
||||||
"Render simple counters in room header": "Render simple counters in room header",
|
"Render simple counters in room header": "Render simple counters in room header",
|
||||||
|
"Custom Notification Sounds": "Custom Notification Sounds",
|
||||||
"Edit messages after they have been sent (refresh to apply changes)": "Edit messages after they have been sent (refresh to apply changes)",
|
"Edit messages after they have been sent (refresh to apply changes)": "Edit messages after they have been sent (refresh to apply changes)",
|
||||||
"React to messages with emoji (refresh to apply changes)": "React to messages with emoji (refresh to apply changes)",
|
"React to messages with emoji (refresh to apply changes)": "React to messages with emoji (refresh to apply changes)",
|
||||||
"Enable Emoji suggestions while typing": "Enable Emoji suggestions while typing",
|
"Enable Emoji suggestions while typing": "Enable Emoji suggestions while typing",
|
||||||
@ -624,6 +625,12 @@
|
|||||||
"Room Addresses": "Room Addresses",
|
"Room Addresses": "Room Addresses",
|
||||||
"Publish this room to the public in %(domain)s's room directory?": "Publish this room to the public in %(domain)s's room directory?",
|
"Publish this room to the public in %(domain)s's room directory?": "Publish this room to the public in %(domain)s's room directory?",
|
||||||
"URL Previews": "URL Previews",
|
"URL Previews": "URL Previews",
|
||||||
|
"Uploaded sound": "Uploaded sound",
|
||||||
|
"Sounds": "Sounds",
|
||||||
|
"Notification sound": "Notification sound",
|
||||||
|
"Reset": "Reset",
|
||||||
|
"Set a new custom sound": "Set a new custom sound",
|
||||||
|
"Browse": "Browse",
|
||||||
"Change room avatar": "Change room avatar",
|
"Change room avatar": "Change room avatar",
|
||||||
"Change room name": "Change room name",
|
"Change room name": "Change room name",
|
||||||
"Change main address for the room": "Change main address for the room",
|
"Change main address for the room": "Change main address for the room",
|
||||||
|
@ -27,6 +27,7 @@ import LowBandwidthController from "./controllers/LowBandwidthController";
|
|||||||
|
|
||||||
// These are just a bunch of helper arrays to avoid copy/pasting a bunch of times
|
// These are just a bunch of helper arrays to avoid copy/pasting a bunch of times
|
||||||
const LEVELS_ROOM_SETTINGS = ['device', 'room-device', 'room-account', 'account', 'config'];
|
const LEVELS_ROOM_SETTINGS = ['device', 'room-device', 'room-account', 'account', 'config'];
|
||||||
|
const LEVELS_ROOM_OR_ACCOUNT = ['room-account', 'account'];
|
||||||
const LEVELS_ROOM_SETTINGS_WITH_ROOM = ['device', 'room-device', 'room-account', 'account', 'config', 'room'];
|
const LEVELS_ROOM_SETTINGS_WITH_ROOM = ['device', 'room-device', 'room-account', 'account', 'config', 'room'];
|
||||||
const LEVELS_ACCOUNT_SETTINGS = ['device', 'account', 'config'];
|
const LEVELS_ACCOUNT_SETTINGS = ['device', 'account', 'config'];
|
||||||
const LEVELS_FEATURE = ['device', 'config'];
|
const LEVELS_FEATURE = ['device', 'config'];
|
||||||
@ -322,6 +323,10 @@ export const SETTINGS = {
|
|||||||
default: false,
|
default: false,
|
||||||
controller: new NotificationsEnabledController(),
|
controller: new NotificationsEnabledController(),
|
||||||
},
|
},
|
||||||
|
"notificationSound": {
|
||||||
|
supportedLevels: LEVELS_ROOM_OR_ACCOUNT,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
"notificationBodyEnabled": {
|
"notificationBodyEnabled": {
|
||||||
supportedLevels: LEVELS_DEVICE_ONLY_SETTINGS,
|
supportedLevels: LEVELS_DEVICE_ONLY_SETTINGS,
|
||||||
default: true,
|
default: true,
|
||||||
|
Loading…
Reference in New Issue
Block a user