Merge remote-tracking branch 'origin/develop' into jryans/adjust-chevrons
						commit
						f28bce36e1
					
				
							
								
								
									
										16
									
								
								.eslintrc.js
								
								
								
								
							
							
						
						
									
										16
									
								
								.eslintrc.js
								
								
								
								
							|  | @ -1,15 +1,3 @@ | |||
| const path = require('path'); | ||||
| 
 | ||||
| // get the path of the js-sdk so we can extend the config
 | ||||
| // eslint supports loading extended configs by module,
 | ||||
| // but only if they come from a module that starts with eslint-config-
 | ||||
| // So we load the filename directly (and it could be in node_modules/
 | ||||
| // or or ../node_modules/ etc)
 | ||||
| //
 | ||||
| // We add a `..` to the end because the js-sdk lives out of lib/, but the eslint
 | ||||
| // config is at the project root.
 | ||||
| const matrixJsSdkPath = path.join(path.dirname(require.resolve('matrix-js-sdk')), '..'); | ||||
| 
 | ||||
| module.exports = { | ||||
|     extends: ["matrix-org", "matrix-org/react-legacy"], | ||||
|     parser: "babel-eslint", | ||||
|  | @ -31,7 +19,7 @@ module.exports = { | |||
|     }, | ||||
| 
 | ||||
|     overrides: [{ | ||||
|         files: ["src/**/*.{ts, tsx}"], | ||||
|         "files": ["src/**/*.{ts, tsx}"], | ||||
|         "extends": ["matrix-org/ts"], | ||||
|         "rules": { | ||||
|             // We disable this while we're transitioning
 | ||||
|  | @ -41,6 +29,6 @@ module.exports = { | |||
| 
 | ||||
|             "quotes": "off", | ||||
|             "no-extra-boolean-cast": "off", | ||||
|         } | ||||
|         }, | ||||
|     }], | ||||
| }; | ||||
|  |  | |||
|  | @ -0,0 +1,39 @@ | |||
| # Local echo (developer docs) | ||||
| 
 | ||||
| The React SDK provides some local echo functionality to allow for components to do something | ||||
| quickly and fall back when it fails. This is all available in the `local-echo` directory within | ||||
| `stores`. | ||||
| 
 | ||||
| Echo is handled in EchoChambers, with `GenericEchoChamber` being the base implementation for all | ||||
| chambers. The `EchoChamber` class is provided as semantic access to a `GenericEchoChamber` | ||||
| implementation, such as the `RoomEchoChamber` (which handles echoable details of a room). | ||||
| 
 | ||||
| Anything that can be locally echoed will be provided by the `GenericEchoChamber` implementation. | ||||
| The echo chamber will also need to deal with external changes, and has full control over whether | ||||
| or not something has successfully been echoed.  | ||||
| 
 | ||||
| An `EchoContext` is provided to echo chambers (usually with a matching type: `RoomEchoContext` | ||||
| gets provided to a `RoomEchoChamber` for example) with details about their intended area of | ||||
| effect, as well as manage `EchoTransaction`s. An `EchoTransaction` is simply a unit of work that | ||||
| needs to be locally echoed. | ||||
| 
 | ||||
| The `EchoStore` manages echo chamber instances, builds contexts, and is generally less semantically | ||||
| accessible than the `EchoChamber` class. For separation of concerns, and to try and keep things | ||||
| tidy, this is an intentional design decision. | ||||
| 
 | ||||
| **Note**: The local echo stack uses a "whenable" pattern, which is similar to thenables and  | ||||
| `EventEmitter`. Whenables are ways of actioning a changing condition without having to deal | ||||
| with listeners being torn down. Once the reference count of the Whenable causes garbage collection, | ||||
| the Whenable's listeners will also be torn down. This is accelerated by the `IDestroyable` interface | ||||
| usage. | ||||
| 
 | ||||
| ## Audit functionality | ||||
| 
 | ||||
| The UI supports a "Server isn't responding" dialog which includes a partial audit log-like | ||||
| structure to it. This is partially the reason for added complexity of `EchoTransaction`s | ||||
| and `EchoContext`s - this information feeds the UI states which then provide direct retry | ||||
| mechanisms. | ||||
| 
 | ||||
| The `EchoStore` is responsible for ensuring that the appropriate non-urgent toast (lower left) | ||||
| is set up, where the dialog then drives through the contexts and transactions. | ||||
| 
 | ||||
|  | @ -140,11 +140,9 @@ | |||
|     "enzyme": "^3.11.0", | ||||
|     "enzyme-adapter-react-16": "^1.15.2", | ||||
|     "eslint": "7.5.0", | ||||
|     "eslint-config-google": "^0.14.0", | ||||
|     "eslint-config-matrix-org": "^0.1.2", | ||||
|     "eslint-plugin-babel": "^5.3.1", | ||||
|     "eslint-plugin-flowtype": "^2.50.3", | ||||
|     "eslint-plugin-jest": "^23.18.0", | ||||
|     "eslint-plugin-react": "^7.20.3", | ||||
|     "eslint-plugin-react-hooks": "^2.5.1", | ||||
|     "estree-walker": "^0.9.0", | ||||
|  |  | |||
|  | @ -15,6 +15,7 @@ | |||
| @import "./structures/_MainSplit.scss"; | ||||
| @import "./structures/_MatrixChat.scss"; | ||||
| @import "./structures/_MyGroups.scss"; | ||||
| @import "./structures/_NonUrgentToastContainer.scss"; | ||||
| @import "./structures/_NotificationPanel.scss"; | ||||
| @import "./structures/_RightPanel.scss"; | ||||
| @import "./structures/_RoomDirectory.scss"; | ||||
|  | @ -75,6 +76,7 @@ | |||
| @import "./views/dialogs/_RoomSettingsDialogBridges.scss"; | ||||
| @import "./views/dialogs/_RoomUpgradeDialog.scss"; | ||||
| @import "./views/dialogs/_RoomUpgradeWarningDialog.scss"; | ||||
| @import "./views/dialogs/_ServerOfflineDialog.scss"; | ||||
| @import "./views/dialogs/_SetEmailDialog.scss"; | ||||
| @import "./views/dialogs/_SetMxIdDialog.scss"; | ||||
| @import "./views/dialogs/_SetPasswordDialog.scss"; | ||||
|  | @ -215,6 +217,7 @@ | |||
| @import "./views/settings/tabs/user/_SecurityUserSettingsTab.scss"; | ||||
| @import "./views/settings/tabs/user/_VoiceUserSettingsTab.scss"; | ||||
| @import "./views/terms/_InlineTermsAgreement.scss"; | ||||
| @import "./views/toasts/_NonUrgentEchoFailureToast.scss"; | ||||
| @import "./views/verification/_VerificationShowSas.scss"; | ||||
| @import "./views/voip/_CallContainer.scss"; | ||||
| @import "./views/voip/_CallView.scss"; | ||||
|  |  | |||
|  | @ -54,5 +54,5 @@ limitations under the License. | |||
|     position: absolute; | ||||
|     left: -9px; | ||||
|     border-radius: 0 3px 3px 0; | ||||
|     top: 12px; // just feels right (see comment above about designs needing to be updated) | ||||
|     top: 5px; // just feels right (see comment above about designs needing to be updated) | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,35 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| .mx_NonUrgentToastContainer { | ||||
|     position: absolute; | ||||
|     bottom: 30px; | ||||
|     left: 28px; | ||||
|     z-index: 101; // same level as other toasts | ||||
| 
 | ||||
|     .mx_NonUrgentToastContainer_toast { | ||||
|         padding: 10px 12px; | ||||
|         border-radius: 8px; | ||||
|         width: 320px; | ||||
|         font-size: $font-13px; | ||||
|         margin-top: 8px; | ||||
| 
 | ||||
|         // We don't use variables on the colours because we want it to be the same | ||||
|         // in all themes. | ||||
|         background-color: #17191c; | ||||
|         color: #fff; | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,72 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| .mx_ServerOfflineDialog { | ||||
|     .mx_ServerOfflineDialog_content { | ||||
|         padding-right: 85px; | ||||
|         color: $primary-fg-color; | ||||
| 
 | ||||
|         hr { | ||||
|             border-color: $primary-fg-color; | ||||
|             opacity: 0.1; | ||||
|             border-bottom: none; | ||||
|         } | ||||
| 
 | ||||
|         ul { | ||||
|             padding: 16px; | ||||
| 
 | ||||
|             li:nth-child(n + 2) { | ||||
|                 margin-top: 16px; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         .mx_ServerOfflineDialog_content_context { | ||||
|             .mx_ServerOfflineDialog_content_context_timestamp { | ||||
|                 display: inline-block; | ||||
|                 width: 115px; | ||||
|                 color: $muted-fg-color; | ||||
|                 line-height: 24px; // same as avatar | ||||
|                 vertical-align: top; | ||||
|             } | ||||
| 
 | ||||
|             .mx_ServerOfflineDialog_content_context_timeline { | ||||
|                 display: inline-block; | ||||
|                 width: calc(100% - 155px); // 115px timestamp width + 40px right margin | ||||
| 
 | ||||
|                 .mx_ServerOfflineDialog_content_context_timeline_header { | ||||
|                     span { | ||||
|                         margin-left: 8px; | ||||
|                         vertical-align: middle; | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 .mx_ServerOfflineDialog_content_context_txn { | ||||
|                     position: relative; | ||||
|                     margin-top: 8px; | ||||
| 
 | ||||
|                     .mx_ServerOfflineDialog_content_context_txn_desc { | ||||
|                         width: calc(100% - 100px); // 100px is an arbitrary margin for the button | ||||
|                     } | ||||
| 
 | ||||
|                     .mx_AccessibleButton { | ||||
|                         float: right; | ||||
|                         padding: 0; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -69,6 +69,7 @@ limitations under the License. | |||
|     right: 0; | ||||
|     mask: url('$(res)/img/icon-jump-to-bottom.svg'); | ||||
|     mask-repeat: no-repeat; | ||||
|     mask-position: 9px 14px; | ||||
|     mask-position: center; | ||||
|     mask-size: 50%; | ||||
|     background: $muted-fg-color; | ||||
| } | ||||
|  |  | |||
|  | @ -28,7 +28,7 @@ limitations under the License. | |||
|     content: ""; | ||||
|     position: absolute; | ||||
|     top: -8px; | ||||
|     left: 11px; | ||||
|     left: 10.5px; | ||||
|     width: 4px; | ||||
|     height: 4px; | ||||
|     border-radius: 16px; | ||||
|  | @ -49,11 +49,12 @@ limitations under the License. | |||
| .mx_TopUnreadMessagesBar_scrollUp::before { | ||||
|     content: ""; | ||||
|     position: absolute; | ||||
|     width: 38px; | ||||
|     height: 38px; | ||||
|     width: 36px; | ||||
|     height: 36px; | ||||
|     mask-image: url('$(res)/img/icon-jump-to-first-unread.svg'); | ||||
|     mask-repeat: no-repeat; | ||||
|     mask-position: 9px 13px; | ||||
|     mask-position: center; | ||||
|     mask-size: 50%; | ||||
|     background: $muted-fg-color; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -0,0 +1,37 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| .mx_NonUrgentEchoFailureToast { | ||||
|     .mx_NonUrgentEchoFailureToast_icon { | ||||
|         display: inline-block; | ||||
|         width: $font-18px; | ||||
|         height: $font-18px; | ||||
|         mask-position: center; | ||||
|         mask-size: contain; | ||||
|         mask-repeat: no-repeat; | ||||
|         background-color: #fff; // we know that non-urgent toasts are always styled the same | ||||
|         mask-image: url('$(res)/img/element-icons/cloud-off.svg'); | ||||
|         margin-right: 8px; | ||||
|     } | ||||
| 
 | ||||
|     span { // includes the i18n block | ||||
|         vertical-align: middle; | ||||
|     } | ||||
| 
 | ||||
|     .mx_AccessibleButton { | ||||
|         padding: 0; | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,3 @@ | |||
| <svg width="18" height="18" viewBox="0 0 18 18" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <path fill-rule="evenodd" clip-rule="evenodd" d="M1.53033 0.46967C1.23744 0.176777 0.762563 0.176777 0.46967 0.46967C0.176777 0.762563 0.176777 1.23744 0.46967 1.53033L4.3982 5.45886C3.81109 6.13809 3.38896 7.01315 3.21555 7.99387C1.96379 8.20624 1 9.465 1 10.981C1 12.6455 2.16209 14 3.59014 14H12.9393L16.4697 17.5303C16.7626 17.8232 17.2374 17.8232 17.5303 17.5303C17.8232 17.2374 17.8232 16.7626 17.5303 16.4697L1.53033 0.46967ZM17 10.9817C16.998 11.8303 16.6946 12.5985 16.2081 13.1475L7.07635 4.01569C7.18805 4.00529 7.30083 4 7.41451 4C8.75982 4 9.99711 4.71787 10.8072 5.94503C11.0993 5.85476 11.4011 5.80939 11.7058 5.80939C13.0303 5.80939 14.2138 6.65743 14.8199 8.00337C16.0519 8.23522 17 9.48685 17 10.9817Z" fill="black"/> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 839 B | 
|  | @ -14,7 +14,11 @@ See the License for the specific language governing permissions and | |||
| limitations under the License. | ||||
| */ | ||||
| 
 | ||||
| import { JSXElementConstructor } from "react"; | ||||
| 
 | ||||
| // Based on https://stackoverflow.com/a/53229857/3532235
 | ||||
| export type Without<T, U> = {[P in Exclude<keyof T, keyof U>] ? : never}; | ||||
| export type XOR<T, U> = (T | U) extends object ? (Without<T, U> & U) | (Without<U, T> & T) : T | U; | ||||
| export type Writeable<T> = { -readonly [P in keyof T]: T[P] }; | ||||
| 
 | ||||
| export type ComponentClass = keyof JSX.IntrinsicElements | JSXElementConstructor<any>; | ||||
|  |  | |||
|  | @ -25,6 +25,7 @@ import { PlatformPeg } from "../PlatformPeg"; | |||
| import RoomListLayoutStore from "../stores/room-list/RoomListLayoutStore"; | ||||
| import {IntegrationManagers} from "../integrations/IntegrationManagers"; | ||||
| import {ModalManager} from "../Modal"; | ||||
| import SettingsStore from "../settings/SettingsStore"; | ||||
| 
 | ||||
| declare global { | ||||
|     interface Window { | ||||
|  | @ -43,6 +44,7 @@ declare global { | |||
|         mxPlatformPeg: PlatformPeg; | ||||
|         mxIntegrationManagers: typeof IntegrationManagers; | ||||
|         singletonModalManager: ModalManager; | ||||
|         mxSettingsStore: SettingsStore; | ||||
|     } | ||||
| 
 | ||||
|     // workaround for https://github.com/microsoft/TypeScript/issues/30933
 | ||||
|  |  | |||
|  | @ -62,10 +62,11 @@ import Matrix from 'matrix-js-sdk'; | |||
| import dis from './dispatcher/dispatcher'; | ||||
| import WidgetUtils from './utils/WidgetUtils'; | ||||
| import WidgetEchoStore from './stores/WidgetEchoStore'; | ||||
| import SettingsStore, { SettingLevel } from './settings/SettingsStore'; | ||||
| import SettingsStore from './settings/SettingsStore'; | ||||
| import {generateHumanReadableId} from "./utils/NamingUtils"; | ||||
| import {Jitsi} from "./widgets/Jitsi"; | ||||
| import {WidgetType} from "./widgets/WidgetType"; | ||||
| import {SettingLevel} from "./settings/SettingLevel"; | ||||
| 
 | ||||
| global.mxCalls = { | ||||
|     //room_id: MatrixCall
 | ||||
|  |  | |||
|  | @ -15,7 +15,8 @@ | |||
| */ | ||||
| 
 | ||||
| import * as Matrix from 'matrix-js-sdk'; | ||||
| import SettingsStore, {SettingLevel} from "./settings/SettingsStore"; | ||||
| import SettingsStore from "./settings/SettingsStore"; | ||||
| import {SettingLevel} from "./settings/SettingLevel"; | ||||
| 
 | ||||
| export default { | ||||
|     hasAnyLabeledDevices: async function() { | ||||
|  |  | |||
|  | @ -306,6 +306,11 @@ async function _restoreFromLocalStorage(opts) { | |||
|         } | ||||
| 
 | ||||
|         const pickleKey = await PlatformPeg.get().getPickleKey(userId, deviceId); | ||||
|         if (pickleKey) { | ||||
|             console.log("Got pickle key"); | ||||
|         } else { | ||||
|             console.log("No pickle key available"); | ||||
|         } | ||||
| 
 | ||||
|         console.log(`Restoring session for ${userId}`); | ||||
|         await _doSetLoggedIn({ | ||||
|  | @ -364,6 +369,12 @@ export async function setLoggedIn(credentials) { | |||
|           ? await PlatformPeg.get().createPickleKey(credentials.userId, credentials.deviceId) | ||||
|           : null; | ||||
| 
 | ||||
|     if (pickleKey) { | ||||
|         console.log("Created pickle key"); | ||||
|     } else { | ||||
|         console.log("Pickle key not created"); | ||||
|     } | ||||
| 
 | ||||
|     return _doSetLoggedIn(Object.assign({}, credentials, {pickleKey}), true); | ||||
| } | ||||
| 
 | ||||
|  | @ -501,6 +512,14 @@ function _persistCredentialsToLocalStorage(credentials) { | |||
|     localStorage.setItem("mx_access_token", credentials.accessToken); | ||||
|     localStorage.setItem("mx_is_guest", JSON.stringify(credentials.guest)); | ||||
| 
 | ||||
|     if (credentials.pickleKey) { | ||||
|         localStorage.setItem("mx_has_pickle_key", true); | ||||
|     } else { | ||||
|         if (localStorage.getItem("mx_has_pickle_key")) { | ||||
|             console.error("Expected a pickle key, but none provided.  Encryption may not work."); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // if we didn't get a deviceId from the login, leave mx_device_id unset,
 | ||||
|     // rather than setting it to "undefined".
 | ||||
|     //
 | ||||
|  |  | |||
|  | @ -256,7 +256,7 @@ class _MatrixClientPeg implements IMatrixClientPeg { | |||
|             deviceId: creds.deviceId, | ||||
|             pickleKey: creds.pickleKey, | ||||
|             timelineSupport: true, | ||||
|             forceTURN: !SettingsStore.getValue('webRtcAllowPeerToPeer', false), | ||||
|             forceTURN: !SettingsStore.getValue('webRtcAllowPeerToPeer'), | ||||
|             fallbackICEServerAllowed: !!SettingsStore.getValue('fallbackICEServerAllowed'), | ||||
|             verificationMethods: [ | ||||
|                 verificationMethods.SAS, | ||||
|  |  | |||
|  | @ -27,10 +27,11 @@ import dis from './dispatcher/dispatcher'; | |||
| import * as sdk from './index'; | ||||
| import { _t } from './languageHandler'; | ||||
| import Modal from './Modal'; | ||||
| import SettingsStore, {SettingLevel} from "./settings/SettingsStore"; | ||||
| import SettingsStore from "./settings/SettingsStore"; | ||||
| import { | ||||
|     hideToast as hideNotificationsToast, | ||||
| } from "./toasts/DesktopNotificationsToast"; | ||||
| import {SettingLevel} from "./settings/SettingLevel"; | ||||
| 
 | ||||
| /* | ||||
|  * Dispatches: | ||||
|  |  | |||
|  | @ -20,9 +20,10 @@ import PropTypes from 'prop-types'; | |||
| import dis from "../../../../dispatcher/dispatcher"; | ||||
| import { _t } from '../../../../languageHandler'; | ||||
| 
 | ||||
| import SettingsStore, {SettingLevel} from "../../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../../settings/SettingsStore"; | ||||
| import EventIndexPeg from "../../../../indexing/EventIndexPeg"; | ||||
| import {Action} from "../../../../dispatcher/actions"; | ||||
| import {SettingLevel} from "../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| /* | ||||
|  * Allows the user to disable the Event Index. | ||||
|  |  | |||
|  | @ -19,11 +19,12 @@ import * as sdk from '../../../../index'; | |||
| import PropTypes from 'prop-types'; | ||||
| import { _t } from '../../../../languageHandler'; | ||||
| import SdkConfig from '../../../../SdkConfig'; | ||||
| import SettingsStore, {SettingLevel} from "../../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../../settings/SettingsStore"; | ||||
| 
 | ||||
| import Modal from '../../../../Modal'; | ||||
| import {formatBytes, formatCountLong} from "../../../../utils/FormattingUtils"; | ||||
| import EventIndexPeg from "../../../../indexing/EventIndexPeg"; | ||||
| import {SettingLevel} from "../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| /* | ||||
|  * Allows the user to introspect the event index state and disable it. | ||||
|  |  | |||
|  | @ -54,6 +54,8 @@ import LeftPanel from "./LeftPanel"; | |||
| import CallContainer from '../views/voip/CallContainer'; | ||||
| import { ViewRoomDeltaPayload } from "../../dispatcher/payloads/ViewRoomDeltaPayload"; | ||||
| import RoomListStore from "../../stores/room-list/RoomListStore"; | ||||
| import NonUrgentToastContainer from "./NonUrgentToastContainer"; | ||||
| import { ToggleRightPanelPayload } from "../../dispatcher/payloads/ToggleRightPanelPayload"; | ||||
| 
 | ||||
| // We need to fetch each pinned message individually (if we don't already have it)
 | ||||
| // so each pinned message may trigger a request. Limit the number per room for sanity.
 | ||||
|  | @ -472,8 +474,8 @@ class LoggedInView extends React.Component<IProps, IState> { | |||
| 
 | ||||
|             case Key.PERIOD: | ||||
|                 if (ctrlCmdOnly && (this.props.page_type === "room_view" || this.props.page_type === "group_view")) { | ||||
|                     dis.dispatch({ | ||||
|                         action: 'toggle_right_panel', | ||||
|                     dis.dispatch<ToggleRightPanelPayload>({ | ||||
|                         action: Action.ToggleRightPanel, | ||||
|                         type: this.props.page_type === "room_view" ? "room" : "group", | ||||
|                     }); | ||||
|                     handled = true; | ||||
|  | @ -687,6 +689,7 @@ class LoggedInView extends React.Component<IProps, IState> { | |||
|                     </DragDropContext> | ||||
|                 </div> | ||||
|                 <CallContainer /> | ||||
|                 <NonUrgentToastContainer /> | ||||
|             </MatrixClientContext.Provider> | ||||
|         ); | ||||
|     } | ||||
|  |  | |||
|  | @ -51,7 +51,7 @@ import { getHomePageUrl } from '../../utils/pages'; | |||
| 
 | ||||
| import createRoom from "../../createRoom"; | ||||
| import {_t, _td, getCurrentLanguage} from '../../languageHandler'; | ||||
| import SettingsStore, { SettingLevel } from "../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../settings/SettingsStore"; | ||||
| import ThemeController from "../../settings/controllers/ThemeController"; | ||||
| import { startAnyRegistrationFlow } from "../../Registration.js"; | ||||
| import { messageForSyncError } from '../../utils/ErrorUtils'; | ||||
|  | @ -75,6 +75,7 @@ import {showToast as showNotificationsToast} from "../../toasts/DesktopNotificat | |||
| import { OpenToTabPayload } from "../../dispatcher/payloads/OpenToTabPayload"; | ||||
| import ErrorDialog from "../views/dialogs/ErrorDialog"; | ||||
| import { RoomNotificationStateStore } from "../../stores/notifications/RoomNotificationStateStore"; | ||||
| import { SettingLevel } from "../../settings/SettingLevel"; | ||||
| 
 | ||||
| /** constants for MatrixChat.state.view */ | ||||
| export enum Views { | ||||
|  |  | |||
|  | @ -0,0 +1,63 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 * as React from "react"; | ||||
| import { ComponentClass } from "../../@types/common"; | ||||
| import NonUrgentToastStore from "../../stores/NonUrgentToastStore"; | ||||
| import { UPDATE_EVENT } from "../../stores/AsyncStore"; | ||||
| 
 | ||||
| interface IProps { | ||||
| } | ||||
| 
 | ||||
| interface IState { | ||||
|     toasts: ComponentClass[], | ||||
| } | ||||
| 
 | ||||
| export default class NonUrgentToastContainer extends React.PureComponent<IProps, IState> { | ||||
|     public constructor(props, context) { | ||||
|         super(props, context); | ||||
| 
 | ||||
|         this.state = { | ||||
|             toasts: NonUrgentToastStore.instance.components, | ||||
|         }; | ||||
| 
 | ||||
|         NonUrgentToastStore.instance.on(UPDATE_EVENT, this.onUpdateToasts); | ||||
|     } | ||||
| 
 | ||||
|     public componentWillUnmount() { | ||||
|         NonUrgentToastStore.instance.off(UPDATE_EVENT, this.onUpdateToasts); | ||||
|     } | ||||
| 
 | ||||
|     private onUpdateToasts = () => { | ||||
|         this.setState({toasts: NonUrgentToastStore.instance.components}); | ||||
|     }; | ||||
| 
 | ||||
|     public render() { | ||||
|         const toasts = this.state.toasts.map((t, i) => { | ||||
|             return ( | ||||
|                 <div className="mx_NonUrgentToastContainer_toast" key={`toast-${i}`}> | ||||
|                     {React.createElement(t, {})} | ||||
|                 </div> | ||||
|             ); | ||||
|         }); | ||||
| 
 | ||||
|         return ( | ||||
|             <div className="mx_NonUrgentToastContainer" role="alert"> | ||||
|                 {toasts} | ||||
|             </div> | ||||
|         ); | ||||
|     } | ||||
| } | ||||
|  | @ -26,7 +26,7 @@ import dis from '../../dispatcher/dispatcher'; | |||
| import RateLimitedFunc from '../../ratelimitedfunc'; | ||||
| import { showGroupInviteDialog, showGroupAddRoomDialog } from '../../GroupAddressPicker'; | ||||
| import GroupStore from '../../stores/GroupStore'; | ||||
| import {RIGHT_PANEL_PHASES, RIGHT_PANEL_PHASES_NO_ARGS} from "../../stores/RightPanelStorePhases"; | ||||
| import {RightPanelPhases, RIGHT_PANEL_PHASES_NO_ARGS} from "../../stores/RightPanelStorePhases"; | ||||
| import RightPanelStore from "../../stores/RightPanelStore"; | ||||
| import MatrixClientContext from "../../contexts/MatrixClientContext"; | ||||
| import {Action} from "../../dispatcher/actions"; | ||||
|  | @ -75,8 +75,8 @@ export default class RightPanel extends React.Component { | |||
|         const userForPanel = this._getUserForPanel(); | ||||
|         if (this.props.groupId) { | ||||
|             if (!RIGHT_PANEL_PHASES_NO_ARGS.includes(rps.groupPanelPhase)) { | ||||
|                 dis.dispatch({action: "set_right_panel_phase", phase: RIGHT_PANEL_PHASES.GroupMemberList}); | ||||
|                 return RIGHT_PANEL_PHASES.GroupMemberList; | ||||
|                 dis.dispatch({action: Action.SetRightPanelPhase, phase: RightPanelPhases.GroupMemberList}); | ||||
|                 return RightPanelPhases.GroupMemberList; | ||||
|             } | ||||
|             return rps.groupPanelPhase; | ||||
|         } else if (userForPanel) { | ||||
|  | @ -98,11 +98,11 @@ export default class RightPanel extends React.Component { | |||
|             ) { | ||||
|                 return rps.roomPanelPhase; | ||||
|             } | ||||
|             return RIGHT_PANEL_PHASES.RoomMemberInfo; | ||||
|             return RightPanelPhases.RoomMemberInfo; | ||||
|         } else { | ||||
|             if (!RIGHT_PANEL_PHASES_NO_ARGS.includes(rps.roomPanelPhase)) { | ||||
|                 dis.dispatch({action: "set_right_panel_phase", phase: RIGHT_PANEL_PHASES.RoomMemberList}); | ||||
|                 return RIGHT_PANEL_PHASES.RoomMemberList; | ||||
|                 dis.dispatch({action: Action.SetRightPanelPhase, phase: RightPanelPhases.RoomMemberList}); | ||||
|                 return RightPanelPhases.RoomMemberList; | ||||
|             } | ||||
|             return rps.roomPanelPhase; | ||||
|         } | ||||
|  | @ -149,7 +149,7 @@ export default class RightPanel extends React.Component { | |||
|     onInviteToGroupButtonClick() { | ||||
|         showGroupInviteDialog(this.props.groupId).then(() => { | ||||
|             this.setState({ | ||||
|                 phase: RIGHT_PANEL_PHASES.GroupMemberList, | ||||
|                 phase: RightPanelPhases.GroupMemberList, | ||||
|             }); | ||||
|         }); | ||||
|     } | ||||
|  | @ -165,9 +165,9 @@ export default class RightPanel extends React.Component { | |||
|             return; | ||||
|         } | ||||
|         // redraw the badge on the membership list
 | ||||
|         if (this.state.phase === RIGHT_PANEL_PHASES.RoomMemberList && member.roomId === this.props.roomId) { | ||||
|         if (this.state.phase === RightPanelPhases.RoomMemberList && member.roomId === this.props.roomId) { | ||||
|             this._delayedUpdate(); | ||||
|         } else if (this.state.phase === RIGHT_PANEL_PHASES.RoomMemberInfo && member.roomId === this.props.roomId && | ||||
|         } else if (this.state.phase === RightPanelPhases.RoomMemberInfo && member.roomId === this.props.roomId && | ||||
|                 member.userId === this.state.member.userId) { | ||||
|             // refresh the member info (e.g. new power level)
 | ||||
|             this._delayedUpdate(); | ||||
|  | @ -175,7 +175,7 @@ export default class RightPanel extends React.Component { | |||
|     } | ||||
| 
 | ||||
|     onAction(payload) { | ||||
|         if (payload.action === "after_right_panel_phase_change") { | ||||
|         if (payload.action === Action.AfterRightPanelPhaseChange) { | ||||
|             this.setState({ | ||||
|                 phase: payload.phase, | ||||
|                 groupRoomId: payload.groupRoomId, | ||||
|  | @ -206,7 +206,7 @@ export default class RightPanel extends React.Component { | |||
|             // or the member list if we were in the member panel... phew.
 | ||||
|             dis.dispatch({ | ||||
|                 action: Action.ViewUser, | ||||
|                 member: this.state.phase === RIGHT_PANEL_PHASES.EncryptionPanel ? this.state.member : null, | ||||
|                 member: this.state.phase === RightPanelPhases.EncryptionPanel ? this.state.member : null, | ||||
|             }); | ||||
|         } | ||||
|     }; | ||||
|  | @ -225,21 +225,21 @@ export default class RightPanel extends React.Component { | |||
|         let panel = <div />; | ||||
| 
 | ||||
|         switch (this.state.phase) { | ||||
|             case RIGHT_PANEL_PHASES.RoomMemberList: | ||||
|             case RightPanelPhases.RoomMemberList: | ||||
|                 if (this.props.roomId) { | ||||
|                     panel = <MemberList roomId={this.props.roomId} key={this.props.roomId} />; | ||||
|                 } | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.GroupMemberList: | ||||
|             case RightPanelPhases.GroupMemberList: | ||||
|                 if (this.props.groupId) { | ||||
|                     panel = <GroupMemberList groupId={this.props.groupId} key={this.props.groupId} />; | ||||
|                 } | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.GroupRoomList: | ||||
|             case RightPanelPhases.GroupRoomList: | ||||
|                 panel = <GroupRoomList groupId={this.props.groupId} key={this.props.groupId} />; | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.RoomMemberInfo: | ||||
|             case RIGHT_PANEL_PHASES.EncryptionPanel: | ||||
|             case RightPanelPhases.RoomMemberInfo: | ||||
|             case RightPanelPhases.EncryptionPanel: | ||||
|                 panel = <UserInfo | ||||
|                     user={this.state.member} | ||||
|                     roomId={this.props.roomId} | ||||
|  | @ -250,26 +250,26 @@ export default class RightPanel extends React.Component { | |||
|                     verificationRequestPromise={this.state.verificationRequestPromise} | ||||
|                 />; | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.Room3pidMemberInfo: | ||||
|             case RightPanelPhases.Room3pidMemberInfo: | ||||
|                 panel = <ThirdPartyMemberInfo event={this.state.event} key={this.props.roomId} />; | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.GroupMemberInfo: | ||||
|             case RightPanelPhases.GroupMemberInfo: | ||||
|                 panel = <UserInfo | ||||
|                     user={this.state.member} | ||||
|                     groupId={this.props.groupId} | ||||
|                     key={this.state.member.userId} | ||||
|                     onClose={this.onCloseUserInfo} />; | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.GroupRoomInfo: | ||||
|             case RightPanelPhases.GroupRoomInfo: | ||||
|                 panel = <GroupRoomInfo | ||||
|                     groupRoomId={this.state.groupRoomId} | ||||
|                     groupId={this.props.groupId} | ||||
|                     key={this.state.groupRoomId} />; | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.NotificationPanel: | ||||
|             case RightPanelPhases.NotificationPanel: | ||||
|                 panel = <NotificationPanel />; | ||||
|                 break; | ||||
|             case RIGHT_PANEL_PHASES.FilePanel: | ||||
|             case RightPanelPhases.FilePanel: | ||||
|                 panel = <FilePanel roomId={this.props.roomId} resizeNotifier={this.props.resizeNotifier} />; | ||||
|                 break; | ||||
|         } | ||||
|  |  | |||
|  | @ -48,7 +48,7 @@ import RightPanel from './RightPanel'; | |||
| import RoomViewStore from '../../stores/RoomViewStore'; | ||||
| import RoomScrollStateStore from '../../stores/RoomScrollStateStore'; | ||||
| import WidgetEchoStore from '../../stores/WidgetEchoStore'; | ||||
| import SettingsStore, {SettingLevel} from "../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../settings/SettingsStore"; | ||||
| import AccessibleButton from "../views/elements/AccessibleButton"; | ||||
| import RightPanelStore from "../../stores/RightPanelStore"; | ||||
| import {haveTileForEvent} from "../views/rooms/EventTile"; | ||||
|  | @ -56,6 +56,7 @@ import RoomContext from "../../contexts/RoomContext"; | |||
| import MatrixClientContext from "../../contexts/MatrixClientContext"; | ||||
| import { shieldStatusForRoom } from '../../utils/ShieldUtils'; | ||||
| import {Action} from "../../dispatcher/actions"; | ||||
| import {SettingLevel} from "../../settings/SettingLevel"; | ||||
| 
 | ||||
| const DEBUG = false; | ||||
| let debuglog = function() {}; | ||||
|  |  | |||
|  | @ -26,7 +26,7 @@ import { OpenToTabPayload } from "../../dispatcher/payloads/OpenToTabPayload"; | |||
| import RedesignFeedbackDialog from "../views/dialogs/RedesignFeedbackDialog"; | ||||
| import Modal from "../../Modal"; | ||||
| import LogoutDialog from "../views/dialogs/LogoutDialog"; | ||||
| import SettingsStore, {SettingLevel} from "../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../settings/SettingsStore"; | ||||
| import {getCustomTheme} from "../../theme"; | ||||
| import {getHostingLink} from "../../utils/HostingLink"; | ||||
| import {ButtonEvent} from "../views/elements/AccessibleButton"; | ||||
|  | @ -37,6 +37,7 @@ import { UPDATE_EVENT } from "../../stores/AsyncStore"; | |||
| import BaseAvatar from '../views/avatars/BaseAvatar'; | ||||
| import classNames from "classnames"; | ||||
| import AccessibleTooltipButton from "../views/elements/AccessibleTooltipButton"; | ||||
| import { SettingLevel } from "../../settings/SettingLevel"; | ||||
| 
 | ||||
| interface IProps { | ||||
|     isMinimized: boolean; | ||||
|  |  | |||
|  | @ -72,7 +72,7 @@ export default class SoftLogout extends React.Component { | |||
| 
 | ||||
|         this._initLogin(); | ||||
| 
 | ||||
|         MatrixClientPeg.get().flagAllGroupSessionsForBackup().then(remaining => { | ||||
|         MatrixClientPeg.get().countSessionsNeedingBackup().then(remaining => { | ||||
|             this.setState({keyBackupNeeded: remaining > 0}); | ||||
|         }); | ||||
|     } | ||||
|  |  | |||
|  | @ -16,10 +16,11 @@ limitations under the License. | |||
| 
 | ||||
| import SdkConfig from "../../../SdkConfig"; | ||||
| import {getCurrentLanguage} from "../../../languageHandler"; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import PlatformPeg from "../../../PlatformPeg"; | ||||
| import * as sdk from '../../../index'; | ||||
| import React from 'react'; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| function onChange(newLang) { | ||||
|     if (getCurrentLanguage() !== newLang) { | ||||
|  |  | |||
|  | @ -19,8 +19,8 @@ import PropTypes from 'prop-types'; | |||
| import createReactClass from 'create-react-class'; | ||||
| import * as sdk from '../../../index'; | ||||
| import { _t } from '../../../languageHandler'; | ||||
| import {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default createReactClass({ | ||||
|     propTypes: { | ||||
|  |  | |||
|  | @ -0,0 +1,124 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 * as React from 'react'; | ||||
| import BaseDialog from './BaseDialog'; | ||||
| import { _t } from '../../../languageHandler'; | ||||
| import { EchoStore } from "../../../stores/local-echo/EchoStore"; | ||||
| import { formatTime } from "../../../DateUtils"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import { RoomEchoContext } from "../../../stores/local-echo/RoomEchoContext"; | ||||
| import RoomAvatar from "../avatars/RoomAvatar"; | ||||
| import { TransactionStatus } from "../../../stores/local-echo/EchoTransaction"; | ||||
| import Spinner from "../elements/Spinner"; | ||||
| import AccessibleButton from "../elements/AccessibleButton"; | ||||
| import { UPDATE_EVENT } from "../../../stores/AsyncStore"; | ||||
| import { MatrixClientPeg } from "../../../MatrixClientPeg"; | ||||
| 
 | ||||
| interface IProps { | ||||
|     onFinished: (bool) => void; | ||||
| } | ||||
| 
 | ||||
| export default class ServerOfflineDialog extends React.PureComponent<IProps> { | ||||
|     public componentDidMount() { | ||||
|         EchoStore.instance.on(UPDATE_EVENT, this.onEchosUpdated); | ||||
|     } | ||||
| 
 | ||||
|     public componentWillUnmount() { | ||||
|         EchoStore.instance.off(UPDATE_EVENT, this.onEchosUpdated); | ||||
|     } | ||||
| 
 | ||||
|     private onEchosUpdated = () => { | ||||
|         this.forceUpdate(); // no state to worry about
 | ||||
|     }; | ||||
| 
 | ||||
|     private renderTimeline(): React.ReactElement[] { | ||||
|         return EchoStore.instance.contexts.map((c, i) => { | ||||
|             if (!c.firstFailedTime) return null; // not useful
 | ||||
|             if (!(c instanceof RoomEchoContext)) throw new Error("Cannot render unknown context: " + c); | ||||
|             const header = ( | ||||
|                 <div className="mx_ServerOfflineDialog_content_context_timeline_header"> | ||||
|                     <RoomAvatar width={24} height={24} room={c.room} /> | ||||
|                     <span>{c.room.name}</span> | ||||
|                 </div> | ||||
|             ); | ||||
|             const entries = c.transactions | ||||
|                 .filter(t => t.status === TransactionStatus.DoneError || t.didPreviouslyFail) | ||||
|                 .map((t, j) => { | ||||
|                     let button = <Spinner w={19} h={19} />; | ||||
|                     if (t.status === TransactionStatus.DoneError) { | ||||
|                         button = ( | ||||
|                             <AccessibleButton kind="link" onClick={() => t.run()}>{_t("Resend")}</AccessibleButton> | ||||
|                         ); | ||||
|                     } | ||||
|                     return ( | ||||
|                         <div className="mx_ServerOfflineDialog_content_context_txn" key={`txn-${j}`}> | ||||
|                             <span className="mx_ServerOfflineDialog_content_context_txn_desc"> | ||||
|                                 {t.auditName} | ||||
|                             </span> | ||||
|                             {button} | ||||
|                         </div> | ||||
|                     ); | ||||
|                 }); | ||||
|             return ( | ||||
|                 <div className="mx_ServerOfflineDialog_content_context" key={`context-${i}`}> | ||||
|                     <div className="mx_ServerOfflineDialog_content_context_timestamp"> | ||||
|                         {formatTime(c.firstFailedTime, SettingsStore.getValue("showTwelveHourTimestamps"))} | ||||
|                     </div> | ||||
|                     <div className="mx_ServerOfflineDialog_content_context_timeline"> | ||||
|                         {header} | ||||
|                         {entries} | ||||
|                     </div> | ||||
|                 </div> | ||||
|             ) | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     public render() { | ||||
|         let timeline = this.renderTimeline().filter(c => !!c); // remove nulls for next check
 | ||||
|         if (timeline.length === 0) { | ||||
|             timeline = [<div key={1}>{_t("You're all caught up.")}</div>]; | ||||
|         } | ||||
| 
 | ||||
|         const serverName = MatrixClientPeg.getHomeserverName(); | ||||
|         return <BaseDialog title={_t("Server isn't responding")} | ||||
|             className='mx_ServerOfflineDialog' | ||||
|             contentId='mx_Dialog_content' | ||||
|             onFinished={this.props.onFinished} | ||||
|             hasCancel={true} | ||||
|         > | ||||
|             <div className="mx_ServerOfflineDialog_content"> | ||||
|                 <p>{_t( | ||||
|                     "Your server isn't responding to some of your requests. " + | ||||
|                     "Below are some of the most likely reasons.", | ||||
|                 )}</p> | ||||
|                 <ul> | ||||
|                     <li>{_t("The server (%(serverName)s) took too long to respond.", {serverName})}</li> | ||||
|                     <li>{_t("Your firewall or anti-virus is blocking the request.")}</li> | ||||
|                     <li>{_t("A browser extension is preventing the request.")}</li> | ||||
|                     <li>{_t("The server is offline.")}</li> | ||||
|                     <li>{_t("The server has denied your request.")}</li> | ||||
|                     <li>{_t("Your area is experiencing difficulties connecting to the internet.")}</li> | ||||
|                     <li>{_t("A connection error occurred while trying to contact the server.")}</li> | ||||
|                     <li>{_t("The server is not configured to indicate what the problem is (CORS).")}</li> | ||||
|                 </ul> | ||||
|                 <hr /> | ||||
|                 <h2>{_t("Recent changes that have not yet been received")}</h2> | ||||
|                 {timeline} | ||||
|             </div> | ||||
|         </BaseDialog>; | ||||
|     } | ||||
| } | ||||
|  | @ -17,10 +17,11 @@ limitations under the License. | |||
| import React from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import {_t} from "../../../languageHandler"; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import * as sdk from "../../../index"; | ||||
| import LabelledToggleSwitch from "../elements/LabelledToggleSwitch"; | ||||
| import WidgetUtils from "../../../utils/WidgetUtils"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class WidgetOpenIDPermissionsDialog extends React.Component { | ||||
|     static propTypes = { | ||||
|  |  | |||
|  | @ -35,12 +35,13 @@ import dis from '../../../dispatcher/dispatcher'; | |||
| import ActiveWidgetStore from '../../../stores/ActiveWidgetStore'; | ||||
| import classNames from 'classnames'; | ||||
| import {IntegrationManagers} from "../../../integrations/IntegrationManagers"; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import {aboveLeftOf, ContextMenu, ContextMenuButton} from "../../structures/ContextMenu"; | ||||
| import PersistedElement from "./PersistedElement"; | ||||
| import {WidgetType} from "../../../widgets/WidgetType"; | ||||
| import {Capability} from "../../../widgets/WidgetApi"; | ||||
| import {sleep} from "../../../utils/promise"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| const ALLOWED_APP_URL_SCHEMES = ['https:', 'http:']; | ||||
| const ENABLE_REACT_PERF = false; | ||||
|  |  | |||
|  | @ -15,8 +15,9 @@ limitations under the License. | |||
| */ | ||||
| 
 | ||||
| import React from 'react'; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import Draggable, {ILocationState} from './Draggable'; | ||||
| import { SettingLevel } from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| interface IProps { | ||||
|     // Current room
 | ||||
|  |  | |||
|  | @ -20,11 +20,12 @@ import SettingsStore from "../../../settings/SettingsStore"; | |||
| import { _t } from '../../../languageHandler'; | ||||
| import ToggleSwitch from "./ToggleSwitch"; | ||||
| import StyledCheckbox from "./StyledCheckbox"; | ||||
| import { SettingLevel } from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| interface IProps { | ||||
|     // The setting must be a boolean
 | ||||
|     name: string; | ||||
|     level: string; | ||||
|     level: SettingLevel; | ||||
|     roomId?: string; // for per-room settings
 | ||||
|     label?: string; // untranslated
 | ||||
|     isExplicit?: boolean; | ||||
|  | @ -52,8 +53,8 @@ export default class SettingsFlag extends React.Component<IProps, IState> { | |||
|         }; | ||||
|     } | ||||
| 
 | ||||
|     private onChange = (checked: boolean): void => { | ||||
|         this.save(checked); | ||||
|     private onChange = async (checked: boolean) => { | ||||
|         await this.save(checked); | ||||
|         this.setState({ value: checked }); | ||||
|         if (this.props.onChange) this.props.onChange(checked); | ||||
|     }; | ||||
|  | @ -62,8 +63,8 @@ export default class SettingsFlag extends React.Component<IProps, IState> { | |||
|         this.onChange(e.target.checked); | ||||
|     }; | ||||
| 
 | ||||
|     private save = (val?: boolean): void => { | ||||
|         return SettingsStore.setValue( | ||||
|     private save = async (val?: boolean) => { | ||||
|         await SettingsStore.setValue( | ||||
|             this.props.name, | ||||
|             this.props.roomId, | ||||
|             this.props.level, | ||||
|  |  | |||
|  | @ -24,8 +24,9 @@ import GroupStore from '../../../stores/GroupStore'; | |||
| import PropTypes from 'prop-types'; | ||||
| import { showGroupInviteDialog } from '../../../GroupAddressPicker'; | ||||
| import AccessibleButton from '../elements/AccessibleButton'; | ||||
| import {RIGHT_PANEL_PHASES} from "../../../stores/RightPanelStorePhases"; | ||||
| import {RightPanelPhases} from "../../../stores/RightPanelStorePhases"; | ||||
| import AutoHideScrollbar from "../../structures/AutoHideScrollbar"; | ||||
| import {Action} from "../../../dispatcher/actions"; | ||||
| 
 | ||||
| const INITIAL_LOAD_NUM_MEMBERS = 30; | ||||
| 
 | ||||
|  | @ -164,9 +165,9 @@ export default createReactClass({ | |||
|     onInviteToGroupButtonClick() { | ||||
|         showGroupInviteDialog(this.props.groupId).then(() => { | ||||
|             dis.dispatch({ | ||||
|                 action: 'set_right_panel_phase', | ||||
|                 phase: RIGHT_PANEL_PHASES.GroupMemberList, | ||||
|                 groupId: this.props.groupId, | ||||
|                 action: Action.SetRightPanelPhase, | ||||
|                 phase: RightPanelPhases.GroupMemberList, | ||||
|                 refireParams: { groupId: this.props.groupId }, | ||||
|             }); | ||||
|         }); | ||||
|     }, | ||||
|  |  | |||
|  | @ -22,7 +22,8 @@ import { _t } from '../../../languageHandler'; | |||
| import {getNameForEventRoom, userLabelForEventRoom} | ||||
|     from '../../../utils/KeyVerificationStateObserver'; | ||||
| import dis from "../../../dispatcher/dispatcher"; | ||||
| import {RIGHT_PANEL_PHASES} from "../../../stores/RightPanelStorePhases"; | ||||
| import {RightPanelPhases} from "../../../stores/RightPanelStorePhases"; | ||||
| import {Action} from "../../../dispatcher/actions"; | ||||
| 
 | ||||
| export default class MKeyVerificationRequest extends React.Component { | ||||
|     constructor(props) { | ||||
|  | @ -48,8 +49,8 @@ export default class MKeyVerificationRequest extends React.Component { | |||
|         const {verificationRequest} = this.props.mxEvent; | ||||
|         const member = MatrixClientPeg.get().getUser(verificationRequest.otherUserId); | ||||
|         dis.dispatch({ | ||||
|             action: "set_right_panel_phase", | ||||
|             phase: RIGHT_PANEL_PHASES.EncryptionPanel, | ||||
|             action: Action.SetRightPanelPhase, | ||||
|             phase: RightPanelPhases.EncryptionPanel, | ||||
|             refireParams: {verificationRequest, member}, | ||||
|         }); | ||||
|     }; | ||||
|  |  | |||
|  | @ -15,10 +15,10 @@ limitations under the License. | |||
| */ | ||||
| 
 | ||||
| import React from "react"; | ||||
| import PropTypes from "prop-types"; | ||||
| 
 | ||||
| import * as sdk from "../../../index"; | ||||
| import {_t} from "../../../languageHandler"; | ||||
| import {RoomMember} from "matrix-js-sdk/src/models/room-member"; | ||||
| 
 | ||||
| export const PendingActionSpinner = ({text}) => { | ||||
|     const Spinner = sdk.getComponent('elements.Spinner'); | ||||
|  | @ -28,7 +28,17 @@ export const PendingActionSpinner = ({text}) => { | |||
|     </div>; | ||||
| }; | ||||
| 
 | ||||
| const EncryptionInfo = ({ | ||||
| interface IProps { | ||||
|     waitingForOtherParty: boolean; | ||||
|     waitingForNetwork: boolean; | ||||
|     member: RoomMember; | ||||
|     onStartVerification: () => Promise<void>; | ||||
|     isRoomEncrypted: boolean; | ||||
|     inDialog: boolean; | ||||
|     isSelfVerification: boolean; | ||||
| } | ||||
| 
 | ||||
| const EncryptionInfo: React.FC<IProps> = ({ | ||||
|     waitingForOtherParty, | ||||
|     waitingForNetwork, | ||||
|     member, | ||||
|  | @ -36,10 +46,10 @@ const EncryptionInfo = ({ | |||
|     isRoomEncrypted, | ||||
|     inDialog, | ||||
|     isSelfVerification, | ||||
| }) => { | ||||
|     let content; | ||||
| }: IProps) => { | ||||
|     let content: JSX.Element; | ||||
|     if (waitingForOtherParty || waitingForNetwork) { | ||||
|         let text; | ||||
|         let text: string; | ||||
|         if (waitingForOtherParty) { | ||||
|             if (isSelfVerification) { | ||||
|                 text = _t("Waiting for you to accept on your other session…"); | ||||
|  | @ -61,7 +71,7 @@ const EncryptionInfo = ({ | |||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     let description; | ||||
|     let description: JSX.Element; | ||||
|     if (isRoomEncrypted) { | ||||
|         description = ( | ||||
|             <div> | ||||
|  | @ -97,10 +107,5 @@ const EncryptionInfo = ({ | |||
|         </div> | ||||
|     </React.Fragment>; | ||||
| }; | ||||
| EncryptionInfo.propTypes = { | ||||
|     member: PropTypes.object.isRequired, | ||||
|     onStartVerification: PropTypes.func.isRequired, | ||||
|     request: PropTypes.object, | ||||
| }; | ||||
| 
 | ||||
| export default EncryptionInfo; | ||||
|  | @ -15,7 +15,6 @@ limitations under the License. | |||
| */ | ||||
| 
 | ||||
| import React, {useCallback, useEffect, useState} from "react"; | ||||
| import PropTypes from "prop-types"; | ||||
| 
 | ||||
| import EncryptionInfo from "./EncryptionInfo"; | ||||
| import VerificationPanel from "./VerificationPanel"; | ||||
|  | @ -26,11 +25,23 @@ import Modal from "../../../Modal"; | |||
| import {PHASE_REQUESTED, PHASE_UNSENT} from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; | ||||
| import * as sdk from "../../../index"; | ||||
| import {_t} from "../../../languageHandler"; | ||||
| import {VerificationRequest} from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; | ||||
| import {RoomMember} from "matrix-js-sdk/src/models/room-member"; | ||||
| 
 | ||||
| // cancellation codes which constitute a key mismatch
 | ||||
| const MISMATCHES = ["m.key_mismatch", "m.user_error", "m.mismatched_sas"]; | ||||
| 
 | ||||
| const EncryptionPanel = (props) => { | ||||
| interface IProps { | ||||
|     member: RoomMember; | ||||
|     onClose: () => void; | ||||
|     verificationRequest: VerificationRequest; | ||||
|     verificationRequestPromise: Promise<VerificationRequest>; | ||||
|     layout: string; | ||||
|     inDialog: boolean; | ||||
|     isRoomEncrypted: boolean; | ||||
| } | ||||
| 
 | ||||
| const EncryptionPanel: React.FC<IProps> = (props: IProps) => { | ||||
|     const {verificationRequest, verificationRequestPromise, member, onClose, layout, isRoomEncrypted} = props; | ||||
|     const [request, setRequest] = useState(verificationRequest); | ||||
|     // state to show a spinner immediately after clicking "start verification",
 | ||||
|  | @ -48,10 +59,10 @@ const EncryptionPanel = (props) => { | |||
|     useEffect(() => { | ||||
|         async function awaitPromise() { | ||||
|             setRequesting(true); | ||||
|             const request = await verificationRequestPromise; | ||||
|             const requestFromPromise = await verificationRequestPromise; | ||||
|             setRequesting(false); | ||||
|             setRequest(request); | ||||
|             setPhase(request.phase); | ||||
|             setRequest(requestFromPromise); | ||||
|             setPhase(requestFromPromise.phase); | ||||
|         } | ||||
|         if (verificationRequestPromise) { | ||||
|             awaitPromise(); | ||||
|  | @ -90,7 +101,7 @@ const EncryptionPanel = (props) => { | |||
|         } | ||||
|     }, [request]); | ||||
| 
 | ||||
|     let cancelButton; | ||||
|     let cancelButton: JSX.Element; | ||||
|     if (layout !== "dialog" && request && request.pending) { | ||||
|         const AccessibleButton = sdk.getComponent("elements.AccessibleButton"); | ||||
|         cancelButton = (<AccessibleButton | ||||
|  | @ -104,9 +115,9 @@ const EncryptionPanel = (props) => { | |||
|         setRequesting(true); | ||||
|         const cli = MatrixClientPeg.get(); | ||||
|         const roomId = await ensureDMExists(cli, member.userId); | ||||
|         const verificationRequest = await cli.requestVerificationDM(member.userId, roomId); | ||||
|         setRequest(verificationRequest); | ||||
|         setPhase(verificationRequest.phase); | ||||
|         const verificationRequest_ = await cli.requestVerificationDM(member.userId, roomId); | ||||
|         setRequest(verificationRequest_); | ||||
|         setPhase(verificationRequest_.phase); | ||||
|     }, [member.userId]); | ||||
| 
 | ||||
|     const requested = | ||||
|  | @ -144,12 +155,5 @@ const EncryptionPanel = (props) => { | |||
|         </React.Fragment>); | ||||
|     } | ||||
| }; | ||||
| EncryptionPanel.propTypes = { | ||||
|     member: PropTypes.object.isRequired, | ||||
|     onClose: PropTypes.func.isRequired, | ||||
|     verificationRequest: PropTypes.object, | ||||
|     layout: PropTypes.string, | ||||
|     inDialog: PropTypes.bool, | ||||
| }; | ||||
| 
 | ||||
| export default EncryptionPanel; | ||||
|  | @ -21,65 +21,68 @@ limitations under the License. | |||
| import React from 'react'; | ||||
| import { _t } from '../../../languageHandler'; | ||||
| import HeaderButton from './HeaderButton'; | ||||
| import HeaderButtons, {HEADER_KIND_GROUP} from './HeaderButtons'; | ||||
| import {RIGHT_PANEL_PHASES} from "../../../stores/RightPanelStorePhases"; | ||||
| import HeaderButtons, {HeaderKind} from './HeaderButtons'; | ||||
| import {RightPanelPhases} from "../../../stores/RightPanelStorePhases"; | ||||
| import {Action} from "../../../dispatcher/actions"; | ||||
| import {ActionPayload} from "../../../dispatcher/payloads"; | ||||
| import {ViewUserPayload} from "../../../dispatcher/payloads/ViewUserPayload"; | ||||
| 
 | ||||
| const GROUP_PHASES = [ | ||||
|     RIGHT_PANEL_PHASES.GroupMemberInfo, | ||||
|     RIGHT_PANEL_PHASES.GroupMemberList, | ||||
|     RightPanelPhases.GroupMemberInfo, | ||||
|     RightPanelPhases.GroupMemberList, | ||||
| ]; | ||||
| const ROOM_PHASES = [ | ||||
|     RIGHT_PANEL_PHASES.GroupRoomList, | ||||
|     RIGHT_PANEL_PHASES.GroupRoomInfo, | ||||
|     RightPanelPhases.GroupRoomList, | ||||
|     RightPanelPhases.GroupRoomInfo, | ||||
| ]; | ||||
| 
 | ||||
| interface IProps {} | ||||
| 
 | ||||
| export default class GroupHeaderButtons extends HeaderButtons { | ||||
|     constructor(props) { | ||||
|         super(props, HEADER_KIND_GROUP); | ||||
|         this._onMembersClicked = this._onMembersClicked.bind(this); | ||||
|         this._onRoomsClicked = this._onRoomsClicked.bind(this); | ||||
|     constructor(props: IProps) { | ||||
|         super(props, HeaderKind.Group); | ||||
|         this.onMembersClicked = this.onMembersClicked.bind(this); | ||||
|         this.onRoomsClicked = this.onRoomsClicked.bind(this); | ||||
|     } | ||||
| 
 | ||||
|     onAction(payload: ActionPayload) { | ||||
|     protected onAction(payload: ActionPayload) { | ||||
|         super.onAction(payload); | ||||
| 
 | ||||
|         if (payload.action === Action.ViewUser) { | ||||
|             if (payload.member) { | ||||
|                 this.setPhase(RIGHT_PANEL_PHASES.RoomMemberInfo, {member: payload.member}); | ||||
|             if ((payload as ViewUserPayload).member) { | ||||
|                 this.setPhase(RightPanelPhases.RoomMemberInfo, {member: payload.member}); | ||||
|             } else { | ||||
|                 this.setPhase(RIGHT_PANEL_PHASES.GroupMemberList); | ||||
|                 this.setPhase(RightPanelPhases.GroupMemberList); | ||||
|             } | ||||
|         } else if (payload.action === "view_group") { | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.GroupMemberList); | ||||
|             this.setPhase(RightPanelPhases.GroupMemberList); | ||||
|         } else if (payload.action === "view_group_room") { | ||||
|             this.setPhase( | ||||
|                 RIGHT_PANEL_PHASES.GroupRoomInfo, | ||||
|                 RightPanelPhases.GroupRoomInfo, | ||||
|                 {groupRoomId: payload.groupRoomId, groupId: payload.groupId}, | ||||
|             ); | ||||
|         } else if (payload.action === "view_group_room_list") { | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.GroupRoomList); | ||||
|             this.setPhase(RightPanelPhases.GroupRoomList); | ||||
|         } else if (payload.action === "view_group_member_list") { | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.GroupMemberList); | ||||
|             this.setPhase(RightPanelPhases.GroupMemberList); | ||||
|         } else if (payload.action === "view_group_user") { | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.GroupMemberInfo, {member: payload.member}); | ||||
|             this.setPhase(RightPanelPhases.GroupMemberInfo, {member: payload.member}); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     _onMembersClicked() { | ||||
|         if (this.state.phase === RIGHT_PANEL_PHASES.GroupMemberInfo) { | ||||
|     private onMembersClicked() { | ||||
|         if (this.state.phase === RightPanelPhases.GroupMemberInfo) { | ||||
|             // send the active phase to trigger a toggle
 | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.GroupMemberInfo); | ||||
|             this.setPhase(RightPanelPhases.GroupMemberInfo); | ||||
|         } else { | ||||
|             // This toggles for us, if needed
 | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.GroupMemberList); | ||||
|             this.setPhase(RightPanelPhases.GroupMemberList); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     _onRoomsClicked() { | ||||
|     private onRoomsClicked() { | ||||
|         // This toggles for us, if needed
 | ||||
|         this.setPhase(RIGHT_PANEL_PHASES.GroupRoomList); | ||||
|         this.setPhase(RightPanelPhases.GroupRoomList); | ||||
|     } | ||||
| 
 | ||||
|     renderButtons() { | ||||
|  | @ -87,13 +90,13 @@ export default class GroupHeaderButtons extends HeaderButtons { | |||
|             <HeaderButton key="groupMembersButton" name="groupMembersButton" | ||||
|                 title={_t('Members')} | ||||
|                 isHighlighted={this.isPhase(GROUP_PHASES)} | ||||
|                 onClick={this._onMembersClicked} | ||||
|                 onClick={this.onMembersClicked} | ||||
|                 analytics={['Right Panel', 'Group Member List Button', 'click']} | ||||
|             />, | ||||
|             <HeaderButton key="roomsButton" name="roomsButton" | ||||
|                 title={_t('Rooms')} | ||||
|                 isHighlighted={this.isPhase(ROOM_PHASES)} | ||||
|                 onClick={this._onRoomsClicked} | ||||
|                 onClick={this.onRoomsClicked} | ||||
|                 analytics={['Right Panel', 'Group Room List Button', 'click']} | ||||
|             />, | ||||
|         ]; | ||||
|  | @ -19,23 +19,38 @@ limitations under the License. | |||
| */ | ||||
| 
 | ||||
| import React from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import classNames from 'classnames'; | ||||
| import Analytics from '../../../Analytics'; | ||||
| import AccessibleTooltipButton from "../elements/AccessibleTooltipButton"; | ||||
| 
 | ||||
| export default class HeaderButton extends React.Component { | ||||
|     constructor() { | ||||
|         super(); | ||||
| interface IProps { | ||||
|     // Whether this button is highlighted
 | ||||
|     isHighlighted: boolean; | ||||
|     // click handler
 | ||||
|     onClick: () => void; | ||||
|     // The badge to display above the icon
 | ||||
|     badge?: React.ReactNode; | ||||
|     // The parameters to track the click event
 | ||||
|     analytics: string[]; | ||||
| 
 | ||||
|     // Button name
 | ||||
|     name: string; | ||||
|     // Button title
 | ||||
|     title: string; | ||||
| } | ||||
| 
 | ||||
| export default class HeaderButton extends React.Component<IProps> { | ||||
|     constructor(props: IProps) { | ||||
|         super(props); | ||||
|         this.onClick = this.onClick.bind(this); | ||||
|     } | ||||
| 
 | ||||
|     onClick(ev) { | ||||
|     private onClick() { | ||||
|         Analytics.trackEvent(...this.props.analytics); | ||||
|         this.props.onClick(); | ||||
|     } | ||||
| 
 | ||||
|     render() { | ||||
|     public render() { | ||||
|         const classes = classNames({ | ||||
|             mx_RightPanel_headerButton: true, | ||||
|             mx_RightPanel_headerButton_highlight: this.props.isHighlighted, | ||||
|  | @ -51,19 +66,3 @@ export default class HeaderButton extends React.Component { | |||
|         />; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| HeaderButton.propTypes = { | ||||
|     // Whether this button is highlighted
 | ||||
|     isHighlighted: PropTypes.bool.isRequired, | ||||
|     // click handler
 | ||||
|     onClick: PropTypes.func.isRequired, | ||||
|     // The badge to display above the icon
 | ||||
|     badge: PropTypes.node, | ||||
|     // The parameters to track the click event
 | ||||
|     analytics: PropTypes.arrayOf(PropTypes.string).isRequired, | ||||
| 
 | ||||
|     // Button name
 | ||||
|     name: PropTypes.string.isRequired, | ||||
|     // Button title
 | ||||
|     title: PropTypes.string.isRequired, | ||||
| }; | ||||
|  | @ -1,88 +0,0 @@ | |||
| /* | ||||
| Copyright 2015, 2016 OpenMarket Ltd | ||||
| Copyright 2017 Vector Creations Ltd | ||||
| Copyright 2017 New Vector Ltd | ||||
| Copyright 2018 New Vector Ltd | ||||
| Copyright 2019 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 dis from '../../../dispatcher/dispatcher'; | ||||
| import RightPanelStore from "../../../stores/RightPanelStore"; | ||||
| 
 | ||||
| export const HEADER_KIND_ROOM = "room"; | ||||
| export const HEADER_KIND_GROUP = "group"; | ||||
| 
 | ||||
| const HEADER_KINDS = [HEADER_KIND_GROUP, HEADER_KIND_ROOM]; | ||||
| 
 | ||||
| export default class HeaderButtons extends React.Component { | ||||
|     constructor(props, kind) { | ||||
|         super(props); | ||||
| 
 | ||||
|         if (!HEADER_KINDS.includes(kind)) throw new Error(`Invalid header kind: ${kind}`); | ||||
| 
 | ||||
|         const rps = RightPanelStore.getSharedInstance(); | ||||
|         this.state = { | ||||
|             headerKind: kind, | ||||
|             phase: kind === HEADER_KIND_ROOM ? rps.visibleRoomPanelPhase : rps.visibleGroupPanelPhase, | ||||
|         }; | ||||
|     } | ||||
| 
 | ||||
|     componentDidMount() { | ||||
|         this._storeToken = RightPanelStore.getSharedInstance().addListener(this.onRightPanelUpdate.bind(this)); | ||||
|         this._dispatcherRef = dis.register(this.onAction.bind(this)); // used by subclasses
 | ||||
|     } | ||||
| 
 | ||||
|     componentWillUnmount() { | ||||
|         if (this._storeToken) this._storeToken.remove(); | ||||
|         if (this._dispatcherRef) dis.unregister(this._dispatcherRef); | ||||
|     } | ||||
| 
 | ||||
|     onAction(payload) { | ||||
|         // Ignore - intended to be overridden by subclasses
 | ||||
|     } | ||||
| 
 | ||||
|     setPhase(phase, extras) { | ||||
|         dis.dispatch({ | ||||
|             action: 'set_right_panel_phase', | ||||
|             phase: phase, | ||||
|             refireParams: extras, | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     isPhase(phases: string | string[]) { | ||||
|         if (Array.isArray(phases)) { | ||||
|             return phases.includes(this.state.phase); | ||||
|         } else { | ||||
|             return phases === this.state.phase; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     onRightPanelUpdate() { | ||||
|         const rps = RightPanelStore.getSharedInstance(); | ||||
|         if (this.state.headerKind === HEADER_KIND_ROOM) { | ||||
|             this.setState({phase: rps.visibleRoomPanelPhase}); | ||||
|         } else if (this.state.headerKind === HEADER_KIND_GROUP) { | ||||
|             this.setState({phase: rps.visibleGroupPanelPhase}); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     render() { | ||||
|         // inline style as this will be swapped around in future commits
 | ||||
|         return <div className="mx_HeaderButtons" role="tablist"> | ||||
|             {this.renderButtons()} | ||||
|         </div>; | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,110 @@ | |||
| /* | ||||
| Copyright 2015, 2016 OpenMarket Ltd | ||||
| Copyright 2017 Vector Creations Ltd | ||||
| Copyright 2017 New Vector Ltd | ||||
| Copyright 2018 New Vector Ltd | ||||
| Copyright 2019 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 dis from '../../../dispatcher/dispatcher'; | ||||
| import RightPanelStore from "../../../stores/RightPanelStore"; | ||||
| import {RightPanelPhases} from "../../../stores/RightPanelStorePhases"; | ||||
| import {Action} from '../../../dispatcher/actions'; | ||||
| import {SetRightPanelPhasePayload, SetRightPanelPhaseRefireParams} from '../../../dispatcher/payloads/SetRightPanelPhasePayload'; | ||||
| import {EventSubscription} from "fbemitter"; | ||||
| 
 | ||||
| export enum HeaderKind { | ||||
|   Room = "room", | ||||
|   Group = "group", | ||||
| } | ||||
| 
 | ||||
| interface IState { | ||||
|     headerKind: HeaderKind; | ||||
|     phase: RightPanelPhases; | ||||
| } | ||||
| 
 | ||||
| interface IProps {} | ||||
| 
 | ||||
| export default class HeaderButtons extends React.Component<IProps, IState> { | ||||
|     private storeToken: EventSubscription; | ||||
|     private dispatcherRef: string; | ||||
| 
 | ||||
|     constructor(props: IProps, kind: HeaderKind) { | ||||
|         super(props); | ||||
| 
 | ||||
|         const rps = RightPanelStore.getSharedInstance(); | ||||
|         this.state = { | ||||
|             headerKind: kind, | ||||
|             phase: kind === HeaderKind.Room ? rps.visibleRoomPanelPhase : rps.visibleGroupPanelPhase, | ||||
|         }; | ||||
|     } | ||||
| 
 | ||||
|     public componentDidMount() { | ||||
|         this.storeToken = RightPanelStore.getSharedInstance().addListener(this.onRightPanelUpdate.bind(this)); | ||||
|         this.dispatcherRef = dis.register(this.onAction.bind(this)); // used by subclasses
 | ||||
|     } | ||||
| 
 | ||||
|     public componentWillUnmount() { | ||||
|         if (this.storeToken) this.storeToken.remove(); | ||||
|         if (this.dispatcherRef) dis.unregister(this.dispatcherRef); | ||||
|     } | ||||
| 
 | ||||
|     protected onAction(payload) { | ||||
|         // Ignore - intended to be overridden by subclasses
 | ||||
|     } | ||||
| 
 | ||||
|     public setPhase(phase: RightPanelPhases, extras?: Partial<SetRightPanelPhaseRefireParams>) { | ||||
|         dis.dispatch<SetRightPanelPhasePayload>({ | ||||
|             action: Action.SetRightPanelPhase, | ||||
|             phase: phase, | ||||
|             refireParams: extras, | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     public isPhase(phases: string | string[]) { | ||||
|         if (Array.isArray(phases)) { | ||||
|             return phases.includes(this.state.phase); | ||||
|         } else { | ||||
|             return phases === this.state.phase; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private onRightPanelUpdate() { | ||||
|         const rps = RightPanelStore.getSharedInstance(); | ||||
|         if (this.state.headerKind === HeaderKind.Room) { | ||||
|             this.setState({phase: rps.visibleRoomPanelPhase}); | ||||
|         } else if (this.state.headerKind === HeaderKind.Group) { | ||||
|             this.setState({phase: rps.visibleGroupPanelPhase}); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // XXX: Make renderButtons a prop
 | ||||
|     public renderButtons(): JSX.Element[] { | ||||
|         // Ignore - intended to be overridden by subclasses
 | ||||
|         // Return empty fragment to satisfy the type
 | ||||
|         return [ | ||||
|           <React.Fragment> | ||||
|           </React.Fragment> | ||||
|         ]; | ||||
|     } | ||||
| 
 | ||||
|     public render() { | ||||
|         // inline style as this will be swapped around in future commits
 | ||||
|         return <div className="mx_HeaderButtons" role="tablist"> | ||||
|             {this.renderButtons()} | ||||
|         </div>; | ||||
|     } | ||||
| } | ||||
|  | @ -21,82 +21,82 @@ limitations under the License. | |||
| import React from 'react'; | ||||
| import { _t } from '../../../languageHandler'; | ||||
| import HeaderButton from './HeaderButton'; | ||||
| import HeaderButtons, {HEADER_KIND_ROOM} from './HeaderButtons'; | ||||
| import {RIGHT_PANEL_PHASES} from "../../../stores/RightPanelStorePhases"; | ||||
| import HeaderButtons, {HeaderKind} from './HeaderButtons'; | ||||
| import {RightPanelPhases} from "../../../stores/RightPanelStorePhases"; | ||||
| import {Action} from "../../../dispatcher/actions"; | ||||
| import {ActionPayload} from "../../../dispatcher/payloads"; | ||||
| 
 | ||||
| const MEMBER_PHASES = [ | ||||
|     RIGHT_PANEL_PHASES.RoomMemberList, | ||||
|     RIGHT_PANEL_PHASES.RoomMemberInfo, | ||||
|     RIGHT_PANEL_PHASES.EncryptionPanel, | ||||
|     RIGHT_PANEL_PHASES.Room3pidMemberInfo, | ||||
|     RightPanelPhases.RoomMemberList, | ||||
|     RightPanelPhases.RoomMemberInfo, | ||||
|     RightPanelPhases.EncryptionPanel, | ||||
|     RightPanelPhases.Room3pidMemberInfo, | ||||
| ]; | ||||
| 
 | ||||
| export default class RoomHeaderButtons extends HeaderButtons { | ||||
|     constructor(props) { | ||||
|         super(props, HEADER_KIND_ROOM); | ||||
|         this._onMembersClicked = this._onMembersClicked.bind(this); | ||||
|         this._onFilesClicked = this._onFilesClicked.bind(this); | ||||
|         this._onNotificationsClicked = this._onNotificationsClicked.bind(this); | ||||
|         super(props, HeaderKind.Room); | ||||
|         this.onMembersClicked = this.onMembersClicked.bind(this); | ||||
|         this.onFilesClicked = this.onFilesClicked.bind(this); | ||||
|         this.onNotificationsClicked = this.onNotificationsClicked.bind(this); | ||||
|     } | ||||
| 
 | ||||
|     onAction(payload: ActionPayload) { | ||||
|     protected onAction(payload: ActionPayload) { | ||||
|         super.onAction(payload); | ||||
|         if (payload.action === Action.ViewUser) { | ||||
|             if (payload.member) { | ||||
|                 this.setPhase(RIGHT_PANEL_PHASES.RoomMemberInfo, {member: payload.member}); | ||||
|                 this.setPhase(RightPanelPhases.RoomMemberInfo, {member: payload.member}); | ||||
|             } else { | ||||
|                 this.setPhase(RIGHT_PANEL_PHASES.RoomMemberList); | ||||
|                 this.setPhase(RightPanelPhases.RoomMemberList); | ||||
|             } | ||||
|         } else if (payload.action === "view_3pid_invite") { | ||||
|             if (payload.event) { | ||||
|                 this.setPhase(RIGHT_PANEL_PHASES.Room3pidMemberInfo, {event: payload.event}); | ||||
|                 this.setPhase(RightPanelPhases.Room3pidMemberInfo, {event: payload.event}); | ||||
|             } else { | ||||
|                 this.setPhase(RIGHT_PANEL_PHASES.RoomMemberList); | ||||
|                 this.setPhase(RightPanelPhases.RoomMemberList); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     _onMembersClicked() { | ||||
|         if (this.state.phase === RIGHT_PANEL_PHASES.RoomMemberInfo) { | ||||
|     private onMembersClicked() { | ||||
|         if (this.state.phase === RightPanelPhases.RoomMemberInfo) { | ||||
|             // send the active phase to trigger a toggle
 | ||||
|             // XXX: we should pass refireParams here but then it won't collapse as we desire it to
 | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.RoomMemberInfo); | ||||
|             this.setPhase(RightPanelPhases.RoomMemberInfo); | ||||
|         } else { | ||||
|             // This toggles for us, if needed
 | ||||
|             this.setPhase(RIGHT_PANEL_PHASES.RoomMemberList); | ||||
|             this.setPhase(RightPanelPhases.RoomMemberList); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     _onFilesClicked() { | ||||
|     private onFilesClicked() { | ||||
|         // This toggles for us, if needed
 | ||||
|         this.setPhase(RIGHT_PANEL_PHASES.FilePanel); | ||||
|         this.setPhase(RightPanelPhases.FilePanel); | ||||
|     } | ||||
| 
 | ||||
|     _onNotificationsClicked() { | ||||
|     private onNotificationsClicked() { | ||||
|         // This toggles for us, if needed
 | ||||
|         this.setPhase(RIGHT_PANEL_PHASES.NotificationPanel); | ||||
|         this.setPhase(RightPanelPhases.NotificationPanel); | ||||
|     } | ||||
| 
 | ||||
|     renderButtons() { | ||||
|     public renderButtons() { | ||||
|         return [ | ||||
|             <HeaderButton key="membersButton" name="membersButton" | ||||
|                 title={_t('Members')} | ||||
|                 isHighlighted={this.isPhase(MEMBER_PHASES)} | ||||
|                 onClick={this._onMembersClicked} | ||||
|                 onClick={this.onMembersClicked} | ||||
|                 analytics={['Right Panel', 'Member List Button', 'click']} | ||||
|             />, | ||||
|             <HeaderButton key="filesButton" name="filesButton" | ||||
|                 title={_t('Files')} | ||||
|                 isHighlighted={this.isPhase(RIGHT_PANEL_PHASES.FilePanel)} | ||||
|                 onClick={this._onFilesClicked} | ||||
|                 isHighlighted={this.isPhase(RightPanelPhases.FilePanel)} | ||||
|                 onClick={this.onFilesClicked} | ||||
|                 analytics={['Right Panel', 'File List Button', 'click']} | ||||
|             />, | ||||
|             <HeaderButton key="notifsButton" name="notifsButton" | ||||
|                 title={_t('Notifications')} | ||||
|                 isHighlighted={this.isPhase(RIGHT_PANEL_PHASES.NotificationPanel)} | ||||
|                 onClick={this._onNotificationsClicked} | ||||
|                 isHighlighted={this.isPhase(RightPanelPhases.NotificationPanel)} | ||||
|                 onClick={this.onNotificationsClicked} | ||||
|                 analytics={['Right Panel', 'Notification List Button', 'click']} | ||||
|             />, | ||||
|         ]; | ||||
|  | @ -40,7 +40,7 @@ import E2EIcon from "../rooms/E2EIcon"; | |||
| import {useEventEmitter} from "../../../hooks/useEventEmitter"; | ||||
| import {textualPowerLevel} from '../../../Roles'; | ||||
| import MatrixClientContext from "../../../contexts/MatrixClientContext"; | ||||
| import {RIGHT_PANEL_PHASES} from "../../../stores/RightPanelStorePhases"; | ||||
| import {RightPanelPhases} from "../../../stores/RightPanelStorePhases"; | ||||
| import EncryptionPanel from "./EncryptionPanel"; | ||||
| import { useAsyncMemo } from '../../../hooks/useAsyncMemo'; | ||||
| import { verifyUser, legacyVerifyUser, verifyDevice } from '../../../verification'; | ||||
|  | @ -1480,7 +1480,7 @@ const UserInfoHeader = ({onClose, member, e2eStatus}) => { | |||
|     </React.Fragment>; | ||||
| }; | ||||
| 
 | ||||
| const UserInfo = ({user, groupId, roomId, onClose, phase=RIGHT_PANEL_PHASES.RoomMemberInfo, ...props}) => { | ||||
| const UserInfo = ({user, groupId, roomId, onClose, phase=RightPanelPhases.RoomMemberInfo, ...props}) => { | ||||
|     const cli = useContext(MatrixClientContext); | ||||
| 
 | ||||
|     // Load room if we are given a room id and memoize it
 | ||||
|  | @ -1500,8 +1500,8 @@ const UserInfo = ({user, groupId, roomId, onClose, phase=RIGHT_PANEL_PHASES.Room | |||
| 
 | ||||
|     let content; | ||||
|     switch (phase) { | ||||
|         case RIGHT_PANEL_PHASES.RoomMemberInfo: | ||||
|         case RIGHT_PANEL_PHASES.GroupMemberInfo: | ||||
|         case RightPanelPhases.RoomMemberInfo: | ||||
|         case RightPanelPhases.GroupMemberInfo: | ||||
|             content = ( | ||||
|                 <BasicUserInfo | ||||
|                     room={room} | ||||
|  | @ -1511,7 +1511,7 @@ const UserInfo = ({user, groupId, roomId, onClose, phase=RIGHT_PANEL_PHASES.Room | |||
|                     isRoomEncrypted={isRoomEncrypted} /> | ||||
|             ); | ||||
|             break; | ||||
|         case RIGHT_PANEL_PHASES.EncryptionPanel: | ||||
|         case RightPanelPhases.EncryptionPanel: | ||||
|             classes.push("mx_UserInfo_smallAvatar"); | ||||
|             content = ( | ||||
|                 <EncryptionPanel {...props} member={member} onClose={onClose} isRoomEncrypted={isRoomEncrypted} /> | ||||
|  |  | |||
|  | @ -15,12 +15,15 @@ limitations under the License. | |||
| */ | ||||
| 
 | ||||
| import React from "react"; | ||||
| import PropTypes from "prop-types"; | ||||
| 
 | ||||
| import {MatrixClientPeg} from "../../../MatrixClientPeg"; | ||||
| import * as sdk from '../../../index'; | ||||
| import {verificationMethods} from 'matrix-js-sdk/src/crypto'; | ||||
| import {SCAN_QR_CODE_METHOD} from "matrix-js-sdk/src/crypto/verification/QRCode"; | ||||
| import {VerificationRequest} from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; | ||||
| import {RoomMember} from "matrix-js-sdk/src/models/room-member"; | ||||
| import {ReciprocateQRCode} from "matrix-js-sdk/src/crypto/verification/QRCode"; | ||||
| import {SAS} from "matrix-js-sdk/src/crypto/verification/SAS"; | ||||
| 
 | ||||
| import VerificationQRCode from "../elements/crypto/VerificationQRCode"; | ||||
| import {_t} from "../../../languageHandler"; | ||||
|  | @ -36,37 +39,51 @@ import { | |||
| } from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; | ||||
| import Spinner from "../elements/Spinner"; | ||||
| 
 | ||||
| export default class VerificationPanel extends React.PureComponent { | ||||
|     static propTypes = { | ||||
|         layout: PropTypes.string, | ||||
|         request: PropTypes.object.isRequired, | ||||
|         member: PropTypes.object.isRequired, | ||||
|         phase: PropTypes.oneOf([ | ||||
|             PHASE_UNSENT, | ||||
|             PHASE_REQUESTED, | ||||
|             PHASE_READY, | ||||
|             PHASE_STARTED, | ||||
|             PHASE_CANCELLED, | ||||
|             PHASE_DONE, | ||||
|         ]).isRequired, | ||||
|         onClose: PropTypes.func.isRequired, | ||||
|         isRoomEncrypted: PropTypes.bool, | ||||
|     }; | ||||
| // XXX: Should be defined in matrix-js-sdk
 | ||||
| enum VerificationPhase { | ||||
|     PHASE_UNSENT, | ||||
|     PHASE_REQUESTED, | ||||
|     PHASE_READY, | ||||
|     PHASE_DONE, | ||||
|     PHASE_STARTED, | ||||
|     PHASE_CANCELLED, | ||||
| } | ||||
| 
 | ||||
|     constructor(props) { | ||||
| interface IProps { | ||||
|     layout: string; | ||||
|     request: VerificationRequest; | ||||
|     member: RoomMember; | ||||
|     phase: VerificationPhase; | ||||
|     onClose: () => void; | ||||
|     isRoomEncrypted: boolean; | ||||
|     inDialog: boolean; | ||||
|     key: number; | ||||
| } | ||||
| 
 | ||||
| interface IState { | ||||
|     sasEvent?: SAS; | ||||
|     emojiButtonClicked?: boolean; | ||||
|     reciprocateButtonClicked?: boolean; | ||||
|     reciprocateQREvent?: ReciprocateQRCode; | ||||
| } | ||||
| 
 | ||||
| export default class VerificationPanel extends React.PureComponent<IProps, IState> { | ||||
|     private hasVerifier: boolean; | ||||
| 
 | ||||
|     constructor(props: IProps) { | ||||
|         super(props); | ||||
|         this.state = {}; | ||||
|         this._hasVerifier = false; | ||||
|         this.hasVerifier = false; | ||||
|     } | ||||
| 
 | ||||
|     renderQRPhase() { | ||||
|     private renderQRPhase() { | ||||
|         const {member, request} = this.props; | ||||
|         const showSAS = request.otherPartySupportsMethod(verificationMethods.SAS); | ||||
|         const showQR = request.otherPartySupportsMethod(SCAN_QR_CODE_METHOD); | ||||
|         const showSAS: boolean = request.otherPartySupportsMethod(verificationMethods.SAS); | ||||
|         const showQR: boolean = request.otherPartySupportsMethod(SCAN_QR_CODE_METHOD); | ||||
|         const AccessibleButton = sdk.getComponent('elements.AccessibleButton'); | ||||
|         const brand = SdkConfig.get().brand; | ||||
| 
 | ||||
|         const noCommonMethodError = !showSAS && !showQR ? | ||||
|         const noCommonMethodError: JSX.Element = !showSAS && !showQR ? | ||||
|             <p>{_t( | ||||
|                 "The session you are trying to verify doesn't support scanning a " + | ||||
|                 "QR code or emoji verification, which is what %(brand)s supports. Try " + | ||||
|  | @ -77,41 +94,41 @@ export default class VerificationPanel extends React.PureComponent { | |||
| 
 | ||||
|         if (this.props.layout === 'dialog') { | ||||
|             // HACK: This is a terrible idea.
 | ||||
|             let qrBlock; | ||||
|             let sasBlock; | ||||
|             let qrBlockDialog: JSX.Element; | ||||
|             let sasBlockDialog: JSX.Element; | ||||
|             if (showQR) { | ||||
|                 qrBlock = | ||||
|                 qrBlockDialog = | ||||
|                     <div className='mx_VerificationPanel_QRPhase_startOption'> | ||||
|                         <p>{_t("Scan this unique code")}</p> | ||||
|                         <VerificationQRCode qrCodeData={request.qrCodeData} /> | ||||
|                     </div>; | ||||
|             } | ||||
|             if (showSAS) { | ||||
|                 sasBlock = | ||||
|                 sasBlockDialog = | ||||
|                     <div className='mx_VerificationPanel_QRPhase_startOption'> | ||||
|                         <p>{_t("Compare unique emoji")}</p> | ||||
|                         <span className='mx_VerificationPanel_QRPhase_helpText'>{_t("Compare a unique set of emoji if you don't have a camera on either device")}</span> | ||||
|                         <AccessibleButton disabled={this.state.emojiButtonClicked} onClick={this._startSAS} kind='primary'> | ||||
|                         <AccessibleButton disabled={this.state.emojiButtonClicked} onClick={this.startSAS} kind='primary'> | ||||
|                             {_t("Start")} | ||||
|                         </AccessibleButton> | ||||
|                     </div>; | ||||
|             } | ||||
|             const or = qrBlock && sasBlock ? | ||||
|             const or = qrBlockDialog && sasBlockDialog ? | ||||
|                 <div className='mx_VerificationPanel_QRPhase_betweenText'>{_t("or")}</div> : null; | ||||
|             return ( | ||||
|                 <div> | ||||
|                     {_t("Verify this session by completing one of the following:")} | ||||
|                     <div className='mx_VerificationPanel_QRPhase_startOptions'> | ||||
|                         {qrBlock} | ||||
|                         {qrBlockDialog} | ||||
|                         {or} | ||||
|                         {sasBlock} | ||||
|                         {sasBlockDialog} | ||||
|                         {noCommonMethodError} | ||||
|                     </div> | ||||
|                 </div> | ||||
|             ); | ||||
|         } | ||||
| 
 | ||||
|         let qrBlock; | ||||
|         let qrBlock: JSX.Element; | ||||
|         if (showQR) { | ||||
|             qrBlock = <div className="mx_UserInfo_container"> | ||||
|                 <h3>{_t("Verify by scanning")}</h3> | ||||
|  | @ -125,7 +142,7 @@ export default class VerificationPanel extends React.PureComponent { | |||
|             </div>; | ||||
|         } | ||||
| 
 | ||||
|         let sasBlock; | ||||
|         let sasBlock: JSX.Element; | ||||
|         if (showSAS) { | ||||
|             const disabled = this.state.emojiButtonClicked; | ||||
|             const sasLabel = showQR ? | ||||
|  | @ -140,7 +157,7 @@ export default class VerificationPanel extends React.PureComponent { | |||
|                     disabled={disabled} | ||||
|                     kind="primary" | ||||
|                     className="mx_UserInfo_wideButton mx_VerificationPanel_verifyByEmojiButton" | ||||
|                     onClick={this._startSAS} | ||||
|                     onClick={this.startSAS} | ||||
|                 > | ||||
|                     {_t("Verify by emoji")} | ||||
|                 </AccessibleButton> | ||||
|  | @ -159,22 +176,22 @@ export default class VerificationPanel extends React.PureComponent { | |||
|         </React.Fragment>; | ||||
|     } | ||||
| 
 | ||||
|     _onReciprocateYesClick = () => { | ||||
|     private onReciprocateYesClick = () => { | ||||
|         this.setState({reciprocateButtonClicked: true}); | ||||
|         this.state.reciprocateQREvent.confirm(); | ||||
|     }; | ||||
| 
 | ||||
|     _onReciprocateNoClick = () => { | ||||
|     private onReciprocateNoClick = () => { | ||||
|         this.setState({reciprocateButtonClicked: true}); | ||||
|         this.state.reciprocateQREvent.cancel(); | ||||
|     }; | ||||
| 
 | ||||
|     _getDevice() { | ||||
|     private getDevice() { | ||||
|         const deviceId = this.props.request && this.props.request.channel.deviceId; | ||||
|         return MatrixClientPeg.get().getStoredDevice(MatrixClientPeg.get().getUserId(), deviceId); | ||||
|     } | ||||
| 
 | ||||
|     renderQRReciprocatePhase() { | ||||
|     private renderQRReciprocatePhase() { | ||||
|         const {member, request} = this.props; | ||||
|         let Button; | ||||
|         // a bit of a hack, but the FormButton should only be used in the right panel
 | ||||
|  | @ -189,7 +206,7 @@ export default class VerificationPanel extends React.PureComponent { | |||
|             _t("Almost there! Is %(displayName)s showing the same shield?", { | ||||
|                 displayName: member.displayName || member.name || member.userId, | ||||
|             }); | ||||
|         let body; | ||||
|         let body: JSX.Element; | ||||
|         if (this.state.reciprocateQREvent) { | ||||
|             // riot web doesn't support scanning yet, so assume here we're the client being scanned.
 | ||||
|             //
 | ||||
|  | @ -202,11 +219,11 @@ export default class VerificationPanel extends React.PureComponent { | |||
|                     <Button | ||||
|                         label={_t("No")} kind="danger" | ||||
|                         disabled={this.state.reciprocateButtonClicked} | ||||
|                         onClick={this._onReciprocateNoClick}>{_t("No")}</Button> | ||||
|                         onClick={this.onReciprocateNoClick}>{_t("No")}</Button> | ||||
|                     <Button | ||||
|                         label={_t("Yes")} kind="primary" | ||||
|                         disabled={this.state.reciprocateButtonClicked} | ||||
|                         onClick={this._onReciprocateYesClick}>{_t("Yes")}</Button> | ||||
|                         onClick={this.onReciprocateYesClick}>{_t("Yes")}</Button> | ||||
|                 </div> | ||||
|             </React.Fragment>; | ||||
|         } else { | ||||
|  | @ -218,10 +235,10 @@ export default class VerificationPanel extends React.PureComponent { | |||
|         </div>; | ||||
|     } | ||||
| 
 | ||||
|     renderVerifiedPhase() { | ||||
|     private renderVerifiedPhase() { | ||||
|         const {member, request} = this.props; | ||||
| 
 | ||||
|         let text; | ||||
|         let text: string; | ||||
|         if (!request.isSelfVerification) { | ||||
|             if (this.props.isRoomEncrypted) { | ||||
|                 text = _t("Verify all users in a room to ensure it's secure."); | ||||
|  | @ -230,9 +247,9 @@ export default class VerificationPanel extends React.PureComponent { | |||
|             } | ||||
|         } | ||||
| 
 | ||||
|         let description; | ||||
|         let description: string; | ||||
|         if (request.isSelfVerification) { | ||||
|             const device = this._getDevice(); | ||||
|             const device = this.getDevice(); | ||||
|             if (!device) { | ||||
|                 // This can happen if the device is logged out while we're still showing verification
 | ||||
|                 // UI for it.
 | ||||
|  | @ -264,19 +281,19 @@ export default class VerificationPanel extends React.PureComponent { | |||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     renderCancelledPhase() { | ||||
|     private renderCancelledPhase() { | ||||
|         const {member, request} = this.props; | ||||
| 
 | ||||
|         const AccessibleButton = sdk.getComponent('elements.AccessibleButton'); | ||||
| 
 | ||||
|         let startAgainInstruction; | ||||
|         let startAgainInstruction: string; | ||||
|         if (request.isSelfVerification) { | ||||
|             startAgainInstruction = _t("Start verification again from the notification."); | ||||
|         } else { | ||||
|             startAgainInstruction = _t("Start verification again from their profile."); | ||||
|         } | ||||
| 
 | ||||
|         let text; | ||||
|         let text: string; | ||||
|         if (request.cancellationCode === "m.timeout") { | ||||
|             text = _t("Verification timed out.") + ` ${startAgainInstruction}`; | ||||
|         } else if (request.cancellingUserId === request.otherUserId) { | ||||
|  | @ -304,7 +321,7 @@ export default class VerificationPanel extends React.PureComponent { | |||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     render() { | ||||
|     public render() { | ||||
|         const {member, phase, request} = this.props; | ||||
| 
 | ||||
|         const displayName = member.displayName || member.name || member.userId; | ||||
|  | @ -321,10 +338,10 @@ export default class VerificationPanel extends React.PureComponent { | |||
|                         const emojis = this.state.sasEvent ? | ||||
|                             <VerificationShowSas | ||||
|                                 displayName={displayName} | ||||
|                                 device={this._getDevice()} | ||||
|                                 device={this.getDevice()} | ||||
|                                 sas={this.state.sasEvent.sas} | ||||
|                                 onCancel={this._onSasMismatchesClick} | ||||
|                                 onDone={this._onSasMatchesClick} | ||||
|                                 onCancel={this.onSasMismatchesClick} | ||||
|                                 onDone={this.onSasMatchesClick} | ||||
|                                 inDialog={this.props.inDialog} | ||||
|                                 isSelf={request.isSelfVerification} | ||||
|                             /> : <Spinner />; | ||||
|  | @ -345,7 +362,7 @@ export default class VerificationPanel extends React.PureComponent { | |||
|         return null; | ||||
|     } | ||||
| 
 | ||||
|     _startSAS = async () => { | ||||
|     private startSAS = async () => { | ||||
|         this.setState({emojiButtonClicked: true}); | ||||
|         const verifier = this.props.request.beginKeyVerification(verificationMethods.SAS); | ||||
|         try { | ||||
|  | @ -355,31 +372,31 @@ export default class VerificationPanel extends React.PureComponent { | |||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     _onSasMatchesClick = () => { | ||||
|     private onSasMatchesClick = () => { | ||||
|         this.state.sasEvent.confirm(); | ||||
|     }; | ||||
| 
 | ||||
|     _onSasMismatchesClick = () => { | ||||
|     private onSasMismatchesClick = () => { | ||||
|         this.state.sasEvent.mismatch(); | ||||
|     }; | ||||
| 
 | ||||
|     _updateVerifierState = () => { | ||||
|     private updateVerifierState = () => { | ||||
|         const {request} = this.props; | ||||
|         const {sasEvent, reciprocateQREvent} = request.verifier; | ||||
|         request.verifier.off('show_sas', this._updateVerifierState); | ||||
|         request.verifier.off('show_reciprocate_qr', this._updateVerifierState); | ||||
|         request.verifier.off('show_sas', this.updateVerifierState); | ||||
|         request.verifier.off('show_reciprocate_qr', this.updateVerifierState); | ||||
|         this.setState({sasEvent, reciprocateQREvent}); | ||||
|     }; | ||||
| 
 | ||||
|     _onRequestChange = async () => { | ||||
|     private onRequestChange = async () => { | ||||
|         const {request} = this.props; | ||||
|         const hadVerifier = this._hasVerifier; | ||||
|         this._hasVerifier = !!request.verifier; | ||||
|         if (!hadVerifier && this._hasVerifier) { | ||||
|             request.verifier.on('show_sas', this._updateVerifierState); | ||||
|             request.verifier.on('show_reciprocate_qr', this._updateVerifierState); | ||||
|         const hadVerifier = this.hasVerifier; | ||||
|         this.hasVerifier = !!request.verifier; | ||||
|         if (!hadVerifier && this.hasVerifier) { | ||||
|             request.verifier.on('show_sas', this.updateVerifierState); | ||||
|             request.verifier.on('show_reciprocate_qr', this.updateVerifierState); | ||||
|             try { | ||||
|                 // on the requester side, this is also awaited in _startSAS,
 | ||||
|                 // on the requester side, this is also awaited in startSAS,
 | ||||
|                 // but that's ok as verify should return the same promise.
 | ||||
|                 await request.verifier.verify(); | ||||
|             } catch (err) { | ||||
|  | @ -388,23 +405,22 @@ export default class VerificationPanel extends React.PureComponent { | |||
|         } | ||||
|     }; | ||||
| 
 | ||||
|     componentDidMount() { | ||||
|     public componentDidMount() { | ||||
|         const {request} = this.props; | ||||
|         request.on("change", this._onRequestChange); | ||||
|         request.on("change", this.onRequestChange); | ||||
|         if (request.verifier) { | ||||
|             const {request} = this.props; | ||||
|             const {sasEvent, reciprocateQREvent} = request.verifier; | ||||
|             this.setState({sasEvent, reciprocateQREvent}); | ||||
|         } | ||||
|         this._onRequestChange(); | ||||
|         this.onRequestChange(); | ||||
|     } | ||||
| 
 | ||||
|     componentWillUnmount() { | ||||
|     public componentWillUnmount() { | ||||
|         const {request} = this.props; | ||||
|         if (request.verifier) { | ||||
|             request.verifier.off('show_sas', this._updateVerifierState); | ||||
|             request.verifier.off('show_reciprocate_qr', this._updateVerifierState); | ||||
|             request.verifier.off('show_sas', this.updateVerifierState); | ||||
|             request.verifier.off('show_reciprocate_qr', this.updateVerifierState); | ||||
|         } | ||||
|         request.off("change", this._onRequestChange); | ||||
|         request.off("change", this.onRequestChange); | ||||
|     } | ||||
| } | ||||
|  | @ -20,7 +20,8 @@ import createReactClass from 'create-react-class'; | |||
| 
 | ||||
| import Tinter from '../../../Tinter'; | ||||
| import dis from '../../../dispatcher/dispatcher'; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| const ROOM_COLORS = [ | ||||
|     // magic room default values courtesy of Ribot
 | ||||
|  |  | |||
|  | @ -22,10 +22,11 @@ import PropTypes from 'prop-types'; | |||
| import createReactClass from 'create-react-class'; | ||||
| import * as sdk from "../../../index"; | ||||
| import { _t, _td } from '../../../languageHandler'; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import dis from "../../../dispatcher/dispatcher"; | ||||
| import {MatrixClientPeg} from "../../../MatrixClientPeg"; | ||||
| import {Action} from "../../../dispatcher/actions"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| 
 | ||||
| export default createReactClass({ | ||||
|  |  | |||
|  | @ -42,7 +42,7 @@ import { RoomNotificationStateStore } from "../../../stores/notifications/RoomNo | |||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import CustomRoomTagStore from "../../../stores/CustomRoomTagStore"; | ||||
| import { arrayFastClone, arrayHasDiff } from "../../../utils/arrays"; | ||||
| import { objectShallowClone } from "../../../utils/objects"; | ||||
| import { objectShallowClone, objectWithOnly } from "../../../utils/objects"; | ||||
| 
 | ||||
| interface IProps { | ||||
|     onKeyDown: (ev: React.KeyboardEvent) => void; | ||||
|  | @ -220,7 +220,12 @@ export default class RoomList extends React.PureComponent<IProps, IState> { | |||
|         } | ||||
| 
 | ||||
|         const previousListIds = Object.keys(this.state.sublists); | ||||
|         const newListIds = Object.keys(newLists); | ||||
|         const newListIds = Object.keys(newLists).filter(t => { | ||||
|             if (!isCustomTag(t)) return true; // always include non-custom tags
 | ||||
| 
 | ||||
|             // if the tag is custom though, only include it if it is enabled
 | ||||
|             return CustomRoomTagStore.getTags()[t]; | ||||
|         }); | ||||
| 
 | ||||
|         let doUpdate = arrayHasDiff(previousListIds, newListIds); | ||||
|         if (!doUpdate) { | ||||
|  | @ -240,7 +245,8 @@ export default class RoomList extends React.PureComponent<IProps, IState> { | |||
|         if (doUpdate) { | ||||
|             // We have to break our reference to the room list store if we want to be able to
 | ||||
|             // diff the object for changes, so do that.
 | ||||
|             const sublists = objectShallowClone(newLists, (k, v) => arrayFastClone(v)); | ||||
|             const newSublists = objectWithOnly(newLists, newListIds); | ||||
|             const sublists = objectShallowClone(newSublists, (k, v) => arrayFastClone(v)); | ||||
| 
 | ||||
|             this.setState({sublists}, () => { | ||||
|                 this.props.onResize(); | ||||
|  | @ -288,8 +294,7 @@ export default class RoomList extends React.PureComponent<IProps, IState> { | |||
|         const tagOrder = TAG_ORDER.reduce((p, c) => { | ||||
|             if (c === CUSTOM_TAGS_BEFORE_TAG) { | ||||
|                 const customTags = Object.keys(this.state.sublists) | ||||
|                     .filter(t => isCustomTag(t)) | ||||
|                     .filter(t => CustomRoomTagStore.getTags()[t]); // isSelected
 | ||||
|                     .filter(t => isCustomTag(t)); | ||||
|                 p.push(...customTags); | ||||
|             } | ||||
|             p.push(c); | ||||
|  |  | |||
|  | @ -21,7 +21,8 @@ import * as sdk from "../../../index"; | |||
| import { _t } from "../../../languageHandler"; | ||||
| import Modal from "../../../Modal"; | ||||
| import {MatrixClientPeg} from "../../../MatrixClientPeg"; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class RoomRecoveryReminder extends React.PureComponent { | ||||
|     static propTypes = { | ||||
|  |  | |||
|  | @ -46,8 +46,8 @@ import { Direction } from "re-resizable/lib/resizer"; | |||
| import { polyfillTouchEvent } from "../../../@types/polyfill"; | ||||
| import { RoomNotificationStateStore } from "../../../stores/notifications/RoomNotificationStateStore"; | ||||
| import RoomListLayoutStore from "../../../stores/room-list/RoomListLayoutStore"; | ||||
| import { arrayHasOrderChange } from "../../../utils/arrays"; | ||||
| import { objectExcluding, objectHasValueChange } from "../../../utils/objects"; | ||||
| import { arrayFastClone, arrayHasOrderChange } from "../../../utils/arrays"; | ||||
| import { objectExcluding, objectHasDiff } from "../../../utils/objects"; | ||||
| import TemporaryTile from "./TemporaryTile"; | ||||
| import { ListNotificationState } from "../../../stores/notifications/ListNotificationState"; | ||||
| 
 | ||||
|  | @ -115,7 +115,7 @@ export default class RoomSublist extends React.Component<IProps, IState> { | |||
|             isResizing: false, | ||||
|             isExpanded: this.isBeingFiltered ? this.isBeingFiltered : !this.layout.isCollapsed, | ||||
|             height: 0, // to be fixed in a moment, we need `rooms` to calculate this.
 | ||||
|             rooms: RoomListStore.instance.orderedLists[this.props.tagId] || [], | ||||
|             rooms: arrayFastClone(RoomListStore.instance.orderedLists[this.props.tagId] || []), | ||||
|         }; | ||||
|         // Why Object.assign() and not this.state.height? Because TypeScript says no.
 | ||||
|         this.state = Object.assign(this.state, {height: this.calculateInitialHeight()}); | ||||
|  | @ -181,7 +181,7 @@ export default class RoomSublist extends React.Component<IProps, IState> { | |||
|     } | ||||
| 
 | ||||
|     public shouldComponentUpdate(nextProps: Readonly<IProps>, nextState: Readonly<IState>): boolean { | ||||
|         if (objectHasValueChange(this.props, nextProps)) { | ||||
|         if (objectHasDiff(this.props, nextProps)) { | ||||
|             // Something we don't care to optimize has updated, so update.
 | ||||
|             return true; | ||||
|         } | ||||
|  | @ -189,7 +189,7 @@ export default class RoomSublist extends React.Component<IProps, IState> { | |||
|         // Do the same check used on props for state, without the rooms we're going to no-op
 | ||||
|         const prevStateNoRooms = objectExcluding(this.state, ['rooms']); | ||||
|         const nextStateNoRooms = objectExcluding(nextState, ['rooms']); | ||||
|         if (objectHasValueChange(prevStateNoRooms, nextStateNoRooms)) { | ||||
|         if (objectHasDiff(prevStateNoRooms, nextStateNoRooms)) { | ||||
|             return true; | ||||
|         } | ||||
| 
 | ||||
|  | @ -255,7 +255,7 @@ export default class RoomSublist extends React.Component<IProps, IState> { | |||
|         } | ||||
| 
 | ||||
|         const currentRooms = this.state.rooms; | ||||
|         const newRooms = RoomListStore.instance.orderedLists[this.props.tagId] || []; | ||||
|         const newRooms = arrayFastClone(RoomListStore.instance.orderedLists[this.props.tagId] || []); | ||||
|         if (arrayHasOrderChange(currentRooms, newRooms)) { | ||||
|             stateUpdates.rooms = newRooms; | ||||
|         } | ||||
|  |  | |||
|  | @ -17,12 +17,13 @@ See the License for the specific language governing permissions and | |||
| limitations under the License. | ||||
| */ | ||||
| 
 | ||||
| import React, {createRef} from "react"; | ||||
| import React, { createRef } from "react"; | ||||
| import { Room } from "matrix-js-sdk/src/models/room"; | ||||
| import classNames from "classnames"; | ||||
| import { RovingTabIndexWrapper } from "../../../accessibility/RovingTabIndex"; | ||||
| import AccessibleButton, { ButtonEvent } from "../../views/elements/AccessibleButton"; | ||||
| import dis from '../../../dispatcher/dispatcher'; | ||||
| import defaultDispatcher from '../../../dispatcher/dispatcher'; | ||||
| import { Key } from "../../../Keyboard"; | ||||
| import ActiveRoomObserver from "../../../ActiveRoomObserver"; | ||||
| import { _t } from "../../../languageHandler"; | ||||
|  | @ -30,31 +31,26 @@ import { | |||
|     ChevronFace, | ||||
|     ContextMenu, | ||||
|     ContextMenuTooltipButton, | ||||
|     MenuItemRadio, | ||||
|     MenuItemCheckbox, | ||||
|     MenuItem, | ||||
|     MenuItemCheckbox, | ||||
|     MenuItemRadio, | ||||
| } from "../../structures/ContextMenu"; | ||||
| import { DefaultTagID, TagID } from "../../../stores/room-list/models"; | ||||
| import { MessagePreviewStore, ROOM_PREVIEW_CHANGED } from "../../../stores/room-list/MessagePreviewStore"; | ||||
| import DecoratedRoomAvatar from "../avatars/DecoratedRoomAvatar"; | ||||
| import { | ||||
|     getRoomNotifsState, | ||||
|     setRoomNotifsState, | ||||
|     ALL_MESSAGES, | ||||
|     ALL_MESSAGES_LOUD, | ||||
|     MENTIONS_ONLY, | ||||
|     MUTE, | ||||
| } from "../../../RoomNotifs"; | ||||
| import { ALL_MESSAGES, ALL_MESSAGES_LOUD, MENTIONS_ONLY, MUTE, } from "../../../RoomNotifs"; | ||||
| import { MatrixClientPeg } from "../../../MatrixClientPeg"; | ||||
| import NotificationBadge from "./NotificationBadge"; | ||||
| import { Volume } from "../../../RoomNotifsTypes"; | ||||
| import RoomListStore from "../../../stores/room-list/RoomListStore"; | ||||
| import RoomListActions from "../../../actions/RoomListActions"; | ||||
| import defaultDispatcher from "../../../dispatcher/dispatcher"; | ||||
| import {ActionPayload} from "../../../dispatcher/payloads"; | ||||
| import { ActionPayload } from "../../../dispatcher/payloads"; | ||||
| import { RoomNotificationStateStore } from "../../../stores/notifications/RoomNotificationStateStore"; | ||||
| import { NOTIFICATION_STATE_UPDATE, NotificationState } from "../../../stores/notifications/NotificationState"; | ||||
| import AccessibleTooltipButton from "../elements/AccessibleTooltipButton"; | ||||
| import { EchoChamber } from "../../../stores/local-echo/EchoChamber"; | ||||
| import { CachedRoomKey, RoomEchoChamber } from "../../../stores/local-echo/RoomEchoChamber"; | ||||
| import { PROPERTY_UPDATED } from "../../../stores/local-echo/GenericEchoChamber"; | ||||
| 
 | ||||
| interface IProps { | ||||
|     room: Room; | ||||
|  | @ -112,6 +108,7 @@ export default class RoomTile extends React.PureComponent<IProps, IState> { | |||
|     private dispatcherRef: string; | ||||
|     private roomTileRef = createRef<HTMLDivElement>(); | ||||
|     private notificationState: NotificationState; | ||||
|     private roomProps: RoomEchoChamber; | ||||
| 
 | ||||
|     constructor(props: IProps) { | ||||
|         super(props); | ||||
|  | @ -130,12 +127,19 @@ export default class RoomTile extends React.PureComponent<IProps, IState> { | |||
|         MessagePreviewStore.instance.on(ROOM_PREVIEW_CHANGED, this.onRoomPreviewChanged); | ||||
|         this.notificationState = RoomNotificationStateStore.instance.getRoomState(this.props.room); | ||||
|         this.notificationState.on(NOTIFICATION_STATE_UPDATE, this.onNotificationUpdate); | ||||
|         this.roomProps = EchoChamber.forRoom(this.props.room); | ||||
|         this.roomProps.on(PROPERTY_UPDATED, this.onRoomPropertyUpdate); | ||||
|     } | ||||
| 
 | ||||
|     private onNotificationUpdate = () => { | ||||
|         this.forceUpdate(); // notification state changed - update
 | ||||
|     }; | ||||
| 
 | ||||
|     private onRoomPropertyUpdate = (property: CachedRoomKey) => { | ||||
|         if (property === CachedRoomKey.NotificationVolume) this.onNotificationUpdate(); | ||||
|         // else ignore - not important for this tile
 | ||||
|     }; | ||||
| 
 | ||||
|     private get showContextMenu(): boolean { | ||||
|         return !this.props.isMinimized && this.props.tag !== DefaultTagID.Invite; | ||||
|     } | ||||
|  | @ -307,17 +311,9 @@ export default class RoomTile extends React.PureComponent<IProps, IState> { | |||
|         ev.stopPropagation(); | ||||
|         if (MatrixClientPeg.get().isGuest()) return; | ||||
| 
 | ||||
|         // get key before we go async and React discards the nativeEvent
 | ||||
|         const key = (ev as React.KeyboardEvent).key; | ||||
|         try { | ||||
|             // TODO add local echo - https://github.com/vector-im/riot-web/issues/14280
 | ||||
|             await setRoomNotifsState(this.props.room.roomId, newState); | ||||
|         } catch (error) { | ||||
|             // TODO: some form of error notification to the user to inform them that their state change failed.
 | ||||
|             // See https://github.com/vector-im/riot-web/issues/14281
 | ||||
|             console.error(error); | ||||
|         } | ||||
|         this.roomProps.notificationVolume = newState; | ||||
| 
 | ||||
|         const key = (ev as React.KeyboardEvent).key; | ||||
|         if (key === Key.ENTER) { | ||||
|             // Implements https://www.w3.org/TR/wai-aria-practices/#keyboard-interaction-12
 | ||||
|             this.setState({notificationsMenuPosition: null}); // hide the menu
 | ||||
|  | @ -335,7 +331,7 @@ export default class RoomTile extends React.PureComponent<IProps, IState> { | |||
|             return null; | ||||
|         } | ||||
| 
 | ||||
|         const state = getRoomNotifsState(this.props.room.roomId); | ||||
|         const state = this.roomProps.notificationVolume; | ||||
| 
 | ||||
|         let contextMenu = null; | ||||
|         if (this.state.notificationsMenuPosition) { | ||||
|  |  | |||
|  | @ -28,6 +28,7 @@ import SettingsStore from "../../../settings/SettingsStore"; | |||
| import {ContextMenu} from "../../structures/ContextMenu"; | ||||
| import {WidgetType} from "../../../widgets/WidgetType"; | ||||
| import AccessibleTooltipButton from "../elements/AccessibleTooltipButton"; | ||||
| import {Action} from "../../../dispatcher/actions"; | ||||
| 
 | ||||
| // This should be below the dialog level (4000), but above the rest of the UI (1000-2000).
 | ||||
| // We sit in a context menu, so this should be given to the context menu.
 | ||||
|  | @ -181,7 +182,7 @@ export default class Stickerpicker extends React.Component { | |||
|             case "stickerpicker_close": | ||||
|                 this.setState({showStickers: false}); | ||||
|                 break; | ||||
|             case "after_right_panel_phase_change": | ||||
|             case Action.AfterRightPanelPhaseChange: | ||||
|             case "show_left_panel": | ||||
|             case "hide_left_panel": | ||||
|                 this.setState({showStickers: false}); | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ import React from 'react'; | |||
| 
 | ||||
| import * as sdk from '../../../index'; | ||||
| import {_t} from "../../../languageHandler"; | ||||
| import {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| const SETTING_MANUALLY_VERIFY_ALL_SESSIONS = "e2ee.manuallyVerifyAllSessions"; | ||||
| 
 | ||||
|  |  | |||
|  | @ -20,10 +20,11 @@ import { _t } from '../../../languageHandler'; | |||
| import SdkConfig from "../../../SdkConfig"; | ||||
| import * as sdk from '../../../index'; | ||||
| import Modal from '../../../Modal'; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import AccessibleButton from "../elements/AccessibleButton"; | ||||
| import {formatBytes, formatCountLong} from "../../../utils/FormattingUtils"; | ||||
| import EventIndexPeg from "../../../indexing/EventIndexPeg"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class EventIndexPanel extends React.Component { | ||||
|     constructor() { | ||||
|  |  | |||
|  | @ -20,7 +20,7 @@ import createReactClass from 'create-react-class'; | |||
| import * as sdk from '../../../index'; | ||||
| import { _t } from '../../../languageHandler'; | ||||
| import {MatrixClientPeg} from '../../../MatrixClientPeg'; | ||||
| import SettingsStore, {SettingLevel} from '../../../settings/SettingsStore'; | ||||
| import SettingsStore from '../../../settings/SettingsStore'; | ||||
| import Modal from '../../../Modal'; | ||||
| import { | ||||
|     NotificationUtils, | ||||
|  | @ -31,6 +31,7 @@ import { | |||
| import SdkConfig from "../../../SdkConfig"; | ||||
| import LabelledToggleSwitch from "../elements/LabelledToggleSwitch"; | ||||
| import AccessibleButton from "../elements/AccessibleButton"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| // TODO: this "view" component still has far too much application logic in it,
 | ||||
| // which should be factored out to other files.
 | ||||
|  |  | |||
|  | @ -18,7 +18,8 @@ import React from 'react'; | |||
| import {_t} from "../../../languageHandler"; | ||||
| import {IntegrationManagers} from "../../../integrations/IntegrationManagers"; | ||||
| import * as sdk from '../../../index'; | ||||
| import SettingsStore, {SettingLevel} from "../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../settings/SettingsStore"; | ||||
| import {SettingLevel} from "../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class SetIntegrationManager extends React.Component { | ||||
|     constructor() { | ||||
|  |  | |||
|  | @ -21,7 +21,7 @@ import {MatrixClientPeg} from "../../../../../MatrixClientPeg"; | |||
| import AccessibleButton from "../../../elements/AccessibleButton"; | ||||
| import Notifier from "../../../../../Notifier"; | ||||
| import SettingsStore from '../../../../../settings/SettingsStore'; | ||||
| import { SettingLevel } from '../../../../../settings/SettingsStore'; | ||||
| import {SettingLevel} from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class NotificationsSettingsTab extends React.Component { | ||||
|     static propTypes = { | ||||
|  |  | |||
|  | @ -20,9 +20,9 @@ import {_t} from "../../../../../languageHandler"; | |||
| import {MatrixClientPeg} from "../../../../../MatrixClientPeg"; | ||||
| import * as sdk from "../../../../.."; | ||||
| import LabelledToggleSwitch from "../../../elements/LabelledToggleSwitch"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingsStore"; | ||||
| import Modal from "../../../../../Modal"; | ||||
| import QuestionDialog from "../../../dialogs/QuestionDialog"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class SecurityRoomSettingsTab extends React.Component { | ||||
|     static propTypes = { | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ limitations under the License. | |||
| import React from 'react'; | ||||
| import {_t} from "../../../../../languageHandler"; | ||||
| import SdkConfig from "../../../../../SdkConfig"; | ||||
| import SettingsStore, {SettingLevel} from "../../../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../../../settings/SettingsStore"; | ||||
| import { enumerateThemes } from "../../../../../theme"; | ||||
| import ThemeWatcher from "../../../../../settings/watchers/ThemeWatcher"; | ||||
| import Slider from "../../../elements/Slider"; | ||||
|  | @ -35,6 +35,7 @@ import Field from '../../../elements/Field'; | |||
| import EventTilePreview from '../../../elements/EventTilePreview'; | ||||
| import StyledRadioGroup from "../../../elements/StyledRadioGroup"; | ||||
| import classNames from 'classnames'; | ||||
| import { SettingLevel } from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| interface IProps { | ||||
| } | ||||
|  |  | |||
|  | @ -20,7 +20,6 @@ import React from 'react'; | |||
| import {_t} from "../../../../../languageHandler"; | ||||
| import ProfileSettings from "../../ProfileSettings"; | ||||
| import * as languageHandler from "../../../../../languageHandler"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../../../settings/SettingsStore"; | ||||
| import LanguageDropdown from "../../../elements/LanguageDropdown"; | ||||
| import AccessibleButton from "../../../elements/AccessibleButton"; | ||||
|  | @ -37,6 +36,7 @@ import IdentityAuthClient from "../../../../../IdentityAuthClient"; | |||
| import {abbreviateUrl} from "../../../../../utils/UrlUtils"; | ||||
| import { getThreepidsWithBindStatus } from '../../../../../boundThreepids'; | ||||
| import Spinner from "../../../elements/Spinner"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class GeneralUserSettingsTab extends React.Component { | ||||
|     static propTypes = { | ||||
|  |  | |||
|  | @ -17,9 +17,10 @@ limitations under the License. | |||
| import React from 'react'; | ||||
| import {_t} from "../../../../../languageHandler"; | ||||
| import PropTypes from "prop-types"; | ||||
| import SettingsStore, {SettingLevel} from "../../../../../settings/SettingsStore"; | ||||
| import SettingsStore from "../../../../../settings/SettingsStore"; | ||||
| import LabelledToggleSwitch from "../../../elements/LabelledToggleSwitch"; | ||||
| import * as sdk from "../../../../../index"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export class LabsSettingToggle extends React.Component { | ||||
|     static propTypes = { | ||||
|  |  | |||
|  | @ -17,12 +17,12 @@ limitations under the License. | |||
| 
 | ||||
| import React from 'react'; | ||||
| import {_t} from "../../../../../languageHandler"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingsStore"; | ||||
| import LabelledToggleSwitch from "../../../elements/LabelledToggleSwitch"; | ||||
| import SettingsStore from "../../../../../settings/SettingsStore"; | ||||
| import Field from "../../../elements/Field"; | ||||
| import * as sdk from "../../../../.."; | ||||
| import PlatformPeg from "../../../../../PlatformPeg"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class PreferencesUserSettingsTab extends React.Component { | ||||
|     static ROOM_LIST_SETTINGS = [ | ||||
|  |  | |||
|  | @ -19,7 +19,6 @@ import React from 'react'; | |||
| import PropTypes from 'prop-types'; | ||||
| import {_t} from "../../../../../languageHandler"; | ||||
| import SdkConfig from "../../../../../SdkConfig"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingsStore"; | ||||
| import {MatrixClientPeg} from "../../../../../MatrixClientPeg"; | ||||
| import * as FormattingUtils from "../../../../../utils/FormattingUtils"; | ||||
| import AccessibleButton from "../../../elements/AccessibleButton"; | ||||
|  | @ -29,6 +28,7 @@ import * as sdk from "../../../../.."; | |||
| import {sleep} from "../../../../../utils/promise"; | ||||
| import dis from "../../../../../dispatcher/dispatcher"; | ||||
| import {privateShouldBeEncrypted} from "../../../../../createRoom"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export class IgnoredUser extends React.Component { | ||||
|     static propTypes = { | ||||
|  |  | |||
|  | @ -21,10 +21,10 @@ import SdkConfig from "../../../../../SdkConfig"; | |||
| import CallMediaHandler from "../../../../../CallMediaHandler"; | ||||
| import Field from "../../../elements/Field"; | ||||
| import AccessibleButton from "../../../elements/AccessibleButton"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingsStore"; | ||||
| import {MatrixClientPeg} from "../../../../../MatrixClientPeg"; | ||||
| import * as sdk from "../../../../../index"; | ||||
| import Modal from "../../../../../Modal"; | ||||
| import {SettingLevel} from "../../../../../settings/SettingLevel"; | ||||
| 
 | ||||
| export default class VoiceUserSettingsTab extends React.Component { | ||||
|     constructor() { | ||||
|  |  | |||
|  | @ -0,0 +1,40 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 { _t } from "../../../languageHandler"; | ||||
| import AccessibleButton from "../elements/AccessibleButton"; | ||||
| import Modal from "../../../Modal"; | ||||
| import ServerOfflineDialog from "../dialogs/ServerOfflineDialog"; | ||||
| 
 | ||||
| export default class NonUrgentEchoFailureToast extends React.PureComponent { | ||||
|     private openDialog = () => { | ||||
|         Modal.createTrackedDialog('Local Echo Server Error', '', ServerOfflineDialog, {}); | ||||
|     }; | ||||
| 
 | ||||
|     public render() { | ||||
|         return ( | ||||
|             <div className="mx_NonUrgentEchoFailureToast"> | ||||
|                 <span className="mx_NonUrgentEchoFailureToast_icon" /> | ||||
|                 {_t("Your server isn't responding to some <a>requests</a>.", {}, { | ||||
|                     'a': (sub) => ( | ||||
|                         <AccessibleButton kind="link" onClick={this.openDialog}>{sub}</AccessibleButton> | ||||
|                     ), | ||||
|                 })} | ||||
|             </div> | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | @ -19,7 +19,8 @@ import React from "react"; | |||
| import * as sdk from "../../../index"; | ||||
| import { _t } from '../../../languageHandler'; | ||||
| import {MatrixClientPeg} from '../../../MatrixClientPeg'; | ||||
| import {RIGHT_PANEL_PHASES} from "../../../stores/RightPanelStorePhases"; | ||||
| import {RightPanelPhases} from "../../../stores/RightPanelStorePhases"; | ||||
| import {SetRightPanelPhasePayload} from "../../../dispatcher/payloads/SetRightPanelPhasePayload" | ||||
| import {userLabelForEventRoom} from "../../../utils/KeyVerificationStateObserver"; | ||||
| import dis from "../../../dispatcher/dispatcher"; | ||||
| import ToastStore from "../../../stores/ToastStore"; | ||||
|  | @ -27,6 +28,7 @@ import Modal from "../../../Modal"; | |||
| import GenericToast from "./GenericToast"; | ||||
| import {VerificationRequest} from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; | ||||
| import {DeviceInfo} from "matrix-js-sdk/src/crypto/deviceinfo"; | ||||
| import {Action} from "../../../dispatcher/actions"; | ||||
| 
 | ||||
| interface IProps { | ||||
|     toastKey: string; | ||||
|  | @ -104,9 +106,9 @@ export default class VerificationRequestToast extends React.PureComponent<IProps | |||
|                     room_id: request.channel.roomId, | ||||
|                     should_peek: false, | ||||
|                 }); | ||||
|                 dis.dispatch({ | ||||
|                     action: "set_right_panel_phase", | ||||
|                     phase: RIGHT_PANEL_PHASES.EncryptionPanel, | ||||
|                 dis.dispatch<SetRightPanelPhasePayload>({ | ||||
|                     action: Action.SetRightPanelPhase, | ||||
|                     phase: RightPanelPhases.EncryptionPanel, | ||||
|                     refireParams: { | ||||
|                         verificationRequest: request, | ||||
|                         member: cli.getUser(request.otherUserId), | ||||
|  |  | |||
|  | @ -79,4 +79,19 @@ export enum Action { | |||
|      * Changes room based on room list order and payload parameters. Should be used with ViewRoomDeltaPayload. | ||||
|      */ | ||||
|     ViewRoomDelta = "view_room_delta", | ||||
| 
 | ||||
|     /** | ||||
|      * Sets the phase for the right panel. Should be used with SetRightPanelPhasePayload. | ||||
|      */ | ||||
|     SetRightPanelPhase = "set_right_panel_phase", | ||||
| 
 | ||||
|     /** | ||||
|      * Toggles the right panel. Should be used with ToggleRightPanelPayload. | ||||
|      */ | ||||
|     ToggleRightPanel = "toggle_right_panel", | ||||
| 
 | ||||
|     /** | ||||
|      * Trigged after the phase of the right panel is set. Should be used with AfterRightPanelPhaseChangePayload. | ||||
|      */ | ||||
|     AfterRightPanelPhaseChange = "after_right_panel_phase_change", | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,30 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 { RightPanelPhases } from "../../stores/RightPanelStorePhases"; | ||||
| import { SetRightPanelPhaseRefireParams } from "./SetRightPanelPhasePayload"; | ||||
| import { ActionPayload } from "../payloads"; | ||||
| import { Action } from "../actions"; | ||||
| import { VerificationRequest } from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; | ||||
| 
 | ||||
| interface AfterRightPanelPhaseChangeAction extends ActionPayload { | ||||
|     action: Action.AfterRightPanelPhaseChange; | ||||
|     phase: RightPanelPhases; | ||||
|     verificationRequestPromise?: Promise<VerificationRequest>; | ||||
| } | ||||
| 
 | ||||
| export type AfterRightPanelPhaseChangePayload | ||||
|     = AfterRightPanelPhaseChangeAction & SetRightPanelPhaseRefireParams; | ||||
|  | @ -0,0 +1,37 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 { VerificationRequest } from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; | ||||
| import { RoomMember } from "matrix-js-sdk/src/models/room-member"; | ||||
| import { RightPanelPhases } from "../../stores/RightPanelStorePhases"; | ||||
| import { ActionPayload } from "../payloads"; | ||||
| import { Action } from "../actions"; | ||||
| 
 | ||||
| export interface SetRightPanelPhasePayload extends ActionPayload { | ||||
|     action: Action.SetRightPanelPhase; | ||||
| 
 | ||||
|     phase: RightPanelPhases; | ||||
|     refireParams?: SetRightPanelPhaseRefireParams; | ||||
| } | ||||
| 
 | ||||
| export interface SetRightPanelPhaseRefireParams { | ||||
|     member?: RoomMember; | ||||
|     verificationRequest?: VerificationRequest; | ||||
|     groupId?: string; | ||||
|     groupRoomId?: string; | ||||
|     // XXX: The type for event should 'view_3pid_invite' action's payload
 | ||||
|     event?: any; | ||||
| } | ||||
|  | @ -0,0 +1,27 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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 { ActionPayload } from "../payloads"; | ||||
| import { Action } from "../actions"; | ||||
| 
 | ||||
| export interface ToggleRightPanelPayload extends ActionPayload { | ||||
|     action: Action.ToggleRightPanel; | ||||
| 
 | ||||
|     /** | ||||
|      * The type of room that the panel is toggled in. | ||||
|      */ | ||||
|     type: "group" | "room"; | ||||
| } | ||||
|  | @ -15,8 +15,9 @@ See the License for the specific language governing permissions and | |||
| limitations under the License. | ||||
| */ | ||||
| 
 | ||||
| import SettingsStore, {SettingLevel} from "../settings/SettingsStore"; | ||||
| import SettingsStore from "../settings/SettingsStore"; | ||||
| import {orderBy} from "lodash"; | ||||
| import { SettingLevel } from "../settings/SettingLevel"; | ||||
| 
 | ||||
| interface ILegacyFormat { | ||||
|     [emoji: string]: [number, number]; // [count, date]
 | ||||
|  |  | |||
|  | @ -426,6 +426,8 @@ | |||
|     "There was an error joining the room": "There was an error joining the room", | ||||
|     "Sorry, your homeserver is too old to participate in this room.": "Sorry, your homeserver is too old to participate in this room.", | ||||
|     "Please contact your homeserver administrator.": "Please contact your homeserver administrator.", | ||||
|     "The person who invited you already left the room.": "The person who invited you already left the room.", | ||||
|     "The person who invited you already left the room, or their server is offline.": "The person who invited you already left the room, or their server is offline.", | ||||
|     "Failed to join room": "Failed to join room", | ||||
|     "You joined the call": "You joined the call", | ||||
|     "%(senderName)s joined the call": "%(senderName)s joined the call", | ||||
|  | @ -441,6 +443,7 @@ | |||
|     "%(senderName)s: %(message)s": "%(senderName)s: %(message)s", | ||||
|     "%(senderName)s: %(reaction)s": "%(senderName)s: %(reaction)s", | ||||
|     "%(senderName)s: %(stickerName)s": "%(senderName)s: %(stickerName)s", | ||||
|     "Change notification settings": "Change notification settings", | ||||
|     "New spinner design": "New spinner design", | ||||
|     "Message Pinning": "Message Pinning", | ||||
|     "Custom user status messages": "Custom user status messages", | ||||
|  | @ -611,6 +614,7 @@ | |||
|     "Headphones": "Headphones", | ||||
|     "Folder": "Folder", | ||||
|     "Pin": "Pin", | ||||
|     "Your server isn't responding to some <a>requests</a>.": "Your server isn't responding to some <a>requests</a>.", | ||||
|     "From %(deviceName)s (%(deviceId)s)": "From %(deviceName)s (%(deviceId)s)", | ||||
|     "Decline (%(counter)s)": "Decline (%(counter)s)", | ||||
|     "Accept <policyLink /> to continue:": "Accept <policyLink /> to continue:", | ||||
|  | @ -1743,6 +1747,19 @@ | |||
|     "Upgrading a room is an advanced action and is usually recommended when a room is unstable due to bugs, missing features or security vulnerabilities.": "Upgrading a room is an advanced action and is usually recommended when a room is unstable due to bugs, missing features or security vulnerabilities.", | ||||
|     "This usually only affects how the room is processed on the server. If you're having problems with your %(brand)s, please <a>report a bug</a>.": "This usually only affects how the room is processed on the server. If you're having problems with your %(brand)s, please <a>report a bug</a>.", | ||||
|     "You'll upgrade this room from <oldVersion /> to <newVersion />.": "You'll upgrade this room from <oldVersion /> to <newVersion />.", | ||||
|     "Resend": "Resend", | ||||
|     "You're all caught up.": "You're all caught up.", | ||||
|     "Server isn't responding": "Server isn't responding", | ||||
|     "Your server isn't responding to some of your requests. Below are some of the most likely reasons.": "Your server isn't responding to some of your requests. Below are some of the most likely reasons.", | ||||
|     "The server (%(serverName)s) took too long to respond.": "The server (%(serverName)s) took too long to respond.", | ||||
|     "Your firewall or anti-virus is blocking the request.": "Your firewall or anti-virus is blocking the request.", | ||||
|     "A browser extension is preventing the request.": "A browser extension is preventing the request.", | ||||
|     "The server is offline.": "The server is offline.", | ||||
|     "The server has denied your request.": "The server has denied your request.", | ||||
|     "Your area is experiencing difficulties connecting to the internet.": "Your area is experiencing difficulties connecting to the internet.", | ||||
|     "A connection error occurred while trying to contact the server.": "A connection error occurred while trying to contact the server.", | ||||
|     "The server is not configured to indicate what the problem is (CORS).": "The server is not configured to indicate what the problem is (CORS).", | ||||
|     "Recent changes that have not yet been received": "Recent changes that have not yet been received", | ||||
|     "Sign out and remove encryption keys?": "Sign out and remove encryption keys?", | ||||
|     "Clear Storage and Sign Out": "Clear Storage and Sign Out", | ||||
|     "Send Logs": "Send Logs", | ||||
|  | @ -1850,7 +1867,6 @@ | |||
|     "Reject invitation": "Reject invitation", | ||||
|     "Are you sure you want to reject the invitation?": "Are you sure you want to reject the invitation?", | ||||
|     "Unable to reject invite": "Unable to reject invite", | ||||
|     "Resend": "Resend", | ||||
|     "Resend edit": "Resend edit", | ||||
|     "Resend %(unsentCount)s reaction(s)": "Resend %(unsentCount)s reaction(s)", | ||||
|     "Resend removal": "Resend removal", | ||||
|  |  | |||
|  | @ -1721,8 +1721,8 @@ | |||
|     "To be secure, do this in person or use a trusted way to communicate.": "Por plia sekureco, faru tion persone, aŭ uzu alian fidatan komunikilon.", | ||||
|     "Verify yourself & others to keep your chats safe": "Kontrolu vin mem kaj aliajn por sekurigi viajn babilojn", | ||||
|     "Channel: %(channelName)s": "Kanalo: %(channelName)s", | ||||
|     "Show less": "Montri pli", | ||||
|     "Show more": "Montri malpli", | ||||
|     "Show less": "Montri malpli", | ||||
|     "Show more": "Montri pli", | ||||
|     "Help": "Helpo", | ||||
|     "Session verified": "Salutaĵo kontroliĝis", | ||||
|     "Copy": "Kopii", | ||||
|  | @ -2376,7 +2376,7 @@ | |||
|     "Set a Security Phrase": "Agordi Sekurecan frazon", | ||||
|     "Confirm Security Phrase": "Konfirmi Sekurecan frazon", | ||||
|     "Save your Security Key": "Konservi vian Sekurecan ŝlosilon", | ||||
|     "New spinner design": "", | ||||
|     "New spinner design": "Nova fasono de la turniĝilo", | ||||
|     "Show rooms with unread messages first": "Montri ĉambrojn kun nelegitaj mesaĝoj kiel unuajn", | ||||
|     "Show previews of messages": "Montri antaŭrigardojn al mesaĝoj", | ||||
|     "This room is public": "Ĉi tiu ĉambro estas publika", | ||||
|  | @ -2386,5 +2386,6 @@ | |||
|     "Are you sure you want to cancel entering passphrase?": "Ĉu vi certe volas nuligi enigon de pasfrazo?", | ||||
|     "Enable advanced debugging for the room list": "Ŝalti altnivelan erarserĉadon por la ĉambrobreto", | ||||
|     "* %(senderName)s %(emote)s": "* %(senderName)s %(emote)s", | ||||
|     "Custom Tag": "Propra etikedo" | ||||
|     "Custom Tag": "Propra etikedo", | ||||
|     "Feedback": "Prikomenti" | ||||
| } | ||||
|  |  | |||
|  | @ -1714,10 +1714,10 @@ | |||
|     "<strong>%(role)s</strong> in %(roomName)s": "<strong>%(role)s</strong> jututoas %(roomName)s", | ||||
|     "Failed to change power level": "Õiguste muutmine ei õnnestunud", | ||||
|     "You will not be able to undo this change as you are promoting the user to have the same power level as yourself.": "Sa ei saa seda muudatust hiljem tagasi pöörata, sest annad teisele kasutajale samad õigused, mis sinul on.", | ||||
|     "Deactivate user?": "Kas blokeerime kasutaja?", | ||||
|     "Deactivating this user will log them out and prevent them from logging back in. Additionally, they will leave all the rooms they are in. This action cannot be reversed. Are you sure you want to deactivate this user?": "Kasutaja blokeerimisel logitakse ta automaatselt välja ning ei lubata enam sisse logida. Lisaks lahkub ta kõikidest jututubadest, mille liige ta parasjagu on. Seda tegevust ei saa tagasi pöörata. Kas sa oled ikka kindel, et soovid selle kasutaja blokeerida?", | ||||
|     "Deactivate user": "Blokeeri kasutaja", | ||||
|     "Failed to deactivate user": "Kasutaja blokeerimine ei õnnestunud", | ||||
|     "Deactivate user?": "Kas deaktiveerime kasutajakonto?", | ||||
|     "Deactivating this user will log them out and prevent them from logging back in. Additionally, they will leave all the rooms they are in. This action cannot be reversed. Are you sure you want to deactivate this user?": "Kasutaja deaktiveerimisel logitakse ta automaatselt välja ning ei lubata enam sisse logida. Lisaks lahkub ta kõikidest jututubadest, mille liige ta parasjagu on. Seda tegevust ei saa tagasi pöörata. Kas sa oled ikka kindel, et soovid selle kasutaja kõijkalt eemaldada?", | ||||
|     "Deactivate user": "Deaktiveeri kasutaja", | ||||
|     "Failed to deactivate user": "Kasutaja deaktiveerimine ei õnnestunud", | ||||
|     "This client does not support end-to-end encryption.": "See klient ei toeta läbivat krüptimist.", | ||||
|     "Security": "Turvalisus", | ||||
|     "Using this widget may share data <helpIcon /> with %(widgetDomain)s & your Integration Manager.": "Selle vidina kasutamisel võidakse jagada andmeid <helpIcon /> saitidega %(widgetDomain)s ning sinu vidinahalduriga.", | ||||
|  | @ -2368,5 +2368,22 @@ | |||
|     "You've previously used a newer version of %(brand)s with this session. To use this version again with end to end encryption, you will need to sign out and back in again.": "Sa oled selle sessiooni jaoks varem kasutanud %(brand)s'i uuemat versiooni. Selle versiooni kasutamiseks läbiva krüptimisega, pead sa esmalt logima välja ja siis uuesti logima tagasi sisse.", | ||||
|     "Without setting up Secure Message Recovery, you'll lose your secure message history when you log out.": "Kui sa pole seadistanud krüptitud sõnumite taastamise meetodeid, siis väljalogimisel sa kaotad võimaluse neid krüptitud sõnumeid lugeda.", | ||||
|     "If you don't want to set this up now, you can later in Settings.": "Kui sa ei soovi seda teha kohe, siis vastava võimaluse leiad hiljem rakenduse seadistustest.", | ||||
|     "If you didn't set the new recovery method, an attacker may be trying to access your account. Change your account password and set a new recovery method immediately in Settings.": "Kui sa ei ole ise uusi taastamise meetodeid lisanud, siis võib olla tegemist ründega sinu konto vastu. Palun vaheta koheselt oma kasutajakonto salasõna ning määra seadistustes uus taastemeetod." | ||||
|     "If you didn't set the new recovery method, an attacker may be trying to access your account. Change your account password and set a new recovery method immediately in Settings.": "Kui sa ei ole ise uusi taastamise meetodeid lisanud, siis võib olla tegemist ründega sinu konto vastu. Palun vaheta koheselt oma kasutajakonto salasõna ning määra seadistustes uus taastemeetod.", | ||||
|     "%(senderDisplayName)s enabled flair for %(groups)s in this room.": "%(senderDisplayName)s võttis selles jututoas kasutusele %(groups)s kogukonna rinnamärgi.", | ||||
|     "%(senderDisplayName)s disabled flair for %(groups)s in this room.": "%(senderDisplayName)s eemaldas selles jututoas kasutuselt %(groups)s kogukonna rinnamärgi.", | ||||
|     "%(senderDisplayName)s enabled flair for %(newGroups)s and disabled flair for %(oldGroups)s in this room.": "%(senderDisplayName)s võttis selles jututoas kasutusele %(newGroups)s kogukonna rinnamärgi ning eemaldas rinnamärgi %(oldGroups)s kogukonnalt.", | ||||
|     "Connect this session to key backup before signing out to avoid losing any keys that may only be on this session.": "Enne väljalogimist seo see sessioon krüptovõtmete varundusega. Kui sa seda ei tee, siis võid kaotada võtmed, mida kasutatakse vaid siin sessioonis.", | ||||
|     "Flair": "Kogukonna rinnasilt", | ||||
|     "Error updating flair": "Viga kogukonna rinnasildi uuendamisel", | ||||
|     "There was an error updating the flair for this room. The server may not allow it or a temporary error occurred.": "Kogukonna rinnasildi uuendamisel tekkis viga. See kas on serveri poolt keelatud või tekkis mingi ajutine viga.", | ||||
|     "Showing flair for these communities:": "Näidatakse nende kogukondade rinnasilte:", | ||||
|     "This room is not showing flair for any communities": "Sellele jututoale ei ole jagatud ühtegi kogukonna rinnasilti", | ||||
|     "Display your community flair in rooms configured to show it.": "Näita oma kogukonna rinnasilti nendes jututubades, kus selle kuvamine on seadistatud.", | ||||
|     "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use %(brand)s with an existing Matrix account on a different homeserver.": "Kohandatud serveriseadistusi saad kasutada selleks, et logida sisse sinu valitud koduserverisse. See võimaldab sinul kasutada %(brand)s'i mõnes teises koduserveri hallatava kasutajakontoga.", | ||||
|     "Did you know: you can use communities to filter your %(brand)s experience!": "Kas sa teadsid, et sa võid %(brand)s'i parema kasutuskogemuse nimel pruukida kogukondi!", | ||||
|     "Without setting up Secure Message Recovery, you won't be able to restore your encrypted message history if you log out or use another session.": "Kui sa pole seadistanud krüptitud sõnumite taastamise meetodeid, siis väljalogimisel või muu sessiooni kasutamisel sa kaotad võimaluse oma krüptitud sõnumeid lugeda.", | ||||
|     "Set up Secure Message Recovery": "Võta kasutusele turvaline sõnumivõtmete varundus", | ||||
|     "Secure your backup with a recovery passphrase": "Krüpti oma varukoopia taastamiseks mõeldud paroolifraasiga", | ||||
|     "The person who invited you already left the room.": "See, kes sind jututoa liikmeks kutsus, on juba jututoast lahkunud.", | ||||
|     "The person who invited you already left the room, or their server is offline.": "See, kes sind jututoa liikmeks kutsus, kas juba on jututoast lahkunud või tema koduserver on võrgust väljas." | ||||
| } | ||||
|  |  | |||
|  | @ -326,7 +326,7 @@ | |||
|     "Upload an avatar:": "Igo abatarra:", | ||||
|     "This server does not support authentication with a phone number.": "Zerbitzari honek ez du telefono zenbakia erabiliz autentifikatzea onartzen.", | ||||
|     "An error occurred: %(error_string)s": "Errore bat gertatu da: %(error_string)s", | ||||
|     "There are no visible files in this room": "Ez dago fitxategi ikusgairik gela honetan", | ||||
|     "There are no visible files in this room": "Ez dago fitxategirik ikusgai gela honetan", | ||||
|     "Sent messages will be stored until your connection has returned.": "Bidalitako mezuak zure konexioa berreskuratu arte gordeko dira.", | ||||
|     "(~%(count)s results)|one": "(~%(count)s emaitza)", | ||||
|     "(~%(count)s results)|other": "(~%(count)s emaitza)", | ||||
|  | @ -734,7 +734,7 @@ | |||
|     "All Rooms": "Gela guztiak", | ||||
|     "Wednesday": "Asteazkena", | ||||
|     "You cannot delete this message. (%(code)s)": "Ezin duzu mezu hau ezabatu. (%(code)s)", | ||||
|     "Quote": "Aipua", | ||||
|     "Quote": "Aipatu", | ||||
|     "Send logs": "Bidali egunkariak", | ||||
|     "All messages": "Mezu guztiak", | ||||
|     "Call invitation": "Dei gonbidapena", | ||||
|  | @ -1539,7 +1539,7 @@ | |||
|     "View": "Ikusi", | ||||
|     "Find a room…": "Bilatu gela bat…", | ||||
|     "Find a room… (e.g. %(exampleRoom)s)": "Bilatu gela bat… (adib. %(exampleRoom)s)", | ||||
|     "If you can't find the room you're looking for, ask for an invite or <a>Create a new room</a>.": "Ezin baduzu bila ari zaren gela aurkitu, eskatu gonbidapen bat edo <a>Sortu gela berri bat</a>.", | ||||
|     "If you can't find the room you're looking for, ask for an invite or <a>Create a new room</a>.": "Ez baduzu bilatzen duzuna aurkitzen, eskatu gonbidapen bat edo <a>Sortu gela berri bat</a>.", | ||||
|     "Explore rooms": "Arakatu gelak", | ||||
|     "Community Autocomplete": "Komunitate osatze automatikoa", | ||||
|     "Emoji Autocomplete": "Emoji osatze automatikoa", | ||||
|  | @ -1897,7 +1897,7 @@ | |||
|     "If your other sessions do not have the key for this message you will not be able to decrypt them.": "Zure beste saioek ez badute mezu honen gakoa ezin izango duzu deszifratu.", | ||||
|     "<requestLink>Re-request encryption keys</requestLink> from your other sessions.": "<requestLink>Eskatu berriro zifratze gakoak</requestLink> zure beste saioei.", | ||||
|     "Waiting for %(displayName)s to accept…": "%(displayName)s(e)k onartu bitartean zain…", | ||||
|     "Your messages are secured and only you and the recipient have the unique keys to unlock them.": "Zuon mezuak babestuta daude eta soilik zuk eta hartzaileak dituzue hauek irekitzeko giltza.", | ||||
|     "Your messages are secured and only you and the recipient have the unique keys to unlock them.": "Zuen mezuak babestuta daude eta soilik zuk eta hartzaileak dituzue hauek desblokeatzeko gakoak.", | ||||
|     "One of the following may be compromised:": "Hauetakoren bat konprometituta egon daiteke:", | ||||
|     "Verify this device to mark it as trusted. Trusting this device gives you and other users extra peace of mind when using end-to-end encrypted messages.": "Egiztatu gailu hau fidagarri gisa markatzeko. Gailu hau fidagarritzat jotzeak lasaitasuna ematen du muturretik-muturrera zifratutako mezuak erabiltzean.", | ||||
|     "Verifying this device will mark it as trusted, and users who have verified with you will trust this device.": "Gailu hau egiaztatzean fidagarri gisa markatuko da, eta egiaztatu zaituzten erabiltzaileek fidagarri gisa ikusiko dute.", | ||||
|  | @ -2248,5 +2248,23 @@ | |||
|     "Security & privacy": "Segurtasuna eta pribatutasuna", | ||||
|     "All settings": "Ezarpen guztiak", | ||||
|     "Feedback": "Iruzkinak", | ||||
|     "Use a different passphrase?": "Erabili pasa-esaldi desberdin bat?" | ||||
|     "Use a different passphrase?": "Erabili pasa-esaldi desberdin bat?", | ||||
|     "We’re excited to announce Riot is now Element": "Pozik jakinarazten dizugu: Riot orain Element deitzen da", | ||||
|     "Riot is now Element!": "Riot orain Element da!", | ||||
|     "Learn More": "Ikasi gehiago", | ||||
|     "Light": "Argia", | ||||
|     "The person who invited you already left the room.": "Gonbidatu zaituen pertsonak dagoeneko gela utzi du.", | ||||
|     "The person who invited you already left the room, or their server is offline.": "Gonbidatu zaituen pertsonak dagoeneko gela utzi du edo bere zerbitzaria lineaz kanpo dago.", | ||||
|     "You joined the call": "Deira batu zara", | ||||
|     "%(senderName)s joined the call": "%(senderName)s deira batu da", | ||||
|     "You left the call": "Deitik atera zara", | ||||
|     "%(senderName)s left the call": "%(senderName)s(e) deitik atera da", | ||||
|     "Call ended": "Deia amaitu da", | ||||
|     "You started a call": "Dei bat hasi duzu", | ||||
|     "%(senderName)s started a call": "%(senderName)s(e)k dei bat hasi du", | ||||
|     "%(senderName)s is calling": "%(senderName)s deitzen ari da", | ||||
|     "%(brand)s Web": "%(brand)s web", | ||||
|     "%(brand)s Desktop": "%(brand)s Desktop", | ||||
|     "%(brand)s iOS": "%(brand)s iOS", | ||||
|     "%(brand)s X for Android": "%(brand)s X for Android" | ||||
| } | ||||
|  |  | |||
|  | @ -2388,5 +2388,7 @@ | |||
|     "Click to view edits": "Preme para ver as edicións", | ||||
|     "Are you sure you want to cancel entering passphrase?": "¿Estás seguro de que non queres escribir a frase de paso?", | ||||
|     "Custom Tag": "Etiqueta personal", | ||||
|     "* %(senderName)s %(emote)s": "* %(senderName)s %(emote)s" | ||||
|     "* %(senderName)s %(emote)s": "* %(senderName)s %(emote)s", | ||||
|     "The person who invited you already left the room.": "A persoa que te convidou xa deixou a sala.", | ||||
|     "The person who invited you already left the room, or their server is offline.": "A persoa que te convidou xa deixou a sala, ou o seu servidor non está a funcionar." | ||||
| } | ||||
|  |  | |||
|  | @ -2387,5 +2387,22 @@ | |||
|     "Click to view edits": "A szerkesztések megtekintéséhez kattints", | ||||
|     "You’re already signed in and good to go here, but you can also grab the latest versions of the app on all platforms at <a>element.io/get-started</a>.": "Már be vagy jelentkezve és ez rendben van, de minden platformon az alkalmazás legfrissebb verziójának beszerzéséhez látogass el ide: <a>element.io/get-started</a>.", | ||||
|     "You can use the custom server options to sign into other Matrix servers by specifying a different homeserver URL. This allows you to use %(brand)s with an existing Matrix account on a different homeserver.": "Használhatod a más szerver opciót, hogy egy másik matrix szerverre jelentkezz be amihez megadod a szerver url címét. Ezzel használhatod %(brand)s klienst egy már létező Matrix fiókkal egy másik matrix szerveren.", | ||||
|     "Enter the location of your Element Matrix Services homeserver. It may use your own domain name or be a subdomain of <a>element.io</a>.": "Add meg az Element Matrix Services matrix szerveredet. Használhatod a saját domain-edet vagy az <a>element.io</a> al-domain-jét." | ||||
|     "Enter the location of your Element Matrix Services homeserver. It may use your own domain name or be a subdomain of <a>element.io</a>.": "Add meg az Element Matrix Services matrix szerveredet. Használhatod a saját domain-edet vagy az <a>element.io</a> al-domain-jét.", | ||||
|     "* %(senderName)s %(emote)s": "* %(senderName)s %(emote)s", | ||||
|     "The person who invited you already left the room.": "Aki meghívott a szobába már távozott.", | ||||
|     "The person who invited you already left the room, or their server is offline.": "Aki meghívott a szobába már távozott, vagy a szervere elérhetetlen.", | ||||
|     "Change notification settings": "Értesítési beállítások megváltoztatása", | ||||
|     "Your server isn't responding to some <a>requests</a>.": "A szervered nem válaszol néhány <a>kérésre</a>.", | ||||
|     "You're all caught up.": "Mindent elolvastál.", | ||||
|     "Server isn't responding": "A szerver nem válaszol", | ||||
|     "Your server isn't responding to some of your requests. Below are some of the most likely reasons.": "A szervered néhány kérésre nem válaszol. Alább felsorolunk pár lehetséges okot.", | ||||
|     "The server (%(serverName)s) took too long to respond.": "A szervernek (%(serverName)s) túl hosszú időbe telt válaszolni.", | ||||
|     "Your firewall or anti-virus is blocking the request.": "A tűzfalad vagy víruskeresőd blokkolja a kérést.", | ||||
|     "A browser extension is preventing the request.": "Egy böngésző kiterjesztés megakadályozza a kérést.", | ||||
|     "The server is offline.": "A szerver nem működik.", | ||||
|     "The server has denied your request.": "A szerver elutasította a kérést.", | ||||
|     "Your area is experiencing difficulties connecting to the internet.": "Probléma az Internet elérésben.", | ||||
|     "A connection error occurred while trying to contact the server.": "Kapcsolati hiba lépett fel miközben a szervert próbáltad elérni.", | ||||
|     "The server is not configured to indicate what the problem is (CORS).": "A szerver nincs beállítva, hogy megmutassa mi okozhatta a hibát (CORS).", | ||||
|     "Recent changes that have not yet been received": "Legutóbbi változások amik még nem érkeztek meg" | ||||
| } | ||||
|  |  | |||
|  | @ -298,7 +298,7 @@ | |||
|     "Low Priority": "Lítill forgangur", | ||||
|     "Direct Chat": "Beint spjall", | ||||
|     "View Community": "Skoða samfélag", | ||||
|     "I understand the risks and wish to continue": "Ég skil áhættuna og vil halda áfram", | ||||
|     "I understand the risks and wish to continue": "Ég skil áhættuna og óska að halda áfram", | ||||
|     "Name": "Nafn", | ||||
|     "Failed to upload image": "Gat ekki sent inn mynd", | ||||
|     "Add rooms to this community": "Bæta spjallrásum í þetta samfélag", | ||||
|  | @ -456,5 +456,7 @@ | |||
|     "Notify the whole room": "Tilkynna öllum á spjallrásinni", | ||||
|     "Room Notification": "Tilkynning á spjallrás", | ||||
|     "Passphrases must match": "Lykilfrasar verða að stemma", | ||||
|     "Passphrase must not be empty": "Lykilfrasi má ekki vera auður" | ||||
|     "Passphrase must not be empty": "Lykilfrasi má ekki vera auður", | ||||
|     "Create Account": "Stofna Reikning", | ||||
|     "Please install <chromeLink>Chrome</chromeLink>, <firefoxLink>Firefox</firefoxLink>, or <safariLink>Safari</safariLink> for the best experience.": "vinsamlegast setja upp <chromeLink>Chrome</chromeLink>, <firefoxLink>Firefox</firefoxLink>, eða <safariLink>Safari</safariLink> fyrir besta reynsluna." | ||||
| } | ||||
|  |  | |||
|  | @ -1360,5 +1360,15 @@ | |||
|     "Leave Room": "部屋を退出", | ||||
|     "Failed to connect to integration manager": "インテグレーションマネージャへの接続に失敗しました", | ||||
|     "Start verification again from their profile.": "プロフィールから再度検証を開始してください。", | ||||
|     "Integration Manager": "インテグレーションマネージャ" | ||||
|     "Integration Manager": "インテグレーションマネージャ", | ||||
|     "Do not use an identity server": "ID サーバーを使用しない", | ||||
|     "Composer": "入力欄", | ||||
|     "Sort by": "並び替え", | ||||
|     "List options": "一覧の設定", | ||||
|     "Use Single Sign On to continue": "シングルサインオンを使用して続行", | ||||
|     "Accept <policyLink /> to continue:": "<policyLink /> に同意して続行:", | ||||
|     "Always show the window menu bar": "常にウィンドウメニューバーを表示する", | ||||
|     "Create room": "部屋を作成", | ||||
|     "Show %(count)s more|other": "さらに %(count)s 件を表示", | ||||
|     "Show %(count)s more|one": "さらに %(count)s 件を表示" | ||||
| } | ||||
|  |  | |||
|  | @ -3,7 +3,7 @@ | |||
|     "This phone number is already in use": ".i xa'o pilno fa da le fonxa judri", | ||||
|     "Failed to verify email address: make sure you clicked the link in the email": ".i da nabmi fi lo nu facki le du'u do ponse le te samymri .i ko birti le du'u do samcu'a le judrysni pe le se samymri", | ||||
|     "The platform you're on": "jicmu vau je se pilno do", | ||||
|     "Your language of choice": "se cuxna fi le ka bangu", | ||||
|     "Your language of choice": "se cuxna fo le ka bangu", | ||||
|     "Which officially provided instance you are using, if any": "samtcise'u vau je catni jai te selfu vau je se pilno do", | ||||
|     "Whether or not you're using the Richtext mode of the Rich Text Editor": "jei do pilno le se jadni ciska tadji pe le notci ciska tutci", | ||||
|     "Your homeserver's URL": "judri le samtcise'u", | ||||
|  | @ -30,7 +30,7 @@ | |||
|     "Unable to capture screen": ".i na ka'e facki le du'u vidvi fi le vidni", | ||||
|     "Existing Call": ".i xa'o ca'o fonjo'e", | ||||
|     "Server may be unavailable, overloaded, or you hit a bug.": ".i la'a cu'i gi ja le samtcise'u cu spofu vau ja mutce le ka gunka gi da samcfi", | ||||
|     "Send": "benji", | ||||
|     "Send": "nu zilbe'i", | ||||
|     "Sun": "jy. dy. ze", | ||||
|     "Mon": "jy. dy. pa", | ||||
|     "Tue": "jy. dy. re", | ||||
|  | @ -100,7 +100,7 @@ | |||
|     "/ddg is not a command": "zoi ny. /ddg .ny. na nu minde", | ||||
|     "Changes your display nickname": "", | ||||
|     "Invites user with given id to current room": ".i vi'ecpe lo pilno poi se judri ti ku le kumfa pe'a", | ||||
|     "Leave room": "nu cliva le ve zilbe'i", | ||||
|     "Leave room": "nu do zilvi'u le se zilbe'i", | ||||
|     "Kicks user with given id": ".i rinka lo nu lo pilno poi se judri ti cu cliva", | ||||
|     "Bans user with given id": ".i rinka lo nu lo pilno poi se judri ti cu vitno cliva", | ||||
|     "Ignores a user, hiding their messages from you": ".i rinka lo nu no'e jundi lo pilno gi'e mipri lo notci be fi py. do", | ||||
|  | @ -200,8 +200,8 @@ | |||
|     "Accept": "nu fonjo'e", | ||||
|     "Error": "nabmi", | ||||
|     "Incorrect verification code": ".i na'e drani ke lacri lerpoi", | ||||
|     "Submit": "benji", | ||||
|     "Phone": "lo fonxa", | ||||
|     "Submit": "nu zilbe'i", | ||||
|     "Phone": "fonxa", | ||||
|     "Add": "jmina", | ||||
|     "Failed to upload profile picture!": ".i da nabmi lo nu kibdu'a le pixra sinxa", | ||||
|     "No display name": ".i na da cmene", | ||||
|  |  | |||
|  | @ -9,7 +9,7 @@ | |||
|     "Are you sure you want to reject the invitation?": "Você tem certeza que deseja rejeitar este convite?", | ||||
|     "Banned users": "Usuárias/os banidas/os", | ||||
|     "Bans user with given id": "Banir usuários com o identificador informado", | ||||
|     "%(senderDisplayName)s changed the topic to \"%(topic)s\".": "%(senderDisplayName)s mudou o tópico para \"%(topic)s\".", | ||||
|     "%(senderDisplayName)s changed the topic to \"%(topic)s\".": "%(senderDisplayName)s mudou a descrição para \"%(topic)s\".", | ||||
|     "Changes your display nickname": "Troca o seu apelido", | ||||
|     "Click here to fix": "Clique aqui para resolver isso", | ||||
|     "Commands": "Comandos", | ||||
|  | @ -29,7 +29,7 @@ | |||
|     "Failed to leave room": "Falha ao tentar deixar a sala", | ||||
|     "Failed to reject invitation": "Falha ao tentar rejeitar convite", | ||||
|     "Failed to unban": "Não foi possível desfazer o banimento", | ||||
|     "Favourite": "Favorito", | ||||
|     "Favourite": "Favoritar", | ||||
|     "Favourites": "Favoritos", | ||||
|     "Filter room members": "Filtrar integrantes da sala", | ||||
|     "Forget room": "Esquecer sala", | ||||
|  | @ -66,7 +66,7 @@ | |||
|     "Privileged Users": "Usuárias/os privilegiadas/os", | ||||
|     "Profile": "Perfil", | ||||
|     "Reject invitation": "Rejeitar convite", | ||||
|     "Remove": "Remover", | ||||
|     "Remove": "Apagar", | ||||
|     "Return to login screen": "Retornar à tela de login", | ||||
|     "Room Colour": "Cores da sala", | ||||
|     "Rooms": "Salas", | ||||
|  | @ -275,7 +275,7 @@ | |||
|     "Failed to invite": "Falha ao enviar o convite", | ||||
|     "Failed to invite the following users to the %(roomName)s room:": "Falha ao convidar as(os) seguintes usuárias(os) para a sala %(roomName)s:", | ||||
|     "Confirm Removal": "Confirmar a remoção", | ||||
|     "Are you sure you wish to remove (delete) this event? Note that if you delete a room name or topic change, it could undo the change.": "Você tem certeza que quer apagar este evento? Note que se você apaga o nome de uma sala ou uma mudança de tópico, esta ação não poderá ser desfeita.", | ||||
|     "Are you sure you wish to remove (delete) this event? Note that if you delete a room name or topic change, it could undo the change.": "Tem certeza de que deseja apagar este evento? Observe que, se você apagar o nome ou alterar a descrição de uma sala, pode desfazer a alteração.", | ||||
|     "Unknown error": "Erro desconhecido", | ||||
|     "Incorrect password": "Senha incorreta", | ||||
|     "Unable to restore session": "Não foi possível restaurar a sessão", | ||||
|  | @ -311,17 +311,17 @@ | |||
|     "Invited": "Convidada(o)", | ||||
|     "Results from DuckDuckGo": "Resultados de DuckDuckGo", | ||||
|     "Verified key": "Chave verificada", | ||||
|     "%(senderDisplayName)s removed the room avatar.": "%(senderDisplayName)s removeu a imagem da sala.", | ||||
|     "%(senderDisplayName)s removed the room avatar.": "%(senderDisplayName)s removeu a foto da sala.", | ||||
|     "%(senderDisplayName)s changed the avatar for %(roomName)s": "%(senderDisplayName)s alterou a imagem da sala %(roomName)s", | ||||
|     "%(senderDisplayName)s changed the room avatar to <img/>": "%(senderDisplayName)s alterou a imagem da sala para <img/>", | ||||
|     "%(senderDisplayName)s changed the room avatar to <img/>": "%(senderDisplayName)s alterou a foto da sala para <img/>", | ||||
|     "No Microphones detected": "Não foi detectado nenhum microfone", | ||||
|     "No Webcams detected": "Não foi detectada nenhuma Webcam", | ||||
|     "No media permissions": "Não há permissões de uso de vídeo/áudio no seu navegador", | ||||
|     "You may need to manually permit %(brand)s to access your microphone/webcam": "Você talvez precise autorizar manualmente que o %(brand)s acesse seu microfone e webcam", | ||||
|     "Default Device": "Dispositivo padrão", | ||||
|     "Default Device": "Aparelho padrão", | ||||
|     "Microphone": "Microfone", | ||||
|     "Camera": "Câmera de vídeo", | ||||
|     "Add a topic": "Adicionar um tópico", | ||||
|     "Add a topic": "Adicionar uma descrição", | ||||
|     "Anyone": "Qualquer pessoa", | ||||
|     "Are you sure you want to leave the room '%(roomName)s'?": "Você tem certeza que deseja sair da sala '%(roomName)s'?", | ||||
|     "Custom level": "Nível personalizado", | ||||
|  | @ -400,7 +400,7 @@ | |||
|     "Edit": "Editar", | ||||
|     "Unpin Message": "Desafixar Mensagem", | ||||
|     "Add rooms to this community": "Adicionar salas na comunidade", | ||||
|     "The version of %(brand)s": "A Versão do %(brand)s", | ||||
|     "The version of %(brand)s": "A versão do %(brand)s", | ||||
|     "The platform you're on": "A plataforma que você está usando", | ||||
|     "Your language of choice": "O idioma que você selecionou", | ||||
|     "Which officially provided instance you are using, if any": "Qual instância oficial você está usando, se for o caso", | ||||
|  | @ -746,7 +746,7 @@ | |||
|     "Off": "Desativado", | ||||
|     "Mentions only": "Apenas menções", | ||||
|     "Wednesday": "Quarta", | ||||
|     "You can now return to your account after signing out, and sign in on other devices.": "Você pode retornar agora para a sua conta depois de fazer logout, e então fazer login em outros dispositivos.", | ||||
|     "You can now return to your account after signing out, and sign in on other devices.": "Agora você pode retornar à sua conta depois de sair, e fazer login em outros aparelhos.", | ||||
|     "Enable email notifications": "Ativar notificações por email", | ||||
|     "Event Type": "Tipo do Evento", | ||||
|     "Download this file": "Baixar este arquivo", | ||||
|  | @ -762,7 +762,7 @@ | |||
|     "Checking for an update...": "Verificando se há atualizações...", | ||||
|     "Every page you use in the app": "Toda a página que você usa no aplicativo", | ||||
|     "e.g. <CurrentPageURL>": "ex. <CurrentPageURL>", | ||||
|     "Your device resolution": "Sua resolução de dispositivo", | ||||
|     "Your device resolution": "A resolução do seu aparelho", | ||||
|     "Call in Progress": "Chamada em andamento", | ||||
|     "A call is currently being placed!": "Uma chamada está sendo feita atualmente!", | ||||
|     "A call is already in progress!": "Uma chamada já está em andamento!", | ||||
|  | @ -825,7 +825,7 @@ | |||
|     "Unable to load key backup status": "Não é possível carregar o status da chave de backup", | ||||
|     "Backup version: ": "Versão do Backup: ", | ||||
|     "Algorithm: ": "Algoritmo: ", | ||||
|     "This event could not be displayed": "O evento não pôde ser exibido", | ||||
|     "This event could not be displayed": "Este evento não pôde ser exibido", | ||||
|     "Use a longer keyboard pattern with more turns": "Use um padrão de teclas em diferentes direções e sentido", | ||||
|     "Share Link to User": "Compartilhar Link com Usuário", | ||||
|     "This room has been replaced and is no longer active.": "Esta sala foi substituída e não está mais ativa.", | ||||
|  | @ -865,9 +865,9 @@ | |||
|     "Failed to send logs: ": "Falha ao enviar registros: ", | ||||
|     "Submit debug logs": "Submeter registros de depuração", | ||||
|     "Debug logs contain application usage data including your username, the IDs or aliases of the rooms or groups you have visited and the usernames of other users. They do not contain messages.": "Os registros de depuração contêm dados de uso do aplicativo, incluindo seu nome de usuário, os IDs ou aliases das salas ou grupos que você visitou e os nomes de usuários de outros usuários. Eles não contêm mensagens.", | ||||
|     "Before submitting logs, you must <a>create a GitHub issue</a> to describe your problem.": "Antes de enviar os registros, você deve <a>criar uma questão no GitHub</a> para descrever seu problema.", | ||||
|     "Before submitting logs, you must <a>create a GitHub issue</a> to describe your problem.": "Antes de enviar os registros, você deve <a>criar um bilhete de erro no GitHub</a> para descrever seu problema.", | ||||
|     "Unable to load commit detail: %(msg)s": "Não é possível carregar os detalhes do commit: %(msg)s", | ||||
|     "To avoid losing your chat history, you must export your room keys before logging out. You will need to go back to the newer version of %(brand)s to do this": "Para evitar perder seu histórico de bate-papo, você deve exportar as chaves do seu quarto antes de fazer logout. Você precisará voltar para a versão mais recente do %(brand)s para fazer isso", | ||||
|     "To avoid losing your chat history, you must export your room keys before logging out. You will need to go back to the newer version of %(brand)s to do this": "Para evitar perder seu histórico de bate-papo, você deve exportar as chaves da sua sala antes de se desconectar. Para fazer isso, você precisará retornar na versão mais atual do %(brand)s", | ||||
|     "Incompatible Database": "Banco de dados incompatível", | ||||
|     "Continue With Encryption Disabled": "Continuar com criptografia desativada", | ||||
|     "This will make your account permanently unusable. You will not be able to log in, and no one will be able to re-register the same user ID. This will cause your account to leave all rooms it is participating in, and it will remove your account details from your identity server. <b>This action is irreversible.</b>": "Isso tornará sua conta permanentemente inutilizável. Você não poderá efetuar login e ninguém poderá registrar novamente o mesmo ID de usuário. Isso fará com que sua conta deixe todas as salas nas quais está participando e removerá os detalhes da sua conta do seu servidor de identidade. <b>Esta ação é irreversível.</ b>", | ||||
|  | @ -933,7 +933,7 @@ | |||
|     "You can't send any messages until you review and agree to <consentLink>our terms and conditions</consentLink>.": "Você não pode enviar nenhuma mensagem até revisar e concordar com <consentLink>nossos termos e condições</consentLink>.", | ||||
|     "Your message wasn't sent because this homeserver has hit its Monthly Active User Limit. Please <a>contact your service administrator</a> to continue using the service.": "Sua mensagem não foi enviada porque este homeserver atingiu seu Limite de usuário ativo mensal. Por favor, <a>entre em contato com o seu administrador de serviços</a> para continuar usando o serviço.", | ||||
|     "Your message wasn't sent because this homeserver has exceeded a resource limit. Please <a>contact your service administrator</a> to continue using the service.": "Sua mensagem não foi enviada porque este homeserver excedeu o limite de recursos. Por favor, <a>entre em contato com o seu administrador de serviços</a> para continuar usando o serviço.", | ||||
|     "If you've submitted a bug via GitHub, debug logs can help us track down the problem. Debug logs contain application usage data including your username, the IDs or aliases of the rooms or groups you have visited and the usernames of other users. They do not contain messages.": "Se você enviou um bug por meio do GitHub, os logs de depuração podem nos ajudar a rastrear o problema. Os logs de depuração contêm dados de uso do aplicativo, incluindo seu nome de usuário, os IDs ou aliases das salas ou grupos que você visitou e os nomes de usuários de outros usuários. Eles não contêm mensagens.", | ||||
|     "If you've submitted a bug via GitHub, debug logs can help us track down the problem. Debug logs contain application usage data including your username, the IDs or aliases of the rooms or groups you have visited and the usernames of other users. They do not contain messages.": "Se você enviou um bug por meio do GitHub, os logs de depuração podem nos ajudar a rastrear o problema. Os logs de depuração contêm dados de uso do aplicativo, incluindo seu nome de usuário, os IDs ou apelidos das salas ou grupos que você visitou e os nomes de usuários de outros usuários. Eles não contêm mensagens.", | ||||
|     "Legal": "Legal", | ||||
|     "No Audio Outputs detected": "Nenhuma saída de áudio detectada", | ||||
|     "Audio Output": "Saída de áudio", | ||||
|  | @ -945,7 +945,7 @@ | |||
|     "Sign in with single sign-on": "Entre com o logon único", | ||||
|     "That matches!": "Isto corresponde!", | ||||
|     "That doesn't match.": "Isto não corresponde.", | ||||
|     "Go back to set it again.": "Volte e configure novamente.", | ||||
|     "Go back to set it again.": "Voltar para configurar novamente.", | ||||
|     "Download": "Baixar", | ||||
|     "<b>Print it</b> and store it somewhere safe": "<b>Imprima-o</ b> e armazene-o em algum lugar seguro", | ||||
|     "<b>Save it</b> on a USB key or backup drive": "<b>Salve isto</ b> em uma chave USB ou unidade de backup", | ||||
|  | @ -968,8 +968,8 @@ | |||
|     "Invite anyway": "Convide mesmo assim", | ||||
|     "Whether or not you're logged in (we don't record your username)": "Se você está logado ou não (não gravamos seu nome de usuário)", | ||||
|     "Upgrades a room to a new version": "Atualiza uma sala para uma nova versão", | ||||
|     "Gets or sets the room topic": "Obtém ou define o tópico da sala", | ||||
|     "This room has no topic.": "Esta sala não tem assunto.", | ||||
|     "Gets or sets the room topic": "Consultar ou definir a descrição da sala", | ||||
|     "This room has no topic.": "Esta sala não tem descrição.", | ||||
|     "Sets the room name": "Define o nome da sala", | ||||
|     "Group & filter rooms by custom tags (refresh to apply changes)": "Agrupar e filtrar salas por tags personalizadas (atualize para aplicar as alterações)", | ||||
|     "Render simple counters in room header": "Renderizar contadores simples no cabeçalho da sala", | ||||
|  | @ -1127,19 +1127,19 @@ | |||
|     "Room version:": "Versão da sala:", | ||||
|     "Developer options": "Opções de desenvolvedor", | ||||
|     "Room Addresses": "Endereços da sala", | ||||
|     "Change room avatar": "Alterar avatar da sala", | ||||
|     "Change room avatar": "Alterar a foto da sala", | ||||
|     "Change room name": "Alterar nome da sala", | ||||
|     "Change main address for the room": "Alterar o endereço principal da sala", | ||||
|     "Change history visibility": "Alterar a visibilidade do histórico", | ||||
|     "Change permissions": "Alterar permissões", | ||||
|     "Change topic": "Alterar o tópico", | ||||
|     "Change topic": "Alterar a descrição", | ||||
|     "Modify widgets": "Modificar widgets", | ||||
|     "Default role": "Papel padrão", | ||||
|     "Send messages": "Enviar mensagens", | ||||
|     "Invite users": "Convidar usuários", | ||||
|     "Use Single Sign On to continue": "Use \"Single Sign On\" para continuar", | ||||
|     "Confirm adding this email address by using Single Sign On to prove your identity.": "Confirme a inclusão deste endereço de correio eletrônico usando o Single Sign On para comprovar sua identidade.", | ||||
|     "Single Sign On": "Single Sign On", | ||||
|     "Single Sign On": "Autenticação Única", | ||||
|     "Confirm adding email": "Confirmar a inclusão de email", | ||||
|     "Click the button below to confirm adding this email address.": "Clique no botão abaixo para confirmar a adição deste endereço de email.", | ||||
|     "Confirm": "Confirmar", | ||||
|  | @ -1147,17 +1147,17 @@ | |||
|     "Confirm adding phone number": "Confirmar adição de número de telefone", | ||||
|     "Add Phone Number": "Adicionar número de telefone", | ||||
|     "Whether you're using %(brand)s on a device where touch is the primary input mechanism": "Se estiver usando %(brand)s em um aparelho onde touch é o mecanismo primário de entrada de dados", | ||||
|     "Whether or not you're using the 'breadcrumbs' feature (avatars above the room list)": "Se você está usando a funcionalidade 'breadcrumbs' (imagens acima da lista de salas)", | ||||
|     "Whether or not you're using the 'breadcrumbs' feature (avatars above the room list)": "Se você está usando ou não a funcionalidade 'breadcrumbs' (fotos acima da lista de salas)", | ||||
|     "Whether you're using %(brand)s as an installed Progressive Web App": "Se estiver usando %(brand)s como uma Progressive Web App (PWA)", | ||||
|     "Your user agent": "Seu agente de usuária(o)", | ||||
|     "Call failed due to misconfigured server": "A chamada caiu por conta de má configuração do servidor", | ||||
|     "Please ask the administrator of your homeserver (<code>%(homeserverDomain)s</code>) to configure a TURN server in order for calls to work reliably.": "Por favor, peça aos administradores do seu servidor (<code>%(homeserverDomain)s</code>) para configurar um servidor TURN para que as chamadas funcionem de forma estável.", | ||||
|     "Alternatively, you can try to use the public server at <code>turn.matrix.org</code>, but this will not be as reliable, and it will share your IP address with that server. You can also manage this in Settings.": "Alternativamente, pode tentar usar o servidor público em <code>turn.matrix.org</code>, mas não será tão fiável e partilhará o seu IP com esse servidor. Também pode gerir isso nas definições.", | ||||
|     "Try using turn.matrix.org": "Tentar utilizar turn.matrix.org", | ||||
|     "Call failed due to misconfigured server": "A chamada caiu por conta de má configuração no servidor", | ||||
|     "Please ask the administrator of your homeserver (<code>%(homeserverDomain)s</code>) to configure a TURN server in order for calls to work reliably.": "Por favor, peça ao administrador do seu servidor (<code>%(homeserverDomain)s</code>) para configurar um servidor TURN, de modo que as chamadas funcionem de maneira estável.", | ||||
|     "Alternatively, you can try to use the public server at <code>turn.matrix.org</code>, but this will not be as reliable, and it will share your IP address with that server. You can also manage this in Settings.": "Alternativamente, você pode tentar usar o servidor público em <code>turn.matrix.org</code>. No entanto, ele não é tão confiável e compartilhará o seu IP com esse servidor. Você também pode configurar isso nas Configurações.", | ||||
|     "Try using turn.matrix.org": "Tente utilizar turn.matrix.org", | ||||
|     "Replying With Files": "Responder com arquivos", | ||||
|     "At this time it is not possible to reply with a file. Would you like to upload this file without replying?": "Neste momento não é possível responder com um arquivo. Você quer fazer upload deste arquivo sem responder à mensagem?", | ||||
|     "The file '%(fileName)s' failed to upload.": "O arquivo '%(fileName)s' não pôde ser enviado.", | ||||
|     "The server does not support the room version specified.": "Este servidor não suporta a versão de sala especificada.", | ||||
|     "At this time it is not possible to reply with a file. Would you like to upload this file without replying?": "Momentaneamente, não é possível responder com um arquivo. Você quer fazer o envio deste arquivo sem responder a mensagem?", | ||||
|     "The file '%(fileName)s' failed to upload.": "O envio do arquivo '%(fileName)s' falhou.", | ||||
|     "The server does not support the room version specified.": "O servidor não suporta a versão da sala especificada.", | ||||
|     "Cancel entering passphrase?": "Cancelar a introdução da frase de senha?", | ||||
|     "Are you sure you want to cancel entering passphrase?": "Tem certeza que quer cancelar a introdução da frase de senha?", | ||||
|     "Go Back": "Voltar", | ||||
|  | @ -1189,7 +1189,7 @@ | |||
|     "Changes the avatar of the current room": "Altera a imagem da sala atual", | ||||
|     "Changes your avatar in this current room only": "Muda sua imagem de perfil apenas nesta sala", | ||||
|     "Changes your avatar in all rooms": "Muda sua imagem de perfil em todas as salas", | ||||
|     "Failed to set topic": "Não foi possível definir o tópico", | ||||
|     "Failed to set topic": "Não foi possível definir a descrição", | ||||
|     "Use an identity server": "Usar um servidor de identidade", | ||||
|     "Use an identity server to invite by email. Manage in Settings.": "Use um servidor de identidade para convidar pessoas por email. Gerencie nas Configurações.", | ||||
|     "Joins room with given address": "Entra em uma sala com o endereço fornecido", | ||||
|  | @ -1215,7 +1215,7 @@ | |||
|     "Opens chat with the given user": "Abre um chat com determinada pessoa", | ||||
|     "Sends a message to the given user": "Envia uma mensagem com determinada pessoa", | ||||
|     "%(senderName)s made no change.": "%(senderName)s não fez nenhuma alteração.", | ||||
|     "%(senderDisplayName)s changed the room name from %(oldRoomName)s to %(newRoomName)s.": "%(senderDisplayName)s alterou o nome da sala de %(oldRoomName)s para", | ||||
|     "%(senderDisplayName)s changed the room name from %(oldRoomName)s to %(newRoomName)s.": "%(senderDisplayName)s alterou o nome da sala de %(oldRoomName)s para %(newRoomName)s.", | ||||
|     "%(senderName)s added the alternative addresses %(addresses)s for this room.|other": "%(senderName)s adicionou os endereços alternativos %(addresses)s para esta sala.", | ||||
|     "%(senderName)s added the alternative addresses %(addresses)s for this room.|one": "%(senderName)s adicionou o endereço alternativo %(addresses)s para esta sala.", | ||||
|     "%(senderName)s removed the alternative addresses %(addresses)s for this room.|other": "%(senderName)s removeu os endereços alternativos %(addresses)s para esta sala.", | ||||
|  | @ -1228,10 +1228,10 @@ | |||
|     "%(senderName)s placed a video call.": "%(senderName)s iniciou uma chamada de vídeo.", | ||||
|     "%(senderName)s placed a video call. (not supported by this browser)": "%(senderName)s iniciou uma chamada de vídeo. (não suportada por este navegador)", | ||||
|     "%(senderName)s revoked the invitation for %(targetDisplayName)s to join the room.": "%(senderName)s cancelou o convite a %(targetDisplayName)s para entrar na sala.", | ||||
|     "%(senderName)s removed the rule banning users matching %(glob)s": "%(senderName)s removeu a regra banindo usuárias(os) correspondendo a", | ||||
|     "%(senderName)s removed the rule banning rooms matching %(glob)s": "%(senderName)s removeu uma regra banindo salas correspondendo a", | ||||
|     "%(senderName)s removed the rule banning servers matching %(glob)s": "%(senderName)s removeu a regra banindo servidores correspondendo a", | ||||
|     "%(senderName)s removed a ban rule matching %(glob)s": "%(senderName)s removeu uma regra de banimento correspondendo a", | ||||
|     "%(senderName)s removed the rule banning users matching %(glob)s": "%(senderName)s removeu a regra que bane usuárias(os) que correspondem a %(glob)s", | ||||
|     "%(senderName)s removed the rule banning rooms matching %(glob)s": "%(senderName)s removeu a regra que bane salas que correspondem a %(glob)s", | ||||
|     "%(senderName)s removed the rule banning servers matching %(glob)s": "%(senderName)s removeu a regra que bane servidores que correspondem a %(glob)s", | ||||
|     "%(senderName)s removed a ban rule matching %(glob)s": "%(senderName)s removeu uma regra de banimento correspondendo a %(glob)s", | ||||
|     "%(senderName)s updated an invalid ban rule": "%(senderName)s atualizou uma regra de banimento inválida", | ||||
|     "%(senderName)s updated the rule banning users matching %(glob)s for %(reason)s": "%(senderName)s atualizou a regra de banimento de usuárias(os) correspondendo a %(glob)s por %(reason)s", | ||||
|     "%(senderName)s updated the rule banning rooms matching %(glob)s for %(reason)s": "%(senderName)s atualizou a regra banindo salas correspondendo a %(glob)s por %(reason)s", | ||||
|  | @ -1283,7 +1283,7 @@ | |||
|     "%(num)s days from now": "dentro de %(num)s dias", | ||||
|     "%(name)s (%(userId)s)": "%(name)s (%(userId)s)", | ||||
|     "The user's homeserver does not support the version of the room.": "O servidor desta(e) usuária(o) não suporta a versão desta sala.", | ||||
|     "Help us improve %(brand)s": "Ajude-nos a melhorar o", | ||||
|     "Help us improve %(brand)s": "Ajude-nos a melhorar %(brand)s", | ||||
|     "Send <UsageDataLink>anonymous usage data</UsageDataLink> which helps us improve %(brand)s. This will use a <PolicyLink>cookie</PolicyLink>.": "Envie <UsageDataLink>dados anônimos de uso</UsageDataLink> que nos ajudam a melhorar o %(brand)s. Isso necessitará do uso de um <PolicyLink>cookie</PolicyLink>.", | ||||
|     "I want to help": "Quero ajudar", | ||||
|     "Review where you’re logged in": "Revisar onde você está logada(o)", | ||||
|  | @ -1291,7 +1291,7 @@ | |||
|     "Review": "Revisar", | ||||
|     "Later": "Mais tarde", | ||||
|     "Your homeserver has exceeded its user limit.": "Seu servidor ultrapassou seu limite de usuárias(os).", | ||||
|     "Your homeserver has exceeded one of its resource limits.": "Seu servidor excedeu um de seus limites de recursos", | ||||
|     "Your homeserver has exceeded one of its resource limits.": "Seu servidor excedeu um de seus limites de recursos.", | ||||
|     "Contact your <a>server admin</a>.": "Entre em contato com sua(seu) <a>administrador(a) do servidor</a>.", | ||||
|     "Ok": "Ok", | ||||
|     "Set password": "Definir senha", | ||||
|  | @ -1301,12 +1301,12 @@ | |||
|     "Verify this session": "Verificar esta sessão", | ||||
|     "Upgrade": "Atualizar", | ||||
|     "Verify": "Verificar", | ||||
|     "Verify yourself & others to keep your chats safe": "Faça a sua auto-verificação e verifique seus contatos para manter suas conversas seguras!", | ||||
|     "Verify yourself & others to keep your chats safe": "Verifique a sua conta e as dos seus contatos, para manter suas conversas seguras", | ||||
|     "Other users may not trust it": "Outras(os) usuárias(os) podem não confiar nela", | ||||
|     "New login. Was this you?": "Novo login. Foi você?", | ||||
|     "Verify the new login accessing your account: %(name)s": "Verifique o novo login acessando sua conta:", | ||||
|     "Verify the new login accessing your account: %(name)s": "Verifique o novo login acessando sua conta: %(name)s", | ||||
|     "Restart": "Reiniciar", | ||||
|     "Upgrade your %(brand)s": "Atualize seu", | ||||
|     "Upgrade your %(brand)s": "Atualize o seu %(brand)s", | ||||
|     "A new version of %(brand)s is available!": "Uma nova versão do %(brand)s está disponível!", | ||||
|     "Guest": "Convidada(o)", | ||||
|     "You joined the call": "Você entrou na chamada", | ||||
|  | @ -1319,10 +1319,10 @@ | |||
|     "%(senderName)s started a call": "%(senderName)s iniciou uma chamada", | ||||
|     "Waiting for answer": "Esperando por uma resposta", | ||||
|     "%(senderName)s is calling": "%(senderName)s está chamando", | ||||
|     "* %(senderName)s %(emote)s": "* %(senderName)s", | ||||
|     "%(senderName)s: %(message)s": "%(senderName)s:", | ||||
|     "%(senderName)s: %(reaction)s": "%(senderName)s:", | ||||
|     "%(senderName)s: %(stickerName)s": "%(senderName)s:", | ||||
|     "* %(senderName)s %(emote)s": "* %(senderName)s %(emote)s", | ||||
|     "%(senderName)s: %(message)s": "%(senderName)s: %(message)s", | ||||
|     "%(senderName)s: %(reaction)s": "%(senderName)s: %(reaction)s", | ||||
|     "%(senderName)s: %(stickerName)s": "%(senderName)s: %(stickerName)s", | ||||
|     "New spinner design": "Novo design do spinner", | ||||
|     "Multiple integration managers": "Múltiplos gestores de integrações", | ||||
|     "Try out new ways to ignore people (experimental)": "Tente novas maneiras de ignorar pessoas (experimental)", | ||||
|  | @ -1343,7 +1343,7 @@ | |||
|     "Show shortcuts to recently viewed rooms above the room list": "Mostrar atalhos para salas recentemente visualizadas acima da lista de salas", | ||||
|     "Show hidden events in timeline": "Mostrar eventos ocultos na timeline", | ||||
|     "Low bandwidth mode": "Modo de baixo uso de internet", | ||||
|     "Allow fallback call assist server turn.matrix.org when your homeserver does not offer one (your IP address would be shared during a call)": "Permitir o servidor de respaldo de assistência de chamadas turn.matrix.org quando seu servidor não o ofereça (seu endereço IP será compartilhado numa chamada, neste caso)", | ||||
|     "Allow fallback call assist server turn.matrix.org when your homeserver does not offer one (your IP address would be shared during a call)": "Permitir a assistência do servidor de chamadas reserva turn.matrix.org quando seu servidor não oferecer este serviço (seu endereço IP será transmitido quando você ligar)", | ||||
|     "Send read receipts for messages (requires compatible homeserver to disable)": "Enviar confirmação de leitura para mensagens (necessita um servidor compatível para desativar)", | ||||
|     "Show previews/thumbnails for images": "Mostrar miniaturas e resumos para imagens", | ||||
|     "Enable message search in encrypted rooms": "Ativar busca de mensagens em salas criptografadas", | ||||
|  | @ -1378,14 +1378,14 @@ | |||
|     "Decline (%(counter)s)": "Recusar (%(counter)s)", | ||||
|     "Accept <policyLink /> to continue:": "Aceitar <policyLink /> para continuar:", | ||||
|     "Upload": "Enviar", | ||||
|     "This bridge was provisioned by <user />.": "Esta ponte foi disponibilizada por", | ||||
|     "This bridge was provisioned by <user />.": "Esta ponte foi disponibilizada por <user />.", | ||||
|     "This bridge is managed by <user />.": "Esta ponte é gerida por <user />.", | ||||
|     "Workspace: %(networkName)s": "Espaço de trabalho:", | ||||
|     "Channel: %(channelName)s": "Canal:", | ||||
|     "Workspace: %(networkName)s": "Espaço de trabalho: %(networkName)s", | ||||
|     "Channel: %(channelName)s": "Canal: %(channelName)s", | ||||
|     "Show less": "Mostrar menos", | ||||
|     "Show more": "Mostrar mais", | ||||
|     "Changing password will currently reset any end-to-end encryption keys on all sessions, making encrypted chat history unreadable, unless you first export your room keys and re-import them afterwards. In future this will be improved.": "Ao mudar a senha, você apagará quaisquer chaves de criptografia ponta-a-ponta existentes em todas as sessões, fazendo com que o histórico de conversas criptografadas fique ilegível, a não ser que você exporte as salas das chaves criptografadas antes de mudar a senha e então as importe novamente depois. No futuro, isso será melhorado.", | ||||
|     "Your homeserver does not support cross-signing.": "Seu servidor não suporta assinatura cruzada", | ||||
|     "Your homeserver does not support cross-signing.": "Seu servidor não suporta assinatura cruzada.", | ||||
|     "Cross-signing and secret storage are enabled.": "Assinaturas cruzadas e armazenamento secreto estão habilitadas.", | ||||
|     "Your account has a cross-signing identity in secret storage, but it is not yet trusted by this session.": "Sua conta tem uma identidade de assinatura cruzada em um armazenamento secreto, mas ainda não é considerada confiável por esta sessão.", | ||||
|     "Cross-signing and secret storage are not yet set up.": "A assinatura cruzada e o armazenamento seguro ainda não foram configurados.", | ||||
|  | @ -1439,13 +1439,13 @@ | |||
|     "not stored": "não armazenado", | ||||
|     "Backup has a <validity>valid</validity> signature from this user": "A cópia de segurança (backup) tem uma assinatura <validity>válida</validity> deste(a) usuário(a)", | ||||
|     "Backup has a <validity>invalid</validity> signature from this user": "A cópia de segurança (backup) tem uma assinatura <validity>inválida</validity> deste(a) usuário(a)", | ||||
|     "Backup has a signature from <verify>unknown</verify> user with ID %(deviceId)s": "Fazer cópia de segurança (backup) de usuária(o) <verify>desconhecida(o)</verify> com ID", | ||||
|     "Backup has a signature from <verify>unknown</verify> session with ID %(deviceId)s": "Fazer cópia de segurança (backup) de uma sessão <verify>desconhecida</verify> com ID", | ||||
|     "Backup has a signature from <verify>unknown</verify> user with ID %(deviceId)s": "A cópia de segurança tem uma assinatura de um(a) usuário <verify>desconhecido</verify> com ID %(deviceId)s", | ||||
|     "Backup has a signature from <verify>unknown</verify> session with ID %(deviceId)s": "A cópia de segurança tem uma assinatura de uma sessão <verify>desconhecida</verify> com ID %(deviceId)s", | ||||
|     "Backup has a <validity>valid</validity> signature from this session": "A cópia de segurança (backup) tem uma assinatura <validity>válida</validity> desta sessão", | ||||
|     "Backup has an <validity>invalid</validity> signature from this session": "A cópia de segurança (backup) tem uma assinatura <validity>inválida</validity> desta sessão", | ||||
|     "Backup has a <validity>valid</validity> signature from <verify>verified</verify> session <device></device>": "A cópia de segurança (backup) tem uma assinatura <validity>válida</validity> da sessão <verify>verificada</verify> <device></device>", | ||||
|     "Backup has a <validity>valid</validity> signature from <verify>unverified</verify> session <device></device>": "A cópia de segurança (backup) tem uma assinatura <validity>válida</validity> da sessão <verify>não verificada</verify>", | ||||
|     "Backup has an <validity>invalid</validity> signature from <verify>verified</verify> session <device></device>": "A cópia de segurança (backup) tem uma assinatura <validity>inválida</validity> de uma sessão <verify>verificada</verify>", | ||||
|     "Backup has a <validity>valid</validity> signature from <verify>unverified</verify> session <device></device>": "A cópia de segurança tem uma assinatura <validity>válida</validity> de uma sessão <verify>não verificada</verify> <device></device>", | ||||
|     "Backup has an <validity>invalid</validity> signature from <verify>verified</verify> session <device></device>": "A cópia de segurança tem uma assinatura <validity>inválida</validity> de uma sessão <verify>verificada</verify> <device></device>", | ||||
|     "Backup has an <validity>invalid</validity> signature from <verify>unverified</verify> session <device></device>": "A cópia de segurança (backup) tem uma assinatura <validity>inválida</validity> de uma sessão <verify>não verificada</verify> <device></device>", | ||||
|     "Backup is not signed by any of your sessions": "A cópia de segurança (backup) não foi assinada por nenhuma de suas sessões", | ||||
|     "This backup is trusted because it has been restored on this session": "Esta cópia de segurança (backup) é confiável, pois foi restaurada nesta sessão", | ||||
|  | @ -1509,9 +1509,9 @@ | |||
|     "%(name)s wants to verify": "%(name)s deseja verificar", | ||||
|     "Smileys & People": "Emoticons e Pessoas", | ||||
|     "Widgets do not use message encryption.": "Widgets não usam criptografia de mensagens.", | ||||
|     "Please <newIssueLink>create a new issue</newIssueLink> on GitHub so that we can investigate this bug.": "Por favor, <newIssueLink>crie uma nova issue</newIssueLink> no GitHub para que possamos investigar esta falha.", | ||||
|     "Please <newIssueLink>create a new issue</newIssueLink> on GitHub so that we can investigate this bug.": "Por favor, <newIssueLink>crie um novo bilhete de erro</newIssueLink> no GitHub para que possamos investigar esta falha.", | ||||
|     "Enter the name of a new server you want to explore.": "Entre com o nome do novo servidor que você quer explorar.", | ||||
|     "Please tell us what went wrong or, better, create a GitHub issue that describes the problem.": "Por favor, diga-nos o que aconteceu de errado ou, ainda melhor, crie uma issue no GitHub que descreva o problema.", | ||||
|     "Please tell us what went wrong or, better, create a GitHub issue that describes the problem.": "Por favor, diga-nos o que aconteceu de errado ou, ainda melhor, crie um bilhete de erro no GitHub que descreva o problema.", | ||||
|     "Clearing all data from this session is permanent. Encrypted messages will be lost unless their keys have been backed up.": "Apagar todos os dados desta sessão é uma ação permanente. Mensagens criptografadas serão perdidas, a não ser que suas chaves tenham sido copiadas para o backup.", | ||||
|     "Set a room address to easily share your room with other people.": "Defina um endereço de sala para facilmente compartilhar sua sala com outras pessoas.", | ||||
|     "You can’t disable this later. Bridges & most bots won’t work yet.": "Você não poderá desabilitar depois. Pontes e a maioria dos bots não funcionarão no momento.", | ||||
|  | @ -1531,7 +1531,7 @@ | |||
|     "You'll lose access to your encrypted messages": "Você perderá acesso às suas mensagens criptografadas", | ||||
|     "Session key": "Chave da sessão", | ||||
|     "Verify session": "Verificar sessão", | ||||
|     "We recommend you change your password and recovery key in Settings immediately": "Nós recomendamos que você altere imediatamente sua senha e chave de recuperação nas configurações.", | ||||
|     "We recommend you change your password and recovery key in Settings immediately": "Nós recomendamos que você altere imediatamente sua senha e chave de recuperação nas Configurações", | ||||
|     "Use this session to verify your new one, granting it access to encrypted messages:": "Use esta sessão para verificar a sua nova sessão, dando a ela acesso às mensagens criptografadas:", | ||||
|     "You’re already signed in and good to go here, but you can also grab the latest versions of the app on all platforms at <a>element.io/get-started</a>.": "Você já está logada(o) e pode começar a usar à vontade, mas você também pode buscar pelas últimas versões do app em todas as plataformas em <a>element.io/get-started</a>.", | ||||
|     "Go to Element": "Ir a Element", | ||||
|  | @ -1560,7 +1560,7 @@ | |||
|     "Missing captcha public key in homeserver configuration. Please report this to your homeserver administrator.": "Está faltando a chave pública do captcha no Servidor (homeserver). Por favor, reporte isso aos(às) administradores(as) do servidor.", | ||||
|     "Enter the location of your Element Matrix Services homeserver. It may use your own domain name or be a subdomain of <a>element.io</a>.": "Entre com a localização do seu Servidor Matrix. Pode ser seu próprio domínio ou ser um subdomínio de <a>element.io</a>.", | ||||
|     "Create your Matrix account on %(serverName)s": "Criar sua conta Matrix em %(serverName)s", | ||||
|     "Create your Matrix account on <underlinedServerName />": "Criar sua conta Matrix em", | ||||
|     "Create your Matrix account on <underlinedServerName />": "Crie sua conta Matrix em <underlinedServerName />", | ||||
|     "Welcome to %(appName)s": "Desejamos boas vindas ao %(appName)s", | ||||
|     "Liberate your communication": "Liberte sua comunicação", | ||||
|     "Send a Direct Message": "Envie uma mensagem direta", | ||||
|  | @ -1700,5 +1700,454 @@ | |||
|     "Verification Requests": "Solicitações de verificação", | ||||
|     "Integrations are disabled": "As integrações estão desativadas", | ||||
|     "Integrations not allowed": "As integrações não estão permitidas", | ||||
|     "End": "Fim" | ||||
|     "End": "Fim", | ||||
|     "List options": "Opções da Lista", | ||||
|     "Jump to first unread room.": "Ir para a primeira sala não lida.", | ||||
|     "Jump to first invite.": "Ir para o primeiro convite.", | ||||
|     "Add room": "Adicionar sala", | ||||
|     "Show %(count)s more|other": "Mostrar %(count)s a mais", | ||||
|     "Show %(count)s more|one": "Mostrar %(count)s a mais", | ||||
|     "Use default": "Usar o padrão", | ||||
|     "Room options": "Opções da Sala", | ||||
|     "%(count)s unread messages including mentions.|other": "%(count)s mensagens não lidas, incluindo menções.", | ||||
|     "%(count)s unread messages including mentions.|one": "1 menção não lida.", | ||||
|     "%(count)s unread messages.|other": "%(count)s mensagens não lidas.", | ||||
|     "%(count)s unread messages.|one": "1 mensagem não lida.", | ||||
|     "Unread messages.": "Mensagens não lidas.", | ||||
|     "This room is public": "Esta sala é pública", | ||||
|     "Away": "Ausente", | ||||
|     "This room has already been upgraded.": "Esta sala já foi atualizada.", | ||||
|     "This room is running room version <roomVersion />, which this homeserver has marked as <i>unstable</i>.": "Esta sala está executando a versão <roomVersion />, que este servidor marcou como <i>instável</i>.", | ||||
|     "Local address": "Endereço local", | ||||
|     "Published Addresses": "Endereços publicados", | ||||
|     "Published addresses can be used by anyone on any server to join your room. To publish an address, it needs to be set as a local address first.": "Os endereços publicados podem ser usados por qualquer pessoa em qualquer servidor para entrar na sala. Para publicar um endereço, primeiramente ele precisa ser definido como um endereço local.", | ||||
|     "Other published addresses:": "Outros endereços publicados:", | ||||
|     "New published address (e.g. #alias:server)": "Novo endereço publicado (por exemplo, #apelido:server)", | ||||
|     "Local Addresses": "Endereços locais", | ||||
|     "%(name)s cancelled verifying": "%(name)s cancelou a verificação", | ||||
|     "Your display name": "Seu nome de exibição", | ||||
|     "Your avatar URL": "A URL da sua foto de perfil", | ||||
|     "Your user ID": "Sua ID de usuário", | ||||
|     "%(brand)s URL": "URL de %(brand)s", | ||||
|     "Using this widget may share data <helpIcon /> with %(widgetDomain)s & your Integration Manager.": "Se você usar esse widget, os dados poderão ser compartilhados <helpIcon /> com %(widgetDomain)s & seu Gerenciador de Integrações.", | ||||
|     "Using this widget may share data <helpIcon /> with %(widgetDomain)s.": "Se você usar esse widget, os dados <helpIcon /> poderão ser compartilhados com %(widgetDomain)s.", | ||||
|     "%(severalUsers)smade no changes %(count)s times|other": "%(severalUsers)s não fizeram alterações %(count)s vezes", | ||||
|     "%(severalUsers)smade no changes %(count)s times|one": "%(severalUsers)s não fizeram alterações", | ||||
|     "%(oneUser)smade no changes %(count)s times|other": "%(oneUser)s não fez alteraçõe s%(count)s vezes", | ||||
|     "%(oneUser)smade no changes %(count)s times|one": "%(oneUser)s não fez alterações", | ||||
|     "Power level": "Nível de permissão", | ||||
|     "Please provide a room address": "Digite um endereço para a sala", | ||||
|     "Looks good": "Muito bem", | ||||
|     "Are you sure you want to remove <b>%(serverName)s</b>": "Tem certeza de que deseja remover <b>%(serverName)s</b>", | ||||
|     "%(networkName)s rooms": "Salas em %(networkName)s", | ||||
|     "Matrix rooms": "Salas em Matrix", | ||||
|     "Close dialog": "Fechar caixa de diálogo", | ||||
|     "GitHub issue": "Bilhete de erro no GitHub", | ||||
|     "If there is additional context that would help in analysing the issue, such as what you were doing at the time, room IDs, user IDs, etc., please include those things here.": "Se houver um contexto adicional que ajude a analisar o problema, tal como o que você estava fazendo no momento, IDs de salas, IDs de usuários etc, inclua essas coisas aqui.", | ||||
|     "Topic (optional)": "Descrição (opcional)", | ||||
|     "There was a problem communicating with the server. Please try again.": "Ocorreu um problema na comunicação com o servidor. Por favor, tente novamente.", | ||||
|     "Server did not require any authentication": "O servidor não exigiu autenticação", | ||||
|     "Verifying this user will mark their session as trusted, and also mark your session as trusted to them.": "Se você verificar esse usuário, a sessão será marcada como confiável para você e para ele.", | ||||
|     "Verifying this device will mark it as trusted, and users who have verified with you will trust this device.": "Verificar este aparelho o marcará como confiável, e os usuários que confirmaram com você também confiarão neste aparelho.", | ||||
|     "Keep going...": "Continue...", | ||||
|     "The username field must not be blank.": "O campo do nome de usuário não pode ficar em branco.", | ||||
|     "Username": "Nome de usuário", | ||||
|     "Use an email address to recover your account": "Use um endereço de e-mail para recuperar sua conta", | ||||
|     "Enter email address (required on this homeserver)": "Digite o endereço de e-mail (necessário neste servidor)", | ||||
|     "Doesn't look like a valid email address": "Este não parece ser um endereço de email válido", | ||||
|     "Passwords don't match": "As senhas não correspondem", | ||||
|     "Other users can invite you to rooms using your contact details": "Outros usuários podem convidá-lo para salas usando seus detalhes de contato", | ||||
|     "Enter phone number (required on this homeserver)": "Digite o número de celular (necessário neste servidor)", | ||||
|     "Doesn't look like a valid phone number": "Este não parece ser um número de telefone válido", | ||||
|     "Use lowercase letters, numbers, dashes and underscores only": "Use apenas letras minúsculas, números, traços e sublinhados", | ||||
|     "Enter username": "Digite o nome de usuário", | ||||
|     "Email (optional)": "E-mail (opcional)", | ||||
|     "Phone (optional)": "Número de celular (opcional)", | ||||
|     "Set an email for account recovery. Use email or phone to optionally be discoverable by existing contacts.": "Defina um e-mail para recuperação da conta. Opcionalmente, use e-mail ou número de celular para ser encontrado por seus contatos.", | ||||
|     "You cannot sign in to your account. Please contact your homeserver admin for more information.": "Você não pôde se conectar na sua conta. Entre em contato com o administrador do servidor para obter mais informações.", | ||||
|     "Confirm adding this phone number by using Single Sign On to prove your identity.": "Confirme a adição deste número de telefone usando o Login Único para provar sua identidade.", | ||||
|     "Use an identity server to invite by email. Click continue to use the default identity server (%(defaultIdentityServerName)s) or manage in Settings.": "Use um servidor de identidade para convidar por e-mail. Clique em continuar para usar o servidor de identidade padrão (%(defaultIdentityServerName)s) ou gerencie nas Configurações.", | ||||
|     "You might have configured them in a client other than %(brand)s. You cannot tune them in %(brand)s but they still apply.": "Você pode ter configurado estas opções em um cliente que não seja %(brand)s. Você não pode ajustar essas opções no %(brand)s, mas elas ainda se aplicam.", | ||||
|     "Enable audible notifications for this session": "Ativar notificações sonoras para esta sessão", | ||||
|     "Display Name": "Nome em exibição", | ||||
|     "Identity Server URL must be HTTPS": "O URL do Servidor de Identidade deve ser HTTPS", | ||||
|     "Not a valid Identity Server (status code %(code)s)": "Servidor de Identidade inválido (código de status %(code)s)", | ||||
|     "Could not connect to Identity Server": "Não foi possível conectar-se ao Servidor de Identidade", | ||||
|     "Checking server": "Verificando servidor", | ||||
|     "Change identity server": "Mudar o servidor de identidade", | ||||
|     "Disconnect from the identity server <current /> and connect to <new /> instead?": "Desconectar-se do servidor de identidade <current /> e conectar-se em <new /> em vez disso?", | ||||
|     "Terms of service not accepted or the identity server is invalid.": "Termos de serviço não aceitos ou o servidor de identidade é inválido.", | ||||
|     "The identity server you have chosen does not have any terms of service.": "O servidor de identidade que você escolheu não possui nenhum termo de serviço.", | ||||
|     "Disconnect identity server": "Desconectar servidor de identidade", | ||||
|     "Disconnect from the identity server <idserver />?": "Desconectar-se do servidor de identidade <idserver />?", | ||||
|     "Disconnect": "Desconectar", | ||||
|     "You should <b>remove your personal data</b> from identity server <idserver /> before disconnecting. Unfortunately, identity server <idserver /> is currently offline or cannot be reached.": "Você deve <b>remover seus dados pessoais</b> do servidor de identidade <idserver /> antes de desconectar. Infelizmente, o servidor de identidade <idserver /> está atualmente offline ou não pode ser acessado.", | ||||
|     "You should:": "Você deveria:", | ||||
|     "check your browser plugins for anything that might block the identity server (such as Privacy Badger)": "verifique se há extensões no seu navegador que possam bloquear o servidor de identidade (por exemplo, Privacy Badger)", | ||||
|     "contact the administrators of identity server <idserver />": "entre em contato com os administradores do servidor de identidade <idserver />", | ||||
|     "Disconnect anyway": "Desconectar de qualquer maneira", | ||||
|     "You are still <b>sharing your personal data</b> on the identity server <idserver />.": "Você ainda está <b>compartilhando seus dados pessoais</b> no servidor de identidade <idserver />.", | ||||
|     "We recommend that you remove your email addresses and phone numbers from the identity server before disconnecting.": "Recomendamos que você remova seus endereços de e-mail e números de telefone do servidor de identidade antes de desconectar.", | ||||
|     "Go back": "Voltar", | ||||
|     "Identity Server (%(server)s)": "Servidor de identidade (%(server)s)", | ||||
|     "You are currently using <server></server> to discover and be discoverable by existing contacts you know. You can change your identity server below.": "No momento, você está usando <server></server> para descobrir e ser descoberto pelos contatos existentes que você conhece. Você pode alterar seu servidor de identidade abaixo.", | ||||
|     "If you don't want to use <server /> to discover and be discoverable by existing contacts you know, enter another identity server below.": "Se você não quiser usar <server /> para descobrir e ser detectável pelos contatos existentes, digite outro servidor de identidade abaixo.", | ||||
|     "Identity Server": "Servidor de identidade", | ||||
|     "You are not currently using an identity server. To discover and be discoverable by existing contacts you know, add one below.": "No momento, você não está usando um servidor de identidade. Para descobrir e ser descoberto pelos contatos existentes, adicione um abaixo.", | ||||
|     "Disconnecting from your identity server will mean you won't be discoverable by other users and you won't be able to invite others by email or phone.": "Desconectar-se do servidor de identidade significa que você não poderá ser descoberto por outros usuários e não poderá convidar outras pessoas por e-mail ou número de celular.", | ||||
|     "Using an identity server is optional. If you choose not to use an identity server, you won't be discoverable by other users and you won't be able to invite others by email or phone.": "Usar um servidor de identidade é opcional. Se você optar por não usar um servidor de identidade, não poderá ser descoberto por outros usuários e não poderá convidar outras pessoas por e-mail ou por número de celular.", | ||||
|     "Do not use an identity server": "Não usar um servidor de identidade", | ||||
|     "Enter a new identity server": "Digitar um novo servidor de identidade", | ||||
|     "Change": "Alterar", | ||||
|     "Manage integrations": "Gerenciar integrações", | ||||
|     "New version available. <a>Update now.</a>": "Nova versão disponível. <a>Atualize agora.</a>", | ||||
|     "Hey you. You're the best!": "Ei, você aí. Você é incrível!", | ||||
|     "Size must be a number": "O tamanho deve ser um número", | ||||
|     "Custom font size can only be between %(min)s pt and %(max)s pt": "O tamanho da fonte personalizada só pode estar entre %(min)s pt e %(max)s pt", | ||||
|     "Use between %(min)s pt and %(max)s pt": "Use entre %(min)s pt e %(max)s pt", | ||||
|     "Invalid theme schema.": "Esquema inválido de tema.", | ||||
|     "Error downloading theme information.": "Erro ao baixar as informações do tema.", | ||||
|     "Theme added!": "Tema adicionado!", | ||||
|     "Custom theme URL": "URL do tema personalizado", | ||||
|     "Add theme": "Adicionar tema", | ||||
|     "Message layout": "Aparência da mensagem", | ||||
|     "Compact": "Compacto", | ||||
|     "Modern": "Moderno", | ||||
|     "Set the name of a font installed on your system & %(brand)s will attempt to use it.": "Defina o nome de uma fonte instalada no seu sistema e %(brand)s tentará usá-la.", | ||||
|     "Customise your appearance": "Personalize sua aparência", | ||||
|     "Appearance Settings only affect this %(brand)s session.": "As Configurações de aparência afetam apenas esta sessão do %(brand)s.", | ||||
|     "Your password was successfully changed. You will not receive push notifications on other sessions until you log back in to them": "Sua senha foi alterada com sucesso. Você não receberá notificações por push em outras sessões até fazer login novamente nelas", | ||||
|     "Agree to the identity server (%(serverName)s) Terms of Service to allow yourself to be discoverable by email address or phone number.": "Concordar com os Termos de Serviço do servidor de identidade (%(serverName)s), para permitir aos seus contatos encontrarem-na(no) por endereço de e-mail ou por número de celular.", | ||||
|     "Discovery": "Contatos", | ||||
|     "Deactivate account": "Desativar conta", | ||||
|     "Clear cache and reload": "Limpar cache e recarregar", | ||||
|     "To report a Matrix-related security issue, please read the Matrix.org <a>Security Disclosure Policy</a>.": "Para relatar um problema de segurança relacionado à tecnologia Matrix, leia a <a>Política de Divulgação de Segurança</a> da Matrix.org.", | ||||
|     "Always show the window menu bar": "Sempre mostrar a barra de menu na janela", | ||||
|     "Show tray icon and minimize window to it on close": "Mostrar ícone na barra de tarefas, que permanece visível ao fechar a janela", | ||||
|     "Read Marker lifetime (ms)": "Duração do marcador de leitura (ms)", | ||||
|     "Read Marker off-screen lifetime (ms)": "Vida útil do marcador de leitura fora da tela (ms)", | ||||
|     "Change settings": "Alterar configurações", | ||||
|     "Send %(eventType)s events": "Enviar eventos de %(eventType)s", | ||||
|     "Roles & Permissions": "Papeis & Permissões", | ||||
|     "Select the roles required to change various parts of the room": "Selecione as permissões necessárias para alterar várias partes da sala", | ||||
|     "Emoji picker": "Seletor de emoji", | ||||
|     "Room %(name)s": "Sala %(name)s", | ||||
|     "No recently visited rooms": "Não há salas visitadas recentemente", | ||||
|     "Custom Tag": "Etiqueta personalizada", | ||||
|     "Joining room …": "Entrando na sala…", | ||||
|     "Loading …": "Carregando…", | ||||
|     "Rejecting invite …": "Rejeitando convite…", | ||||
|     "Join the conversation with an account": "Participar da conversa com uma conta", | ||||
|     "Sign Up": "Inscrever-se", | ||||
|     "Loading room preview": "Carregando visualização da sala", | ||||
|     "You were kicked from %(roomName)s by %(memberName)s": "Você foi removida(o) de %(roomName)s por %(memberName)s", | ||||
|     "Reason: %(reason)s": "Razão: %(reason)s", | ||||
|     "Forget this room": "Esquecer esta sala", | ||||
|     "Re-join": "Entrar novamente", | ||||
|     "You were banned from %(roomName)s by %(memberName)s": "Você foi banida(o) de %(roomName)s por %(memberName)s", | ||||
|     "Something went wrong with your invite to %(roomName)s": "Ocorreu um erro no seu convite para %(roomName)s", | ||||
|     "An error (%(errcode)s) was returned while trying to validate your invite. You could try to pass this information on to a room admin.": "Ocorreu um erro (%(errcode)s) ao validar seu convite. Você pode passar essas informações para um administrador da sala.", | ||||
|     "You can only join it with a working invite.": "Você só pode participar com um convite válido.", | ||||
|     "Try to join anyway": "Tentar entrar mesmo assim", | ||||
|     "You can still join it because this is a public room.": "Você ainda pode entrar, porque esta é uma sala pública.", | ||||
|     "Join the discussion": "Participar da discussão", | ||||
|     "This invite to %(roomName)s was sent to %(email)s which is not associated with your account": "Este convite para %(roomName)s foi enviado para %(email)s, que não está associado à sua conta", | ||||
|     "Link this email with your account in Settings to receive invites directly in %(brand)s.": "Vincule esse e-mail à sua conta em Configurações, para receber convites diretamente em %(brand)s.", | ||||
|     "This invite to %(roomName)s was sent to %(email)s": "Este convite para %(roomName)s foi enviado para %(email)s", | ||||
|     "Use an identity server in Settings to receive invites directly in %(brand)s.": "Use um servidor de identidade em Configurações para receber convites diretamente em %(brand)s.", | ||||
|     "Share this email in Settings to receive invites directly in %(brand)s.": "Compartilhe este e-mail em Configurações para receber convites diretamente em %(brand)s.", | ||||
|     "Do you want to chat with %(user)s?": "Deseja conversar com %(user)s?", | ||||
|     "<userName/> wants to chat": "<userName/> quer conversar", | ||||
|     "Do you want to join %(roomName)s?": "Deseja se juntar a %(roomName)s?", | ||||
|     "<userName/> invited you": "<userName/> convidou você", | ||||
|     "Reject & Ignore user": "Rejeitar e ignorar usuário", | ||||
|     "You're previewing %(roomName)s. Want to join it?": "Você está visualizando %(roomName)s. Deseja participar?", | ||||
|     "%(roomName)s can't be previewed. Do you want to join it?": "%(roomName)s não pode ser visualizado. Deseja participar?", | ||||
|     "This room doesn't exist. Are you sure you're at the right place?": "Esta sala não existe. Tem certeza de que você está no lugar certo?", | ||||
|     "Securely back up your keys to avoid losing them. <a>Learn more.</a>": "Faça backup de suas chaves com segurança para evitar perdê-las. <a>Saiba mais.</a>", | ||||
|     "Not now": "Agora não", | ||||
|     "Don't ask me again": "Não me pergunte novamente", | ||||
|     "Appearance": "Aparência", | ||||
|     "Show rooms with unread messages first": "Mostrar salas com mensagens não lidas primeiro", | ||||
|     "Show previews of messages": "Mostrar pré-visualizações de mensagens", | ||||
|     "Sort by": "Ordenar por", | ||||
|     "Activity": "Atividade", | ||||
|     "A-Z": "A-Z", | ||||
|     "Unknown Command": "Comando desconhecido", | ||||
|     "Unrecognised command: %(commandText)s": "Comando não reconhecido: %(commandText)s", | ||||
|     "You can use <code>/help</code> to list available commands. Did you mean to send this as a message?": "Você pode usar <code>/help</code> para listar os comandos disponíveis. Você quis enviar isso como uma mensagem?", | ||||
|     "Send as message": "Enviar como mensagem", | ||||
|     "Room Topic": "Descrição da sala", | ||||
|     "React": "Reagir", | ||||
|     "If you run into any bugs or have feedback you'd like to share, please let us know on GitHub.": "Se você encontrar algum erro ou tiver um comentário que gostaria de compartilhar, informe-nos no GitHub.", | ||||
|     "Resend %(unsentCount)s reaction(s)": "Reenviar %(unsentCount)s reações", | ||||
|     "Notification settings": "Configurar notificações", | ||||
|     "Want more than a community? <a>Get your own server</a>": "Quer mais do que uma comunidade? <a>Obtenha seu próprio servidor</a>", | ||||
|     "Switch to light mode": "Alternar para o modo claro", | ||||
|     "Switch to dark mode": "Alternar para o modo escuro", | ||||
|     "Security & privacy": "Segurança & privacidade", | ||||
|     "All settings": "Todas as configurações", | ||||
|     "You're signed out": "Você foi desconectada(o)", | ||||
|     "Clear personal data": "Limpar dados pessoais", | ||||
|     "Command Autocomplete": "Preenchimento automático de comandos", | ||||
|     "Community Autocomplete": "Preenchimento automático da comunidade", | ||||
|     "DuckDuckGo Results": "Resultados no DuckDuckGo", | ||||
|     "If you didn't remove the recovery method, an attacker may be trying to access your account. Change your account password and set a new recovery method immediately in Settings.": "Se você não excluiu o método de recuperação, um invasor pode estar tentando acessar sua conta. Altere a senha da sua conta e defina imediatamente um novo método de recuperação nas Configurações.", | ||||
|     "Room List": "Lista de salas", | ||||
|     "Autocomplete": "Autocompletar", | ||||
|     "Alt": "Alt", | ||||
|     "Alt Gr": "Alt Gr", | ||||
|     "Shift": "Shift", | ||||
|     "Super": "Super", | ||||
|     "Ctrl": "Ctrl", | ||||
|     "Toggle Bold": "Negrito", | ||||
|     "Toggle Italics": "Itálico", | ||||
|     "Toggle Quote": "Citar", | ||||
|     "New line": "Nova linha", | ||||
|     "Navigate recent messages to edit": "Navegue pelas mensagens recentes para editar", | ||||
|     "Cancel replying to a message": "Cancelar resposta à mensagem", | ||||
|     "Toggle microphone mute": "Ativar/desativar som do microfone", | ||||
|     "Toggle video on/off": "Ativar/desativar o vídeo", | ||||
|     "Scroll up/down in the timeline": "Rolar para cima/baixo na linha do tempo", | ||||
|     "Dismiss read marker and jump to bottom": "Ignorar o marcador de leitura e ir para o final", | ||||
|     "Jump to oldest unread message": "Ir para a mensagem não lida mais antiga", | ||||
|     "Upload a file": "Enviar um arquivo", | ||||
|     "Jump to room search": "Ir para a pesquisa de salas", | ||||
|     "Navigate up/down in the room list": "Navegue para cima/baixo na lista de salas", | ||||
|     "Select room from the room list": "Selecionar sala da lista de salas", | ||||
|     "Collapse room list section": "Esconder seção da lista de salas", | ||||
|     "Expand room list section": "Mostrar seção da lista de salas", | ||||
|     "The person who invited you already left the room.": "A pessoa que convidou você já saiu da sala.", | ||||
|     "The person who invited you already left the room, or their server is offline.": "A pessoa que convidou você já saiu da sala, ou o servidor dela está offline.", | ||||
|     "Use an Integration Manager <b>(%(serverName)s)</b> to manage bots, widgets, and sticker packs.": "Use o Gerenciador de Integrações em <b>(%(serverName)s)</b> para gerenciar bots, widgets e pacotes de adesivos.", | ||||
|     "Use an Integration Manager to manage bots, widgets, and sticker packs.": "Use o Gerenciador de Integrações para gerenciar bots, widgets e pacotes de adesivos.", | ||||
|     "Integration Managers receive configuration data, and can modify widgets, send room invites, and set power levels on your behalf.": "O Gerenciador de Integrações recebe dados de configuração e pode modificar widgets, enviar convites para salas e definir níveis de permissão em seu nome.", | ||||
|     "Keyboard Shortcuts": "Atalhos do teclado", | ||||
|     "Customise your experience with experimental labs features. <a>Learn more</a>.": "Personalize sua experiência com os recursos experimentais. <a>Saiba mais</a>.", | ||||
|     "Ignored/Blocked": "Ignorado/Bloqueado", | ||||
|     "Error adding ignored user/server": "Erro ao adicionar usuário/servidor ignorado", | ||||
|     "Something went wrong. Please try again or view your console for hints.": "Algo deu errado. Por favor, tente novamente ou veja seu console para obter dicas.", | ||||
|     "Error subscribing to list": "Erro ao inscrever-se na lista", | ||||
|     "Error removing ignored user/server": "Erro ao remover usuário/servidor ignorado", | ||||
|     "Error unsubscribing from list": "Erro ao cancelar a inscrição da lista", | ||||
|     "Please try again or view your console for hints.": "Por favor, tente novamente ou veja seu console para obter dicas.", | ||||
|     "None": "Nenhum", | ||||
|     "Server rules": "Regras do servidor", | ||||
|     "User rules": "Regras do usuário", | ||||
|     "You have not ignored anyone.": "Você não ignorou ninguém.", | ||||
|     "You are currently ignoring:": "Você está atualmente ignorando:", | ||||
|     "You are not subscribed to any lists": "Você não está inscrito em nenhuma lista", | ||||
|     "Unsubscribe": "Desinscrever-se", | ||||
|     "View rules": "Ver regras", | ||||
|     "You are currently subscribed to:": "No momento, você está inscrito em:", | ||||
|     "⚠ These settings are meant for advanced users.": "⚠ Essas configurações são destinadas a usuários avançados.", | ||||
|     "Add users and servers you want to ignore here. Use asterisks to have %(brand)s match any characters. For example, <code>@bot:*</code> would ignore all users that have the name 'bot' on any server.": "Adicione aqui os usuários e servidores que você deseja ignorar. Use asteriscos para fazer com que o %(brand)s corresponda a qualquer caractere. Por exemplo, <code>@bot:*</code> ignorará todos os usuários em qualquer servidor que tenham 'bot' no nome.", | ||||
|     "Server or user ID to ignore": "Servidor ou ID de usuário para ignorar", | ||||
|     "Subscribe": "Inscrever-se", | ||||
|     "Session ID:": "ID da sessão:", | ||||
|     "Message search": "Pesquisa de mensagens", | ||||
|     "Where you’re logged in": "Onde você está conectado", | ||||
|     "Set a new custom sound": "Definir um novo som personalizado", | ||||
|     "Browse": "Buscar", | ||||
|     "Upgrade the room": "Atualizar a sala", | ||||
|     "Kick users": "Remover usuários", | ||||
|     "Ban users": "Banir usuários", | ||||
|     "Remove messages": "Remover mensagens", | ||||
|     "Notify everyone": "Notificar todo mundo", | ||||
|     "Your email address hasn't been verified yet": "Seu endereço de e-mail ainda não foi verificado", | ||||
|     "Revoke": "Revogar", | ||||
|     "Share": "Compartilhar", | ||||
|     "Unable to revoke sharing for phone number": "Não foi possível revogar o compartilhamento do número de celular", | ||||
|     "Unable to share phone number": "Não foi possível compartilhar o número de celular", | ||||
|     "Please enter verification code sent via text.": "Digite o código de verificação enviado por mensagem de texto.", | ||||
|     "Remove %(email)s?": "Remover %(email)s?", | ||||
|     "Remove %(phone)s?": "Remover %(phone)s?", | ||||
|     "A text message has been sent to +%(msisdn)s. Please enter the verification code it contains.": "Digite o código de verificação enviado por mensagem de texto para +%(msisdn)s.", | ||||
|     "This user has not verified all of their sessions.": "Este usuário não verificou todas suas próprias sessões.", | ||||
|     "You have not verified this user.": "Você não verificou este usuário.", | ||||
|     "You have verified this user. This user has verified all of their sessions.": "Você confirmou este usuário. Este usuário verificou todas as próprias sessões.", | ||||
|     "Someone is using an unknown session": "Alguém está usando uma sessão desconhecida", | ||||
|     "Everyone in this room is verified": "Todo mundo nesta sala está verificado", | ||||
|     "Edit message": "Editar mensagem", | ||||
|     "Mod": "Moderador", | ||||
|     "Scroll to most recent messages": "Ir para as mensagens mais recentes", | ||||
|     "Close preview": "Fechar a visualização", | ||||
|     "Send a reply…": "Enviar uma resposta…", | ||||
|     "Send a message…": "Enviar uma mensagem…", | ||||
|     "Bold": "Negrito", | ||||
|     "Italics": "Itálico", | ||||
|     "Strikethrough": "Riscado", | ||||
|     "Code block": "Bloco de código", | ||||
|     "Failed to connect to integration manager": "Falha ao conectar-se ao gerenciador de integrações", | ||||
|     "Failed to revoke invite": "Falha ao revogar o convite", | ||||
|     "Could not revoke the invite. The server may be experiencing a temporary problem or you do not have sufficient permissions to revoke the invite.": "Não foi possível revogar o convite. O servidor pode estar com um problema temporário ou você não tem permissões suficientes para revogar o convite.", | ||||
|     "Revoke invite": "Revogar o convite", | ||||
|     "Invited by %(sender)s": "Convidado por %(sender)s", | ||||
|     "Mark all as read": "Marcar tudo como lido", | ||||
|     "Error updating main address": "Erro ao atualizar o endereço principal", | ||||
|     "There was an error updating the room's main address. It may not be allowed by the server or a temporary failure occurred.": "Ocorreu um erro ao atualizar o endereço principal da sala. Isso pode não ser permitido pelo servidor ou houve um problema temporário.", | ||||
|     "There was an error updating the room's alternative addresses. It may not be allowed by the server or a temporary failure occurred.": "Ocorreu um erro ao atualizar o endereço alternativo da sala. Isso pode não ser permitido pelo servidor ou houve um problema temporário.", | ||||
|     "Error creating address": "Erro ao criar o endereço", | ||||
|     "There was an error creating that address. It may not be allowed by the server or a temporary failure occurred.": "Ocorreu um erro ao criar esse endereço. Isso pode não ser permitido pelo servidor ou houve um problema temporário.", | ||||
|     "You don't have permission to delete the address.": "Você não tem permissão para excluir este endereço.", | ||||
|     "There was an error removing that address. It may no longer exist or a temporary error occurred.": "Ocorreu um erro ao remover esse endereço. Ele pode não mais existir ou houve um problema temporário.", | ||||
|     "Error removing address": "Erro ao remover o endereço", | ||||
|     "Main address": "Endereço principal", | ||||
|     "Room Name": "Nome da sala", | ||||
|     "Room avatar": "Foto da sala", | ||||
|     "Waiting for you to accept on your other session…": "Aguardando sua confirmação na sua outra sessão…", | ||||
|     "Waiting for %(displayName)s to accept…": "Aguardando %(displayName)s aceitar…", | ||||
|     "Accepting…": "Aceitando…", | ||||
|     "Your messages are secured and only you and the recipient have the unique keys to unlock them.": "Suas mensagens são protegidas e somente você e o destinatário têm as chaves exclusivas para desbloqueá-las.", | ||||
|     "Your messages are not secure": "Suas mensagens não estão seguras", | ||||
|     "Your homeserver": "Seu servidor local", | ||||
|     "Trusted": "Confiável", | ||||
|     "Not trusted": "Não confiável", | ||||
|     "%(count)s verified sessions|other": "%(count)s sessões verificadas", | ||||
|     "%(count)s verified sessions|one": "1 sessão verificada", | ||||
|     "Hide verified sessions": "Esconder sessões verificadas", | ||||
|     "%(count)s sessions|other": "%(count)s sessões", | ||||
|     "%(count)s sessions|one": "%(count)s sessão", | ||||
|     "Hide sessions": "Esconder sessões", | ||||
|     "No recent messages by %(user)s found": "Nenhuma mensagem recente de %(user)s foi encontrada", | ||||
|     "Remove recent messages by %(user)s": "Remover mensagens recentes de %(user)s", | ||||
|     "Remove %(count)s messages|other": "Remover %(count)s mensagens", | ||||
|     "Remove %(count)s messages|one": "Remover 1 mensagem", | ||||
|     "Remove recent messages": "Remover mensagens recentes", | ||||
|     "<strong>%(role)s</strong> in %(roomName)s": "<strong>%(role)s</strong> em %(roomName)s", | ||||
|     "Deactivate user?": "Desativar usuário?", | ||||
|     "Deactivate user": "Desativar usuário", | ||||
|     "Failed to deactivate user": "Falha ao desativar o usuário", | ||||
|     "Security": "Segurança", | ||||
|     "You've successfully verified your device!": "Você verificou o seu aparelho com êxito!", | ||||
|     "Verification timed out.": "O tempo de verificação se esgotou.", | ||||
|     "You cancelled verification on your other session.": "Você cancelou a verificação em sua outra sessão.", | ||||
|     "%(displayName)s cancelled verification.": "%(displayName)s cancelou a verificação.", | ||||
|     "You cancelled verification.": "Você cancelou a verificação.", | ||||
|     "Verification cancelled": "Verificação cancelada", | ||||
|     "Compare emoji": "Comparar emojis", | ||||
|     "Show image": "Mostrar imagem", | ||||
|     "You have ignored this user, so their message is hidden. <a>Show anyways.</a>": "Você ignorou este usuário, portanto, a mensagem dele está oculta. <a>Mostrar mesmo assim.</a>", | ||||
|     "You verified %(name)s": "Você verificou %(name)s", | ||||
|     "Use an identity server to invite by email. <default>Use the default (%(defaultIdentityServerName)s)</default> or manage in <settings>Settings</settings>.": "Use um servidor de identidade para convidar por e-mail. <default>Use o padrão (%(defaultIdentityServerName)s)</default> ou um servidor personalizado em <settings>Configurações</settings>.", | ||||
|     "Use an identity server to invite by email. Manage in <settings>Settings</settings>.": "Use um servidor de identidade para convidar por e-mail. Gerencie o servidor em <settings>Configurações</settings>.", | ||||
|     "Destroy cross-signing keys?": "Destruir chaves de assinatura cruzada?", | ||||
|     "Waiting for partner to confirm...": "Esperando a outra pessoa confirmar...", | ||||
|     "Enable 'Manage Integrations' in Settings to do this.": "Para fazer isso, ative 'Gerenciar Integrações' nas Configurações.", | ||||
|     "Your %(brand)s doesn't allow you to use an Integration Manager to do this. Please contact an admin.": "Seu %(brand)s não permite que você use o Gerenciador de Integrações para fazer isso. Entre em contato com um administrador.", | ||||
|     "Confirm to continue": "Confirme para continuar", | ||||
|     "Click the button below to confirm your identity.": "Clique no botão abaixo para confirmar sua identidade.", | ||||
|     "Failed to invite the following users to chat: %(csvUsers)s": "Falha ao convidar os seguintes usuários para a conversa: %(csvUsers)s", | ||||
|     "Something went wrong trying to invite the users.": "Ocorreu um erro ao tentar convidar os usuários.", | ||||
|     "We couldn't invite those users. Please check the users you want to invite and try again.": "Não foi possível convidar esses usuários. Por favor, tente novamente.", | ||||
|     "Failed to find the following users": "Falha ao encontrar os seguintes usuários", | ||||
|     "The following users might not exist or are invalid, and cannot be invited: %(csvNames)s": "Os seguintes usuários não puderam ser convidados porque não existem ou são inválidos: %(csvNames)s", | ||||
|     "Recent Conversations": "Conversas recentes", | ||||
|     "Suggestions": "Sugestões", | ||||
|     "Invite someone using their name, username (like <userId/>), email address or <a>share this room</a>.": "Convide alguém com seu nome, nome de usuário (por exemplo: <userId/>), endereço de e-mail ou <a>compartilhe esta sala</a>.", | ||||
|     "Use your account to sign in to the latest version of the app at <a />": "Use sua conta para fazer login na versão mais recente do aplicativo em <a />", | ||||
|     "Report bugs & give feedback": "Relatar erros & enviar comentários", | ||||
|     "Room Settings - %(roomName)s": "Configurações da sala - %(roomName)s", | ||||
|     "Automatically invite users": "Convidar usuários automaticamente", | ||||
|     "Upgrade private room": "Atualizar a sala privada", | ||||
|     "Upgrade public room": "Atualizar a sala pública", | ||||
|     "Upgrading a room is an advanced action and is usually recommended when a room is unstable due to bugs, missing features or security vulnerabilities.": "Atualizar uma sala é uma ação avançada e geralmente é recomendada quando uma sala está instável devido a erros, recursos ausentes ou vulnerabilidades de segurança.", | ||||
|     "This usually only affects how the room is processed on the server. If you're having problems with your %(brand)s, please <a>report a bug</a>.": "Isso geralmente afeta apenas como a sala é processada no servidor. Se você tiver problemas com o %(brand)s, <a>informe um erro</a>.", | ||||
|     "You'll upgrade this room from <oldVersion /> to <newVersion />.": "Você atualizará esta sala de <oldVersion /> para <newVersion />.", | ||||
|     "A username can only contain lower case letters, numbers and '=_-./'": "Um nome de usuário só pode ter letras minúsculas, números e '=_-./'", | ||||
|     "Command Help": "Ajuda com Comandos", | ||||
|     "To help us prevent this in future, please <a>send us logs</a>.": "Para nos ajudar a evitar isso no futuro, <a>envie-nos os registros</a>.", | ||||
|     "Your browser likely removed this data when running low on disk space.": "O seu navegador provavelmente removeu esses dados quando o espaço de armazenamento ficou insuficiente.", | ||||
|     "Integration Manager": "Gerenciador de Integrações", | ||||
|     "Find others by phone or email": "Encontre outras pessoas por telefone ou e-mail", | ||||
|     "Use bots, bridges, widgets and sticker packs": "Use bots, pontes, widgets e pacotes de adesivos", | ||||
|     "Terms of Service": "Termos de serviço", | ||||
|     "To continue you need to accept the terms of this service.": "Para continuar, você precisa aceitar os termos deste serviço.", | ||||
|     "Service": "Serviço", | ||||
|     "Summary": "Resumo", | ||||
|     "Document": "Documento", | ||||
|     "Upload files (%(current)s of %(total)s)": "Enviar arquivos (%(current)s de %(total)s)", | ||||
|     "Upload files": "Enviar arquivos", | ||||
|     "Upload all": "Enviar tudo", | ||||
|     "This file is <b>too large</b> to upload. The file size limit is %(limit)s but this file is %(sizeOfThisFile)s.": "Este arquivo é <b>muito grande</b> para ser enviado. O limite do tamanho de arquivos é %(limit)s, enquanto que o tamanho desse arquivo é %(sizeOfThisFile)s.", | ||||
|     "These files are <b>too large</b> to upload. The file size limit is %(limit)s.": "Esses arquivos são <b>muito grandes</b> para serem enviados. O limite do tamanho de arquivos é %(limit)s.", | ||||
|     "Some files are <b>too large</b> to be uploaded. The file size limit is %(limit)s.": "Alguns arquivos são <b>muito grandes</b> para serem enviados. O limite do tamanho de arquivos é %(limit)s.", | ||||
|     "Upload %(count)s other files|other": "Enviar %(count)s outros arquivos", | ||||
|     "Upload %(count)s other files|one": "Enviar %(count)s outros arquivos", | ||||
|     "Cancel All": "Cancelar tudo", | ||||
|     "Upload Error": "Erro no envio", | ||||
|     "Verification Request": "Solicitação de verificação", | ||||
|     "Remember my selection for this widget": "Lembrar minha escolha para este widget", | ||||
|     "Deny": "Rejeitar", | ||||
|     "Wrong file type": "Tipo errado de arquivo", | ||||
|     "Address (optional)": "Endereço (opcional)", | ||||
|     "Report Content": "Reportar conteúdo", | ||||
|     "Update status": "Atualizar status", | ||||
|     "Set status": "Definir status", | ||||
|     "Hide": "Esconder", | ||||
|     "Help": "Ajuda", | ||||
|     "Remove for everyone": "Remover para todo mundo", | ||||
|     "Remove for me": "Remover para mim", | ||||
|     "User Status": "Status do usuário", | ||||
|     "This homeserver would like to make sure you are not a robot.": "Este servidor local gostaria se certificar de que você não é um robô.", | ||||
|     "Confirm your identity by entering your account password below.": "Confirme sua identidade digitando sua senha abaixo.", | ||||
|     "Unable to validate homeserver/identity server": "Não foi possível validar seu servidor local/servidor de identidade", | ||||
|     "Server Name": "Nome do servidor", | ||||
|     "Enter password": "Digite a senha", | ||||
|     "Nice, strong password!": "Muito bem, uma senha forte!", | ||||
|     "Password is allowed, but unsafe": "Esta senha é permitida, mas não é segura", | ||||
|     "Not sure of your password? <a>Set a new one</a>": "Esqueceu sua senha? <a>Defina uma nova</a>", | ||||
|     "Set an email for account recovery. Use email to optionally be discoverable by existing contacts.": "Defina um e-mail para poder recuperar a conta. Este e-mail também pode ser usado para encontrar seus contatos.", | ||||
|     "Enter your custom homeserver URL <a>What does this mean?</a>": "Digite o URL de um servidor local <a>O que isso significa?</a>", | ||||
|     "Homeserver URL": "URL do servidor local", | ||||
|     "Identity Server URL": "URL do servidor de identidade", | ||||
|     "Other servers": "Outros servidores", | ||||
|     "Free": "Gratuito", | ||||
|     "Find other public servers or use a custom server": "Encontre outros servidores públicos ou use um servidor personalizado", | ||||
|     "Sign in to your Matrix account on %(serverName)s": "Faça login com sua conta Matrix em %(serverName)s", | ||||
|     "Sign in to your Matrix account on <underlinedServerName />": "Faça login com sua conta Matrix em <underlinedServerName />", | ||||
|     "Sign in with SSO": "Faça login com SSO (Login Único)", | ||||
|     "Please install <chromeLink>Chrome</chromeLink>, <firefoxLink>Firefox</firefoxLink>, or <safariLink>Safari</safariLink> for the best experience.": "Por favor, instale o <chromeLink>Chrome</chromeLink>, o <firefoxLink>Firefox</firefoxLink> ou o <safariLink>Safari</safariLink> para obter a melhor experiência de uso.", | ||||
|     "Couldn't load page": "Não foi possível carregar a página", | ||||
|     "This homeserver does not support communities": "Este servidor local não suporta o recurso de comunidades", | ||||
|     "%(brand)s failed to get the protocol list from the homeserver. The homeserver may be too old to support third party networks.": "%(brand)s não conseguiu obter a lista de protocolos do servidor local. O servidor local pode ser muito antigo para suportar redes de terceiros.", | ||||
|     "%(brand)s failed to get the public room list.": "%(brand)s não conseguiu obter a lista de salas públicas.", | ||||
|     "The homeserver may be unavailable or overloaded.": "O servidor local pode estar indisponível ou sobrecarregado.", | ||||
|     "Delete the room address %(alias)s and remove %(name)s from the directory?": "Excluir o endereço da sala %(alias)s e remover %(name)s da lista de salas?", | ||||
|     "delete the address.": "exclui o endereço.", | ||||
|     "%(brand)s does not know how to join a room on this network": "%(brand)s não sabe como entrar em uma sala desta rede", | ||||
|     "View": "Ver", | ||||
|     "Find a room…": "Encontrar uma sala…", | ||||
|     "Find a room… (e.g. %(exampleRoom)s)": "Encontrar uma sala… (por exemplo: %(exampleRoom)s)", | ||||
|     "Search rooms": "Buscar salas", | ||||
|     "You have %(count)s unread notifications in a prior version of this room.|other": "Você tem %(count)s notificações não lidas em uma versão anterior desta sala.", | ||||
|     "You have %(count)s unread notifications in a prior version of this room.|one": "Você tem %(count)s notificações não lidas em uma versão anterior desta sala.", | ||||
|     "Feedback": "Feedback", | ||||
|     "User menu": "Menu do usuário", | ||||
|     "Could not load user profile": "Não foi possível carregar o perfil do usuário", | ||||
|     "Session verified": "Sessão verificada", | ||||
|     "Your Matrix account on %(serverName)s": "Sua conta Matrix em %(serverName)s", | ||||
|     "Your Matrix account on <underlinedServerName />": "Sua conta Matrix em <underlinedServerName />", | ||||
|     "No identity server is configured: add one in server settings to reset your password.": "Nenhum servidor de identidade está configurado: adicione um nas configurações do servidor para redefinir sua senha.", | ||||
|     "A verification email will be sent to your inbox to confirm setting your new password.": "Um e-mail de verificação será enviado para sua caixa de entrada para confirmar sua nova senha.", | ||||
|     "Your password has been reset.": "Sua senha foi alterada.", | ||||
|     "You have been logged out of all sessions and will no longer receive push notifications. To re-enable notifications, sign in again on each device.": "Você foi desconectado de todas as sessões e não receberá mais notificações. Para reativar as notificações, faça login novamente em cada aparelho.", | ||||
|     "Set a new password": "Digite uma nova senha", | ||||
|     "Invalid base_url for m.homeserver": "base_url inválido para m.homeserver", | ||||
|     "Invalid base_url for m.identity_server": "base_url inválido para m.identity_server", | ||||
|     "This account has been deactivated.": "Esta conta foi desativada.", | ||||
|     "Syncing...": "Sincronizando...", | ||||
|     "%(brand)s Web": "%(brand)s Web", | ||||
|     "Your new session is now verified. Other users will see it as trusted.": "Sua nova sessão agora está confirmada. Para outros usuários ela será vista como confiável.", | ||||
|     "Forgotten your password?": "Esqueceu sua senha?", | ||||
|     "Restore": "Restaurar", | ||||
|     "Copy": "Copiar", | ||||
|     "For maximum security, this should be different from your account password.": "Para segurança máxima, essa deve ser diferente da senha da sua conta.", | ||||
|     "Success!": "Pronto!", | ||||
|     "Disable": "Desativar", | ||||
|     "Space used:": "Espaço usado:", | ||||
|     "Navigation": "Navegação", | ||||
|     "Calls": "Chamadas", | ||||
|     "Esc": "Esc", | ||||
|     "Enter": "Enter" | ||||
| } | ||||
|  |  | |||
|  | @ -1547,8 +1547,8 @@ | |||
|     "%(creator)s created and configured the room.": "%(creator)s создал и настроил комнату.", | ||||
|     "Preview": "Заглянуть", | ||||
|     "View": "Просмотр", | ||||
|     "Find a room…": "Найди комнату…", | ||||
|     "Find a room… (e.g. %(exampleRoom)s)": "Найди комнату... (напр. %(exampleRoom)s)", | ||||
|     "Find a room…": "Поиск комнат…", | ||||
|     "Find a room… (e.g. %(exampleRoom)s)": "Поиск комнат... (напр. %(exampleRoom)s)", | ||||
|     "Explore rooms": "Список комнат", | ||||
|     "No identity server is configured: add one in server settings to reset your password.": "Идентификационный сервер не настроен: добавьте его в настройки сервера, чтобы сбросить пароль.", | ||||
|     "Command Autocomplete": "Автозаполнение команды", | ||||
|  | @ -1609,8 +1609,8 @@ | |||
|     "Verifies a user, session, and pubkey tuple": "Проверяет пользователя, сессию и публичные ключи", | ||||
|     "Unknown (user, session) pair:": "Неизвестная (пользователь:сессия) пара:", | ||||
|     "Session already verified!": "Сессия уже подтверждена!", | ||||
|     "Never send encrypted messages to unverified sessions from this session": "Никогда не отправляйте зашифрованные сообщения в непроверенные сессий из этой сессии", | ||||
|     "Never send encrypted messages to unverified sessions in this room from this session": "Никогда не отправляйте зашифрованные сообщения в непроверенные сессии в эту комнату из этой сессии", | ||||
|     "Never send encrypted messages to unverified sessions from this session": "Никогда не отправлять зашифрованные сообщения непроверенным сессиям в этой сессии", | ||||
|     "Never send encrypted messages to unverified sessions in this room from this session": "Никогда не отправлять зашифрованные сообщения непроверенным сессиям в этой комнате и в этой сессии", | ||||
|     "Your keys are <b>not being backed up from this session</b>.": "Ваши ключи <b>не резервируются с этой сессии</b>.", | ||||
|     "Server or user ID to ignore": "Сервер или ID пользователя для игнорирования", | ||||
|     "Subscribed lists": "Подписанные списки", | ||||
|  | @ -2172,5 +2172,8 @@ | |||
|     "This address is already in use": "Этот адрес уже используется", | ||||
|     "Are you sure you want to remove <b>%(serverName)s</b>": "Вы уверены, что хотите удалить <b>%(serverName)s</b>", | ||||
|     "Enter the name of a new server you want to explore.": "Введите имя нового сервера для просмотра.", | ||||
|     "Reminder: Your browser is unsupported, so your experience may be unpredictable.": "Напоминание: ваш браузер не поддерживается, возможны непредвиденные проблемы." | ||||
|     "Reminder: Your browser is unsupported, so your experience may be unpredictable.": "Напоминание: ваш браузер не поддерживается, возможны непредвиденные проблемы.", | ||||
|     "Notification settings": "Настройки уведомлений", | ||||
|     "Switch to light mode": "Переключить в светлый режим", | ||||
|     "Switch to dark mode": "Переключить в тёмный режим" | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,8 @@ | |||
| { | ||||
|     "This email address is already in use": "මෙම විද්යුත් තැපැල් ලිපිනය දැනටමත් භාවිතයේ පවතී", | ||||
|     "This phone number is already in use": "මෙම දුරකථන අංකය දැනටමත් භාවිතයේ පවතී", | ||||
|     "Use Single Sign On to continue": "ඉදිරියට යාමට තනි පුරනය වීම භාවිතා කරන්න", | ||||
|     "Confirm adding this email address by using Single Sign On to prove your identity.": "ඔබගේ අනන්යතාවය සනාථ කිරීම සඳහා තනි පුරනය භාවිතා කිරීමෙන් මෙම විද්යුත් තැපැල් ලිපිනය එක් කිරීම තහවුරු කරන්න.", | ||||
|     "Confirm": "තහවුරු කරන්න", | ||||
|     "Add Email Address": "විද්යුත් තැපැල් ලිපිනය එක් කරන්න" | ||||
| } | ||||
|  | @ -2387,5 +2387,7 @@ | |||
|     "%(brand)s iOS": "%(brand)s iOS", | ||||
|     "%(brand)s X for Android": "%(brand)s X për Android", | ||||
|     "* %(senderName)s %(emote)s": "* %(senderName)s %(emote)s", | ||||
|     "Custom Tag": "Etiketë Vetjake" | ||||
|     "Custom Tag": "Etiketë Vetjake", | ||||
|     "The person who invited you already left the room.": "Personi që ju ftoi ka dalë nga dhoma tashmë.", | ||||
|     "The person who invited you already left the room, or their server is offline.": "Personi që ju ftoi, ka dalë nga dhoma tashmë, ose shërbyesi i tij është jashtë funksionimi." | ||||
| } | ||||
|  |  | |||
|  | @ -6,7 +6,7 @@ | |||
|     "Dismiss": "Відхилити", | ||||
|     "Error": "Помилка", | ||||
|     "Failed to forget room %(errCode)s": "Не вдалось видалити кімнату %(errCode)s", | ||||
|     "Favourite": "Вибране", | ||||
|     "Favourite": "Улюблені", | ||||
|     "Mute": "Стишити", | ||||
|     "Notifications": "Сповіщення", | ||||
|     "Operation failed": "Не вдалося виконати дію", | ||||
|  | @ -78,7 +78,7 @@ | |||
|     "Unpin Message": "Відкріпити повідомлення", | ||||
|     "Register": "Зареєструватися", | ||||
|     "Rooms": "Кімнати", | ||||
|     "Add rooms to this community": "Добавити кімнати в це суспільство", | ||||
|     "Add rooms to this community": "Додати кімнати в цю спільноту", | ||||
|     "This email address is already in use": "Ця е-пошта вже використовується", | ||||
|     "This phone number is already in use": "Цей телефонний номер вже використовується", | ||||
|     "Fetching third party location failed": "Не вдалось отримати стороннє місцеперебування", | ||||
|  | @ -240,33 +240,33 @@ | |||
|     "You cannot place a call with yourself.": "Ви не можете подзвонити самим собі.", | ||||
|     "Warning!": "Увага!", | ||||
|     "Upload Failed": "Помилка відвантаження", | ||||
|     "Sun": "Нд", | ||||
|     "Mon": "Пн", | ||||
|     "Tue": "Вт", | ||||
|     "Wed": "Ср", | ||||
|     "Thu": "Чт", | ||||
|     "Fri": "Пт", | ||||
|     "Sat": "Сб", | ||||
|     "Jan": "Січ", | ||||
|     "Feb": "Лют", | ||||
|     "Mar": "Бер", | ||||
|     "Apr": "Квіт", | ||||
|     "May": "Трав", | ||||
|     "Jun": "Чер", | ||||
|     "Jul": "Лип", | ||||
|     "Aug": "Сер", | ||||
|     "Sep": "Вер", | ||||
|     "Oct": "Жов", | ||||
|     "Nov": "Лис", | ||||
|     "Dec": "Гру", | ||||
|     "PM": "PM", | ||||
|     "AM": "AM", | ||||
|     "Sun": "нд", | ||||
|     "Mon": "пн", | ||||
|     "Tue": "вт", | ||||
|     "Wed": "ср", | ||||
|     "Thu": "чт", | ||||
|     "Fri": "пт", | ||||
|     "Sat": "сб", | ||||
|     "Jan": "січ.", | ||||
|     "Feb": "лют.", | ||||
|     "Mar": "бер.", | ||||
|     "Apr": "квіт.", | ||||
|     "May": "трав.", | ||||
|     "Jun": "черв.", | ||||
|     "Jul": "лип.", | ||||
|     "Aug": "серп.", | ||||
|     "Sep": "вер.", | ||||
|     "Oct": "жовт.", | ||||
|     "Nov": "лист.", | ||||
|     "Dec": "груд.", | ||||
|     "PM": "пп", | ||||
|     "AM": "дп", | ||||
|     "%(weekDayName)s %(time)s": "%(weekDayName)s %(time)s", | ||||
|     "%(weekDayName)s, %(monthName)s %(day)s %(time)s": "%(weekDayName)s, %(monthName)s, %(day)s, %(time)s", | ||||
|     "%(weekDayName)s, %(monthName)s %(day)s %(fullYear)s": "%(weekDayName)s, %(monthName)s, %(day)s, %(fullYear)s", | ||||
|     "%(weekDayName)s, %(monthName)s %(day)s %(fullYear)s %(time)s": "%(weekDayName)s, %(monthName)s %(day)s %(fullYear)s %(time)s", | ||||
|     "%(weekDayName)s, %(monthName)s %(day)s %(time)s": "%(weekDayName)s, %(day)s %(monthName)s %(time)s", | ||||
|     "%(weekDayName)s, %(monthName)s %(day)s %(fullYear)s": "%(weekDayName)s, %(day)s %(monthName)s %(fullYear)s", | ||||
|     "%(weekDayName)s, %(monthName)s %(day)s %(fullYear)s %(time)s": "%(weekDayName)s, %(day)s %(monthName)s %(fullYear)s %(time)s", | ||||
|     "Who would you like to add to this community?": "Кого ви хочете додати до цієї спільноти?", | ||||
|     "Where this page includes identifiable information, such as a room, user or group ID, that data is removed before being sent to the server.": "Якщо дана сторінка містить особисту інформацію, як то назва кімнати, користувача чи групи, ці дані будуть вилучені перед надсиланням на сервер.", | ||||
|     "Where this page includes identifiable information, such as a room, user or group ID, that data is removed before being sent to the server.": "Там, де ця сторінка містить ототожненну інформацію, як-от назва кімнати, користувача чи групи, ці дані будуть вилучені перед надсиланням на сервер.", | ||||
|     "Call in Progress": "Іде виклик", | ||||
|     "A call is currently being placed!": "Зараз іде виклик!", | ||||
|     "A call is already in progress!": "Вже здійснюється дзвінок!", | ||||
|  | @ -311,9 +311,9 @@ | |||
|     "Changes your display nickname": "Змінює ваш нік", | ||||
|     "Invites user with given id to current room": "Запрошує користувача з вказаним ідентифікатором до кімнати", | ||||
|     "Leave room": "Покинути кімнату", | ||||
|     "Kicks user with given id": "Вилучити з кімнати користувача з вказаним ідентифікатором", | ||||
|     "Ignores a user, hiding their messages from you": "Ігнорує користувача, приховуючи повідомлення від них", | ||||
|     "Ignored user": "Користувача ігноровано", | ||||
|     "Kicks user with given id": "Викидає з кімнати користувача з вказаним ідентифікатором", | ||||
|     "Ignores a user, hiding their messages from you": "Ігнорує користувача, приховуючи його повідомлення від вас", | ||||
|     "Ignored user": "Зігнорований користувач", | ||||
|     "You are now ignoring %(userId)s": "Ви ігноруєте %(userId)s", | ||||
|     "Stops ignoring a user, showing their messages going forward": "Припиняє ігнорувати користувача, від цього моменту показуючи їхні повідомлення", | ||||
|     "Unignored user": "Припинено ігнорування користувача", | ||||
|  | @ -321,8 +321,8 @@ | |||
|     "Define the power level of a user": "Вказати рівень повноважень користувача", | ||||
|     "Deops user with given id": "Знімає права оператора з користувача з вказаним ідентифікатором", | ||||
|     "Opens the Developer Tools dialog": "Відкриває вікно інструментів розробника", | ||||
|     "Verified key": "Перевірений ключ", | ||||
|     "Displays action": "Показує дію", | ||||
|     "Verified key": "Звірений ключ", | ||||
|     "Displays action": "Відбиває дію", | ||||
|     "Reason": "Причина", | ||||
|     "%(senderName)s requested a VoIP conference.": "%(senderName)s бажає розпочати дзвінок-конференцію.", | ||||
|     "%(senderName)s invited %(targetName)s.": "%(senderName)s запросив/ла %(targetName)s.", | ||||
|  | @ -339,9 +339,9 @@ | |||
|     "%(senderName)s unbanned %(targetName)s.": "%(senderName)s розблокував/ла %(targetName)s.", | ||||
|     "%(senderName)s kicked %(targetName)s.": "%(senderName)s викинув/ла %(targetName)s.", | ||||
|     "%(senderName)s withdrew %(targetName)s's invitation.": "%(senderName)s відкликав/ла запрошення %(targetName)s.", | ||||
|     "%(senderDisplayName)s sent an image.": "%(senderDisplayName)s надіслав/ла зображення.", | ||||
|     "%(senderName)s set the main address for this room to %(address)s.": "%(senderName)s призначив/ла основну адресу цієї кімнати: %(address)s.", | ||||
|     "%(senderName)s removed the main address for this room.": "%(senderName)s вилучив/ла основу адресу цієї кімнати.", | ||||
|     "%(senderDisplayName)s sent an image.": "%(senderDisplayName)s надіслав(-ла) зображення.", | ||||
|     "%(senderName)s set the main address for this room to %(address)s.": "%(senderName)s призначив(-ла) основну адресу цієї кімнати: %(address)s.", | ||||
|     "%(senderName)s removed the main address for this room.": "%(senderName)s прибрав(-ла) основу адресу цієї кімнати.", | ||||
|     "Someone": "Хтось", | ||||
|     "(not supported by this browser)": "(не підтримується цією веб-переглядачкою)", | ||||
|     "(could not connect media)": "(не можливо під'єднати медіа)", | ||||
|  | @ -403,7 +403,7 @@ | |||
|     "Password": "Пароль", | ||||
|     "New Password": "Новий пароль", | ||||
|     "Confirm password": "Підтвердження пароля", | ||||
|     "Last seen": "Востаннє з'являвся", | ||||
|     "Last seen": "Востаннє в мережі", | ||||
|     "Failed to set display name": "Не вдалося встановити ім'я для показу", | ||||
|     "The maximum permitted number of widgets have already been added to this room.": "Максимально дозволену кількість віджетів уже додано до цієї кімнати.", | ||||
|     "Drop File Here": "Киньте файл сюди", | ||||
|  | @ -460,7 +460,7 @@ | |||
|     "Sends a message as plain text, without interpreting it as markdown": "Надсилає повідомлення як чистий текст, не використовуючи markdown", | ||||
|     "Upgrades a room to a new version": "Покращує кімнату до нової версії", | ||||
|     "You do not have the required permissions to use this command.": "Вам бракує дозволу на використання цієї команди.", | ||||
|     "<b>Warning</b>: Upgrading a room will <i>not automatically migrate room members to the new version of the room.</i> We'll post a link to the new room in the old version of the room - room members will have to click this link to join the new room.": "<b>Увага!</b>: Покращення кімнати <i>не перенесе автоматично усіх учасників до нової версії кімнати.</i> Ми опублікуємо посилання на нову кімнату у старій версії кімнати, а учасники мають власноруч клацнути це посилання, щоб приєднатися до нової кімнати.", | ||||
|     "<b>Warning</b>: Upgrading a room will <i>not automatically migrate room members to the new version of the room.</i> We'll post a link to the new room in the old version of the room - room members will have to click this link to join the new room.": "<b>Увага!</b>: Поліпшення кімнати <i>не перенесе автоматично усіх учасників до нової версії кімнати.</i> Ми опублікуємо посилання на нову кімнату у старій версії кімнати, а учасники мають власноруч клацнути це посилання, щоб приєднатися до нової кімнати.", | ||||
|     "Changes your display nickname in the current room only": "Змінює ваше псевдо тільки для поточної кімнати", | ||||
|     "Changes the avatar of the current room": "Змінює аватар поточної кімнати", | ||||
|     "Changes your avatar in this current room only": "Змінює ваш аватар для поточної кімнати", | ||||
|  | @ -469,13 +469,13 @@ | |||
|     "This room has no topic.": "Ця кімната не має теми.", | ||||
|     "Sets the room name": "Встановлює назву кімнати", | ||||
|     "Use an identity server": "Використовувати сервер ідентифікації", | ||||
|     "Use an identity server to invite by email. Manage in Settings.": "Використовувати сервер ідентифікації для запрошень через е-пошту. Керується у налаштуваннях.", | ||||
|     "Use an identity server to invite by email. Manage in Settings.": "Використовувати сервер ідентифікації для запрошень через е-пошту. Керуйте у налаштуваннях.", | ||||
|     "Unbans user with given ID": "Розблоковує користувача з вказаним ідентифікатором", | ||||
|     "Adds a custom widget by URL to the room": "Додає власний віджет до кімнати за посиланням", | ||||
|     "Please supply a https:// or http:// widget URL": "Вкажіть посилання на віджет — https:// або http://", | ||||
|     "You cannot modify widgets in this room.": "Ви не можете змінювати віджети у цій кімнаті.", | ||||
|     "Forces the current outbound group session in an encrypted room to be discarded": "Примусово відкидає поточний вихідний груповий сеанс у шифрованій кімнаті", | ||||
|     "Sends the given message coloured as a rainbow": "Надсилає вказане повідомлення розфарбоване веселкою", | ||||
|     "Forces the current outbound group session in an encrypted room to be discarded": "Примусово відкидає поточний вихідний груповий сеанс у зашифрованій кімнаті", | ||||
|     "Sends the given message coloured as a rainbow": "Надсилає вказане повідомлення, розфарбоване веселкою", | ||||
|     "Your %(brand)s is misconfigured": "Ваш %(brand)s налаштовано неправильно", | ||||
|     "Join the discussion": "Приєднатися до обговорення", | ||||
|     "Upload": "Обрати", | ||||
|  | @ -639,47 +639,47 @@ | |||
|     "Room Settings - %(roomName)s": "Налаштування кімнати - %(roomName)s", | ||||
|     "A verification email will be sent to your inbox to confirm setting your new password.": "Ми відправимо перевіряльний електронний лист до вас для підтвердження зміни пароля.", | ||||
|     "To return to your account in future you need to set a password": "Щоб повернутися до своєї обліківки в майбутньому, вам потрібно встановити пароль", | ||||
|     "Use an identity server to invite by email. Click continue to use the default identity server (%(defaultIdentityServerName)s) or manage in Settings.": "Використайте сервер ідентифікації, щоб запросити е-поштою. Нажміть продовжити, щоб використовувати звичайний сервер ідентифікації(%(defaultIdentityServerName)s) або змініть у Налаштуваннях.", | ||||
|     "Use an identity server to invite by email. Click continue to use the default identity server (%(defaultIdentityServerName)s) or manage in Settings.": "Використовувати сервер ідентифікації, щоб запрошувати через е-пошту. Натисніть \"Продовжити\", щоб використовувати типовий сервер ідентифікації (%(defaultIdentityServerName)s) або змініть його у налаштуваннях.", | ||||
|     "Joins room with given address": "Приєднатися до кімнати зі вказаною адресою", | ||||
|     "Unrecognised room address:": "Не вдалося знайти адресу кімнати:", | ||||
|     "Unrecognised room address:": "Невпізнана адреса кімнати:", | ||||
|     "Command failed": "Не вдалося виконати команду", | ||||
|     "Could not find user in room": "Не вдалося знайти користувача в кімнаті", | ||||
|     "Please supply a widget URL or embed code": "Вкажіть URL або код вставки віджету", | ||||
|     "Verifies a user, session, and pubkey tuple": "Перевіряє користувача, сесію та публічні ключі", | ||||
|     "Unknown (user, session) pair:": "Невідома (користувач:сесія) пара:", | ||||
|     "Session already verified!": "Сесія вже підтверджена!", | ||||
|     "WARNING: Session already verified, but keys do NOT MATCH!": "УВАГА: Сесія вже підтверджена, проте ключі НЕ ЗБІГАЮТЬСЯ!", | ||||
|     "WARNING: KEY VERIFICATION FAILED! The signing key for %(userId)s and session %(deviceId)s is \"%(fprint)s\" which does not match the provided key \"%(fingerprint)s\". This could mean your communications are being intercepted!": "УВАГА: НЕ ВДАЛОСЯ ЗДІЙСНИТИ ПЕРЕВІРКУ КЛЮЧА! Ключом для %(userId)s та сесії %(devicerId)s являється \"%(fprint)s\", що не відповідає вказаному ключу \"%(fingerprint)s\". Це може означати, що ваші повідомлення перехоплюються!", | ||||
|     "The signing key you provided matches the signing key you received from %(userId)s's session %(deviceId)s. Session marked as verified.": "Ключ підпису, який ви надали, відповідає ключу підпису, який ви отримали від %(userId)s's сесії %(deviceId)s. Сесія відзначена як підтверджена.", | ||||
|     "Sends the given emote coloured as a rainbow": "Надсилає заданий смайлик, пофарбований у вигляді веселки", | ||||
|     "Displays list of commands with usages and descriptions": "Відображає список команд із описом та користуванням", | ||||
|     "Displays information about a user": "Показати інформацію про користувача", | ||||
|     "Send a bug report with logs": "Відправити звіт про помилку з логами", | ||||
|     "Opens chat with the given user": "Відкрити чат з даним користувачем", | ||||
|     "Sends a message to the given user": "Відправити повідомлення даному користувачу", | ||||
|     "%(senderName)s made no change.": "%(senderName)s не вніс змін.", | ||||
|     "Verifies a user, session, and pubkey tuple": "Звіряє користувача, сеанс та кортеж відкритого ключа", | ||||
|     "Unknown (user, session) pair:": "Невідома пара (користувача, сеансу):", | ||||
|     "Session already verified!": "Сеанс вже підтверджений!", | ||||
|     "WARNING: Session already verified, but keys do NOT MATCH!": "УВАГА: Сеанс вже підтверджений, проте ключі НЕ ЗБІГАЮТЬСЯ!", | ||||
|     "WARNING: KEY VERIFICATION FAILED! The signing key for %(userId)s and session %(deviceId)s is \"%(fprint)s\" which does not match the provided key \"%(fingerprint)s\". This could mean your communications are being intercepted!": "УВАГА: НЕ ВДАЛОСЯ ЗДІЙСНИТИ ЗВІРЯННЯ КЛЮЧА! Ключем для %(userId)s та сеансу %(deviceId)s є \"%(fprint)s\", що не відповідає наданому ключу \"%(fingerprint)s\". Це може означати, що ваші повідомлення перехоплюють!", | ||||
|     "The signing key you provided matches the signing key you received from %(userId)s's session %(deviceId)s. Session marked as verified.": "Наданий вами ключ підпису збігається з ключем підпису, що ви отримали від сеансу %(deviceId)s %(userId)s. Сеанс позначено як звірений.", | ||||
|     "Sends the given emote coloured as a rainbow": "Надсилає вказаний смайлик, розфарбований веселкою", | ||||
|     "Displays list of commands with usages and descriptions": "Відбиває перелік команд із прикладами вжитку та описом", | ||||
|     "Displays information about a user": "Відбиває інформацію про користувача", | ||||
|     "Send a bug report with logs": "Надіслати звіт про ваду разом з журналами", | ||||
|     "Opens chat with the given user": "Відкриває балачку з вказаним користувачем", | ||||
|     "Sends a message to the given user": "Надсилає повідомлення вказаному користувачеві", | ||||
|     "%(senderName)s made no change.": "%(senderName)s не запровадив(-ла) жодних змін.", | ||||
|     "%(senderDisplayName)s changed the room name from %(oldRoomName)s to %(newRoomName)s.": "%(senderDisplayName)s змінив(ла) назву кімнати з %(oldRoomName)s на %(newRoomName)s.", | ||||
|     "%(senderDisplayName)s upgraded this room.": "%(senderDisplayName)s модернізував цю кімнату.", | ||||
|     "%(senderDisplayName)s made the room public to whoever knows the link.": "%(senderDisplayName)s робить кімнату публічною для всіх, хто знає посилання.", | ||||
|     "%(senderDisplayName)s made the room invite only.": "%(senderDisplayName)s робить кімнату доступною лише по запрошенню.", | ||||
|     "%(senderDisplayName)s changed the join rule to %(rule)s": "%(senderDisplayName)s змінює правило входу на \"%(rule)s\"", | ||||
|     "%(senderDisplayName)s has allowed guests to join the room.": "%(senderDisplayName)s дозволяє гостям приєднуватися до кімнати.", | ||||
|     "%(senderDisplayName)s has prevented guests from joining the room.": "%(senderDisplayName)s забороняє гостям приєднуватися до кімнати.", | ||||
|     "%(senderDisplayName)s changed guest access to %(rule)s": "%(senderDisplayName)s змінює гостьовий доступ на \"%(rule)s\"", | ||||
|     "%(senderDisplayName)s enabled flair for %(groups)s in this room.": "%(senderDisplayName)s увімкнуто для %(groups)s у цій кімнаті.", | ||||
|     "%(senderDisplayName)s disabled flair for %(groups)s in this room.": "%(senderDisplayName)s вимкнуто для %(groups)s в цій кімнаті.", | ||||
|     "%(senderName)s added the alternative addresses %(addresses)s for this room.|other": "%(senderName)s додав(ла) альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s added the alternative addresses %(addresses)s for this room.|one": "%(senderName)s добавив(ла)альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s removed the alternative addresses %(addresses)s for this room.|other": "%(senderName)s видалив(ла) альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s removed the alternative addresses %(addresses)s for this room.|one": "%(senderName)s видалив(ла) альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s changed the alternative addresses for this room.": "%(senderName)s змінив(ла) альтернативні адреси для цієї кімнати.", | ||||
|     "%(senderName)s changed the main and alternative addresses for this room.": "%(senderName)s змінив(ла) головні та альтернативні адреси для цієї кімнати.", | ||||
|     "%(senderName)s changed the addresses for this room.": "%(senderName)s змінив(ла) адреси для цієї кімнати.", | ||||
|     "%(senderName)s placed a voice call.": "%(senderName)s зробив голосовий виклик.", | ||||
|     "%(senderName)s placed a voice call. (not supported by this browser)": "%(senderName)s зробив голосовий виклик. (не підтримується вашим браузером)", | ||||
|     "%(senderName)s placed a video call.": "%(senderName)s здійснив відео-виклик.", | ||||
|     "%(senderName)s placed a video call. (not supported by this browser)": "%(senderName)s здійснив відео-виклик. (не підтримується вашим браузером)", | ||||
|     "%(senderName)s revoked the invitation for %(targetDisplayName)s to join the room.": "%(senderName)s відкликав(ла) запрошення %(targetDisplayName)s приєднання до кімнати.", | ||||
|     "%(senderDisplayName)s upgraded this room.": "%(senderDisplayName)s поліпшив(-ла) цю кімнату.", | ||||
|     "%(senderDisplayName)s made the room public to whoever knows the link.": "%(senderDisplayName)s зробив(-ла) кімнату відкритою для всіх, хто знає посилання.", | ||||
|     "%(senderDisplayName)s made the room invite only.": "%(senderDisplayName)s зробив(-ла) кімнату доступною лише за запрошеннями.", | ||||
|     "%(senderDisplayName)s changed the join rule to %(rule)s": "%(senderDisplayName)s змінив(-ла) правило приєднування на \"%(rule)s\"", | ||||
|     "%(senderDisplayName)s has allowed guests to join the room.": "%(senderDisplayName)s дозволив(-ла) гостям приєднуватися до кімнати.", | ||||
|     "%(senderDisplayName)s has prevented guests from joining the room.": "%(senderDisplayName)s заборонив(-ла) гостям приєднуватися до кімнати.", | ||||
|     "%(senderDisplayName)s changed guest access to %(rule)s": "%(senderDisplayName)s змінив(-ла) гостьовий доступ на \"%(rule)s\"", | ||||
|     "%(senderDisplayName)s enabled flair for %(groups)s in this room.": "%(senderDisplayName)s увімкнув(-ла) значок для %(groups)s у цій кімнаті.", | ||||
|     "%(senderDisplayName)s disabled flair for %(groups)s in this room.": "%(senderDisplayName)s вимкнув(-ла) значок для %(groups)s в цій кімнаті.", | ||||
|     "%(senderName)s added the alternative addresses %(addresses)s for this room.|other": "%(senderName)s додав(-ла) альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s added the alternative addresses %(addresses)s for this room.|one": "%(senderName)s додав(-ла) альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s removed the alternative addresses %(addresses)s for this room.|other": "%(senderName)s прибрав(-ла) альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s removed the alternative addresses %(addresses)s for this room.|one": "%(senderName)s прибрав(-ла) альтернативні адреси %(addresses)s для цієї кімнати.", | ||||
|     "%(senderName)s changed the alternative addresses for this room.": "%(senderName)s змінив(-ла) альтернативні адреси для цієї кімнати.", | ||||
|     "%(senderName)s changed the main and alternative addresses for this room.": "%(senderName)s змінив(-ла) головні та альтернативні адреси для цієї кімнати.", | ||||
|     "%(senderName)s changed the addresses for this room.": "%(senderName)s змінив(-ла) адреси для цієї кімнати.", | ||||
|     "%(senderName)s placed a voice call.": "%(senderName)s розпочав(-ла) голосовий виклик.", | ||||
|     "%(senderName)s placed a voice call. (not supported by this browser)": "%(senderName)s розпочав(-ла) голосовий виклик. (не підтримується цим переглядачем)", | ||||
|     "%(senderName)s placed a video call.": "%(senderName)s розпочав(-ла) відеовиклик.", | ||||
|     "%(senderName)s placed a video call. (not supported by this browser)": "%(senderName)s розпочав(-ла) відеовиклик. (не підтримується цим переглядачем)", | ||||
|     "%(senderName)s revoked the invitation for %(targetDisplayName)s to join the room.": "%(senderName)s відкликав(-ла) запрошення %(targetDisplayName)s приєднання до кімнати.", | ||||
|     "%(senderName)s removed the rule banning users matching %(glob)s": "%(senderName)s видалив(ла) правило блокування користувачів по шаблону %(glob)s", | ||||
|     "%(senderName)s removed the rule banning rooms matching %(glob)s": "%(senderName)s видалив(ла) правило блокування кімнат по шаблону %(glob)s", | ||||
|     "%(senderName)s removed the rule banning servers matching %(glob)s": "%(senderName)s видалив(ла) правило блокування серверів по шаблону %(glob)s", | ||||
|  | @ -765,13 +765,13 @@ | |||
|     "Review where you’re logged in": "Перевірте, де ви ввійшли", | ||||
|     "Your homeserver has exceeded its user limit.": "Ваш домашній сервер перевищив свій ліміт користувачів.", | ||||
|     "Your homeserver has exceeded one of its resource limits.": "Ваш домашній сервер перевищив один із своїх ресурсних лімітів.", | ||||
|     "Contact your <a>server admin</a>.": "Зверніться до <a>адміністратора серверу<a>.", | ||||
|     "Contact your <a>server admin</a>.": "Зверніться до <a>адміністратора серверу</a>.", | ||||
|     "Ok": "Гаразд", | ||||
|     "Set password": "Встановити пароль", | ||||
|     "Set up encryption": "Налаштування шифрування", | ||||
|     "Encryption upgrade available": "Доступне оновлення шифрування", | ||||
|     "Encryption upgrade available": "Доступне поліпшене шифрування", | ||||
|     "Set up": "Налаштувати", | ||||
|     "Upgrade": "Оновлення", | ||||
|     "Upgrade": "Поліпшити", | ||||
|     "Other users may not trust it": "Інші користувачі можуть не довіряти цьому", | ||||
|     "New login. Was this you?": "Новий вхід у вашу обліківку. Це були Ви?", | ||||
|     "Restart": "Перезапустити", | ||||
|  | @ -855,7 +855,7 @@ | |||
|     "Preferences": "Параметри", | ||||
|     "Room list": "Перелік кімнат", | ||||
|     "Composer": "Редактор", | ||||
|     "Security & Privacy": "Безпека та конфіденціальність", | ||||
|     "Security & Privacy": "Безпека та конфіденційність", | ||||
|     "Where you’re logged in": "Де ви ввійшли", | ||||
|     "Skip": "Пропустити", | ||||
|     "Notification settings": "Налаштування сповіщень", | ||||
|  | @ -882,7 +882,7 @@ | |||
|     "Enable 'Manage Integrations' in Settings to do this.": "Щоб зробити це увімкніть \"Керувати інтеграціями\" у налаштуваннях.", | ||||
|     "Confirm by comparing the following with the User Settings in your other session:": "Підтвердьте шляхом порівняння наступного рядка з рядком у користувацьких налаштуваннях вашого іншого сеансу:", | ||||
|     "Confirm this user's session by comparing the following with their User Settings:": "Підтвердьте сеанс цього користувача шляхом порівняння наступного рядка з рядком з їхніх користувацьких налаштувань:", | ||||
|     "We recommend you change your password and recovery key in Settings immediately": "Ми радимо невідкладно змінити ваші пароль та ключ відновлення у налаштуваннях", | ||||
|     "We recommend you change your password and recovery key in Settings immediately": "Ми радимо невідкладно змінити ваші пароль та відновлювальний ключ у налаштуваннях", | ||||
|     "Share Message": "Поширити повідомлення", | ||||
|     "Community Settings": "Налаштування спільноти", | ||||
|     "All settings": "Усі налаштування", | ||||
|  | @ -968,7 +968,7 @@ | |||
|     "Cross-signing and secret storage are enabled.": "Кросс-підпис та секретне сховище дозволені.", | ||||
|     "well formed": "добре сформований", | ||||
|     "unexpected type": "несподіваний тип", | ||||
|     "Cross-signing public keys:": "Публічні ключі для кросс-підпису:", | ||||
|     "Cross-signing public keys:": "Перехресно-підписувальні відкриті ключі:", | ||||
|     "in memory": "у пам'яті", | ||||
|     "not found": "не знайдено", | ||||
|     "Cross-signing private keys:": "Приватні ключі для кросс-підпису:", | ||||
|  | @ -1004,5 +1004,101 @@ | |||
|     "Enter a new identity server": "Введіть новий сервер ідентифікації", | ||||
|     "Change": "Змінити", | ||||
|     "Manage integrations": "Керування інтеграціями", | ||||
|     "Size must be a number": "Розмір повинен бути числом" | ||||
|     "Size must be a number": "Розмір повинен бути числом", | ||||
|     "Incoming voice call": "Входовий голосовий виклик", | ||||
|     "Incoming video call": "Входовий відеовиклик", | ||||
|     "<a>Upgrade</a> to your own domain": "<a>Поліпшити</a> до свого власного домену", | ||||
|     "No Audio Outputs detected": "Звуковий вивід не виявлено", | ||||
|     "Audio Output": "Звуковий вивід", | ||||
|     "Voice & Video": "Голос та відео", | ||||
|     "Upgrade this room to the recommended room version": "Поліпшити цю кімнату до рекомендованої версії", | ||||
|     "this room": "ця кімната", | ||||
|     "Upgrade the room": "Поліпшити кімнату", | ||||
|     "Unable to revoke sharing for email address": "Не вдалось відкликати оприлюднювання адреси е-пошти", | ||||
|     "Revoke": "Відкликати", | ||||
|     "Unable to revoke sharing for phone number": "Не вдалось відкликати оприлюднювання телефонного номеру", | ||||
|     "Filter room members": "Відфільтрувати учасників кімнати", | ||||
|     "Voice call": "Голосовий виклик", | ||||
|     "Video call": "Відеовиклик", | ||||
|     "Not now": "Не зараз", | ||||
|     "Don't ask me again": "Не запитувати мене знову", | ||||
|     "Appearance": "Вигляд", | ||||
|     "Show rooms with unread messages first": "Показувати вгорі кімнати з непрочитаними повідомленнями", | ||||
|     "Show previews of messages": "Показувати попередній перегляд повідомлень", | ||||
|     "Sort by": "Упорядкувати за", | ||||
|     "Activity": "Активністю", | ||||
|     "A-Z": "А-Я", | ||||
|     "List options": "Параметри переліку", | ||||
|     "Use default": "Типово", | ||||
|     "Mentions & Keywords": "Згадки та ключові слова", | ||||
|     "Notification options": "Параметри сповіщень", | ||||
|     "Leave Room": "Вийти з кімнати", | ||||
|     "Forget Room": "Забути кімнату", | ||||
|     "Favourited": "Улюблено", | ||||
|     "%(count)s unread messages including mentions.|other": "%(count)s непрочитаних повідомлень включно зі згадками.", | ||||
|     "%(count)s unread messages including mentions.|one": "1 непрочитана згадка.", | ||||
|     "%(count)s unread messages.|other": "%(count)s непрочитаних повідомлень.", | ||||
|     "%(count)s unread messages.|one": "1 непрочитане повідомлення.", | ||||
|     "Unread messages.": "Непрочитані повідомлення.", | ||||
|     "This room is public": "Ця кімната є прилюдною", | ||||
|     "Show Stickers": "Показати наліпки", | ||||
|     "Failed to revoke invite": "Не вдалось відкликати запрошення", | ||||
|     "Could not revoke the invite. The server may be experiencing a temporary problem or you do not have sufficient permissions to revoke the invite.": "Не вдалось відкликати запрошення. Сервер може мати тимчасові збої або у вас немає достатніх дозволів щоб відкликати запрошення.", | ||||
|     "Revoke invite": "Відкликати запрошення", | ||||
|     "Security": "Безпека", | ||||
|     "Report bugs & give feedback": "Відзвітувати про вади та залишити відгук", | ||||
|     "Report Content to Your Homeserver Administrator": "Поскаржитись на зміст адміністратору вашого домашнього сервера", | ||||
|     "Failed to upgrade room": "Не вдалось поліпшити кімнату", | ||||
|     "The room upgrade could not be completed": "Поліпшення кімнати не може бути завершене", | ||||
|     "Upgrade this room to version %(version)s": "Поліпшити цю кімнату до версії %(version)s", | ||||
|     "Upgrade Room Version": "Поліпшити версію кімнати", | ||||
|     "Upgrade private room": "Поліпшити закриту кімнату", | ||||
|     "You'll upgrade this room from <oldVersion /> to <newVersion />.": "Ви поліпшите цю кімнату з <oldVersion /> до <newVersion /> версії.", | ||||
|     "Share Room Message": "Поширити повідомлення кімнати", | ||||
|     "Report Content": "Поскаржитись на зміст", | ||||
|     "Feedback": "Зворотній зв'язок", | ||||
|     "General failure": "Загальний збій", | ||||
|     "Enter your account password to confirm the upgrade:": "Введіть пароль вашого облікового запису щоб підтвердити поліпшення:", | ||||
|     "Security & privacy": "Безпека та конфіденційність", | ||||
|     "Secret storage public key:": "Таємне сховище відкритого ключа:", | ||||
|     "Key backup": "Резервне копіювання ключів", | ||||
|     "Message search": "Пошук повідомлень", | ||||
|     "Cross-signing": "Перехресне підписування", | ||||
|     "Your server admin has disabled end-to-end encryption by default in private rooms & Direct Messages.": "Адміністратором вашого сервера було вимкнено початкове наскрізне шифрування у закритих кімнатах та особистих повідомленнях.", | ||||
|     "Something went wrong!": "Щось пішло не так!", | ||||
|     "expand": "розгорнути", | ||||
|     "Wrong Recovery Key": "Неправильний відновлювальний ключ", | ||||
|     "Invalid Recovery Key": "Нечинний відновлювальний ключ", | ||||
|     "Recovery key mismatch": "Незбіг відновлювального ключа", | ||||
|     "Backup could not be decrypted with this recovery key: please verify that you entered the correct recovery key.": "Резервна копія не може бути дешифрована з цим відновлювальним ключем: переконайтесь, будь ласка, що ви ввели правильний відновлювальний ключ.", | ||||
|     "If you've forgotten your recovery passphrase you can <button1>use your recovery key</button1> or <button2>set up new recovery options</button2>": "Якщо ви забули відновлювальну парольну фразу, ви можете <button1>скористатись вашим відновлювальним ключем</button1> або <button2>налаштувати нові параметри відновлювання</button2>", | ||||
|     "Enter recovery key": "Введіть відновлювальний ключ", | ||||
|     "This looks like a valid recovery key!": "Це скидається на чинний відновлювальний ключ!", | ||||
|     "Not a valid recovery key": "Нечинний відновлювальний ключ", | ||||
|     "Access your secure message history and set up secure messaging by entering your recovery key.": "Доступіться до вашої захищеної історії повідомлень та налаштуйте захищене листування шляхом вводження вашого відновлювального ключа.", | ||||
|     "If you've forgotten your recovery key you can <button>set up new recovery options</button>": "Якщо ви забули ваш відновлювальний ключ, ви можете <button>наново налаштувати параметри відновлювання</button>", | ||||
|     "Switch to light mode": "Світла тема", | ||||
|     "Switch to dark mode": "Темна тема", | ||||
|     "Use Recovery Key or Passphrase": "Скористуйтесь відновлювальними ключем або парольною фразою", | ||||
|     "Use Recovery Key": "Скористуйтесь відновлювальним ключем", | ||||
|     "Set up with a recovery key": "Налаштувати з відновлювальним ключем", | ||||
|     "Your recovery key is a safety net - you can use it to restore access to your encrypted messages if you forget your recovery passphrase.": "Ваш відновлювальний ключ — це убезпека. Ви можете використовувати його щоб доступитись до ваших зашифрованих повідомлень у разі втрати вашої відновлювальної парольної фрази.", | ||||
|     "Your recovery key": "Ваш відновлювальний ключ", | ||||
|     "Your recovery key has been <b>copied to your clipboard</b>, paste it to:": "Ваш відновлювальний ключ було <b>скопійовано до буферу обміну</b>, вставте його у:", | ||||
|     "Your recovery key is in your <b>Downloads</b> folder.": "Ваш відновлювальний ключ у вашій теці <b>Завантаження</b>.", | ||||
|     "Make a copy of your recovery key": "Зробити копію вашого відновлювального ключа", | ||||
|     "Don't ask again": "Не запитувати знову", | ||||
|     "New Recovery Method": "Новий відновлювальний засіб", | ||||
|     "A new recovery passphrase and key for Secure Messages have been detected.": "Було виявлено нові відновлювальні парольну фразу та ключ від захищених повідомлень.", | ||||
|     "This session is encrypting history using the new recovery method.": "Цей сеанс зашифровує історію новим відновлювальним засобом.", | ||||
|     "Set up Secure Messages": "Налаштувати захищені повідомлення", | ||||
|     "Recovery Method Removed": "Відновлювальний засіб було видалено", | ||||
|     "This session has detected that your recovery passphrase and key for Secure Messages have been removed.": "Цей сеанс виявив, що ваші відновлювальні парольна фраза та ключ від захищених повідомлень були видалені.", | ||||
|     "%(senderDisplayName)s enabled flair for %(newGroups)s and disabled flair for %(oldGroups)s in this room.": "%(senderDisplayName)s увімкнув(-ла) значок для %(newGroups)s та вимкнув(-ла) значок для %(oldGroups)s у цій кімнаті.", | ||||
|     "New version available. <a>Update now.</a>": "Доступна нова версія. <a>Оновити зараз</a>", | ||||
|     "Upgrade public room": "Поліпшити відкриту кімнату", | ||||
|     "Restore your key backup to upgrade your encryption": "Відновіть резервну копію вашого ключа щоб поліпшити шифрування", | ||||
|     "You'll need to authenticate with the server to confirm the upgrade.": "Ви матимете пройти розпізнання на сервері щоб підтвердити поліпшування.", | ||||
|     "Upgrade this session to allow it to verify other sessions, granting them access to encrypted messages and marking them as trusted for other users.": "Поліпште цей сеанс щоб уможливити звіряння інших сеансів, надаючи їм доступ до зашифрованих повідомлень та позначуючи їх довіреними для інших користувачів.", | ||||
|     "Upgrade your encryption": "Поліпшити ваше шифрування" | ||||
| } | ||||
|  |  | |||
|  | @ -18,8 +18,9 @@ import PlatformPeg from "../PlatformPeg"; | |||
| import {MatrixClientPeg} from "../MatrixClientPeg"; | ||||
| import {EventTimeline, RoomMember} from 'matrix-js-sdk'; | ||||
| import {sleep} from "../utils/promise"; | ||||
| import SettingsStore, {SettingLevel} from "../settings/SettingsStore"; | ||||
| import SettingsStore from "../settings/SettingsStore"; | ||||
| import {EventEmitter} from "events"; | ||||
| import {SettingLevel} from "../settings/SettingLevel"; | ||||
| 
 | ||||
| /* | ||||
|  * Event indexing class that wraps the platform specific event indexing. | ||||
|  |  | |||
|  | @ -21,7 +21,8 @@ limitations under the License. | |||
| 
 | ||||
| import PlatformPeg from "../PlatformPeg"; | ||||
| import EventIndex from "../indexing/EventIndex"; | ||||
| import SettingsStore, {SettingLevel} from '../settings/SettingsStore'; | ||||
| import SettingsStore from '../settings/SettingsStore'; | ||||
| import {SettingLevel} from "../settings/SettingLevel"; | ||||
| 
 | ||||
| const INDEX_VERSION = 1; | ||||
| 
 | ||||
|  |  | |||
|  | @ -21,11 +21,12 @@ import request from 'browser-request'; | |||
| import counterpart from 'counterpart'; | ||||
| import React from 'react'; | ||||
| 
 | ||||
| import SettingsStore, {SettingLevel} from "./settings/SettingsStore"; | ||||
| import SettingsStore from "./settings/SettingsStore"; | ||||
| import PlatformPeg from "./PlatformPeg"; | ||||
| 
 | ||||
| // @ts-ignore - $webapp is a webpack resolve alias pointing to the output directory, see webpack config
 | ||||
| import webpackLangJsonUrl from "$webapp/i18n/languages.json"; | ||||
| import { SettingLevel } from "./settings/SettingLevel"; | ||||
| 
 | ||||
| const i18nFolder = 'i18n/'; | ||||
| 
 | ||||
|  |  | |||
|  | @ -16,9 +16,10 @@ limitations under the License. | |||
| 
 | ||||
| import {MatrixClientPeg} from "../MatrixClientPeg"; | ||||
| import {ALL_RULE_TYPES, BanList} from "./BanList"; | ||||
| import SettingsStore, {SettingLevel} from "../settings/SettingsStore"; | ||||
| import SettingsStore from "../settings/SettingsStore"; | ||||
| import {_t} from "../languageHandler"; | ||||
| import dis from "../dispatcher/dispatcher"; | ||||
| import {SettingLevel} from "../settings/SettingLevel"; | ||||
| 
 | ||||
| // TODO: Move this and related files to the js-sdk or something once finalized.
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -141,7 +141,7 @@ export default async function sendBugReport(bugReportEndpoint: string, opts: IOp | |||
|     } | ||||
| 
 | ||||
|     // add labs options
 | ||||
|     const enabledLabs = SettingsStore.getLabsFeatures().filter(SettingsStore.isFeatureEnabled); | ||||
|     const enabledLabs = SettingsStore.getLabsFeatures().filter(f => SettingsStore.isFeatureEnabled(f)); | ||||
|     if (enabledLabs.length) { | ||||
|         body.append('enabled_labs', enabledLabs.join(', ')); | ||||
|     } | ||||
|  |  | |||
|  | @ -0,0 +1,29 @@ | |||
| /* | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| /** | ||||
|  * Represents the various setting levels supported by the SettingsStore. | ||||
|  */ | ||||
| export enum SettingLevel { | ||||
|     // TODO: [TS] Follow naming convention
 | ||||
|     DEVICE = "device", | ||||
|     ROOM_DEVICE = "room-device", | ||||
|     ROOM_ACCOUNT = "room-account", | ||||
|     ACCOUNT = "account", | ||||
|     ROOM = "room", | ||||
|     CONFIG = "config", | ||||
|     DEFAULT = "default", | ||||
| } | ||||
|  | @ -1,7 +1,6 @@ | |||
| /* | ||||
| Copyright 2017 Travis Ralston | ||||
| Copyright 2018, 2019 New Vector Ltd. | ||||
| Copyright 2019, 2020 The Matrix.org Foundation C.I.C. | ||||
| Copyright 2018, 2019, 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
|  | @ -16,9 +15,9 @@ See the License for the specific language governing permissions and | |||
| limitations under the License. | ||||
| */ | ||||
| 
 | ||||
| import {MatrixClient} from 'matrix-js-sdk'; | ||||
| import { MatrixClient } from 'matrix-js-sdk/src/client'; | ||||
| 
 | ||||
| import {_td} from '../languageHandler'; | ||||
| import { _td } from '../languageHandler'; | ||||
| import { | ||||
|     AudioNotificationsEnabledController, | ||||
|     NotificationBodyEnabledController, | ||||
|  | @ -28,75 +27,89 @@ import CustomStatusController from "./controllers/CustomStatusController"; | |||
| import ThemeController from './controllers/ThemeController'; | ||||
| import PushToMatrixClientController from './controllers/PushToMatrixClientController'; | ||||
| import ReloadOnChangeController from "./controllers/ReloadOnChangeController"; | ||||
| import {RIGHT_PANEL_PHASES} from "../stores/RightPanelStorePhases"; | ||||
| import FontSizeController from './controllers/FontSizeController'; | ||||
| import SystemFontController from './controllers/SystemFontController'; | ||||
| import UseSystemFontController from './controllers/UseSystemFontController'; | ||||
| import { SettingLevel } from "./SettingLevel"; | ||||
| import SettingController from "./controllers/SettingController"; | ||||
| import { RightPanelPhases } from "../stores/RightPanelStorePhases"; | ||||
| 
 | ||||
| // 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_OR_ACCOUNT = ['room-account', 'account']; | ||||
| const LEVELS_ROOM_SETTINGS_WITH_ROOM = ['device', 'room-device', 'room-account', 'account', 'config', 'room']; | ||||
| const LEVELS_ACCOUNT_SETTINGS = ['device', 'account', 'config']; | ||||
| const LEVELS_FEATURE = ['device', 'config']; | ||||
| const LEVELS_DEVICE_ONLY_SETTINGS = ['device']; | ||||
| const LEVELS_DEVICE_ONLY_SETTINGS_WITH_CONFIG = ['device', 'config']; | ||||
| const LEVELS_ROOM_SETTINGS = [ | ||||
|     SettingLevel.DEVICE, | ||||
|     SettingLevel.ROOM_DEVICE, | ||||
|     SettingLevel.ROOM_ACCOUNT, | ||||
|     SettingLevel.ACCOUNT, | ||||
|     SettingLevel.CONFIG, | ||||
| ]; | ||||
| const LEVELS_ROOM_OR_ACCOUNT = [ | ||||
|     SettingLevel.ROOM_ACCOUNT, | ||||
|     SettingLevel.ACCOUNT, | ||||
| ]; | ||||
| const LEVELS_ROOM_SETTINGS_WITH_ROOM = [ | ||||
|     SettingLevel.DEVICE, | ||||
|     SettingLevel.ROOM_DEVICE, | ||||
|     SettingLevel.ROOM_ACCOUNT, | ||||
|     SettingLevel.ACCOUNT, | ||||
|     SettingLevel.CONFIG, | ||||
|     SettingLevel.ROOM, | ||||
| ]; | ||||
| const LEVELS_ACCOUNT_SETTINGS = [ | ||||
|     SettingLevel.DEVICE, | ||||
|     SettingLevel.ACCOUNT, | ||||
|     SettingLevel.CONFIG, | ||||
| ]; | ||||
| const LEVELS_FEATURE = [ | ||||
|     SettingLevel.DEVICE, | ||||
|     SettingLevel.CONFIG, | ||||
| ]; | ||||
| const LEVELS_DEVICE_ONLY_SETTINGS = [ | ||||
|     SettingLevel.DEVICE, | ||||
| ]; | ||||
| const LEVELS_DEVICE_ONLY_SETTINGS_WITH_CONFIG = [ | ||||
|     SettingLevel.DEVICE, | ||||
|     SettingLevel.CONFIG, | ||||
| ]; | ||||
| 
 | ||||
| export const SETTINGS = { | ||||
|     // EXAMPLE SETTING:
 | ||||
|     // "my-setting": {
 | ||||
|     //     // Must be set to true for features. Default is 'false'.
 | ||||
|     //     isFeature: false,
 | ||||
|     //
 | ||||
|     //     // Display names are strongly recommended for clarity.
 | ||||
|     //     displayName: _td("Cool Name"),
 | ||||
|     //
 | ||||
|     //     // Display name can also be an object for different levels.
 | ||||
|     //     //displayName: {
 | ||||
|     //     //    "device": _td("Name for when the setting is used at 'device'"),
 | ||||
|     //     //    "room": _td("Name for when the setting is used at 'room'"),
 | ||||
|     //     //    "default": _td("The name for all other levels"),
 | ||||
|     //     //}
 | ||||
|     //
 | ||||
|     //     // The supported levels are required. Preferably, use the preset arrays
 | ||||
|     //     // at the top of this file to define this rather than a custom array.
 | ||||
|     //     supportedLevels: [
 | ||||
|     //         // The order does not matter.
 | ||||
|     //
 | ||||
|     //         "device",        // Affects the current device only
 | ||||
|     //         "room-device",   // Affects the current room on the current device
 | ||||
|     //         "room-account",  // Affects the current room for the current account
 | ||||
|     //         "account",       // Affects the current account
 | ||||
|     //         "room",          // Affects the current room (controlled by room admins)
 | ||||
|     //         "config",        // Affects the current application
 | ||||
|     //
 | ||||
|     //         // "default" is always supported and does not get listed here.
 | ||||
|     //     ],
 | ||||
|     //
 | ||||
|     //     // Required. Can be any data type. The value specified here should match
 | ||||
|     //     // the data being stored (ie: if a boolean is used, the setting should
 | ||||
|     //     // represent a boolean).
 | ||||
|     //     default: {
 | ||||
|     //         your: "value",
 | ||||
|     //     },
 | ||||
|     //
 | ||||
|     //     // Optional settings controller. See SettingsController for more information.
 | ||||
|     //     controller: new MySettingController(),
 | ||||
|     //
 | ||||
|     //     // Optional flag to make supportedLevels be respected as the order to handle
 | ||||
|     //     // settings. The first element is treated as "most preferred". The "default"
 | ||||
|     //     // level is always appended to the end.
 | ||||
|     //     supportedLevelsAreOrdered: false,
 | ||||
|     //
 | ||||
|     //     // Optional value to invert a boolean setting's value. The string given will
 | ||||
|     //     // be read as the setting's ID instead of the one provided as the key for the
 | ||||
|     //     // setting definition. By setting this, the returned value will automatically
 | ||||
|     //     // be inverted, except for when the default value is returned. Inversion will
 | ||||
|     //     // occur after the controller is asked for an override. This should be used by
 | ||||
|     //     // historical settings which we don't want existing user's values be wiped. Do
 | ||||
|     //     // not use this for new settings.
 | ||||
|     //     invertedSettingName: "my-negative-setting",
 | ||||
|     // },
 | ||||
| export interface ISetting { | ||||
|     // Must be set to true for features. Default is 'false'.
 | ||||
|     isFeature?: boolean; | ||||
| 
 | ||||
|     // Display names are strongly recommended for clarity.
 | ||||
|     // Display name can also be an object for different levels.
 | ||||
|     displayName?: string | { | ||||
|         // @ts-ignore - TS wants the key to be a string, but we know better
 | ||||
|         [level: SettingLevel]: string; | ||||
|     }; | ||||
| 
 | ||||
|     // The supported levels are required. Preferably, use the preset arrays
 | ||||
|     // at the top of this file to define this rather than a custom array.
 | ||||
|     supportedLevels?: SettingLevel[]; | ||||
| 
 | ||||
|     // Required. Can be any data type. The value specified here should match
 | ||||
|     // the data being stored (ie: if a boolean is used, the setting should
 | ||||
|     // represent a boolean).
 | ||||
|     default: any; | ||||
| 
 | ||||
|     // Optional settings controller. See SettingsController for more information.
 | ||||
|     controller?: SettingController; | ||||
| 
 | ||||
|     // Optional flag to make supportedLevels be respected as the order to handle
 | ||||
|     // settings. The first element is treated as "most preferred". The "default"
 | ||||
|     // level is always appended to the end.
 | ||||
|     supportedLevelsAreOrdered?: boolean; | ||||
| 
 | ||||
|     // Optional value to invert a boolean setting's value. The string given will
 | ||||
|     // be read as the setting's ID instead of the one provided as the key for the
 | ||||
|     // setting definition. By setting this, the returned value will automatically
 | ||||
|     // be inverted, except for when the default value is returned. Inversion will
 | ||||
|     // occur after the controller is asked for an override. This should be used by
 | ||||
|     // historical settings which we don't want existing user's values be wiped. Do
 | ||||
|     // not use this for new settings.
 | ||||
|     invertedSettingName?: string; | ||||
| } | ||||
| 
 | ||||
| export const SETTINGS: {[setting: string]: ISetting} = { | ||||
|     "feature_new_spinner": { | ||||
|         isFeature: true, | ||||
|         displayName: _td("New spinner design"), | ||||
|  | @ -153,11 +166,11 @@ export const SETTINGS = { | |||
|         default: false, | ||||
|     }, | ||||
|     "mjolnirRooms": { | ||||
|         supportedLevels: ['account'], | ||||
|         supportedLevels: [SettingLevel.ACCOUNT], | ||||
|         default: [], | ||||
|     }, | ||||
|     "mjolnirPersonalRoom": { | ||||
|         supportedLevels: ['account'], | ||||
|         supportedLevels: [SettingLevel.ACCOUNT], | ||||
|         default: null, | ||||
|     }, | ||||
|     "feature_bridge_state": { | ||||
|  | @ -354,24 +367,24 @@ export const SETTINGS = { | |||
|     }, | ||||
|     "breadcrumb_rooms": { | ||||
|         // not really a setting
 | ||||
|         supportedLevels: ['account'], | ||||
|         supportedLevels: [SettingLevel.ACCOUNT], | ||||
|         default: [], | ||||
|     }, | ||||
|     "recent_emoji": { | ||||
|         // not really a setting
 | ||||
|         supportedLevels: ['account'], | ||||
|         supportedLevels: [SettingLevel.ACCOUNT], | ||||
|         default: [], | ||||
|     }, | ||||
|     "room_directory_servers": { | ||||
|         supportedLevels: ['account'], | ||||
|         supportedLevels: [SettingLevel.ACCOUNT], | ||||
|         default: [], | ||||
|     }, | ||||
|     "integrationProvisioning": { | ||||
|         supportedLevels: ['account'], | ||||
|         supportedLevels: [SettingLevel.ACCOUNT], | ||||
|         default: true, | ||||
|     }, | ||||
|     "allowedWidgets": { | ||||
|         supportedLevels: ['room-account'], | ||||
|         supportedLevels: [SettingLevel.ROOM_ACCOUNT], | ||||
|         default: {}, // none allowed
 | ||||
|     }, | ||||
|     "analyticsOptIn": { | ||||
|  | @ -398,7 +411,7 @@ export const SETTINGS = { | |||
|     "blacklistUnverifiedDevices": { | ||||
|         // We specifically want to have room-device > device so that users may set a device default
 | ||||
|         // with a per-room override.
 | ||||
|         supportedLevels: ['room-device', 'device'], | ||||
|         supportedLevels: [SettingLevel.ROOM_DEVICE, SettingLevel.DEVICE], | ||||
|         supportedLevelsAreOrdered: true, | ||||
|         displayName: { | ||||
|             "default": _td('Never send encrypted messages to unverified sessions from this session'), | ||||
|  | @ -416,7 +429,7 @@ export const SETTINGS = { | |||
|         default: true, | ||||
|     }, | ||||
|     "urlPreviewsEnabled_e2ee": { | ||||
|         supportedLevels: ['room-device', 'room-account'], | ||||
|         supportedLevels: [SettingLevel.ROOM_DEVICE, SettingLevel.ROOM_ACCOUNT], | ||||
|         displayName: { | ||||
|             "room-account": _td("Enable URL previews for this room (only affects you)"), | ||||
|         }, | ||||
|  | @ -455,7 +468,7 @@ export const SETTINGS = { | |||
|         default: false, | ||||
|     }, | ||||
|     "PinnedEvents.isOpen": { | ||||
|         supportedLevels: ['room-device'], | ||||
|         supportedLevels: [SettingLevel.ROOM_DEVICE], | ||||
|         default: false, | ||||
|     }, | ||||
|     "promptBeforeInviteUnknownUsers": { | ||||
|  | @ -534,11 +547,11 @@ export const SETTINGS = { | |||
|     }, | ||||
|     "lastRightPanelPhaseForRoom": { | ||||
|         supportedLevels: LEVELS_DEVICE_ONLY_SETTINGS, | ||||
|         default: RIGHT_PANEL_PHASES.RoomMemberInfo, | ||||
|         default: RightPanelPhases.RoomMemberInfo, | ||||
|     }, | ||||
|     "lastRightPanelPhaseForGroup": { | ||||
|         supportedLevels: LEVELS_DEVICE_ONLY_SETTINGS, | ||||
|         default: RIGHT_PANEL_PHASES.GroupMemberList, | ||||
|         default: RightPanelPhases.GroupMemberList, | ||||
|     }, | ||||
|     "enableEventIndexing": { | ||||
|         supportedLevels: LEVELS_DEVICE_ONLY_SETTINGS, | ||||
|  | @ -565,7 +578,8 @@ export const SETTINGS = { | |||
|     "ircDisplayNameWidth": { | ||||
|         // We specifically want to have room-device > device so that users may set a device default
 | ||||
|         // with a per-room override.
 | ||||
|         supportedLevels: ['room-device', 'device'], | ||||
|         supportedLevels: [SettingLevel.ROOM_DEVICE, SettingLevel.DEVICE], | ||||
|         supportedLevelsAreOrdered: true, | ||||
|         displayName: _td("IRC display name width"), | ||||
|         default: 80, | ||||
|     }, | ||||
|  | @ -1,6 +1,6 @@ | |||
| /* | ||||
| Copyright 2017 Travis Ralston | ||||
| Copyright 2019 New Vector Ltd. | ||||
| Copyright 2019, 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
|  | @ -22,27 +22,14 @@ import RoomAccountSettingsHandler from "./handlers/RoomAccountSettingsHandler"; | |||
| import AccountSettingsHandler from "./handlers/AccountSettingsHandler"; | ||||
| import RoomSettingsHandler from "./handlers/RoomSettingsHandler"; | ||||
| import ConfigSettingsHandler from "./handlers/ConfigSettingsHandler"; | ||||
| import {_t} from '../languageHandler'; | ||||
| import { _t } from '../languageHandler'; | ||||
| import SdkConfig from "../SdkConfig"; | ||||
| import dis from '../dispatcher/dispatcher'; | ||||
| import {SETTINGS} from "./Settings"; | ||||
| import { ISetting, SETTINGS } from "./Settings"; | ||||
| import LocalEchoWrapper from "./handlers/LocalEchoWrapper"; | ||||
| import {WatchManager} from "./WatchManager"; | ||||
| 
 | ||||
| /** | ||||
|  * Represents the various setting levels supported by the SettingsStore. | ||||
|  */ | ||||
| export const SettingLevel = { | ||||
|     // Note: This enum is not used in this class or in the Settings file
 | ||||
|     // This should always be used elsewhere in the project.
 | ||||
|     DEVICE: "device", | ||||
|     ROOM_DEVICE: "room-device", | ||||
|     ROOM_ACCOUNT: "room-account", | ||||
|     ACCOUNT: "account", | ||||
|     ROOM: "room", | ||||
|     CONFIG: "config", | ||||
|     DEFAULT: "default", | ||||
| }; | ||||
| import { WatchManager } from "./WatchManager"; | ||||
| import { SettingLevel } from "./SettingLevel"; | ||||
| import SettingsHandler from "./handlers/SettingsHandler"; | ||||
| 
 | ||||
| const defaultWatchManager = new WatchManager(); | ||||
| 
 | ||||
|  | @ -61,13 +48,13 @@ for (const key of Object.keys(SETTINGS)) { | |||
| } | ||||
| 
 | ||||
| const LEVEL_HANDLERS = { | ||||
|     "device": new DeviceSettingsHandler(featureNames, defaultWatchManager), | ||||
|     "room-device": new RoomDeviceSettingsHandler(defaultWatchManager), | ||||
|     "room-account": new RoomAccountSettingsHandler(defaultWatchManager), | ||||
|     "account": new AccountSettingsHandler(defaultWatchManager), | ||||
|     "room": new RoomSettingsHandler(defaultWatchManager), | ||||
|     "config": new ConfigSettingsHandler(), | ||||
|     "default": new DefaultSettingsHandler(defaultSettings, invertedDefaultSettings), | ||||
|     [SettingLevel.DEVICE]: new DeviceSettingsHandler(featureNames, defaultWatchManager), | ||||
|     [SettingLevel.ROOM_DEVICE]: new RoomDeviceSettingsHandler(defaultWatchManager), | ||||
|     [SettingLevel.ROOM_ACCOUNT]: new RoomAccountSettingsHandler(defaultWatchManager), | ||||
|     [SettingLevel.ACCOUNT]: new AccountSettingsHandler(defaultWatchManager), | ||||
|     [SettingLevel.ROOM]: new RoomSettingsHandler(defaultWatchManager), | ||||
|     [SettingLevel.CONFIG]: new ConfigSettingsHandler(), | ||||
|     [SettingLevel.DEFAULT]: new DefaultSettingsHandler(defaultSettings, invertedDefaultSettings), | ||||
| }; | ||||
| 
 | ||||
| // Wrap all the handlers with local echo
 | ||||
|  | @ -76,20 +63,41 @@ for (const key of Object.keys(LEVEL_HANDLERS)) { | |||
| } | ||||
| 
 | ||||
| const LEVEL_ORDER = [ | ||||
|     'device', 'room-device', 'room-account', 'account', 'room', 'config', 'default', | ||||
|     SettingLevel.DEVICE, | ||||
|     SettingLevel.ROOM_DEVICE, | ||||
|     SettingLevel.ROOM_ACCOUNT, | ||||
|     SettingLevel.ACCOUNT, | ||||
|     SettingLevel.ROOM, | ||||
|     SettingLevel.CONFIG, | ||||
|     SettingLevel.DEFAULT, | ||||
| ]; | ||||
| 
 | ||||
| export type CallbackFn = ( | ||||
|     settingName: string, | ||||
|     roomId: string, | ||||
|     atLevel: SettingLevel, | ||||
|     newValAtLevel: any, | ||||
|     newVal: any, | ||||
| ) => void; | ||||
| 
 | ||||
| interface IHandlerMap { | ||||
|     // @ts-ignore - TS wants this to be a string key but we know better
 | ||||
|     [level: SettingLevel]: SettingsHandler; | ||||
| } | ||||
| 
 | ||||
| export type LabsFeatureState = "labs" | "disable" | "enable" | string; | ||||
| 
 | ||||
| /** | ||||
|  * Controls and manages application settings by providing varying levels at which the | ||||
|  * setting value may be specified. The levels are then used to determine what the setting | ||||
|  * value should be given a set of circumstances. The levels, in priority order, are: | ||||
|  * - "device"         - Values are determined by the current device | ||||
|  * - "room-device"    - Values are determined by the current device for a particular room | ||||
|  * - "room-account"   - Values are determined by the current account for a particular room | ||||
|  * - "account"        - Values are determined by the current account | ||||
|  * - "room"           - Values are determined by a particular room (by the room admins) | ||||
|  * - "config"         - Values are determined by the config.json | ||||
|  * - "default"        - Values are determined by the hardcoded defaults | ||||
|  * - SettingLevel.DEVICE         - Values are determined by the current device | ||||
|  * - SettingLevel.ROOM_DEVICE    - Values are determined by the current device for a particular room | ||||
|  * - SettingLevel.ROOM_ACCOUNT   - Values are determined by the current account for a particular room | ||||
|  * - SettingLevel.ACCOUNT        - Values are determined by the current account | ||||
|  * - SettingLevel.ROOM           - Values are determined by a particular room (by the room admins) | ||||
|  * - SettingLevel.CONFIG         - Values are determined by the config.json | ||||
|  * - SettingLevel.DEFAULT        - Values are determined by the hardcoded defaults | ||||
|  * | ||||
|  * Each level has a different method to storing the setting value. For implementation | ||||
|  * specific details, please see the handlers. The "config" and "default" levels are | ||||
|  | @ -110,11 +118,11 @@ export default class SettingsStore { | |||
|     // We also maintain a list of monitors which are special watchers: they cause dispatches
 | ||||
|     // when the setting changes. We track which rooms we're monitoring though to ensure we
 | ||||
|     // don't duplicate updates on the bus.
 | ||||
|     static _watchers = {}; // { callbackRef => { callbackFn } }
 | ||||
|     static _monitors = {}; // { settingName => { roomId => callbackRef } }
 | ||||
|     private static watchers = {}; // { callbackRef => { callbackFn } }
 | ||||
|     private static monitors = {}; // { settingName => { roomId => callbackRef } }
 | ||||
| 
 | ||||
|     // Counter used for generation of watcher IDs
 | ||||
|     static _watcherCount = 1; | ||||
|     private static watcherCount = 1; | ||||
| 
 | ||||
|     /** | ||||
|      * Watches for changes in a particular setting. This is done without any local echo | ||||
|  | @ -132,7 +140,7 @@ export default class SettingsStore { | |||
|      * if the change in value is worthwhile enough to react upon. | ||||
|      * @returns {string} A reference to the watcher that was employed. | ||||
|      */ | ||||
|     static watchSetting(settingName, roomId, callbackFn) { | ||||
|     public static watchSetting(settingName: string, roomId: string, callbackFn: CallbackFn): string { | ||||
|         const setting = SETTINGS[settingName]; | ||||
|         const originalSettingName = settingName; | ||||
|         if (!setting) throw new Error(`${settingName} is not a setting`); | ||||
|  | @ -141,14 +149,14 @@ export default class SettingsStore { | |||
|             settingName = setting.invertedSettingName; | ||||
|         } | ||||
| 
 | ||||
|         const watcherId = `${new Date().getTime()}_${SettingsStore._watcherCount++}_${settingName}_${roomId}`; | ||||
|         const watcherId = `${new Date().getTime()}_${SettingsStore.watcherCount++}_${settingName}_${roomId}`; | ||||
| 
 | ||||
|         const localizedCallback = (changedInRoomId, atLevel, newValAtLevel) => { | ||||
|             const newValue = SettingsStore.getValue(originalSettingName); | ||||
|             callbackFn(originalSettingName, changedInRoomId, atLevel, newValAtLevel, newValue); | ||||
|         }; | ||||
| 
 | ||||
|         SettingsStore._watchers[watcherId] = localizedCallback; | ||||
|         SettingsStore.watchers[watcherId] = localizedCallback; | ||||
|         defaultWatchManager.watchSetting(settingName, roomId, localizedCallback); | ||||
| 
 | ||||
|         return watcherId; | ||||
|  | @ -160,14 +168,14 @@ export default class SettingsStore { | |||
|      * @param {string} watcherReference The watcher reference (received from #watchSetting) | ||||
|      * to cancel. | ||||
|      */ | ||||
|     static unwatchSetting(watcherReference) { | ||||
|         if (!SettingsStore._watchers[watcherReference]) { | ||||
|     public static unwatchSetting(watcherReference: string) { | ||||
|         if (!SettingsStore.watchers[watcherReference]) { | ||||
|             console.warn(`Ending non-existent watcher ID ${watcherReference}`); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         defaultWatchManager.unwatchSetting(SettingsStore._watchers[watcherReference]); | ||||
|         delete SettingsStore._watchers[watcherReference]; | ||||
|         defaultWatchManager.unwatchSetting(SettingsStore.watchers[watcherReference]); | ||||
|         delete SettingsStore.watchers[watcherReference]; | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|  | @ -178,13 +186,13 @@ export default class SettingsStore { | |||
|      * @param {string} settingName The setting name to monitor. | ||||
|      * @param {String} roomId The room ID to monitor for changes in. Use null for all rooms. | ||||
|      */ | ||||
|     static monitorSetting(settingName, roomId) { | ||||
|     public static monitorSetting(settingName: string, roomId: string) { | ||||
|         roomId = roomId || null; // the thing wants null specifically to work, so appease it.
 | ||||
| 
 | ||||
|         if (!this._monitors[settingName]) this._monitors[settingName] = {}; | ||||
|         if (!this.monitors[settingName]) this.monitors[settingName] = {}; | ||||
| 
 | ||||
|         const registerWatcher = () => { | ||||
|             this._monitors[settingName][roomId] = SettingsStore.watchSetting( | ||||
|             this.monitors[settingName][roomId] = SettingsStore.watchSetting( | ||||
|                 settingName, roomId, (settingName, inRoomId, level, newValueAtLevel, newValue) => { | ||||
|                     dis.dispatch({ | ||||
|                         action: 'setting_updated', | ||||
|  | @ -198,16 +206,16 @@ export default class SettingsStore { | |||
|             ); | ||||
|         }; | ||||
| 
 | ||||
|         const hasRoom = Object.keys(this._monitors[settingName]).find((r) => r === roomId || r === null); | ||||
|         const hasRoom = Object.keys(this.monitors[settingName]).find((r) => r === roomId || r === null); | ||||
|         if (!hasRoom) { | ||||
|             registerWatcher(); | ||||
|         } else { | ||||
|             if (roomId === null) { | ||||
|                 // Unregister all existing watchers and register the new one
 | ||||
|                 for (const roomId of Object.keys(this._monitors[settingName])) { | ||||
|                     SettingsStore.unwatchSetting(this._monitors[settingName][roomId]); | ||||
|                 for (const roomId of Object.keys(this.monitors[settingName])) { | ||||
|                     SettingsStore.unwatchSetting(this.monitors[settingName][roomId]); | ||||
|                 } | ||||
|                 this._monitors[settingName] = {}; | ||||
|                 this.monitors[settingName] = {}; | ||||
|                 registerWatcher(); | ||||
|             } // else a watcher is already registered for the room, so don't bother registering it again
 | ||||
|         } | ||||
|  | @ -216,11 +224,11 @@ export default class SettingsStore { | |||
|     /** | ||||
|      * Gets the translated display name for a given setting | ||||
|      * @param {string} settingName The setting to look up. | ||||
|      * @param {"device"|"room-device"|"room-account"|"account"|"room"|"config"|"default"} atLevel | ||||
|      * @param {SettingLevel} atLevel | ||||
|      * The level to get the display name for; Defaults to 'default'. | ||||
|      * @return {String} The display name for the setting, or null if not found. | ||||
|      */ | ||||
|     static getDisplayName(settingName, atLevel = "default") { | ||||
|     public static getDisplayName(settingName: string, atLevel = SettingLevel.DEFAULT) { | ||||
|         if (!SETTINGS[settingName] || !SETTINGS[settingName].displayName) return null; | ||||
| 
 | ||||
|         let displayName = SETTINGS[settingName].displayName; | ||||
|  | @ -229,20 +237,20 @@ export default class SettingsStore { | |||
|             else displayName = displayName["default"]; | ||||
|         } | ||||
| 
 | ||||
|         return _t(displayName); | ||||
|         return _t(displayName as string); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Returns a list of all available labs feature names | ||||
|      * @returns {string[]} The list of available feature names | ||||
|      */ | ||||
|     static getLabsFeatures() { | ||||
|     public static getLabsFeatures(): string[] { | ||||
|         const possibleFeatures = Object.keys(SETTINGS).filter((s) => SettingsStore.isFeature(s)); | ||||
| 
 | ||||
|         const enableLabs = SdkConfig.get()["enableLabs"]; | ||||
|         if (enableLabs) return possibleFeatures; | ||||
| 
 | ||||
|         return possibleFeatures.filter((s) => SettingsStore._getFeatureState(s) === "labs"); | ||||
|         return possibleFeatures.filter((s) => SettingsStore.getFeatureState(s) === "labs"); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|  | @ -250,7 +258,7 @@ export default class SettingsStore { | |||
|      * @param {string} settingName The setting to look up. | ||||
|      * @return {boolean} True if the setting is a feature. | ||||
|      */ | ||||
|     static isFeature(settingName) { | ||||
|     public static isFeature(settingName: string) { | ||||
|         if (!SETTINGS[settingName]) return false; | ||||
|         return SETTINGS[settingName].isFeature; | ||||
|     } | ||||
|  | @ -262,7 +270,7 @@ export default class SettingsStore { | |||
|      * @param {String} roomId The optional room ID to validate in, may be null. | ||||
|      * @return {boolean} True if the feature is enabled, false otherwise | ||||
|      */ | ||||
|     static isFeatureEnabled(settingName, roomId = null) { | ||||
|     public static isFeatureEnabled(settingName: string, roomId: string = null) { | ||||
|         if (!SettingsStore.isFeature(settingName)) { | ||||
|             throw new Error("Setting " + settingName + " is not a feature"); | ||||
|         } | ||||
|  | @ -276,7 +284,7 @@ export default class SettingsStore { | |||
|      * @param {boolean} value True to enable the feature, false otherwise. | ||||
|      * @returns {Promise} Resolves when the setting has been set. | ||||
|      */ | ||||
|     static setFeatureEnabled(settingName, value) { | ||||
|     public static setFeatureEnabled(settingName: string, value: any): Promise<void> { | ||||
|         // Verify that the setting is actually a setting
 | ||||
|         if (!SETTINGS[settingName]) { | ||||
|             throw new Error("Setting '" + settingName + "' does not appear to be a setting."); | ||||
|  | @ -285,7 +293,7 @@ export default class SettingsStore { | |||
|             throw new Error("Setting " + settingName + " is not a feature"); | ||||
|         } | ||||
| 
 | ||||
|         return SettingsStore.setValue(settingName, null, "device", value); | ||||
|         return SettingsStore.setValue(settingName, null, SettingLevel.DEVICE, value); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|  | @ -296,7 +304,7 @@ export default class SettingsStore { | |||
|      * @param {boolean} excludeDefault True to disable using the default value. | ||||
|      * @return {*} The value, or null if not found | ||||
|      */ | ||||
|     static getValue(settingName, roomId = null, excludeDefault = false) { | ||||
|     public static getValue(settingName: string, roomId: string = null, excludeDefault = false): any { | ||||
|         // Verify that the setting is actually a setting
 | ||||
|         if (!SETTINGS[settingName]) { | ||||
|             throw new Error("Setting '" + settingName + "' does not appear to be a setting."); | ||||
|  | @ -310,7 +318,7 @@ export default class SettingsStore { | |||
| 
 | ||||
|     /** | ||||
|      * Gets a setting's value at a particular level, ignoring all levels that are more specific. | ||||
|      * @param {"device"|"room-device"|"room-account"|"account"|"room"|"config"|"default"} level The | ||||
|      * @param {SettingLevel|"config"|"default"} level The | ||||
|      * level to look at. | ||||
|      * @param {string} settingName The name of the setting to read. | ||||
|      * @param {String} roomId The room ID to read the setting value in, may be null. | ||||
|  | @ -319,7 +327,13 @@ export default class SettingsStore { | |||
|      * @param {boolean} excludeDefault True to disable using the default value. | ||||
|      * @return {*} The value, or null if not found. | ||||
|      */ | ||||
|     static getValueAt(level, settingName, roomId = null, explicit = false, excludeDefault = false) { | ||||
|     public static getValueAt( | ||||
|         level: SettingLevel, | ||||
|         settingName: string, | ||||
|         roomId: string = null, | ||||
|         explicit = false, | ||||
|         excludeDefault = false, | ||||
|     ): any { | ||||
|         // Verify that the setting is actually a setting
 | ||||
|         const setting = SETTINGS[settingName]; | ||||
|         if (!setting) { | ||||
|  | @ -327,19 +341,19 @@ export default class SettingsStore { | |||
|         } | ||||
| 
 | ||||
|         const levelOrder = (setting.supportedLevelsAreOrdered ? setting.supportedLevels : LEVEL_ORDER); | ||||
|         if (!levelOrder.includes("default")) levelOrder.push("default"); // always include default
 | ||||
|         if (!levelOrder.includes(SettingLevel.DEFAULT)) levelOrder.push(SettingLevel.DEFAULT); // always include default
 | ||||
| 
 | ||||
|         const minIndex = levelOrder.indexOf(level); | ||||
|         if (minIndex === -1) throw new Error("Level " + level + " is not prioritized"); | ||||
| 
 | ||||
|         if (SettingsStore.isFeature(settingName)) { | ||||
|             const configValue = SettingsStore._getFeatureState(settingName); | ||||
|             const configValue = SettingsStore.getFeatureState(settingName); | ||||
|             if (configValue === "enable") return true; | ||||
|             if (configValue === "disable") return false; | ||||
|             // else let it fall through the default process
 | ||||
|         } | ||||
| 
 | ||||
|         const handlers = SettingsStore._getHandlers(settingName); | ||||
|         const handlers = SettingsStore.getHandlers(settingName); | ||||
| 
 | ||||
|         // Check if we need to invert the setting at all. Do this after we get the setting
 | ||||
|         // handlers though, otherwise we'll fail to read the value.
 | ||||
|  | @ -351,10 +365,10 @@ export default class SettingsStore { | |||
|         if (explicit) { | ||||
|             const handler = handlers[level]; | ||||
|             if (!handler) { | ||||
|                 return SettingsStore._getFinalValue(setting, level, roomId, null, null); | ||||
|                 return SettingsStore.getFinalValue(setting, level, roomId, null, null); | ||||
|             } | ||||
|             const value = handler.getValue(settingName, roomId); | ||||
|             return SettingsStore._getFinalValue(setting, level, roomId, value, level); | ||||
|             return SettingsStore.getFinalValue(setting, level, roomId, value, level); | ||||
|         } | ||||
| 
 | ||||
|         for (let i = minIndex; i < levelOrder.length; i++) { | ||||
|  | @ -364,10 +378,10 @@ export default class SettingsStore { | |||
| 
 | ||||
|             const value = handler.getValue(settingName, roomId); | ||||
|             if (value === null || value === undefined) continue; | ||||
|             return SettingsStore._getFinalValue(setting, level, roomId, value, levelOrder[i]); | ||||
|             return SettingsStore.getFinalValue(setting, level, roomId, value, levelOrder[i]); | ||||
|         } | ||||
| 
 | ||||
|         return SettingsStore._getFinalValue(setting, level, roomId, null, null); | ||||
|         return SettingsStore.getFinalValue(setting, level, roomId, null, null); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|  | @ -376,7 +390,7 @@ export default class SettingsStore { | |||
|      * @param {String} roomId The room ID to read the setting value in, may be null. | ||||
|      * @return {*} The default value | ||||
|      */ | ||||
|     static getDefaultValue(settingName) { | ||||
|     public static getDefaultValue(settingName: string): any { | ||||
|         // Verify that the setting is actually a setting
 | ||||
|         if (!SETTINGS[settingName]) { | ||||
|             throw new Error("Setting '" + settingName + "' does not appear to be a setting."); | ||||
|  | @ -385,7 +399,13 @@ export default class SettingsStore { | |||
|         return SETTINGS[settingName].default; | ||||
|     } | ||||
| 
 | ||||
|     static _getFinalValue(setting, level, roomId, calculatedValue, calculatedAtLevel) { | ||||
|     private static getFinalValue( | ||||
|         setting: ISetting, | ||||
|         level: SettingLevel, | ||||
|         roomId: string, | ||||
|         calculatedValue: any, | ||||
|         calculatedAtLevel: SettingLevel, | ||||
|     ): any { | ||||
|         let resultingValue = calculatedValue; | ||||
| 
 | ||||
|         if (setting.controller) { | ||||
|  | @ -404,20 +424,21 @@ export default class SettingsStore { | |||
|      * to indicate that the level should no longer have an override. | ||||
|      * @param {string} settingName The name of the setting to change. | ||||
|      * @param {String} roomId The room ID to change the value in, may be null. | ||||
|      * @param {"device"|"room-device"|"room-account"|"account"|"room"} level The level | ||||
|      * @param {SettingLevel} level The level | ||||
|      * to change the value at. | ||||
|      * @param {*} value The new value of the setting, may be null. | ||||
|      * @return {Promise} Resolves when the setting has been changed. | ||||
|      */ | ||||
| 
 | ||||
|     /* eslint-enable valid-jsdoc */ | ||||
|     static async setValue(settingName, roomId, level, value) { | ||||
|     public static async setValue(settingName: string, roomId: string, level: SettingLevel, value: any): Promise<void> { | ||||
|         // Verify that the setting is actually a setting
 | ||||
|         const setting = SETTINGS[settingName]; | ||||
|         if (!setting) { | ||||
|             throw new Error("Setting '" + settingName + "' does not appear to be a setting."); | ||||
|         } | ||||
| 
 | ||||
|         const handler = SettingsStore._getHandler(settingName, level); | ||||
|         const handler = SettingsStore.getHandler(settingName, level); | ||||
|         if (!handler) { | ||||
|             throw new Error("Setting " + settingName + " does not have a handler for " + level); | ||||
|         } | ||||
|  | @ -449,28 +470,28 @@ export default class SettingsStore { | |||
|      * set for a particular room, otherwise it should be supplied. | ||||
|      * @param {string} settingName The name of the setting to check. | ||||
|      * @param {String} roomId The room ID to check in, may be null. | ||||
|      * @param {"device"|"room-device"|"room-account"|"account"|"room"} level The level to | ||||
|      * @param {SettingLevel} level The level to | ||||
|      * check at. | ||||
|      * @return {boolean} True if the user may set the setting, false otherwise. | ||||
|      */ | ||||
|     static canSetValue(settingName, roomId, level) { | ||||
|     public static canSetValue(settingName: string, roomId: string, level: SettingLevel): boolean { | ||||
|         // Verify that the setting is actually a setting
 | ||||
|         if (!SETTINGS[settingName]) { | ||||
|             throw new Error("Setting '" + settingName + "' does not appear to be a setting."); | ||||
|         } | ||||
| 
 | ||||
|         const handler = SettingsStore._getHandler(settingName, level); | ||||
|         const handler = SettingsStore.getHandler(settingName, level); | ||||
|         if (!handler) return false; | ||||
|         return handler.canSetValue(settingName, roomId); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Determines if the given level is supported on this device. | ||||
|      * @param {"device"|"room-device"|"room-account"|"account"|"room"} level The level | ||||
|      * @param {SettingLevel} level The level | ||||
|      * to check the feasibility of. | ||||
|      * @return {boolean} True if the level is supported, false otherwise. | ||||
|      */ | ||||
|     static isLevelSupported(level) { | ||||
|     public static isLevelSupported(level: SettingLevel): boolean { | ||||
|         if (!LEVEL_HANDLERS[level]) return false; | ||||
|         return LEVEL_HANDLERS[level].isSupported(); | ||||
|     } | ||||
|  | @ -482,7 +503,7 @@ export default class SettingsStore { | |||
|      * @param {string} realSettingName The setting name to try and read. | ||||
|      * @param {string} roomId Optional room ID to test the setting in. | ||||
|      */ | ||||
|     static debugSetting(realSettingName, roomId) { | ||||
|     public static debugSetting(realSettingName: string, roomId: string) { | ||||
|         console.log(`--- DEBUG ${realSettingName}`); | ||||
| 
 | ||||
|         // Note: we intentionally use JSON.stringify here to avoid the console masking the
 | ||||
|  | @ -570,13 +591,13 @@ export default class SettingsStore { | |||
|         console.log(`--- END DEBUG`); | ||||
|     } | ||||
| 
 | ||||
|     static _getHandler(settingName, level) { | ||||
|         const handlers = SettingsStore._getHandlers(settingName); | ||||
|     private static getHandler(settingName: string, level: SettingLevel): SettingsHandler { | ||||
|         const handlers = SettingsStore.getHandlers(settingName); | ||||
|         if (!handlers[level]) return null; | ||||
|         return handlers[level]; | ||||
|     } | ||||
| 
 | ||||
|     static _getHandlers(settingName) { | ||||
|     private static getHandlers(settingName: string): IHandlerMap { | ||||
|         if (!SETTINGS[settingName]) return {}; | ||||
| 
 | ||||
|         const handlers = {}; | ||||
|  | @ -591,7 +612,7 @@ export default class SettingsStore { | |||
|         return handlers; | ||||
|     } | ||||
| 
 | ||||
|     static _getFeatureState(settingName) { | ||||
|     private static getFeatureState(settingName: string): LabsFeatureState { | ||||
|         const featuresConfig = SdkConfig.get()['features']; | ||||
|         const enableLabs = SdkConfig.get()['enableLabs']; // we'll honour the old flag
 | ||||
| 
 | ||||
|  | @ -611,4 +632,4 @@ export default class SettingsStore { | |||
| } | ||||
| 
 | ||||
| // For debugging purposes
 | ||||
| global.mxSettingsStore = SettingsStore; | ||||
| window.mxSettingsStore = SettingsStore; | ||||
|  | @ -14,41 +14,52 @@ See the License for the specific language governing permissions and | |||
| limitations under the License. | ||||
| */ | ||||
| 
 | ||||
| import { SettingLevel } from "./SettingLevel"; | ||||
| 
 | ||||
| export type CallbackFn = (changedInRoomId: string, atLevel: SettingLevel, newValAtLevel: any) => void; | ||||
| 
 | ||||
| const IRRELEVANT_ROOM = Symbol("any room"); | ||||
| 
 | ||||
| interface RoomWatcherMap { | ||||
|     // @ts-ignore - TS wants string-only keys but we know better - https://github.com/Microsoft/TypeScript/issues/1863
 | ||||
|     [roomId: string | symbol]: CallbackFn[]; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Generalized management class for dealing with watchers on a per-handler (per-level) | ||||
|  * basis without duplicating code. Handlers are expected to push updates through this | ||||
|  * class, which are then proxied outwards to any applicable watchers. | ||||
|  */ | ||||
| export class WatchManager { | ||||
|     _watchers = {}; // { settingName: { roomId: callbackFns[] } }
 | ||||
|     private watchers: {[settingName: string]: RoomWatcherMap} = {}; | ||||
| 
 | ||||
|     // Proxy for handlers to delegate changes to this manager
 | ||||
|     watchSetting(settingName, roomId, cb) { | ||||
|         if (!this._watchers[settingName]) this._watchers[settingName] = {}; | ||||
|         if (!this._watchers[settingName][roomId]) this._watchers[settingName][roomId] = []; | ||||
|         this._watchers[settingName][roomId].push(cb); | ||||
|     public watchSetting(settingName: string, roomId: string | null, cb: CallbackFn) { | ||||
|         if (!this.watchers[settingName]) this.watchers[settingName] = {}; | ||||
|         if (!this.watchers[settingName][roomId]) this.watchers[settingName][roomId] = []; | ||||
|         this.watchers[settingName][roomId].push(cb); | ||||
|     } | ||||
| 
 | ||||
|     // Proxy for handlers to delegate changes to this manager
 | ||||
|     unwatchSetting(cb) { | ||||
|         for (const settingName of Object.keys(this._watchers)) { | ||||
|             for (const roomId of Object.keys(this._watchers[settingName])) { | ||||
|     public unwatchSetting(cb: CallbackFn) { | ||||
|         for (const settingName of Object.keys(this.watchers)) { | ||||
|             for (const roomId of Object.keys(this.watchers[settingName])) { | ||||
|                 let idx; | ||||
|                 while ((idx = this._watchers[settingName][roomId].indexOf(cb)) !== -1) { | ||||
|                     this._watchers[settingName][roomId].splice(idx, 1); | ||||
|                 while ((idx = this.watchers[settingName][roomId].indexOf(cb)) !== -1) { | ||||
|                     this.watchers[settingName][roomId].splice(idx, 1); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     notifyUpdate(settingName, inRoomId, atLevel, newValueAtLevel) { | ||||
|     public notifyUpdate(settingName: string, inRoomId: string | null, atLevel: SettingLevel, newValueAtLevel: any) { | ||||
|         // Dev note: We could avoid raising changes for ultimately inconsequential changes, but
 | ||||
|         // we also don't have a reliable way to get the old value of a setting. Instead, we'll just
 | ||||
|         // let it fall through regardless and let the receiver dedupe if they want to.
 | ||||
| 
 | ||||
|         if (!this._watchers[settingName]) return; | ||||
|         if (!this.watchers[settingName]) return; | ||||
| 
 | ||||
|         const roomWatchers = this._watchers[settingName]; | ||||
|         const roomWatchers = this.watchers[settingName]; | ||||
|         const callbacks = []; | ||||
| 
 | ||||
|         if (inRoomId !== null && roomWatchers[inRoomId]) { | ||||
|  | @ -59,8 +70,8 @@ export class WatchManager { | |||
|             // Fire updates to all the individual room watchers too, as they probably
 | ||||
|             // care about the change higher up.
 | ||||
|             callbacks.push(...Object.values(roomWatchers).reduce((r, a) => [...r, ...a], [])); | ||||
|         } else if (roomWatchers[null]) { | ||||
|             callbacks.push(...roomWatchers[null]); | ||||
|         } else if (roomWatchers[IRRELEVANT_ROOM]) { | ||||
|             callbacks.push(...roomWatchers[IRRELEVANT_ROOM]); | ||||
|         } | ||||
| 
 | ||||
|         for (const callback of callbacks) { | ||||
|  | @ -1,5 +1,5 @@ | |||
| /* | ||||
| Copyright 2019 New Vector Ltd | ||||
| Copyright 2019, 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
|  | @ -16,9 +16,10 @@ limitations under the License. | |||
| 
 | ||||
| import SettingController from "./SettingController"; | ||||
| import dis from "../../dispatcher/dispatcher"; | ||||
| import { SettingLevel } from "../SettingLevel"; | ||||
| 
 | ||||
| export default class CustomStatusController extends SettingController { | ||||
|     onChange(level, roomId, newValue) { | ||||
|     public onChange(level: SettingLevel, roomId: string, newValue: any) { | ||||
|         // Dispatch setting change so that some components that are still visible when the
 | ||||
|         // Settings page is open (such as RoomTiles) can reflect the change.
 | ||||
|         dis.dispatch({ | ||||
|  | @ -18,13 +18,14 @@ import SettingController from "./SettingController"; | |||
| import dis from "../../dispatcher/dispatcher"; | ||||
| import { UpdateFontSizePayload } from "../../dispatcher/payloads/UpdateFontSizePayload"; | ||||
| import { Action } from "../../dispatcher/actions"; | ||||
| import { SettingLevel } from "../SettingLevel"; | ||||
| 
 | ||||
| export default class FontSizeController extends SettingController { | ||||
|     constructor() { | ||||
|         super(); | ||||
|     } | ||||
| 
 | ||||
|     onChange(level, roomId, newValue) { | ||||
|     public onChange(level: SettingLevel, roomId: string, newValue: any) { | ||||
|         // Dispatch font size change so that everything open responds to the change.
 | ||||
|         dis.dispatch<UpdateFontSizePayload>({ | ||||
|             action: Action.UpdateFontSize, | ||||
|  |  | |||
|  | @ -1,5 +1,6 @@ | |||
| /* | ||||
| Copyright 2017 Travis Ralston | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
|  | @ -16,13 +17,14 @@ limitations under the License. | |||
| 
 | ||||
| import SettingController from "./SettingController"; | ||||
| import {MatrixClientPeg} from '../../MatrixClientPeg'; | ||||
| import { SettingLevel } from "../SettingLevel"; | ||||
| 
 | ||||
| // XXX: This feels wrong.
 | ||||
| import {PushProcessor} from "matrix-js-sdk/src/pushprocessor"; | ||||
| 
 | ||||
| // .m.rule.master being enabled means all events match that push rule
 | ||||
| // default action on this rule is dont_notify, but it could be something else
 | ||||
| function isPushNotifyDisabled() { | ||||
| function isPushNotifyDisabled(): boolean { | ||||
|     // Return the value of the master push rule as a default
 | ||||
|     const processor = new PushProcessor(MatrixClientPeg.get()); | ||||
|     const masterRule = processor.getPushRuleById(".m.rule.master"); | ||||
|  | @ -36,14 +38,20 @@ function isPushNotifyDisabled() { | |||
|     return masterRule.enabled && !masterRule.actions.includes("notify"); | ||||
| } | ||||
| 
 | ||||
| function getNotifier() { | ||||
| function getNotifier(): any { // TODO: [TS] Formal type that doesn't cause a cyclical reference.
 | ||||
|     // eslint-disable-next-line @typescript-eslint/no-var-requires
 | ||||
|     let Notifier = require('../../Notifier'); // avoids cyclical references
 | ||||
|     if (Notifier.default) Notifier = Notifier.default; // correct for webpack require() weirdness
 | ||||
|     return Notifier; | ||||
| } | ||||
| 
 | ||||
| export class NotificationsEnabledController extends SettingController { | ||||
|     getValueOverride(level, roomId, calculatedValue, calculatedAtLevel) { | ||||
|     public getValueOverride( | ||||
|         level: SettingLevel, | ||||
|         roomId: string, | ||||
|         calculatedValue: any, | ||||
|         calculatedAtLevel: SettingLevel, | ||||
|     ): any { | ||||
|         if (!getNotifier().isPossible()) return false; | ||||
| 
 | ||||
|         if (calculatedValue === null || calculatedAtLevel === "default") { | ||||
|  | @ -53,7 +61,7 @@ export class NotificationsEnabledController extends SettingController { | |||
|         return calculatedValue; | ||||
|     } | ||||
| 
 | ||||
|     onChange(level, roomId, newValue) { | ||||
|     public onChange(level: SettingLevel, roomId: string, newValue: any) { | ||||
|         if (getNotifier().supportsDesktopNotifications()) { | ||||
|             getNotifier().setEnabled(newValue); | ||||
|         } | ||||
|  | @ -61,7 +69,7 @@ export class NotificationsEnabledController extends SettingController { | |||
| } | ||||
| 
 | ||||
| export class NotificationBodyEnabledController extends SettingController { | ||||
|     getValueOverride(level, roomId, calculatedValue) { | ||||
|     public getValueOverride(level: SettingLevel, roomId: string, calculatedValue: any): any { | ||||
|         if (!getNotifier().isPossible()) return false; | ||||
| 
 | ||||
|         if (calculatedValue === null) { | ||||
|  | @ -73,7 +81,7 @@ export class NotificationBodyEnabledController extends SettingController { | |||
| } | ||||
| 
 | ||||
| export class AudioNotificationsEnabledController extends SettingController { | ||||
|     getValueOverride(level, roomId, calculatedValue) { | ||||
|     public getValueOverride(level: SettingLevel, roomId: string, calculatedValue: any): any { | ||||
|         if (!getNotifier().isPossible()) return false; | ||||
| 
 | ||||
|         // Note: Audio notifications are *not* enabled by default.
 | ||||
|  | @ -15,23 +15,20 @@ limitations under the License. | |||
| */ | ||||
| 
 | ||||
| import { MatrixClientPeg } from '../../MatrixClientPeg'; | ||||
| import { SettingLevel } from "../SettingLevel"; | ||||
| import SettingController from "./SettingController"; | ||||
| 
 | ||||
| /** | ||||
|  * When the value changes, call a setter function on the matrix client with the new value | ||||
|  */ | ||||
| export default class PushToMatrixClientController { | ||||
|     constructor(setter, inverse) { | ||||
|         this._setter = setter; | ||||
|         this._inverse = inverse; | ||||
| export default class PushToMatrixClientController extends SettingController { | ||||
|     constructor(private setter: Function, private inverse: boolean) { | ||||
|         super(); | ||||
|     } | ||||
| 
 | ||||
|     getValueOverride(level, roomId, calculatedValue, calculatedAtLevel) { | ||||
|         return null; // no override
 | ||||
|     } | ||||
| 
 | ||||
|     onChange(level, roomId, newValue) { | ||||
|     public onChange(level: SettingLevel, roomId: string, newValue: any) { | ||||
|         // XXX does this work? This surely isn't necessarily the effective value,
 | ||||
|         // but it's what NotificationsEnabledController does...
 | ||||
|         this._setter.call(MatrixClientPeg.get(), this._inverse ? !newValue : newValue); | ||||
|         this.setter.call(MatrixClientPeg.get(), this.inverse ? !newValue : newValue); | ||||
|     } | ||||
| } | ||||
|  | @ -1,5 +1,5 @@ | |||
| /* | ||||
| Copyright 2019 The Matrix.org Foundation C.I.C. | ||||
| Copyright 2019, 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
|  | @ -16,9 +16,10 @@ limitations under the License. | |||
| 
 | ||||
| import SettingController from "./SettingController"; | ||||
| import PlatformPeg from "../../PlatformPeg"; | ||||
| import { SettingLevel } from "../SettingLevel"; | ||||
| 
 | ||||
| export default class ReloadOnChangeController extends SettingController { | ||||
|     onChange(level, roomId, newValue) { | ||||
|     public onChange(level: SettingLevel, roomId: string, newValue: any) { | ||||
|         PlatformPeg.get().reload(); | ||||
|     } | ||||
| } | ||||
|  | @ -1,5 +1,6 @@ | |||
| /* | ||||
| Copyright 2017 Travis Ralston | ||||
| Copyright 2020 The Matrix.org Foundation C.I.C. | ||||
| 
 | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
|  | @ -14,6 +15,8 @@ See the License for the specific language governing permissions and | |||
| limitations under the License. | ||||
| */ | ||||
| 
 | ||||
| import { SettingLevel } from "../SettingLevel"; | ||||
| 
 | ||||
| /** | ||||
|  * Represents a controller for individual settings to alter the reading behaviour | ||||
|  * based upon environmental conditions, or to react to changes and therefore update | ||||
|  | @ -22,7 +25,7 @@ limitations under the License. | |||
|  * This is not intended to replace the functionality of a SettingsHandler, it is only | ||||
|  * intended to handle environmental factors for specific settings. | ||||
|  */ | ||||
| export default class SettingController { | ||||
| export default abstract class SettingController { | ||||
|     /** | ||||
|      * Gets the overridden value for the setting, if any. This must return null if the | ||||
|      * value is not to be overridden, otherwise it must return the new value. | ||||
|  | @ -30,11 +33,16 @@ export default class SettingController { | |||
|      * @param {String} roomId The room ID, may be null. | ||||
|      * @param {*} calculatedValue The value that the handlers think the setting should be, | ||||
|      * may be null. | ||||
|      * @param {string} calculatedAtLevel The level for which the calculated value was | ||||
|      * @param {SettingLevel} calculatedAtLevel The level for which the calculated value was | ||||
|      * calculated at. May be null. | ||||
|      * @return {*} The value that should be used, or null if no override is applicable. | ||||
|      */ | ||||
|     getValueOverride(level, roomId, calculatedValue, calculatedAtLevel) { | ||||
|     public getValueOverride( | ||||
|         level: SettingLevel, | ||||
|         roomId: string, | ||||
|         calculatedValue: any, | ||||
|         calculatedAtLevel: SettingLevel, | ||||
|     ): any { | ||||
|         return null; // no override
 | ||||
|     } | ||||
| 
 | ||||
|  | @ -44,7 +52,7 @@ export default class SettingController { | |||
|      * @param {String} roomId The room ID, may be null. | ||||
|      * @param {*} newValue The new value for the setting, may be null. | ||||
|      */ | ||||
|     onChange(level, roomId, newValue) { | ||||
|     public onChange(level: SettingLevel, roomId: string, newValue: any) { | ||||
|         // do nothing by default
 | ||||
|     } | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue
	
	 J. Ryan Stinnett
						J. Ryan Stinnett